From 184b8403234d328dd337a9eb387db327cf254fb2 Mon Sep 17 00:00:00 2001 From: Chen Cen Date: Mon, 24 Apr 2023 18:18:35 -0700 Subject: [PATCH] Stripe SDK 23.7.0 --- CHANGELOG.md | 1028 ++++++ MIGRATING.md | 308 ++ Package.swift | 155 + README.md | 156 +- .../Stripe Tests-Debug.xcconfig | 8 + .../Stripe Tests-Release.xcconfig | 8 + .../BuildConfigurations/Stripe-Debug.xcconfig | 7 + .../Stripe-Release.xcconfig | 7 + Stripe/Project.swift | 145 + Stripe/Stripe.xcodeproj/project.pbxproj | 2581 +++++++++++++ .../contents.xcworkspacedata | 7 + .../xcshareddata/xcschemes/StripeiOS.xcscheme | 83 + .../xcschemes/StripeiOSTestHostApp.xcscheme | 89 + Stripe/StripeiOS/Info.plist | 24 + .../Cards/stp_card_form_amex_cvc@3x.png | Bin 0 -> 4550 bytes .../Images/Cards/stp_card_form_back@3x.png | Bin 0 -> 4810 bytes .../Images/Cards/stp_card_form_front@3x.png | Bin 0 -> 3414 bytes .../Images/FPX/stp_bank_fpx_affin_bank@3x.png | Bin 0 -> 2603 bytes .../FPX/stp_bank_fpx_alliance_bank@3x.png | Bin 0 -> 858 bytes .../Images/FPX/stp_bank_fpx_ambank@3x.png | Bin 0 -> 1351 bytes .../Images/FPX/stp_bank_fpx_bank_islam@3x.png | Bin 0 -> 1214 bytes .../FPX/stp_bank_fpx_bank_muamalat@3x.png | Bin 0 -> 2207 bytes .../FPX/stp_bank_fpx_bank_rakyat@3x.png | Bin 0 -> 1297 bytes .../Images/FPX/stp_bank_fpx_bsn@3x.png | Bin 0 -> 2235 bytes .../Images/FPX/stp_bank_fpx_cimb@3x.png | Bin 0 -> 358 bytes .../FPX/stp_bank_fpx_hong_leong_bank@3x.png | Bin 0 -> 2647 bytes .../Images/FPX/stp_bank_fpx_hsbc@3x.png | Bin 0 -> 922 bytes .../Images/FPX/stp_bank_fpx_kfh@3x.png | Bin 0 -> 2634 bytes .../Images/FPX/stp_bank_fpx_maybank2e@3x.png | Bin 0 -> 4570 bytes .../Images/FPX/stp_bank_fpx_maybank2u@3x.png | Bin 0 -> 4570 bytes .../Images/FPX/stp_bank_fpx_ocbc@3x.png | Bin 0 -> 2054 bytes .../FPX/stp_bank_fpx_public_bank@3x.png | Bin 0 -> 1050 bytes .../Images/FPX/stp_bank_fpx_rhb@3x.png | Bin 0 -> 1566 bytes .../stp_bank_fpx_standard_chartered@3x.png | Bin 0 -> 2596 bytes .../Images/FPX/stp_bank_fpx_uob@3x.png | Bin 0 -> 603 bytes .../Images/FPX/stp_fpx_big_logo@3x.png | Bin 0 -> 12646 bytes .../Resources/Images/FPX/stp_fpx_logo@3x.png | Bin 0 -> 1445 bytes .../Resources/Images/stp_icon_add@3x.png | Bin 0 -> 114 bytes .../Resources/Images/stp_icon_bank@3x.png | Bin 0 -> 586 bytes .../Images/stp_icon_checkmark@3x.png | Bin 0 -> 653 bytes .../Resources/Images/stp_shipping_form@3x.png | Bin 0 -> 9438 bytes .../bg-BG.lproj/Localizable.strings | 39 + .../ca-ES.lproj/Localizable.strings | 39 + .../cs-CZ.lproj/Localizable.strings | 39 + .../da.lproj/Localizable.strings | 39 + .../de.lproj/Localizable.strings | 39 + .../el-GR.lproj/Localizable.strings | 39 + .../en-GB.lproj/Localizable.strings | 39 + .../en.lproj/Localizable.strings | 60 + .../es-419.lproj/Localizable.strings | 39 + .../es.lproj/Localizable.strings | 39 + .../et-EE.lproj/Localizable.strings | 39 + .../fi.lproj/Localizable.strings | 39 + .../fil.lproj/Localizable.strings | 39 + .../fr-CA.lproj/Localizable.strings | 39 + .../fr.lproj/Localizable.strings | 39 + .../hr.lproj/Localizable.strings | 37 + .../hu.lproj/Localizable.strings | 39 + .../id.lproj/Localizable.strings | 39 + .../it.lproj/Localizable.strings | 39 + .../ja.lproj/Localizable.strings | 39 + .../ko.lproj/Localizable.strings | 39 + .../lt-LT.lproj/Localizable.strings | 39 + .../lv-LV.lproj/Localizable.strings | 39 + .../ms-MY.lproj/Localizable.strings | 39 + .../mt.lproj/Localizable.strings | 39 + .../nb.lproj/Localizable.strings | 39 + .../nl.lproj/Localizable.strings | 39 + .../nn-NO.lproj/Localizable.strings | 39 + .../pl-PL.lproj/Localizable.strings | 39 + .../pt-BR.lproj/Localizable.strings | 39 + .../pt-PT.lproj/Localizable.strings | 39 + .../ro-RO.lproj/Localizable.strings | 39 + .../ru.lproj/Localizable.strings | 39 + .../sk-SK.lproj/Localizable.strings | 39 + .../sl-SI.lproj/Localizable.strings | 39 + .../sv.lproj/Localizable.strings | 39 + .../tk.lproj/Localizable.strings | 0 .../tr.lproj/Localizable.strings | 39 + .../vi.lproj/Localizable.strings | 39 + .../zh-HK.lproj/Localizable.strings | 39 + .../zh-Hans.lproj/Localizable.strings | 39 + .../zh-Hant.lproj/Localizable.strings | 39 + .../Enums+CustomStringConvertible.swift | 63 + ...PKAddPaymentPassRequest+Stripe_Error.swift | 30 + ...rizationViewController+Stripe_Blocks.swift | 187 + .../Source/STPAPIClient+BasicUI.swift | 178 + .../STPAPIClient+PushProvisioning.swift | 40 + .../Source/STPAddCardViewController.swift | 972 +++++ .../StripeiOS/Source/STPAddress+BasicUI.swift | 212 ++ .../Source/STPAddressFieldTableViewCell.swift | 509 +++ .../Source/STPAddressViewModel.swift | 434 +++ .../Source/STPAnalyticsClient+BasicUI.swift | 84 + .../Source/STPAnalyticsClient+Payments.swift | 28 + .../Source/STPApplePayContextDelegate.swift | 98 + .../Source/STPApplePayPaymentOption.swift | 51 + .../Source/STPBackendAPIAdapter.swift | 86 + .../STPBankSelectionTableViewCell.swift | 131 + .../STPBankSelectionViewController.swift | 300 ++ Stripe/StripeiOS/Source/STPBlocks.swift | 44 + Stripe/StripeiOS/Source/STPCameraView.swift | 77 + Stripe/StripeiOS/Source/STPCard+BasicUI.swift | 30 + Stripe/StripeiOS/Source/STPCardScanner.swift | 518 +++ .../Source/STPCardScannerTableViewCell.swift | 67 + .../Source/STPCardValidationState.swift | 9 + .../Source/STPCoreScrollViewController.swift | 62 + .../Source/STPCoreTableViewController.swift | 54 + .../Source/STPCoreViewController.swift | 162 + .../StripeiOS/Source/STPCustomerContext.swift | 419 +++ Stripe/StripeiOS/Source/STPEphemeralKey.swift | 104 + .../Source/STPEphemeralKeyManager.swift | 179 + .../Source/STPEphemeralKeyProvider.swift | 74 + .../Source/STPFPXBankStatusResponse.swift | 44 + .../STPFakeAddPaymentPassViewController.swift | 287 ++ Stripe/StripeiOS/Source/STPImageLibrary.swift | 94 + ...PIntentActionLinkAuthenticateAccount.swift | 33 + .../StripeiOS/Source/STPLocalizedString.swift | 16 + .../STPPaymentActivityIndicatorView.swift | 135 + .../Source/STPPaymentCardTextFieldCell.swift | 91 + .../Source/STPPaymentConfiguration.swift | 244 ++ .../StripeiOS/Source/STPPaymentContext.swift | 1206 ++++++ .../Source/STPPaymentContextAmountModel.swift | 96 + .../STPPaymentIntentParams+BasicUI.swift | 22 + .../Source/STPPaymentMethod+BasicUI.swift | 79 + .../STPPaymentMethodParams+BasicUI.swift | 48 + .../StripeiOS/Source/STPPaymentOption.swift | 36 + .../STPPaymentOptionTableViewCell.swift | 338 ++ .../Source/STPPaymentOptionTuple.swift | 88 + ...PaymentOptionsInternalViewController.swift | 584 +++ .../STPPaymentOptionsViewController.swift | 655 ++++ .../StripeiOS/Source/STPPaymentResult.swift | 43 + .../Source/STPPinManagementService.swift | 143 + .../Source/STPPushProvisioningContext.swift | 144 + .../Source/STPPushProvisioningDetails.swift | 97 + .../STPPushProvisioningDetailsParams.swift | 73 + .../Source/STPSectionHeaderView.swift | 161 + .../STPShippingAddressViewController.swift | 674 ++++ .../STPShippingMethodTableViewCell.swift | 147 + .../STPShippingMethodsViewController.swift | 211 ++ .../StripeiOS/Source/STPSource+BasicUI.swift | 74 + Stripe/StripeiOS/Source/STPTheme.swift | 260 ++ .../StripeiOS/Source/STPUserInformation.swift | 42 + .../StripeiOS/Source/String+Localized.swift | 22 + Stripe/StripeiOS/Source/Stripe+Exports.swift | 13 + .../Source/StripeBundleLocator.swift | 20 + .../Source/UIBarButtonItem+Stripe.swift | 54 + .../Source/UINavigationBar+Stripe_Theme.swift | 144 + ...vigationController+Stripe_Completion.swift | 61 + .../UITableViewCell+Stripe_Borders.swift | 103 + .../UIToolbar+Stripe_InputAccessory.swift | 38 + Stripe/StripeiOS/Source/UIView+Helpers.swift | 35 + .../Source/UIView+Stripe_FirstResponder.swift | 24 + .../Source/UIView+Stripe_SafeAreaBounds.swift | 24 + ...ewController+Stripe_KeyboardAvoiding.swift | 166 + ...ontroller+Stripe_NavigationItemProxy.swift | 38 + ...ntroller+Stripe_ParentViewController.swift | 50 + Stripe/StripeiOS/Stripe-umbrella.h | 11 + Stripe/StripeiOS/Stripe.modulemap | 6 + Stripe/StripeiOSAppHostedTests/Info.plist | 22 + .../LinkSecureCookieStoreTests.swift | 76 + Stripe/StripeiOSTestHostApp/AppDelegate.swift | 18 + Stripe/StripeiOSTestHostApp/Info.plist | 66 + .../AccentColor.colorset/Contents.json | 11 + .../AppIcon.appiconset/Contents.json | 98 + .../Resources/Assets.xcassets/Contents.json | 6 + .../Base.lproj/LaunchScreen.storyboard | 25 + .../Resources/Base.lproj/Main.storyboard | 24 + .../StripeiOSTestHostApp/ViewController.swift | 18 + Stripe/StripeiOSTests.xctestplan | 40 + Stripe/StripeiOSTests/APIRequestTest.swift | 287 ++ ...entMethodViewControllerSnapshotTests.swift | 44 + .../AddressViewControllerSnapshotTests.swift | 138 + ...erpayPriceBreakdownViewSnapshotTests.swift | 56 + .../StripeiOSTests/AnalyticsHelperTests.swift | 60 + ...oCompleteViewControllerSnapshotTests.swift | 150 + .../ButtonLinkSnapshotTests.swift | 79 + .../StripeiOSTests/CardExpiryDateTests.swift | 73 + .../CircularButtonSnapshotTests.swift | 66 + .../ConfirmButtonSnapshotTests.swift | 86 + .../StripeiOSTests/ConfirmButtonTests.swift | 91 + .../StripeiOSTests/ConsumerSessionTests.swift | 490 +++ .../Error+PaymentSheetTests.swift | 54 + ...hotTestCase+STPViewControllerLoading.swift | 79 + .../StripeiOSTests/FormSpecProviderTest.swift | 577 +++ .../FraudDetectionDataTest.swift | 31 + Stripe/StripeiOSTests/ImageTest.swift | 27 + Stripe/StripeiOSTests/Info.plist | 24 + Stripe/StripeiOSTests/KlarnaHelperTest.swift | 97 + .../LinkAccountServiceTests.swift | 42 + .../LinkBadgeViewSnapshotTest.swift | 59 + .../LinkCardEditElementSnapshotTests.swift | 75 + .../LinkInMemoryCookieStoreTests.swift | 81 + ...LinkInlineSignupElementSnapshotTests.swift | 123 + ...InstantDebitMandateViewSnapshotTests.swift | 78 + .../LinkLegalTermsViewSnapshotTests.swift | 94 + .../LinkNavigationBarSnapshotTests.swift | 89 + .../LinkNoticeViewSnapshotTests.swift | 51 + ...LinkPaymentMethodPickerSnapshotTests.swift | 108 + .../LinkSignupViewModelTests.swift | 179 + Stripe/StripeiOSTests/LinkStubs.swift | 99 + .../LinkToastSnapshotTests.swift | 41 + .../LinkVerificationViewSnapshotTests.swift | 84 + .../MKPlacemark+PaymentSheetTests.swift | 209 ++ .../StripeiOSTests/NSArray+StripeTest.swift | 67 + .../NSDecimalNumber+StripeTest.swift | 98 + .../NSDictionary+StripeTest.swift | 254 ++ Stripe/StripeiOSTests/NSLocale+STPSwizzling.h | 15 + Stripe/StripeiOSTests/NSLocale+STPSwizzling.m | 87 + .../StripeiOSTests/NSString+StripeTest.swift | 95 + .../NSURLComponents_StripeTest.swift | 40 + .../OneTimeCodeTextFieldSnapshotTests.swift | 46 + .../OneTimeCodeTextFieldTests.swift | 322 ++ .../OperationDebouncerTests.swift | 45 + .../StripeiOSTests/PKPayment+StripeTest.swift | 36 + .../PayWithLinkButtonSnapshotTests.swift | 111 + ...kViewController-WalletViewModelTests.swift | 144 + .../StripeiOSTests/PaymentAnalyticTest.swift | 33 + ...entMethodMessagingViewFunctionalTest.swift | 81 + ...mentMethodMessagingViewSnapshotTests.swift | 106 + .../StripeiOSTests/PaymentSheet+APITest.swift | 671 ++++ .../PaymentSheetAddressTests.swift | 312 ++ .../PaymentSheetFormFactorySnapshotTest.swift | 437 +++ .../PaymentSheetFormFactoryTest.swift | 1907 ++++++++++ .../PaymentSheetLinkAccountTests.swift | 82 + .../PaymentSheetPaymentMethodTypeTest.swift | 731 ++++ .../PaymentSheetTestUtils.swift | 53 + .../PaymentTypeCellSnapshotTests.swift | 81 + .../StripeiOSTests/Resources/3DSSource.json | 57 + .../Resources/AlipaySource.json | 34 + .../Resources/ApplePayPaymentMethod.json | 30 + .../Resources/BacsDebitPaymentMethod.json | 28 + .../Resources/BancontactSource.json | 38 + .../StripeiOSTests/Resources/BankAccount.json | 18 + Stripe/StripeiOSTests/Resources/Card.json | 26 + .../Resources/CardPaymentMethod.json | 61 + .../StripeiOSTests/Resources/CardSource.json | 33 + Stripe/StripeiOSTests/Resources/Customer.json | 26 + .../StripeiOSTests/Resources/EPSSource.json | 34 + .../Resources/ElementsSession.json | 199 + .../Resources/EphemeralKey.json | 14 + .../StripeiOSTests/Resources/FileUpload.json | 10 + .../Resources/GiropaySource.json | 36 + .../Resources/Images.xcassets/Contents.json | 6 + .../MockFiles/paymentIntentResponse.json | 53 + .../Resources/MultibancoSource.json | 42 + .../StripeiOSTests/Resources/P24Source.json | 33 + .../Resources/PaymentIntent.json | 82 + .../Resources/SEPADebitSource.json | 38 + .../StripeiOSTests/Resources/SetupIntent.json | 71 + .../Resources/SofortSource.json | 39 + .../Resources/WeChatPaySource.json | 40 + .../StripeiOSTests/Resources/iDEALSource.json | 32 + .../get_content_0.tail | 21 + ...-srv_assets_afterpay_logo_black.png_1.tail | 22 + ...cs-srv_assets_klarna_logo_black.png_2.tail | 23 + .../get_content_0.tail | 18 + .../post_v1_tokens_0.tail | 25 + .../post_v1_tokens_0.tail | 25 + .../post_v1_tokens_0.tail | 25 + .../post_create_payment_intent_0.tail | 18 + .../post_v1_3ds2_authenticate_2.tail | 47 + ...pi_3KgG1XFY0qyl6XeW1TLgmwD3_confirm_1.tail | 119 + .../post_v1_payment_methods_0.tail | 48 + .../get_v1_issuing_cards_ic_token_pin_0.tail | 86 + .../get_v1_issuing_cards_ic_token_pin_0.tail | 26 + .../post_v1_issuing_cards_ic_token_pin_0.tail | 88 + ...K91WoXa9u_push_provisioning_details_0.tail | 27 + .../Resources/stp_test_upload_image.jpeg | Bin 0 -> 3340 bytes .../RotatingCardBrandsViewSnapshotTests.swift | 38 + .../RotatingCardBrandsViewTests.swift | 42 + .../STPAPIClientNetworkBridgeTest.m | 374 ++ .../STPAPIClientStubbedTest.swift | 291 ++ Stripe/StripeiOSTests/STPAPIClientTest.swift | 148 + .../StripeiOSTests/STPAPISettingsBridgeTest.m | 86 + .../STPAUBECSDebitFormViewSnapshotTests.swift | 118 + .../STPAUBECSFormViewModelTests.swift | 584 +++ ...dCardViewControllerLocalizationTests.swift | 106 + .../STPAddCardViewControllerTest.swift | 292 ++ Stripe/StripeiOSTests/STPAddressTests.m | 525 +++ .../STPAddressViewModelTest.swift | 205 ++ .../STPAnalyticsClientPaymentSheetTest.swift | 268 ++ .../STPAnalyticsClientPaymentsTest.swift | 236 ++ .../STPApplePayContextFunctionalTest.m | 385 ++ ...PApplePayContextFunctionalTestExtras.swift | 44 + .../STPApplePayContextTest.swift | 163 + .../STPApplePayFunctionalTest.swift | 110 + .../STPApplePayPaymentOptionTest.m | 50 + Stripe/StripeiOSTests/STPApplePayTest.m | 62 + Stripe/StripeiOSTests/STPApplePayTest.swift | 34 + ...BECSDebitAccountNumberValidatorTests.swift | 240 ++ .../STPBSBNumberValidatorTests.swift | 92 + .../STPBankAccountFunctionalTest.m | 69 + .../StripeiOSTests/STPBankAccountParamsTest.m | 119 + Stripe/StripeiOSTests/STPBankAccountTest.m | 148 + Stripe/StripeiOSTests/STPBinRangeTest.swift | 190 + Stripe/StripeiOSTests/STPBlocks.h | 255 ++ .../STPCardBINMetadataTests.swift | 55 + Stripe/StripeiOSTests/STPCardBrandTest.m | 68 + ...PCardCVCInputTextFieldFormatterTests.swift | 52 + ...TPCardCVCInputTextFieldSnapshotTests.swift | 61 + .../STPCardCVCInputTextFieldTests.swift | 34 + ...PCardCVCInputTextFieldValidatorTests.swift | 54 + ...rdExpiryInputTextFieldFormatterTests.swift | 64 + ...ardExpiryInputTextFieldSnapshotTests.swift | 56 + ...rdExpiryInputTextFieldValidatorTests.swift | 131 + .../STPCardFormViewSnapshotTests.swift | 126 + .../StripeiOSTests/STPCardFormViewTests.swift | 243 ++ Stripe/StripeiOSTests/STPCardFunctionalTest.m | 156 + ...rdNumberInputTextFieldFormatterTests.swift | 73 + ...ardNumberInputTextFieldSnapshotTests.swift | 61 + ...rdNumberInputTextFieldValidatorTests.swift | 207 ++ Stripe/StripeiOSTests/STPCardParamsTest.m | 185 + Stripe/StripeiOSTests/STPCardTest.swift | 263 ++ .../StripeiOSTests/STPCardValidatorTest.swift | 471 +++ Stripe/StripeiOSTests/STPCertTest.swift | 68 + .../STPConfirmCardOptionsTest.m | 36 + .../STPConfirmPaymentMethodOptionsTest.m | 40 + .../STPConnectAccountAddressTest.m | 42 + .../STPConnectAccountFunctionalTest.m | 82 + .../STPConnectAccountParamsTest.m | 62 + ...CountryPickerInputFieldSnapshotTests.swift | 31 + .../STPCustomerContextTest.swift | 676 ++++ Stripe/StripeiOSTests/STPCustomerTest.m | 68 + Stripe/StripeiOSTests/STPE2ETest.swift | 155 + .../STPElementsSessionTest.swift | 54 + .../STPEphemeralKeyManagerTest.m | 185 + .../StripeiOSTests/STPEphemeralKeyTest.swift | 32 + Stripe/StripeiOSTests/STPErrorBridgeTest.m | 39 + Stripe/StripeiOSTests/STPFPXBankBrandTest.m | 131 + Stripe/StripeiOSTests/STPFileFunctionalTest.m | 89 + Stripe/StripeiOSTests/STPFileTest.m | 118 + Stripe/StripeiOSTests/STPFixtures+Swift.swift | 67 + Stripe/StripeiOSTests/STPFixtures.h | 225 ++ Stripe/StripeiOSTests/STPFixtures.m | 361 ++ ...ingPlaceholderTextFieldSnapshotTests.swift | 443 +++ .../StripeiOSTests/STPFormEncoderTest.swift | 210 ++ .../StripeiOSTests/STPFormTextFieldTest.swift | 62 + .../STPFormViewSnapshotTests.swift | 165 + ...GenericInputPickerFieldSnapshotTests.swift | 79 + ...GenericInputPickerFieldValidatorTest.swift | 45 + ...TPGenericInputTextFieldSnapshotTests.swift | 42 + .../StripeiOSTests/STPImageLibraryTest.swift | 371 ++ .../STPInputTextFieldFormatterTests.swift | 81 + .../STPInputTextFieldValidatorTests.swift | 64 + ...IntentActionAlipayHandleRedirectTest.swift | 55 + Stripe/StripeiOSTests/STPIntentActionTest.m | 93 + .../STPIntentActionTypeTest.swift | 48 + ...tentActionWeChatPayRedirectToAppTest.swift | 44 + .../STPIntentWithPreferencesTest.swift | 154 + ...STPLabeledFormTextFieldViewSnapshotTests.m | 36 + ...dMultiFormTextFieldViewSnapshotTests.swift | 44 + ...PMandateCustomerAcceptanceParamsTest.swift | 44 + .../STPMandateDataParamsTest.swift | 42 + .../STPMandateOnlineParamsTest.swift | 40 + Stripe/StripeiOSTests/STPMocks.h | 34 + Stripe/StripeiOSTests/STPMocks.m | 59 + .../STPNetworkStubbingTestCase.h | 24 + .../STPNetworkStubbingTestCase.m | 104 + .../STPNetworkStubbingTestCase.swift | 147 + ...PNumericDigitInputTextFormatterTests.swift | 157 + .../STPNumericStringValidatorTests.swift | 49 + Stripe/StripeiOSTests/STPPIIFunctionalTest.m | 51 + .../STPPaymentCardTextFieldTest.m | 1191 ++++++ .../STPPaymentCardTextFieldTestsSwift.swift | 67 + ...STPPaymentCardTextFieldViewModelTest.swift | 112 + .../STPPaymentConfigurationTest.m | 137 + .../STPPaymentContextApplePayTest.swift | 204 ++ .../STPPaymentContextSnapshotTests.m | 111 + .../STPPaymentHandlerFunctionalTest.m | 76 + ...aymentHandlerStubbedMockedFilesTests.swift | 628 ++++ .../STPPaymentHandlerTests.swift | 259 ++ .../STPPaymentIntentEnumsTest.swift | 184 + .../STPPaymentIntentFunctionalTest.m | 1058 ++++++ .../STPPaymentIntentFunctionalTest.swift | 152 + ...STPPaymentIntentLastPaymentErrorTest.swift | 60 + .../STPPaymentIntentParamsTest.swift | 218 ++ .../StripeiOSTests/STPPaymentIntentTest.swift | 190 + .../STPPaymentMethodAUBECSDebitParamsTests.m | 59 + .../STPPaymentMethodAUBECSDebitTests.swift | 84 + .../STPPaymentMethodAddressTest.m | 44 + .../STPPaymentMethodAffirmParamsTest.swift | 43 + .../STPPaymentMethodAffirmTests.swift | 46 + ...PPaymentMethodAfterpayClearpayParamsTest.m | 65 + .../STPPaymentMethodAfterpayClearpayTest.m | 44 + .../STPPaymentMethodBacsDebitTest.m | 46 + .../STPPaymentMethodBancontactParamsTests.m | 53 + .../STPPaymentMethodBancontactTests.swift | 44 + .../STPPaymentMethodBillingDetailsTest.m | 44 + ...aymentMethodBillingDetailsTests+Link.swift | 41 + .../STPPaymentMethodBoletoParamsTests.swift | 71 + .../STPPaymentMethodBoletoTests.swift | 53 + .../STPPaymentMethodCardChecksTest.m | 48 + .../STPPaymentMethodCardParamsTest.swift | 68 + .../STPPaymentMethodCardTest.swift | 125 + ...STPPaymentMethodCardWalletMasterpassTest.m | 31 + .../STPPaymentMethodCardWalletTest.m | 46 + ...PPaymentMethodCardWalletVisaCheckoutTest.m | 30 + .../STPPaymentMethodCashAppParamsTests.swift | 44 + .../STPPaymentMethodCashAppTests.swift | 40 + .../STPPaymentMethodEPSParamsTests.m | 55 + .../STPPaymentMethodEPSTests.swift | 43 + .../StripeiOSTests/STPPaymentMethodFPXTest.m | 42 + .../STPPaymentMethodFunctionalTest.m | 167 + .../STPPaymentMethodGiropayParamsTests.m | 53 + .../STPPaymentMethodGiropayTests.swift | 43 + .../STPPaymentMethodGrabPayParamsTest.m | 55 + .../STPPaymentMethodKlarnaParamsTests.swift | 71 + .../STPPaymentMethodKlarnaTests.swift | 46 + .../STPPaymentMethodNetBankingParamsTest.m | 49 + .../STPPaymentMethodNetBankingTests.swift | 44 + .../STPPaymentMethodOXXOParamsTests.m | 57 + .../STPPaymentMethodOXXOTests.m | 46 + .../STPPaymentMethodOptionsTest.swift | 131 + .../STPPaymentMethodParamsTest.m | 42 + .../STPPaymentMethodPayPalParamsTests.m | 53 + .../STPPaymentMethodPayPalTests.m | 45 + .../STPPaymentMethodPrzelewy24ParamsTests.m | 54 + .../STPPaymentMethodPrzelewy24Tests.swift | 43 + .../STPPaymentMethodSEPADebitTest.m | 50 + .../STPPaymentMethodSofortParamsTests.m | 55 + .../STPPaymentMethodSofortTests.swift | 43 + .../StripeiOSTests/STPPaymentMethodTest.swift | 161 + .../STPPaymentMethodThreeDSecureUsageTest.m | 31 + .../STPPaymentMethodUPIParamsTest.m | 55 + .../STPPaymentMethodUPITests.swift | 44 + ...MethodUSBankAccountParamsStubbedTest.swift | 302 ++ ...PaymentMethodUSBankAccountParamsTest.swift | 233 ++ .../STPPaymentMethodUSBankAccountTest.swift | 52 + .../STPPaymentMethodiDEALTest.m | 45 + ...tionsViewControllerLocalizationTests.swift | 106 + .../STPPaymentOptionsViewControllerTest.swift | 351 ++ .../STPPhoneNumberValidatorTest.swift | 137 + ...TPPinManagementServiceFunctionalTest.swift | 103 + ...stalCodeInputTextFieldFormatterTests.swift | 58 + ...ostalCodeInputTextFieldSnapshotTests.swift | 77 + .../STPPostalCodeInputTextFieldTests.swift | 69 + ...stalCodeInputTextFieldValidatorTests.swift | 79 + .../STPPostalCodeValidatorTest.swift | 103 + ...ushProvisioningDetailsFunctionalTest.swift | 56 + .../STPRadarSessionFunctionalTest.swift | 56 + .../StripeiOSTests/STPRedirectContextTest.m | 698 ++++ .../STPSTPViewWithSeparatorSnapshotTests.m | 38 + .../STPSetupIntentConfirmParamsTest.swift | 156 + .../STPSetupIntentFunctionalTest.m | 154 + .../STPSetupIntentFunctionalTest.swift | 126 + .../STPSetupIntentLastSetupErrorTest.m | 47 + Stripe/StripeiOSTests/STPSetupIntentTest.m | 108 + ...dressViewControllerLocalizationTests.swift | 116 + ...STPShippingAddressViewControllerTest.swift | 114 + ...thodsViewControllerLocalizationTests.swift | 80 + .../STPSourceCardDetailsTest.swift | 116 + .../StripeiOSTests/STPSourceFunctionalTest.m | 652 ++++ Stripe/StripeiOSTests/STPSourceOwnerTest.m | 63 + .../StripeiOSTests/STPSourceParamsTest.swift | 317 ++ Stripe/StripeiOSTests/STPSourceReceiverTest.m | 58 + Stripe/StripeiOSTests/STPSourceRedirectTest.m | 119 + .../STPSourceSEPADebitDetailsTest.m | 56 + Stripe/StripeiOSTests/STPSourceTest.m | 573 +++ .../STPSourceVerificationTest.m | 110 + ...PStackViewWithSeparatorSnapshotTests.swift | 218 ++ Stripe/StripeiOSTests/STPStringUtilsTest.m | 35 + .../StripeiOSTests/STPStringUtilsTest.swift | 95 + Stripe/StripeiOSTests/STPSwiftFixtures.swift | 87 + .../STPTestAPIClient+Swift.swift | 25 + Stripe/StripeiOSTests/STPTestUtils.h | 49 + Stripe/StripeiOSTests/STPTestUtils.m | 72 + Stripe/StripeiOSTests/STPTestingAPIClient.h | 69 + Stripe/StripeiOSTests/STPTestingAPIClient.m | 168 + .../STPTextFieldDelegateProxyTests.swift | 37 + .../STPThreeDSButtonCustomizationTest.swift | 40 + .../STPThreeDSFooterCustomizationTest.swift | 45 + .../STPThreeDSLabelCustomizationTest.swift | 37 + ...hreeDSNavigationBarCustomizationTest.swift | 48 + ...STPThreeDSSelectionCustomizationTest.swift | 42 + ...STPThreeDSTextFieldCustomizationTest.swift | 48 + .../STPThreeDSUICustomizationTest.swift | 137 + Stripe/StripeiOSTests/STPTokenTest.m | 58 + .../STPUIVCStripeParentViewControllerTests.m | 46 + Stripe/StripeiOSTests/SWHttpTrafficRecorder.h | 200 + Stripe/StripeiOSTests/SWHttpTrafficRecorder.m | 513 +++ .../ServerErrorMapperTest.swift | 94 + Stripe/StripeiOSTests/StripeErrorTest.swift | 254 ++ Stripe/StripeiOSTests/StripeTests-Prefix.pch | 21 + .../StripeiOS Tests-Bridging-Header.h | 9 + .../TextFieldElement+CardTest.swift | 366 ++ .../TextFieldElement+IBANTest.swift | 140 + .../UINavigationBar+StripeTest.m | 52 + .../UserDefaults+StripeTest.swift | 33 + .../WalletHeaderViewSnapshotTests.swift | 198 + .../Project-Debug.xcconfig | 15 + .../Project-Release.xcconfig | 12 + .../Project-Shared.xcconfig | 63 + .../Stripe3DS2-Debug.xcconfig | 12 + .../Stripe3DS2-Release.xcconfig | 12 + .../Stripe3DS2-Shared.xcconfig | 23 + .../Stripe3DS2DemoUI-Debug.xcconfig | 12 + .../Stripe3DS2DemoUI-Release.xcconfig | 12 + .../Stripe3DS2DemoUI-Shared.xcconfig | 13 + .../Stripe3DS2DemoUITests-Debug.xcconfig | 12 + .../Stripe3DS2DemoUITests-Release.xcconfig | 12 + .../Stripe3DS2DemoUITests-Shared.xcconfig | 13 + .../Stripe3DS2Tests-Debug.xcconfig | 12 + .../Stripe3DS2Tests-Release.xcconfig | 12 + .../Stripe3DS2Tests-Shared.xcconfig | 12 + Stripe3DS2/Project.swift | 126 + .../Stripe3DS2.xcodeproj/project.pbxproj | 1748 +++++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/Stripe3DS2.xcscheme | 86 + .../xcschemes/Stripe3DS2DemoUI.xcscheme | 107 + Stripe3DS2/Stripe3DS2/Info.plist | 22 + Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.h | 20 + Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.m | 35 + .../Stripe3DS2/NSDictionary+DecodingHelpers.h | 47 + .../Stripe3DS2/NSDictionary+DecodingHelpers.m | 147 + Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.h | 32 + Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.m | 38 + .../NSLayoutConstraint+LayoutSupport.h | 53 + .../NSLayoutConstraint+LayoutSupport.m | 30 + .../Stripe3DS2/NSString+EmptyChecking.h | 19 + .../Stripe3DS2/NSString+EmptyChecking.m | 25 + Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.h | 21 + Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.m | 54 + .../Resources/CertificateFiles/amex.der | Bin 0 -> 1237 bytes .../CertificateFiles/cartes-bancaires.der | Bin 0 -> 1212 bytes .../Resources/CertificateFiles/discover.der | Bin 0 -> 1118 bytes .../Resources/CertificateFiles/ec_test.der | Bin 0 -> 525 bytes .../Resources/CertificateFiles/mastercard.der | Bin 0 -> 1465 bytes .../Resources/CertificateFiles/ul-test.der | Bin 0 -> 1578 bytes .../Resources/CertificateFiles/visa.der | Bin 0 -> 1458 bytes .../Resources/Images/Chevron@3x.png | Bin 0 -> 454 bytes .../Resources/Images/amex-logo@3x.png | Bin 0 -> 4291 bytes .../Images/cartes-bancaires-logo.png | Bin 0 -> 14407 bytes .../Resources/Images/discover-logo.png | Bin 0 -> 3220 bytes .../Stripe3DS2/Resources/Images/error@3x.png | Bin 0 -> 1020 bytes .../Resources/Images/mastercard-logo@3x.png | Bin 0 -> 3860 bytes .../Resources/Images/visa-logo@3x.png | Bin 0 -> 3114 bytes .../Resources/Images/visa-white-logo@3x.png | Bin 0 -> 2088 bytes .../Resources/bg-BG.lproj/Localizable.strings | 45 + .../Resources/ca-ES.lproj/Localizable.strings | 45 + .../Resources/cs-CZ.lproj/Localizable.strings | 45 + .../Resources/da.lproj/Localizable.strings | 39 + .../Resources/de.lproj/Localizable.strings | 39 + .../Resources/el-GR.lproj/Localizable.strings | 45 + .../Resources/en-GB.lproj/Localizable.strings | 39 + .../Resources/en.lproj/Localizable.strings | 45 + .../es-419.lproj/Localizable.strings | 39 + .../Resources/es.lproj/Localizable.strings | 39 + .../Resources/et-EE.lproj/Localizable.strings | 45 + .../Resources/fi.lproj/Localizable.strings | 39 + .../Resources/fil.lproj/Localizable.strings | 45 + .../Resources/fr-CA.lproj/Localizable.strings | 39 + .../Resources/fr.lproj/Localizable.strings | 39 + .../Resources/hr.lproj/Localizable.strings | 45 + .../Resources/hu.lproj/Localizable.strings | 39 + .../Resources/id.lproj/Localizable.strings | 45 + .../Resources/it.lproj/Localizable.strings | 39 + .../Resources/ja.lproj/Localizable.strings | 39 + .../Resources/ko.lproj/Localizable.strings | 39 + .../Resources/lt-LT.lproj/Localizable.strings | 45 + .../Resources/lv-LV.lproj/Localizable.strings | 45 + .../Resources/ms-MY.lproj/Localizable.strings | 45 + .../Resources/mt.lproj/Localizable.strings | 39 + .../Resources/nb.lproj/Localizable.strings | 39 + .../Resources/nl.lproj/Localizable.strings | 39 + .../Resources/nn-NO.lproj/Localizable.strings | 39 + .../Resources/pl-PL.lproj/Localizable.strings | 45 + .../Resources/pt-BR.lproj/Localizable.strings | 39 + .../Resources/pt-PT.lproj/Localizable.strings | 39 + .../Resources/ro-RO.lproj/Localizable.strings | 45 + .../Resources/ru.lproj/Localizable.strings | 39 + .../Resources/sk-SK.lproj/Localizable.strings | 45 + .../Resources/sl-SI.lproj/Localizable.strings | 45 + .../Resources/sv.lproj/Localizable.strings | 39 + .../Resources/tr.lproj/Localizable.strings | 39 + .../Resources/vi.lproj/Localizable.strings | 45 + .../Resources/zh-HK.lproj/Localizable.strings | 39 + .../zh-Hans.lproj/Localizable.strings | 39 + .../zh-Hant.lproj/Localizable.strings | 39 + .../Stripe3DS2/STDSACSNetworkingManager.h | 30 + .../Stripe3DS2/STDSACSNetworkingManager.m | 169 + .../STDSAuthenticationResponseObject.h | 20 + .../STDSAuthenticationResponseObject.m | 100 + Stripe3DS2/Stripe3DS2/STDSBrandingView.h | 23 + Stripe3DS2/Stripe3DS2/STDSBrandingView.m | 132 + Stripe3DS2/Stripe3DS2/STDSBundleLocator.h | 15 + Stripe3DS2/Stripe3DS2/STDSBundleLocator.m | 109 + .../Stripe3DS2/STDSChallengeInformationView.h | 25 + .../Stripe3DS2/STDSChallengeInformationView.m | 137 + .../STDSChallengeRequestParameters.h | 138 + .../STDSChallengeRequestParameters.m | 105 + Stripe3DS2/Stripe3DS2/STDSChallengeResponse.h | 145 + .../Stripe3DS2/STDSChallengeResponseImage.h | 27 + .../STDSChallengeResponseImageObject.h | 23 + .../STDSChallengeResponseImageObject.m | 53 + .../STDSChallengeResponseMessageExtension.h | 30 + ...SChallengeResponseMessageExtensionObject.h | 21 + ...SChallengeResponseMessageExtensionObject.m | 72 + .../Stripe3DS2/STDSChallengeResponseObject.h | 49 + .../Stripe3DS2/STDSChallengeResponseObject.m | 321 ++ .../STDSChallengeResponseSelectionInfo.h | 24 + ...STDSChallengeResponseSelectionInfoObject.h | 21 + ...STDSChallengeResponseSelectionInfoObject.m | 46 + .../STDSChallengeResponseViewController.h | 80 + .../STDSChallengeResponseViewController.m | 571 +++ .../Stripe3DS2/STDSChallengeSelectionView.h | 35 + .../Stripe3DS2/STDSChallengeSelectionView.m | 255 ++ Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.h | 19 + Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.m | 54 + Stripe3DS2/Stripe3DS2/STDSDeviceInformation.h | 21 + Stripe3DS2/Stripe3DS2/STDSDeviceInformation.m | 30 + .../Stripe3DS2/STDSDeviceInformationManager.h | 23 + .../Stripe3DS2/STDSDeviceInformationManager.m | 65 + .../STDSDeviceInformationParameter+Private.h | 76 + .../STDSDeviceInformationParameter.h | 24 + .../STDSDeviceInformationParameter.m | 449 +++ Stripe3DS2/Stripe3DS2/STDSDirectoryServer.h | 134 + .../STDSDirectoryServerCertificate+Internal.h | 22 + .../STDSDirectoryServerCertificate.h | 43 + .../STDSDirectoryServerCertificate.m | 350 ++ .../Stripe3DS2/STDSEllipticCurvePoint.h | 26 + .../Stripe3DS2/STDSEllipticCurvePoint.m | 90 + .../Stripe3DS2/STDSEphemeralKeyPair+Testing.h | 19 + Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.h | 39 + Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.m | 108 + .../Stripe3DS2/STDSErrorMessage+Internal.h | 40 + .../Stripe3DS2/STDSErrorMessage+Internal.m | 91 + .../Stripe3DS2/STDSException+Internal.h | 19 + .../STDSExpandableInformationView.h | 23 + .../STDSExpandableInformationView.m | 150 + Stripe3DS2/Stripe3DS2/STDSIPAddress.h | 15 + Stripe3DS2/Stripe3DS2/STDSIPAddress.m | 43 + Stripe3DS2/Stripe3DS2/STDSImageLoader.h | 36 + Stripe3DS2/Stripe3DS2/STDSImageLoader.m | 50 + Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.h | 19 + Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.m | 41 + Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.h | 45 + Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.m | 407 +++ Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.h | 41 + Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.m | 88 + Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.h | 19 + Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.m | 31 + Stripe3DS2/Stripe3DS2/STDSLocalizedString.h | 18 + Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.h | 19 + Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.m | 25 + Stripe3DS2/Stripe3DS2/STDSProcessingView.h | 26 + Stripe3DS2/Stripe3DS2/STDSProcessingView.m | 98 + .../Stripe3DS2/STDSProgressViewController.h | 23 + .../Stripe3DS2/STDSProgressViewController.m | 55 + Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.h | 49 + Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.m | 366 ++ Stripe3DS2/Stripe3DS2/STDSSelectionButton.h | 26 + Stripe3DS2/Stripe3DS2/STDSSelectionButton.m | 167 + Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.h | 19 + Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.m | 25 + Stripe3DS2/Stripe3DS2/STDSSpacerView.h | 20 + Stripe3DS2/Stripe3DS2/STDSSpacerView.m | 34 + Stripe3DS2/Stripe3DS2/STDSStackView.h | 56 + Stripe3DS2/Stripe3DS2/STDSStackView.m | 164 + .../STDSSynchronousLocationManager.h | 26 + .../STDSSynchronousLocationManager.m | 110 + Stripe3DS2/Stripe3DS2/STDSTextChallengeView.h | 26 + Stripe3DS2/Stripe3DS2/STDSTextChallengeView.m | 133 + .../STDSThreeDSProtocolVersion+Private.h | 53 + .../Stripe3DS2/STDSThreeDSProtocolVersion.m | 15 + .../Stripe3DS2/STDSTransaction+Private.h | 41 + Stripe3DS2/Stripe3DS2/STDSWebView.h | 22 + Stripe3DS2/Stripe3DS2/STDSWebView.m | 37 + Stripe3DS2/Stripe3DS2/STDSWhitelistView.h | 25 + Stripe3DS2/Stripe3DS2/STDSWhitelistView.m | 103 + .../Stripe3DS2/Stripe3DS2-Bridging-Header.h | 12 + .../UIButton+CustomInitialization.h | 21 + .../UIButton+CustomInitialization.m | 64 + Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.h | 21 + Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.m | 31 + .../Stripe3DS2/UIColor+ThirteenSupport.h | 24 + .../Stripe3DS2/UIColor+ThirteenSupport.m | 53 + Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.h | 23 + Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.m | 41 + Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.h | 24 + Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.m | 35 + .../Stripe3DS2/UIViewController+Stripe3DS2.h | 21 + .../Stripe3DS2/UIViewController+Stripe3DS2.m | 49 + .../include/STDSAlreadyInitializedException.h | 22 + .../include/STDSAlreadyInitializedException.m | 17 + .../STDSAuthenticationRequestParameters.h | 68 + .../STDSAuthenticationRequestParameters.m | 63 + .../include/STDSAuthenticationResponse.h | 115 + .../include/STDSButtonCustomization.h | 83 + .../include/STDSButtonCustomization.m | 69 + .../include/STDSChallengeParameters.h | 61 + .../include/STDSChallengeParameters.m | 31 + .../include/STDSChallengeStatusReceiver.h | 67 + .../Stripe3DS2/include/STDSCompletionEvent.h | 40 + .../Stripe3DS2/include/STDSCompletionEvent.m | 26 + .../Stripe3DS2/include/STDSConfigParameters.h | 96 + .../Stripe3DS2/include/STDSConfigParameters.m | 113 + .../Stripe3DS2/include/STDSCustomization.h | 25 + .../Stripe3DS2/include/STDSCustomization.m | 25 + .../Stripe3DS2/include/STDSErrorMessage.h | 103 + .../Stripe3DS2/include/STDSErrorMessage.m | 94 + Stripe3DS2/Stripe3DS2/include/STDSException.h | 25 + Stripe3DS2/Stripe3DS2/include/STDSException.m | 27 + .../include/STDSFooterCustomization.h | 41 + .../include/STDSFooterCustomization.m | 45 + .../include/STDSInvalidInputException.h | 21 + .../include/STDSInvalidInputException.m | 17 + .../Stripe3DS2/include/STDSJSONDecodable.h | 33 + .../Stripe3DS2/include/STDSJSONEncodable.h | 22 + .../Stripe3DS2/include/STDSJSONEncoder.h | 27 + .../Stripe3DS2/include/STDSJSONEncoder.m | 51 + .../include/STDSLabelCustomization.h | 31 + .../include/STDSLabelCustomization.m | 43 + .../include/STDSNavigationBarCustomization.h | 59 + .../include/STDSNavigationBarCustomization.m | 44 + .../include/STDSNotInitializedException.h | 23 + .../include/STDSNotInitializedException.m | 17 + .../include/STDSProtocolErrorEvent.h | 42 + .../include/STDSProtocolErrorEvent.m | 28 + .../include/STDSRuntimeErrorEvent.h | 53 + .../include/STDSRuntimeErrorEvent.m | 37 + .../Stripe3DS2/include/STDSRuntimeException.h | 21 + .../Stripe3DS2/include/STDSRuntimeException.m | 17 + .../include/STDSSelectionCustomization.h | 48 + .../include/STDSSelectionCustomization.m | 77 + .../Stripe3DS2/include/STDSStripe3DS2Error.h | 68 + .../Stripe3DS2/include/STDSStripe3DS2Error.m | 15 + .../Stripe3DS2/include/STDSSwiftTryCatch.h | 50 + .../Stripe3DS2/include/STDSSwiftTryCatch.m | 59 + .../include/STDSTextFieldCustomization.h | 47 + .../include/STDSTextFieldCustomization.m | 50 + .../Stripe3DS2/include/STDSThreeDS2Service.h | 84 + .../Stripe3DS2/include/STDSThreeDS2Service.m | 191 + .../include/STDSThreeDSProtocolVersion.h | 15 + .../Stripe3DS2/include/STDSTransaction.h | 94 + .../Stripe3DS2/include/STDSTransaction.m | 531 +++ .../Stripe3DS2/include/STDSUICustomization.h | 109 + .../Stripe3DS2/include/STDSUICustomization.m | 83 + Stripe3DS2/Stripe3DS2/include/STDSWarning.h | 69 + Stripe3DS2/Stripe3DS2/include/STDSWarning.m | 30 + .../Stripe3DS2/include/Stripe3DS2-Prefix.pch | 14 + Stripe3DS2/Stripe3DS2/include/Stripe3DS2.h | 52 + Stripe3DS2/Stripe3DS2DemoUI/Info.plist | 45 + .../Resources/acs_challenge.html | 178 + .../Stripe3DS2DemoUI/Sources/AppDelegate.h | 15 + .../Stripe3DS2DemoUI/Sources/AppDelegate.m | 36 + .../STDSChallengeResponseObject+TestObjects.h | 23 + .../STDSChallengeResponseObject+TestObjects.m | 197 + .../Sources/STDSDemoViewController.h | 20 + .../Sources/STDSDemoViewController.m | 225 ++ Stripe3DS2/Stripe3DS2DemoUI/Sources/main.m | 16 + Stripe3DS2/Stripe3DS2DemoUITests/Info.plist | 24 + ...lengeResponseViewControllerSnapshotTests.m | 117 + Stripe3DS2/Stripe3DS2Resources/Info.plist | 26 + Stripe3DS2/Stripe3DS2Tests/Info.plist | 22 + Stripe3DS2/Stripe3DS2Tests/JSON/ARes.json | 16 + Stripe3DS2/Stripe3DS2Tests/JSON/CRes.json | 31 + .../Stripe3DS2Tests/JSON/ErrorMessage.json | 13 + .../NSDictionary+DecodingHelpersTest.m | 312 ++ .../NSString+EmptyCheckingTests.m | 32 + .../STDSACSNetworkingManagerTest.m | 54 + .../STDSAuthenticationRequestParametersTest.m | 41 + .../STDSAuthenticationResponseTests.m | 32 + .../STDSBase64URLEncodingTests.m | 59 + .../STDSChallengeParametersTests.m | 56 + .../STDSChallengeRequestParametersTest.m | 71 + .../STDSChallengeResponseObjectTest.m | 79 + .../STDSConfigParametersTests.m | 67 + .../STDSDeviceInformationManagerTests.m | 33 + .../STDSDeviceInformationParameterTests.m | 214 ++ .../STDSDirectoryServerCertificateTests.m | 811 ++++ .../STDSEllipticCurvePointTests.m | 42 + .../STDSEphemeralKeyPairTests.m | 41 + .../Stripe3DS2Tests/STDSErrorMessageTest.m | 61 + .../Stripe3DS2Tests/STDSJSONEncoderTest.m | 201 + .../STDSJSONWebEncryptionTests.m | 119 + .../STDSJSONWebSignatureTests.m | 78 + .../STDSSecTypeUtilitiesTests.m | 142 + .../STDSSynchronousLocationManagerTests.m | 28 + .../Stripe3DS2Tests/STDSTestJSONUtils.h | 19 + .../Stripe3DS2Tests/STDSTestJSONUtils.m | 54 + .../STDSThreeDS2ServiceTests.m | 62 + .../Stripe3DS2Tests/STDSTransactionTest.m | 157 + .../STDSUICustomizationTests.m | 249 ++ Stripe3DS2/Stripe3DS2Tests/STDSWarningTests.m | 26 + Stripe3DS2/exported_symbols.txt | 13 + StripeApplePay/Project.swift | 15 + StripeApplePay/README.md | 36 + .../StripeApplePay.xcodeproj/project.pbxproj | 610 ++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripeApplePay.xcscheme | 95 + StripeApplePay/StripeApplePay/Info.plist | 22 + .../STPAPIClient+ApplePay.swift | 44 + .../STPApplePayContext+LegacySupport.swift | 55 + .../ApplePayContext/STPApplePayContext.swift | 746 ++++ .../StripeApplePay/Source/Blocks.swift | 18 + .../Extensions/BillingDetails+ApplePay.swift | 101 + .../Source/Extensions/PKContact+Stripe.swift | 26 + .../Source/Extensions/PKPayment+Stripe.swift | 32 + .../PaymentsCore/API/Models/Address.swift | 43 + .../API/Models/BillingDetails.swift | 67 + .../PaymentsCore/API/Models/CardBrand.swift | 33 + .../API/Models/PaymentIntent.swift | 149 + .../API/Models/PaymentIntentParams.swift | 101 + .../API/Models/PaymentMethod.swift | 176 + .../API/Models/PaymentMethodParams.swift | 73 + .../PaymentsCore/API/Models/SetupIntent.swift | 58 + .../API/Models/SetupIntentParams.swift | 66 + .../API/Models/ShippingDetails.swift | 93 + .../PaymentsCore/API/Models/Token.swift | 129 + .../PaymentsCore/API/PaymentIntent+API.swift | 85 + .../PaymentsCore/API/PaymentMethod+API.swift | 61 + .../PaymentsCore/API/SetupIntent+API.swift | 84 + .../Source/PaymentsCore/API/Token+API.swift | 91 + .../STPAnalyticsClient+Payments.swift | 28 + .../STPAnalyticsClient+PaymentsAPI.swift | 72 + .../STPAPIClient+PaymentsCore.swift | 20 + .../Source/StripeCore+Import.swift | 10 + .../StripeApplePay/StripeApplePay.h | 18 + StripeApplePay/StripeApplePayTests/Info.plist | 22 + .../STPTelemetryClientFunctionalTest.swift | 67 + .../PaymentsCore/STPTelemetryClientTest.swift | 75 + .../STPAnalyticsClient+ApplePayTest.swift | 30 + .../STPPaymentMethodFunctionalTest.swift | 96 + .../TelemetryInjectionTest.swift | 95 + StripeCameraCore/Project.swift | 11 + .../project.pbxproj | 640 ++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripeCameraCore.xcscheme | 95 + StripeCameraCore/StripeCameraCore/Info.plist | 22 + .../Source/CameraExifMetadata.swift | 46 + .../Categories/CGRect+StripeCameraCore.swift | 80 + .../CVPixelBuffer+StripeCameraCore.swift | 18 + ...UIDeviceOrientation+StripeCameraCore.swift | 26 + .../Source/Categories/UIImage+Buffer.swift | 118 + .../Coordinators/AppSettingsHelper.swift | 44 + .../CameraPermissionsManager.swift | 85 + .../Source/Coordinators/CameraSession.swift | 514 +++ .../MockSimulatorCameraSession.swift | 220 ++ .../Source/Coordinators/Torch.swift | 53 + .../Source/Views/CameraPreviewView.swift | 81 + .../StripeCameraCore/StripeCameraCore.h | 18 + .../StripeCameraCoreTestUtils/Info.plist | 22 + .../Mocks/MockAppSettingsHelper.swift | 21 + .../Mocks/MockCameraPermissionsManager.swift | 41 + .../Mocks/MockTestCameraSession.swift | 180 + .../StripeCameraCoreTestUtils.h | 18 + .../StripeCameraCoreTests/Info.plist | 22 + .../CGRect_StripeCameraCoreTest.swift | 68 + .../StripeCardScan-Debug.xcconfig | 7 + .../StripeCardScan-Release.xcconfig | 7 + StripeCardScan/Project.swift | 32 + StripeCardScan/README.md | 87 + .../StripeCardScan.xcodeproj/project.pbxproj | 1212 ++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripeCardScan.xcscheme | 95 + StripeCardScan/StripeCardScan/Info.plist | 22 + .../SSDOcr.mlmodelc/analytics/coremldata.bin | Bin 0 -> 438 bytes .../SSDOcr.mlmodelc/coremldata.bin | Bin 0 -> 589 bytes .../SSDOcr.mlmodelc/metadata.json | 92 + .../SSDOcr.mlmodelc/model.espresso.net | 2489 +++++++++++++ .../SSDOcr.mlmodelc/model.espresso.shape | 661 ++++ .../SSDOcr.mlmodelc/model.espresso.weights | Bin 0 -> 1653056 bytes .../SSDOcr.mlmodelc/model/coremldata.bin | Bin 0 -> 168 bytes .../neural_network_optionals/coremldata.bin | Bin 0 -> 40 bytes .../UxModel.mlmodelc/analytics/coremldata.bin | Bin 0 -> 439 bytes .../UxModel.mlmodelc/coremldata.bin | Bin 0 -> 189 bytes .../UxModel.mlmodelc/metadata.json | 66 + .../UxModel.mlmodelc/model.espresso.net | 3252 +++++++++++++++++ .../UxModel.mlmodelc/model.espresso.shape | 1066 ++++++ .../UxModel.mlmodelc/model.espresso.weights | Bin 0 -> 1381120 bytes .../UxModel.mlmodelc/model/coremldata.bin | Bin 0 -> 208 bytes .../neural_network_optionals/coremldata.bin | Bin 0 -> 40 bytes .../bg-BG.lproj/Localizable.strings | 9 + .../ca-ES.lproj/Localizable.strings | 9 + .../cs-CZ.lproj/Localizable.strings | 9 + .../da.lproj/Localizable.strings | 9 + .../de.lproj/Localizable.strings | 9 + .../el-GR.lproj/Localizable.strings | 9 + .../en-GB.lproj/Localizable.strings | 9 + .../en.lproj/Localizable.strings | 15 + .../es-419.lproj/Localizable.strings | 9 + .../es.lproj/Localizable.strings | 9 + .../et-EE.lproj/Localizable.strings | 9 + .../fi.lproj/Localizable.strings | 9 + .../fil.lproj/Localizable.strings | 9 + .../fr-CA.lproj/Localizable.strings | 9 + .../fr.lproj/Localizable.strings | 9 + .../hr.lproj/Localizable.strings | 9 + .../hu.lproj/Localizable.strings | 9 + .../id.lproj/Localizable.strings | 9 + .../it.lproj/Localizable.strings | 9 + .../ja.lproj/Localizable.strings | 9 + .../ko.lproj/Localizable.strings | 9 + .../lt-LT.lproj/Localizable.strings | 9 + .../lv-LV.lproj/Localizable.strings | 9 + .../ms-MY.lproj/Localizable.strings | 9 + .../mt.lproj/Localizable.strings | 9 + .../nb.lproj/Localizable.strings | 9 + .../nl.lproj/Localizable.strings | 9 + .../nn-NO.lproj/Localizable.strings | 9 + .../pl-PL.lproj/Localizable.strings | 9 + .../pt-BR.lproj/Localizable.strings | 9 + .../pt-PT.lproj/Localizable.strings | 9 + .../ro-RO.lproj/Localizable.strings | 9 + .../ru.lproj/Localizable.strings | 9 + .../sk-SK.lproj/Localizable.strings | 9 + .../sl-SI.lproj/Localizable.strings | 9 + .../sv.lproj/Localizable.strings | 9 + .../tr.lproj/Localizable.strings | 9 + .../vi.lproj/Localizable.strings | 9 + .../zh-HK.lproj/Localizable.strings | 9 + .../zh-Hans.lproj/Localizable.strings | 9 + .../zh-Hant.lproj/Localizable.strings | 9 + .../Source/CardScan/AppleOcr/AppleOcr.swift | 79 + .../CardScan/CardUtils/CardNetwork.swift | 34 + .../Source/CardScan/CardUtils/CardType.swift | 33 + .../CardScan/CardUtils/CreditCardUtils.swift | 390 ++ .../Source/CardScan/CardUtils/Expiry.swift | 20 + .../CreditCardOcr/AppleCreditCardOcr.swift | 96 + .../CreditCardOcrImplementation.swift | 42 + .../CreditCardOcrPrediction.swift | 191 + .../CreditCardOcr/CreditCardOcrResult.swift | 62 + .../CreditCardOcr/ErrorCorrection.swift | 95 + .../CreditCardOcr/MachineLearningResult.swift | 24 + .../CreditCardOcr/MainLoopStateMachine.swift | 119 + .../CardScan/CreditCardOcr/NonNameWords.swift | 43 + .../CardScan/CreditCardOcr/OcrMainLoop.swift | 313 ++ .../CardScan/CreditCardOcr/OcrObject.swift | 32 + .../CreditCardOcr/SSDCreditCardOcr.swift | 50 + .../CardScan/Extensions/Array+utils.swift | 16 + .../CardScan/Extensions/CGRectExtension.swift | 32 + .../CardScan/Extensions/CGrect+utils.swift | 18 + .../CreditCardOcrPrediction+expiry.swift | 14 + .../CardScan/Extensions/Image+utils.swift | 270 ++ .../Extensions/UIImage+pixelBuffer.swift | 248 ++ .../CardScan/MLModels/AsyncModelLoading.swift | 66 + .../CardScan/MLModels/SSDOcr+Utils.swift | 19 + .../Source/CardScan/MLModels/SSDOcr.swift | 320 ++ .../CardScan/MLModels/UxModel+Utils.swift | 19 + .../Source/CardScan/MLModels/UxModel.swift | 306 ++ .../MLRuntime/ActiveStateComputation.swift | 91 + .../Source/CardScan/MLRuntime/AppState.swift | 22 + .../CardScan/MLRuntime/DetectedAllBoxes.swift | 18 + .../MLRuntime/DetectedAllOcrBoxes.swift | 23 + .../CardScan/MLRuntime/DetectedBox.swift | 62 + .../CardScan/MLRuntime/DetectedSSDBox.swift | 51 + .../MLRuntime/DetectedSSDOcrBox.swift | 41 + .../Source/CardScan/MLRuntime/NMS.swift | 75 + .../Source/CardScan/MLRuntime/OcrDD.swift | 27 + .../CardScan/MLRuntime/OcrDDUtils.swift | 177 + .../CardScan/MLRuntime/OcrPriorsGen.swift | 163 + .../MLRuntime/PostDetectionAlgorithm.swift | 221 ++ .../CardScan/MLRuntime/PredictionAPI.swift | 78 + .../CardScan/MLRuntime/PredictionResult.swift | 84 + .../MLRuntime/PredictionUtilOcr.swift | 68 + .../CardScan/MLRuntime/SSDOcrDetect.swift | 207 ++ .../MLRuntime/SSDOcrOutputExtensions.swift | 176 + .../Source/CardScan/MLRuntime/SoftNMS.swift | 80 + .../Source/CardScan/UI/BlurView.swift | 40 + .../Source/CardScan/UI/CardScanSheet.swift | 87 + .../Source/CardScan/UI/CornerView.swift | 72 + .../CardScan/UI/InterfaceOrientation.swift | 60 + .../Source/CardScan/UI/PreviewView.swift | 90 + .../CardScan/UI/ScanBaseViewController.swift | 582 +++ .../CardScan/UI/ScanConfiguration.swift | 11 + .../CardScan/UI/ScanEventsProtocol.swift | 26 + .../Source/CardScan/UI/ScanStats.swift | 84 + .../UI/SimpleScanViewController.swift | 545 +++ .../Source/CardScan/UI/Torch.swift | 48 + .../Source/CardScan/UI/VideoFeed.swift | 244 ++ .../Source/CardScan/Utils/AppInfoUtils.swift | 43 + .../Utils/AtomicPropertyWrapper.swift | 43 + .../Source/CardScan/Utils/DeviceUtils.swift | 82 + ...CardImageVerificationDetailsResponse.swift | 68 + .../ScanStatsPayload+Common.swift | 43 + .../ScanStatsPayload+Tasks.swift | 21 + .../Scan Analytics/ScanStatsPayload.swift | 62 + .../Api/Models/VerificationFramesData.swift | 46 + .../CardVerify/Api/Models/VerifyFrames.swift | 15 + .../STPAPIClient+CardImageVerification.swift | 95 + .../Source/CardVerify/Bouncer.swift | 22 + .../CancellationReason.swift | 18 + .../CardImageVerificationController.swift | 160 + .../CardImageVerificationIntent.swift | 16 + .../CardImageVerificationSheet.swift | 127 + ...dImageVerificationSheetConfiguration.swift | 43 + .../CardScanSheetError.swift | 35 + .../ScanAnalyticsManager+Helpers.swift | 20 + .../ScanAnalyticsManager+Managers.swift | 51 + .../ScanAnalyticsManager+Tasks.swift | 80 + .../ScanAnalyticsManager.swift | 160 + .../Card Image Verification/ScannedCard.swift | 14 + .../ScannedCardImageData+Verification.swift | 137 + .../ScannedCardImageData.swift | 96 + .../StripeCore+Import.swift | 9 + .../Source/CardVerify/CardBase.swift | 46 + .../Source/CardVerify/CardScanFraudData.swift | 180 + .../Source/CardVerify/CardScanMisc.swift | 41 + .../CardVerify/CardVerifyFraudData.swift | 75 + .../CardVerify/CardVerifyStateMachine.swift | 325 ++ .../Source/CardVerify/FadeInAnimation.swift | 67 + .../Source/CardVerify/FrameData.swift | 40 + .../Helpers/EndToEndTestDataSource.swift | 48 + .../Helpers/STPLocalizedString.swift | 16 + .../CardVerify/Helpers/String+Localized.swift | 46 + .../Source/CardVerify/PaymentCard.swift | 76 + .../SimpleScanViewController+Verify.swift | 28 + .../StripeCardScanBundleLocator.swift | 20 + .../Source/CardVerify/UxAnalyzer.swift | 131 + .../Source/CardVerify/UxAndOcrMainLoop.swift | 27 + .../VerifyCardAddViewController.swift | 202 + .../CardVerify/VerifyCardViewController.swift | 310 ++ .../Source/CardVerify/ZoomedInCGImage.swift | 161 + .../StripeCardScan/StripeCardScan.h | 18 + .../Helpers/CardScanMockData.swift | 37 + .../Helpers/Data+Sha256.swift | 26 + .../Helpers/ImageHelpers.swift | 40 + .../Helpers/ScannedCardDetails.swift | 32 + .../Helpers/String+Sha256.swift | 19 + StripeCardScan/StripeCardScanTests/Info.plist | 22 + .../CardImageVerification_CardAdd_200.json | 31 + .../CardImageVerification_CardSet_200.json | 32 + .../Resources/synthetic_test_image.jpg | Bin 0 -> 988407 bytes ...PAPIClient+CardImageVerificationTest.swift | 383 ++ .../ScanStatsPayloadAPIBindingsTests.swift | 357 ++ .../VerifyFramesAPIBindingsTests.swift | 71 + ...CardImageVerificationControllerTests.swift | 170 + ...ImageVerificationDetailsResponseTest.swift | 81 + .../Unit/ImageCompressionTests.swift | 133 + .../Unit/ML Models/UxModelTests.swift | 50 + .../Unit/ScanAnalyticsManagerTests.swift | 165 + .../Unit/StrictModeFramesTest.swift | 159 + .../Unit/StringResourceTests.swift | 27 + StripeCore/Project.swift | 13 + .../StripeCore.xcodeproj/project.pbxproj | 1236 +++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripeCore.xcscheme | 95 + StripeCore/StripeCore/Info.plist | 22 + .../bg-BG.lproj/Localizable.strings | 31 + .../ca-ES.lproj/Localizable.strings | 31 + .../cs-CZ.lproj/Localizable.strings | 31 + .../da.lproj/Localizable.strings | 31 + .../de.lproj/Localizable.strings | 31 + .../el-GR.lproj/Localizable.strings | 31 + .../en-GB.lproj/Localizable.strings | 31 + .../en.lproj/Localizable.strings | 48 + .../es-419.lproj/Localizable.strings | 31 + .../es.lproj/Localizable.strings | 31 + .../et-EE.lproj/Localizable.strings | 31 + .../fi.lproj/Localizable.strings | 31 + .../fil.lproj/Localizable.strings | 31 + .../fr-CA.lproj/Localizable.strings | 31 + .../fr.lproj/Localizable.strings | 31 + .../hr.lproj/Localizable.strings | 31 + .../hu.lproj/Localizable.strings | 31 + .../id.lproj/Localizable.strings | 31 + .../it.lproj/Localizable.strings | 31 + .../ja.lproj/Localizable.strings | 31 + .../ko.lproj/Localizable.strings | 31 + .../lt-LT.lproj/Localizable.strings | 31 + .../lv-LV.lproj/Localizable.strings | 31 + .../ms-MY.lproj/Localizable.strings | 31 + .../mt.lproj/Localizable.strings | 31 + .../nb.lproj/Localizable.strings | 31 + .../nl.lproj/Localizable.strings | 31 + .../nn-NO.lproj/Localizable.strings | 31 + .../pl-PL.lproj/Localizable.strings | 31 + .../pt-BR.lproj/Localizable.strings | 31 + .../pt-PT.lproj/Localizable.strings | 31 + .../ro-RO.lproj/Localizable.strings | 31 + .../ru.lproj/Localizable.strings | 31 + .../sk-SK.lproj/Localizable.strings | 31 + .../sl-SI.lproj/Localizable.strings | 31 + .../sv.lproj/Localizable.strings | 31 + .../tr.lproj/Localizable.strings | 31 + .../vi.lproj/Localizable.strings | 31 + .../zh-HK.lproj/Localizable.strings | 31 + .../zh-Hans.lproj/Localizable.strings | 31 + .../zh-Hant.lproj/Localizable.strings | 31 + .../API Bindings/Models/EmptyResponse.swift | 14 + .../API Bindings/Models/StripeFile.swift | 45 + .../STPAPIClient+FileUpload.swift | 234 ++ .../Source/API Bindings/STPAPIClient.swift | 545 +++ .../Source/API Bindings/STPAppInfo.swift | 45 + .../STPMultipartFormDataEncoder.swift | 38 + .../STPMultipartFormDataPart.swift | 63 + .../Source/API Bindings/StripeAPI.swift | 174 + .../StripeAPIConfiguration+Version.swift | 19 + .../API Bindings/StripeAPIConfiguration.swift | 16 + .../Source/API Bindings/StripeError.swift | 80 + .../API Bindings/StripeServiceError.swift | 73 + .../Source/Analytics/Analytic.swift | 37 + .../Analytics/AnalyticLoggableError.swift | 52 + .../Source/Analytics/AnalyticsClientV2.swift | 153 + .../Source/Analytics/PluginDetector.swift | 47 + .../Source/Analytics/STPAnalyticEvent.swift | 139 + .../Source/Analytics/STPAnalyticsClient.swift | 147 + .../Categories/Decimal+StripeCore.swift | 15 + .../Source/Categories/Dictionary+Stripe.swift | 69 + .../Enums+CustomStringConvertible.swift | 29 + .../Source/Categories/NSArray+Stripe.swift | 31 + .../Categories/NSBundle+Stripe_AppName.swift | 32 + .../NSCharacterSet+StripeCore.swift | 21 + .../Source/Categories/NSError+Stripe.swift | 135 + .../Categories/NSError+StripeCore.swift | 18 + .../NSMutableURLRequest+Stripe.swift | 43 + .../Categories/NSURLComponents+Stripe.swift | 63 + .../Source/Categories/String+StripeCore.swift | 36 + .../Categories/UIImage+StripeCore.swift | 162 + .../Source/Coder/StripeCodable.swift | 177 + .../Source/Coder/StripeJSONDecoder.swift | 710 ++++ .../Source/Coder/StripeJSONEncoder.swift | 564 +++ .../Source/Coder/StripeJSONShared.swift | 37 + .../Source/Coder/UnknownFields.swift | 98 + .../ConnectionsSDKInterface.swift | 37 + .../DownloadManager/DownloadManager.swift | 234 ++ .../StripeCore/Source/Helpers/Async.swift | 153 + .../Helpers/BundleLocatorProtocol.swift | 75 + .../Source/Helpers/FileDownloader.swift | 75 + .../Source/Helpers/InstallMethod.swift | 27 + .../Source/Helpers/PaymentsSDKVariant.swift | 57 + .../Source/Helpers/STPDeviceUtils.swift | 25 + .../Source/Helpers/STPDispatchFunctions.swift | 17 + .../StripeCore/Source/Helpers/STPError.swift | 307 ++ .../Helpers/STPNumericStringValidator.swift | 31 + .../Helpers/STPURLCallbackHandler.swift | 92 + .../Source/Helpers/ServerErrorMapper.swift | 95 + .../Helpers/StripeCoreBundleLocator.swift | 18 + .../Source/Helpers/URLEncoder.swift | 147 + .../Source/Helpers/URLSession+Retry.swift | 54 + .../Localization/STPLocalizationUtils.swift | 97 + .../Localization/STPLocalizedString.swift | 14 + .../Localization/String+Localized.swift | 51 + .../Source/Telemetry/FraudDetectionData.swift | 72 + .../Source/Telemetry/STPTelemetryClient.swift | 217 ++ .../Telemetry/UserDefaults+PaymentsCore.swift | 42 + .../UI/UIActivityIndicatorView+Stripe.swift | 35 + .../StripeCore/Source/UI/UIFont+Stripe.swift | 88 + StripeCore/StripeCore/StripeCore.h | 18 + .../APIStubbedTestCase.swift | 53 + .../UIImage+StripeCoreTestingUtils.swift | 30 + .../UIView+StripeCoreTestingUtils.swift | 24 + StripeCore/StripeCoreTestUtils/Info.plist | 22 + .../KeyPathExpectation.swift | 70 + .../Mock Files/File_IdentityDocument.json | 7 + .../Mocks/MockAnalyticsClient.swift | 31 + .../Mocks/MockAnalyticsClientV2.swift | 25 + .../StripeCoreTestUtils/Mocks/MockData.swift | 50 + .../STPSnapshotVerifyView.swift | 32 + .../StripeCoreTestUtils/StripeCoreTestUtils.h | 18 + .../StripeCoreTestUtils/TestConstants.swift | 16 + .../URLRequest+StripeTest.swift | 50 + .../XCTestCase+Stripe.swift | 64 + .../STPAPIClient+EmptyResponseTest.swift | 81 + .../STPAPIClient+ErrorResponseTest.swift | 143 + .../API Bindings/StripeCodableTest.swift | 212 ++ .../Analytics/AnalyticsClientV2Test.swift | 91 + .../Error_SerializeForLoggingTest.swift | 67 + .../Analytics/STPAnalyticsClientTest.swift | 44 + .../Categories/Dictionary+StripeTests.swift | 37 + .../Categories/NSArray+StripeCoreTest.swift | 27 + .../NSMutableURLRequest+StripeTest.swift | 36 + .../Categories/UIImage+StripeCoreTests.swift | 85 + .../DownloadManager/DownloadManagerTest.swift | 336 ++ .../External/TestJSONEncoder.swift | 1760 +++++++++ .../Helpers/URLEncoderTest.swift | 61 + StripeCore/StripeCoreTests/Info.plist | 22 + .../StripeCoreTests/Mock Files/test_image.png | Bin 0 -> 9914 bytes StripeFinancialConnections/Project.swift | 19 + StripeFinancialConnections/README.md | 50 + .../project.pbxproj | 1388 +++++++ .../contents.xcworkspacedata | 7 + .../StripeFinancialConnections.xcscheme | 95 + .../StripeFinancialConnections/Info.plist | 22 + .../Resources/Images/add@3x.png | Bin 0 -> 347 bytes .../Resources/Images/back_arrow@3x.png | Bin 0 -> 642 bytes .../Resources/Images/bank@3x.png | Bin 0 -> 602 bytes .../Resources/Images/bank_check@2x.png | Bin 0 -> 41279 bytes .../Resources/Images/brandicon_default@3x.png | Bin 0 -> 905 bytes .../Resources/Images/bullet@3x.png | Bin 0 -> 327 bytes .../Resources/Images/cancel_circle@3x.png | Bin 0 -> 1208 bytes .../Resources/Images/check@3x.png | Bin 0 -> 633 bytes .../Resources/Images/chevron_down@3x.png | Bin 0 -> 475 bytes .../Resources/Images/close@3x.png | Bin 0 -> 788 bytes .../Resources/Images/ellipsis@3x.png | Bin 0 -> 567 bytes .../Resources/Images/generic_error@3x.png | Bin 0 -> 3439 bytes .../Images/prepane_phone_background@3x.png | Bin 0 -> 213733 bytes .../Resources/Images/search@3x.png | Bin 0 -> 1381 bytes .../Resources/Images/spinner@3x.png | Bin 0 -> 3480 bytes .../Resources/Images/stripe_logo@3x.png | Bin 0 -> 2889 bytes .../Resources/Images/warning_circle@3x.png | Bin 0 -> 1092 bytes .../Resources/Images/warning_triangle@3x.png | Bin 0 -> 914 bytes .../bg-BG.lproj/Localizable.strings | 2 + .../ca-ES.lproj/Localizable.strings | 2 + .../cs-CZ.lproj/Localizable.strings | 2 + .../da.lproj/Localizable.strings | 2 + .../de.lproj/Localizable.strings | 2 + .../el-GR.lproj/Localizable.strings | 2 + .../en-GB.lproj/Localizable.strings | 2 + .../en.lproj/Localizable.strings | 1 + .../es-419.lproj/Localizable.strings | 2 + .../es.lproj/Localizable.strings | 2 + .../et-EE.lproj/Localizable.strings | 2 + .../fi.lproj/Localizable.strings | 2 + .../fil.lproj/Localizable.strings | 2 + .../fr-CA.lproj/Localizable.strings | 2 + .../fr.lproj/Localizable.strings | 2 + .../hr.lproj/Localizable.strings | 2 + .../hu.lproj/Localizable.strings | 2 + .../id.lproj/Localizable.strings | 2 + .../it.lproj/Localizable.strings | 2 + .../ja.lproj/Localizable.strings | 2 + .../ko.lproj/Localizable.strings | 2 + .../lt-LT.lproj/Localizable.strings | 2 + .../lv-LV.lproj/Localizable.strings | 2 + .../ms-MY.lproj/Localizable.strings | 2 + .../mt.lproj/Localizable.strings | 2 + .../nb.lproj/Localizable.strings | 2 + .../nl.lproj/Localizable.strings | 2 + .../nn-NO.lproj/Localizable.strings | 2 + .../pl-PL.lproj/Localizable.strings | 2 + .../pt-BR.lproj/Localizable.strings | 2 + .../pt-PT.lproj/Localizable.strings | 2 + .../ro-RO.lproj/Localizable.strings | 2 + .../ru.lproj/Localizable.strings | 2 + .../sk-SK.lproj/Localizable.strings | 2 + .../sl-SI.lproj/Localizable.strings | 2 + .../sv.lproj/Localizable.strings | 2 + .../tr.lproj/Localizable.strings | 2 + .../vi.lproj/Localizable.strings | 2 + .../zh-HK.lproj/Localizable.strings | 2 + .../zh-Hans.lproj/Localizable.strings | 2 + .../zh-Hant.lproj/Localizable.strings | 2 + .../API Bindings/APIPollingHelper.swift | 108 + .../Source/API Bindings/APIVersion.swift | 26 + .../FinancialConnectionsAPIClient.swift | 418 +++ .../Models/BankAccountToken.swift | 40 + .../Models/FinancialConnectionsAccount.swift | 138 + .../FinancialConnectionsAuthSession.swift | 58 + .../FinancialConnectionsBulletPoint.swift | 20 + .../Models/FinancialConnectionsConsent.swift | 23 + ...FinancialConnectionsDataAccessNotice.swift | 21 + .../Models/FinancialConnectionsImage.swift | 13 + .../FinancialConnectionsInstitution.swift | 32 + ...tionsInstitutionSearchResultResource.swift | 13 + ...nancialConnectionsLegalDetailsNotice.swift | 19 + ...FinancialConnectionsMixedOAuthParams.swift | 13 + .../FinancialConnectionsOAuthPrepane.swift | 75 + .../FinancialConnectionsPartnerAccount.swift | 43 + ...ialConnectionsPaymentAccountResource.swift | 25 + ...inancialConnectionsPaymentMethodType.swift | 15 + .../Models/FinancialConnectionsSession.swift | 166 + .../FinancialConnectionsSessionManifest.swift | 88 + .../FinancialConnectionsSynchronize.swift | 23 + .../FinancialConnectionsAnalyticsClient.swift | 175 + .../FinancialConnectionsSheetAnalytics.swift | 84 + .../Source/Common/ExperimentHelper.swift | 69 + ...ctionsCustomManualEntryRequiredError.swift | 10 + ...ncialConnectionsNavigationController.swift | 204 ++ .../Source/Common/FlowRouter.swift | 91 + .../Source/Common/HostController.swift | 167 + .../Source/Common/HostViewController.swift | 126 + .../Source/Common/LoadingView.swift | 97 + ...dalPresentationWrapperViewController.swift | 50 + ...inancialConnectionsSDKImplementation.swift | 123 + .../Source/FinancialConnectionsSheet.swift | 242 ++ .../FinancialConnectionsSheetError.swift | 43 + .../Source/Helpers/Helpers.swift | 15 + .../Source/Helpers/Image.swift | 34 + .../NSAttributedString+Extensions.swift | 78 + .../Source/Helpers/STPLocalizedString.swift | 16 + .../Source/Helpers/String+Extensions.swift | 123 + .../Source/Helpers/String+Localized.swift | 42 + ...ipeFinancialConnectionsBundleLocator.swift | 18 + .../Source/Helpers/UIColor+Extensions.swift | 106 + .../Source/Helpers/UIFont+Extensions.swift | 94 + .../Helpers/UIViewController+Extensions.swift | 49 + .../AccountPickerAccountLoadErrorView.swift | 146 + .../AccountPickerDataSource.swift | 108 + .../AccountPickerFooterView.swift | 130 + .../AccountPicker/AccountPickerHelpers.swift | 105 + .../AccountPickerLabelRowView.swift | 99 + ...ountPickerNoAccountEligibleErrorView.swift | 228 ++ .../AccountPickerSelectionListView.swift | 132 + .../AccountPickerSelectionRowView.swift | 240 ++ .../AccountPickerSelectionView.swift | 61 + .../AccountPickerViewController.swift | 380 ++ .../Native/AccountPicker/CheckboxView.swift | 57 + .../LinkingAccountsLoadingView.swift | 74 + .../AccountPicker/RadioButtonView.swift | 82 + .../AccountNumberRetrievalErrorView.swift | 114 + ...AttachLinkedPaymentAccountDataSource.swift | 56 + ...chLinkedPaymentAccountViewController.swift | 164 + .../Native/Consent/ConsentBodyView.swift | 127 + .../Native/Consent/ConsentDataSource.swift | 48 + .../Native/Consent/ConsentFooterView.swift | 132 + .../Native/Consent/ConsentLogoView.swift | 76 + .../Consent/ConsentViewController.swift | 182 + .../FeaturedInstitutionGridCell.swift | 102 + .../FeaturedInstitutionGridView.swift | 141 + .../InstitutionDataSource.swift | 58 + .../InstitutionPickerViewController.swift | 364 ++ .../InstitutionSearchBar.swift | 247 ++ .../InstitutionSearchErrorView.swift | 156 + .../InstitutionSearchFooterView.swift | 233 ++ .../InstitutionSearchTableView.swift | 237 ++ .../InstitutionSearchTableViewCell.swift | 103 + .../ManualEntry/ManualEntryCheckView.swift | 54 + .../ManualEntry/ManualEntryDataSource.swift | 50 + .../ManualEntry/ManualEntryErrorView.swift | 58 + .../ManualEntry/ManualEntryFooterView.swift | 51 + .../ManualEntry/ManualEntryFormView.swift | 224 ++ .../ManualEntry/ManualEntryTextField.swift | 256 ++ .../ManualEntry/ManualEntryValidator.swift | 121 + .../ManualEntryViewController.swift | 131 + ...nualEntrySuccessTransactionTableView.swift | 293 ++ .../ManualEntrySuccessViewController.swift | 108 + .../Source/Native/NativeFlowController.swift | 791 ++++ .../Source/Native/NativeFlowDataManager.swift | 126 + .../PartnerAuth/PartnerAuthDataSource.swift | 168 + .../PartnerAuthViewController.swift | 730 ++++ .../Native/PartnerAuth/PrepaneImageView.swift | 104 + .../Native/PartnerAuth/PrepaneView.swift | 283 ++ .../PlaceholderViewController.swift | 103 + .../ResetFlow/ResetFlowDataSource.swift | 36 + .../ResetFlow/ResetFlowViewController.swift | 75 + .../Shared/AlwaysTemplateImageView.swift | 36 + .../Native/Shared/AuthFlowHelpers.swift | 57 + .../Native/Shared/BulletPointLabelView.swift | 52 + .../Native/Shared/Button+Extensions.swift | 41 + .../Source/Native/Shared/ClickableLabel.swift | 171 + .../CloseConfirmationAlertHandler.swift | 69 + .../Shared/ConsentBottomSheetModel.swift | 22 + .../Shared/ConsentBottomSheetView.swift | 301 ++ .../ConsentBottomSheetViewController.swift | 143 + .../Native/Shared/HitTestStackView.swift | 25 + .../Source/Native/Shared/HitTestView.swift | 25 + .../Native/Shared/InstitutionIconView.swift | 160 + .../Shared/MerchantDataAccessView.swift | 268 ++ .../Source/Native/Shared/PaneLayoutView.swift | 51 + .../PaneWithCustomHeaderLayoutView.swift | 80 + .../Shared/PaneWithHeaderLayoutView.swift | 104 + .../Shared/ReusableInformationView.swift | 175 + .../SFSafariViewController+Extensions.swift | 25 + .../Native/Shared/SpinnerIconView.swift | 87 + .../Native/Shared/SuccessIconView.swift | 76 + .../Shared/TimeInterval+Extensions.swift | 16 + .../Native/Shared/UIImage+Extensions.swift | 27 + .../Shared/UIImageView+Extensions.swift | 72 + .../Shared/UITableView+Extensions.swift | 31 + .../UIViewController+KeyboardAvoiding.swift | 179 + .../Success/SuccessAccountListView.swift | 122 + .../Native/Success/SuccessBodyView.swift | 187 + .../Native/Success/SuccessDataSource.swift | 47 + .../Native/Success/SuccessFooterView.swift | 76 + .../Success/SuccessViewController.swift | 141 + .../TerminalErrorViewController.swift | 90 + .../Source/Placeholder.swift | 10 + .../Source/StripeCore+Import.swift | 9 + .../Web/AuthenticationSessionManager.swift | 147 + .../Source/Web/ContinueStateView.swift | 84 + .../FinancialConnectionsAccountFetcher.swift | 69 + .../FinancialConnectionsSessionFetcher.swift | 68 + ...cialConnectionsWebFlowViewController.swift | 327 ++ .../StripeFinancialConnections.h | 18 + .../APIPollingHelperTests.swift | 342 ++ .../AccountFetcherTests.swift | 142 + .../AccountPickerHelpersTests.swift | 25 + .../AuthFlowHelpersTests.swift | 30 + .../EmptyFinancialConnectionsAPIClient.swift | 109 + .../FinancialConnectionsAnalyticsTest.swift | 74 + .../FinancialConnectionsSessionTests.swift | 53 + .../FinancialConnectionsSheetTests.swift | 75 + .../Info.plist | 22 + .../ManualEntryValidatorTests.swift | 76 + .../MarkdownBoldAttributedStringTests.swift | 168 + ...alConnectionsSession_both_accounts_la.json | 204 ++ ...ncialConnectionsSession_only_accounts.json | 129 + ...lConnectionsSession_only_both_missing.json | 9 + .../FinancialConnectionsSession_only_la.json | 129 + .../SessionFetcherTests.swift | 108 + .../SoftLinkTests.swift | 19 + .../StringExtensionsTests.swift | 49 + StripeIdentity/Project.swift | 20 + StripeIdentity/README.md | 58 + .../StripeIdentity.xcodeproj/project.pbxproj | 1774 +++++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripeIdentity.xcscheme | 104 + StripeIdentity/StripeIdentity/Info.plist | 22 + .../Resources/Images/icon_add@3x.png | Bin 0 -> 226 bytes .../Resources/Images/icon_camera@3x.png | Bin 0 -> 1186 bytes .../Resources/Images/icon_checkmark@3x.png | Bin 0 -> 518 bytes .../Resources/Images/icon_checkmark_92@3x.png | Bin 0 -> 2248 bytes .../Resources/Images/icon_clock@3x.png | Bin 0 -> 651 bytes .../Resources/Images/icon_info@3x.png | Bin 0 -> 391 bytes .../Resources/Images/icon_warning@3x.png | Bin 0 -> 2170 bytes .../Resources/Images/icon_warning_92@3x.png | Bin 0 -> 3705 bytes .../bg-BG.lproj/Localizable.strings | 151 + .../ca-ES.lproj/Localizable.strings | 151 + .../cs-CZ.lproj/Localizable.strings | 151 + .../da.lproj/Localizable.strings | 151 + .../de.lproj/Localizable.strings | 151 + .../el-GR.lproj/Localizable.strings | 151 + .../en-GB.lproj/Localizable.strings | 151 + .../en.lproj/Localizable.strings | 256 ++ .../es-419.lproj/Localizable.strings | 151 + .../es.lproj/Localizable.strings | 151 + .../et-EE.lproj/Localizable.strings | 151 + .../fi.lproj/Localizable.strings | 151 + .../fil.lproj/Localizable.strings | 151 + .../fr-CA.lproj/Localizable.strings | 151 + .../fr.lproj/Localizable.strings | 151 + .../hr.lproj/Localizable.strings | 151 + .../hu.lproj/Localizable.strings | 151 + .../id.lproj/Localizable.strings | 151 + .../it.lproj/Localizable.strings | 151 + .../ja.lproj/Localizable.strings | 151 + .../ko.lproj/Localizable.strings | 151 + .../lt-LT.lproj/Localizable.strings | 151 + .../lv-LV.lproj/Localizable.strings | 151 + .../ms-MY.lproj/Localizable.strings | 151 + .../mt.lproj/Localizable.strings | 151 + .../nb.lproj/Localizable.strings | 151 + .../nl.lproj/Localizable.strings | 151 + .../nn-NO.lproj/Localizable.strings | 151 + .../pl-PL.lproj/Localizable.strings | 151 + .../pt-BR.lproj/Localizable.strings | 151 + .../pt-PT.lproj/Localizable.strings | 151 + .../ro-RO.lproj/Localizable.strings | 151 + .../ru.lproj/Localizable.strings | 151 + .../sk-SK.lproj/Localizable.strings | 151 + .../sl-SI.lproj/Localizable.strings | 151 + .../sv.lproj/Localizable.strings | 151 + .../tr.lproj/Localizable.strings | 151 + .../vi.lproj/Localizable.strings | 151 + .../zh-HK.lproj/Localizable.strings | 151 + .../zh-Hans.lproj/Localizable.strings | 151 + .../zh-Hant.lproj/Localizable.strings | 151 + .../API Bindings/DocumentScanner+API.swift | 40 + .../DocumentType+StripeIdentity.swift | 21 + .../API Bindings/DocumentUploader+API.swift | 63 + .../Source/API Bindings/FaceScanner+API.swift | 28 + .../API Bindings/IdentityAPIClient.swift | 123 + .../API Bindings/Models/DocumentType.swift | 16 + .../Models/TruncatedDecimal.swift | 61 + .../VerificationPage/VerificationPage.swift | 46 + .../VerificationPageFieldType.swift | 23 + .../VerificationPageRequirements.swift | 17 + ...ficationPageStaticContentConsentPage.swift | 23 + ...ageStaticContentCountryNotListedPage.swift | 19 + ...geStaticContentDocumentCaptureModels.swift | 19 + ...PageStaticContentDocumentCapturePage.swift | 30 + ...nPageStaticContentDocumentSelectPage.swift | 20 + ...ationPageStaticContentIndividualPage.swift | 21 + ...geStaticContentIndividualWelcomePage.swift | 22 + ...icationPageStaticContentSelfieModels.swift | 19 + ...ificationPageStaticContentSelfiePage.swift | 33 + ...erificationPageStaticContentTextPage.swift | 19 + .../VerificationPageData.swift | 27 + ...VerificationPageDataRequirementError.swift | 21 + .../VerificationPageDataRequirements.swift | 18 + .../RequiredInternationalAddress.swift | 20 + .../VerificationPageClearData.swift | 42 + .../VerificationPageCollectedData.swift | 141 + .../VerificationPageDataDob.swift | 17 + ...VerificationPageDataDocumentFileData.swift | 47 + .../VerificationPageDataFace.swift | 48 + .../VerificationPageDataIdNumber.swift | 17 + .../VerificationPageDataName.swift | 16 + .../VerificationPageDataUpdate.swift | 18 + .../API Bindings/SelfieUploader+API.swift | 64 + .../Analytics/IdentityAnalyticsClient.swift | 487 +++ .../Categories/Array+StripeIdentity.swift | 33 + .../Categories/CGImage+StripeIdentity.swift | 119 + .../MLMultiArray+StripeIdentity.swift | 16 + .../Categories/NSAttributedString+HTML.swift | 245 ++ .../TimeInterval+StripeIdentity.swift | 21 + ...INavigationController+StripeIdentity.swift | 71 + .../VNBarcodeSymbology+StripeIdentity.swift | 158 + .../Source/Elements/IdNumberElement.swift | 98 + .../Elements/IdentityElementsFactory.swift | 184 + .../Elements/IdentityTextButtonElement.swift | 52 + .../Elements/IndividualFormElement.swift | 169 + .../Enums+CustomStringConvertible.swift | 7 + .../StripeIdentity/Source/Helpers/Image.swift | 25 + .../Source/Helpers/STPLocalizedString.swift | 16 + .../Source/Helpers/String+Localized.swift | 201 + .../Helpers/StripeIdentityBundleLocator.swift | 19 + .../Source/IdentityVerificationSheet.swift | 285 ++ .../IdentityVerificationSheetError.swift | 65 + .../IdentityTopLevelDestination.swift | 19 + .../DocumentScanner/DocumentScanner.swift | 142 + .../DocumentScannerConfiguration.swift | 37 + .../DocumentScannerOutput.swift | 37 + .../FaceScanner/FaceCaptureData.swift | 53 + .../FaceScanner/FaceScanner.swift | 69 + .../FaceScannerConfiguration.swift | 33 + .../FaceScanner/FaceScannerOutput.swift | 85 + .../ImageScanner/ImageScanner.swift | 74 + .../ImageScanningConcurrencyManager.swift | 172 + .../ImageScanningSession.swift | 380 ++ .../ImageScanningSessionDelegate.swift | 278 ++ .../ImageUploaders/DocumentUploader.swift | 226 ++ .../IdentityImageUploader.swift | 177 + .../ImageUploaders/SelfieUploader.swift | 138 + .../VerificationSheetController.swift | 490 +++ .../VerificationSheetFlowController.swift | 718 ++++ ...VerificationSheetFlowControllerError.swift | 66 + .../Detectors/BarcodeDetector.swift | 102 + .../Detectors/FaceDetector/FaceDetector.swift | 49 + .../FaceDetector/FaceDetectorOutput.swift | 188 + .../Detectors/IDDetector/IDDetector.swift | 48 + .../IDDetector/IDDetectorConstants.swift | 17 + .../IDDetector/IDDetectorOutput.swift | 217 ++ .../Detectors/MLDetectorConfiguration.swift | 19 + .../Detectors/MLDetectorMetricsTracker.swift | 94 + .../Detectors/MotionBlurDetector.swift | 99 + .../Detectors/VisionBasedDetector.swift | 142 + .../NativeComponents/DocumentSide.swift | 14 + .../IdentityDataCollecting.swift | 34 + .../IdentityFlowNavigationController.swift | 139 + .../Source/NativeComponents/IdentityUI.swift | 82 + .../ML/Helpers/MLModelLoader.swift | 137 + .../MLModelUnexpectedOutputError.swift | 70 + .../ML/Helpers/NonMaxSuppression.swift | 188 + .../ML/IdentityMLModelLoader.swift | 179 + .../BiometricConsentViewController.swift | 222 ++ .../CountryNotListedViewController.swift | 106 + .../ViewControllers/DebugViewController.swift | 71 + ...ocumentCaptureViewController+Strings.swift | 125 + .../DocumentCaptureViewController.swift | 561 +++ ...mentFileUploadViewController+Strings.swift | 167 + .../DocumentFileUploadViewController.swift | 619 ++++ .../DocumentTypeSelectViewController.swift | 191 + .../ViewControllers/ErrorViewController.swift | 158 + .../IdentityFlowViewController.swift | 158 + .../IndividualViewController.swift | 104 + .../IndividualWelcomeViewController.swift | 89 + .../LoadingViewController.swift | 88 + .../SelfieCaptureViewController+Strings.swift | 33 + .../SelfieCaptureViewController.swift | 469 +++ .../SuccessViewController.swift | 82 + .../Views/BottomAlignedLabel.swift | 144 + .../Views/CameraPreviewContainerView.swift | 116 + .../Views/ContentCenteringScrollView.swift | 40 + .../NativeComponents/Views/DebugView.swift | 219 ++ .../DocumentCapture/AnimatedBorderView.swift | 171 + .../DocumentCapture/DocumentCaptureView.swift | 76 + .../DocumentScanningView.swift | 204 ++ .../InstructionalDocumentScanningView.swift | 96 + .../NativeComponents/Views/ErrorView.swift | 99 + .../Views/HeaderIconView.swift | 199 + .../NativeComponents/Views/HeaderView.swift | 136 + .../Views/IdentityFlowView.swift | 384 ++ .../Views/IdentityHTMLView/HTMLTextView.swift | 143 + .../HTMLViewWithIconLabels.swift | 226 ++ .../IdentityHTMLView/IconLabelHTMLView.swift | 124 + .../Views/InstructionListView.swift | 100 + .../Views/ListView/ListItemView.swift | 233 ++ .../Views/ListView/ListView.swift | 136 + .../Views/Selfie/SelfieCaptureView.swift | 98 + .../Views/Selfie/SelfieScanningView.swift | 433 +++ .../Views/ShadowConfiguration.swift | 25 + .../Views/ShadowedCorneredImageView.swift | 80 + .../Source/StripeCore+Import.swift | 10 + .../Source/VerificationClientSecret.swift | 47 + .../Source/VerificationSheetAnalytics.swift | 87 + .../WebWrapper/VerificationFlowWebView.swift | 296 ++ .../VerificationFlowWebViewController.swift | 208 ++ .../WebWrapper/VerifyWebURLHelper.swift | 27 + .../StripeIdentity/StripeIdentity.h | 18 + .../Helpers/DocumentUploaderMock.swift | 57 + .../Helpers/IdentityAPIClientTestMock.swift | 128 + .../IdentityAnalyticsClientTestHelpers.swift | 51 + .../Helpers/IdentityMLModelLoaderMock.swift | 45 + .../Helpers/IdentityMockData.swift | 164 + .../Helpers/ImageScannerMock.swift | 43 + .../ImageScanningConcurrencyManagerMock.swift | 53 + .../MLDetectorMetricsTrackerMock.swift | 39 + .../Helpers/SnapshotTestMockData.swift | 31 + .../VerificationFlowResult+Equatable.swift | 27 + .../VerificationSheetControllerMock.swift | 144 + .../VerificationSheetFlowControllerMock.swift | 88 + StripeIdentity/StripeIdentityTests/Info.plist | 22 + .../Mock Photos/back_drivers_license.jpg | Bin 0 -> 1385618 bytes .../cgimage_stripeidentity_test.png | Bin 0 -> 318214 bytes .../Mock Photos/front_drivers_license.jpg | Bin 0 -> 1524408 bytes .../Mock Files/Mock Photos/header_icon.png | Bin 0 -> 1067 bytes .../VerificationPage_200.json | 144 + .../VerificationPage_200_submitted.json | 144 + .../VerificationPage_200_testMode.json | 144 + .../VerificationPage_no_selfie.json | 123 + ...VerificationPage_require_live_capture.json | 144 + .../VerificationPage_type_address.json | 121 + ...ficationPage_type_doc_require_address.json | 123 + ...icationPage_type_doc_require_idNumber.json | 123 + ...type_doc_require_idNumber_and_address.json | 124 + .../VerificationPage_type_idNumber.json | 121 + .../VerificationPageData_200.json | 17 + .../VerificationPageData_no_errors.json | 10 + ...rificationPageData_no_errors_needback.json | 11 + .../VerificationPageData_submitted.json | 10 + .../StripeIdentityTests/Mock Files/mock.html | 8 + .../AnimatedBorderViewSnapshotTest.swift | 74 + ...ricConsentViewControllerSnapshotTest.swift | 33 + .../CGImage_StripeIdentitySnapshotTest.png | Bin 0 -> 402622 bytes .../CGImage_StripeIdentitySnapshotTest.swift | 163 + .../DebugViewControllerSnapshotTest.swift | 25 + .../DocumentScanningViewSnapshotTest.swift | 71 + .../Snapshot/ErrorViewSnapshotTest.swift | 59 + .../Snapshot/HeaderIconViewSnapshotTest.swift | 77 + .../Snapshot/HeaderViewSnapshotTest.swift | 139 + .../IdentityFlowViewSnapshotTest.swift | 114 + .../IdentityHTMLViewSnapshotTest.swift | 99 + ...ualWelcomeViewControllerSnapshotTest.swift | 31 + .../InstructionListViewSnapshotTest.swift | 72 + ...onalDocumentScanningViewSnapshotTest.swift | 72 + .../Snapshot/ListItemViewSnapshotTest.swift | 146 + .../Snapshot/ListViewSnapshotTest.swift | 103 + .../NSAttributedString_HTMLSnapshotTest.swift | 97 + .../SelfieCaptureViewSnapshotTest.swift | 76 + .../SelfieScanningViewSnapshotTest.swift | 113 + .../SuccessViewControllerSnapshotTest.swift | 35 + ...VerificationFlowWebViewSnapshotTests.swift | 89 + .../API Bindings/IdentityAPIClientTest.swift | 291 ++ .../API Bindings/TruncatedDecimalTest.swift | 69 + .../CGImage+StripeIdentityUnitTest.swift | 209 ++ .../IdentityElementsFactoryTest.swift | 104 + .../Unit/Elements/IndividualElementTest.swift | 83 + .../Coordinators/DocumentUploaderTest.swift | 523 +++ .../IdentityImageUploaderTest.swift | 269 ++ .../IdentityVerificationSheetTest.swift | 268 ++ .../VerificationSheetControllerTest.swift | 594 +++ .../VerificationSheetFlowControllerTest.swift | 596 +++ .../BiometricConsentViewControllerTest.swift | 48 + .../DocumentCaptureViewControllerTest.swift | 844 +++++ ...DocumentFileUploadViewControllerTest.swift | 217 ++ ...DocumentTypeSelectViewControllerTest.swift | 136 + .../ErrorViewControllerTest.swift | 54 + ...IdentityFlowNavigationControllerTest.swift | 40 + .../IndividualViewControllerTest.swift | 41 + .../IndividualWelcomeViewControllerTest.swift | 37 + .../Unit/VerificationClientSecretTest.swift | 89 + .../Unit/VerificationSheetAnalyticsTest.swift | 72 + ...erificationFlowWebViewControllerTest.swift | 73 + .../VerificationFlowWebViewTest.swift | 66 + StripeLinkCore/Project.swift | 14 + .../StripeLinkCore.xcodeproj/project.pbxproj | 389 ++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripeLinkCore.xcscheme | 95 + StripeLinkCore/StripeLinkCore/Info.plist | 22 + .../StripeLinkCore/Source/Placeholder.swift | 13 + .../StripeLinkCore/StripeLinkCore.h | 18 + StripeLinkCore/StripeLinkCoreTests/Info.plist | 22 + .../StripeLinkCoreTests.swift | 36 + StripePaymentSheet/Project.swift | 16 + StripePaymentSheet/README.md | 41 + .../project.pbxproj | 1883 ++++++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripePaymentSheet.xcscheme | 95 + .../StripePaymentSheet/Info.plist | 22 + .../InstantDebitIcons/bank_icon_boa@3x.png | Bin 0 -> 941 bytes .../bank_icon_capitalone@3x.png | Bin 0 -> 432 bytes .../bank_icon_citibank@3x.png | Bin 0 -> 566 bytes .../bank_icon_compass@3x.png | Bin 0 -> 591 bytes .../bank_icon_default@3x.png | Bin 0 -> 541 bytes .../bank_icon_morganchase@3x.png | Bin 0 -> 387 bytes .../InstantDebitIcons/bank_icon_nfcu@3x.png | Bin 0 -> 1333 bytes .../InstantDebitIcons/bank_icon_pnc@3x.png | Bin 0 -> 702 bytes .../InstantDebitIcons/bank_icon_stripe@3x.png | Bin 0 -> 465 bytes .../bank_icon_suntrust@3x.png | Bin 0 -> 958 bytes .../InstantDebitIcons/bank_icon_svb@3x.png | Bin 0 -> 605 bytes .../InstantDebitIcons/bank_icon_td@3x.png | Bin 0 -> 546 bytes .../InstantDebitIcons/bank_icon_usaa@3x.png | Bin 0 -> 705 bytes .../InstantDebitIcons/bank_icon_usbank@3x.png | Bin 0 -> 568 bytes .../bank_icon_wellsfargo@3x.png | Bin 0 -> 581 bytes .../Resources/Images/Link/back_button@3x.png | Bin 0 -> 193 bytes .../Resources/Images/Link/icon-pm-link@3x.png | Bin 0 -> 736 bytes .../Images/Link/icon_add_bordered@3x.png | Bin 0 -> 459 bytes .../Resources/Images/Link/icon_cancel@3x.png | Bin 0 -> 333 bytes .../Images/Link/icon_link_error@3x.png | Bin 0 -> 466 bytes .../Images/Link/icon_link_success@3x.png | Bin 0 -> 461 bytes .../Resources/Images/Link/icon_menu@3x.png | Bin 0 -> 152 bytes .../Images/Link/icon_menu_horizontal@3x.png | Bin 0 -> 172 bytes .../Images/Link/link_carousel_logo@3x.png | Bin 0 -> 768 bytes .../Resources/Images/Link/link_logo@3x.png | Bin 0 -> 720 bytes .../PaymentMethods/icon-pm-affirm@3x.png | Bin 0 -> 1090 bytes .../PaymentMethods/icon-pm-afterpay@3x.png | Bin 0 -> 700 bytes .../PaymentMethods/icon-pm-aubecsdebit@3x.png | Bin 0 -> 261 bytes .../PaymentMethods/icon-pm-bancontact@3x.png | Bin 0 -> 1481 bytes .../Images/PaymentMethods/icon-pm-bank@3x.png | Bin 0 -> 586 bytes .../Images/PaymentMethods/icon-pm-card@3x.png | Bin 0 -> 195 bytes .../PaymentMethods/icon-pm-cashapp@3x.png | Bin 0 -> 3649 bytes .../Images/PaymentMethods/icon-pm-eps@3x.png | Bin 0 -> 940 bytes .../PaymentMethods/icon-pm-giropay@3x.png | Bin 0 -> 918 bytes .../PaymentMethods/icon-pm-ideal@3x.png | Bin 0 -> 811 bytes .../PaymentMethods/icon-pm-klarna@3x.png | Bin 0 -> 454 bytes .../Images/PaymentMethods/icon-pm-p24@3x.png | Bin 0 -> 722 bytes .../PaymentMethods/icon-pm-paypal@3x.png | Bin 0 -> 751 bytes .../PaymentMethods/icon-pm-paypal_dark@3x.png | Bin 0 -> 726 bytes .../Images/PaymentMethods/icon-pm-sepa@3x.png | Bin 0 -> 917 bytes .../Images/PaymentMethods/icon-pm-upi@3x.png | Bin 0 -> 2002 bytes .../Images/PaymentSheet/icon_checkmark@3x.png | Bin 0 -> 242 bytes .../PaymentSheet/icon_chevron_left@3x.png | Bin 0 -> 235 bytes .../icon_chevron_left_standalone@3x.png | Bin 0 -> 172 bytes .../Images/PaymentSheet/icon_lock@3x.png | Bin 0 -> 297 bytes .../Images/PaymentSheet/icon_plus@3x.png | Bin 0 -> 181 bytes .../Images/PaymentSheet/icon_x@3x.png | Bin 0 -> 182 bytes .../PaymentSheet/icon_x_standalone@3x.png | Bin 0 -> 308 bytes .../Resources/Images/affirm_mark@3x.png | Bin 0 -> 1183 bytes .../Resources/Images/affirm_mark_dark@3x.png | Bin 0 -> 1629 bytes .../Images/afterpay_icon_info@3x.png | Bin 0 -> 728 bytes .../Resources/Images/afterpay_mark@3x.png | Bin 0 -> 3221 bytes .../Images/afterpay_mark_dark@3x.png | Bin 0 -> 3290 bytes .../Resources/Images/apple_pay_mark@3x.png | Bin 0 -> 6188 bytes .../Resources/Images/clearpay_mark@3x.png | Bin 0 -> 3884 bytes .../Images/clearpay_mark_dark@3x.png | Bin 0 -> 3616 bytes .../Images/polling_error_icon@3x.png | Bin 0 -> 1131 bytes .../Resources/JSON/form_specs.json | 810 ++++ .../bg-BG.lproj/Localizable.strings | 179 + .../ca-ES.lproj/Localizable.strings | 179 + .../cs-CZ.lproj/Localizable.strings | 179 + .../da.lproj/Localizable.strings | 179 + .../de.lproj/Localizable.strings | 179 + .../el-GR.lproj/Localizable.strings | 179 + .../en-GB.lproj/Localizable.strings | 179 + .../en.lproj/Localizable.strings | 291 ++ .../es-419.lproj/Localizable.strings | 179 + .../es.lproj/Localizable.strings | 179 + .../et-EE.lproj/Localizable.strings | 179 + .../fi.lproj/Localizable.strings | 179 + .../fil.lproj/Localizable.strings | 179 + .../fr-CA.lproj/Localizable.strings | 179 + .../fr.lproj/Localizable.strings | 179 + .../hr.lproj/Localizable.strings | 179 + .../hu.lproj/Localizable.strings | 179 + .../id.lproj/Localizable.strings | 179 + .../it.lproj/Localizable.strings | 179 + .../ja.lproj/Localizable.strings | 179 + .../ko.lproj/Localizable.strings | 179 + .../lt-LT.lproj/Localizable.strings | 179 + .../lv-LV.lproj/Localizable.strings | 179 + .../ms-MY.lproj/Localizable.strings | 179 + .../mt.lproj/Localizable.strings | 179 + .../nb.lproj/Localizable.strings | 179 + .../nl.lproj/Localizable.strings | 179 + .../nn-NO.lproj/Localizable.strings | 179 + .../pl-PL.lproj/Localizable.strings | 179 + .../pt-BR.lproj/Localizable.strings | 179 + .../pt-PT.lproj/Localizable.strings | 179 + .../ro-RO.lproj/Localizable.strings | 179 + .../ru.lproj/Localizable.strings | 179 + .../sk-SK.lproj/Localizable.strings | 179 + .../sl-SI.lproj/Localizable.strings | 179 + .../sv.lproj/Localizable.strings | 179 + .../tk.lproj/Localizable.strings | 0 .../tr.lproj/Localizable.strings | 179 + .../vi.lproj/Localizable.strings | 179 + .../zh-HK.lproj/Localizable.strings | 179 + .../zh-Hans.lproj/Localizable.strings | 179 + .../zh-Hant.lproj/Localizable.strings | 179 + .../Source/Analytics/AnalyticsHelper.swift | 52 + .../STPAnalyticsClient+Address.swift | 93 + .../Analytics/STPAnalyticsClient+LUXE.swift | 32 + .../Source/Categories/Data+SHA256.swift | 26 + .../Source/Categories/Date+Distance.swift | 22 + .../NSAttributedString+Stripe.swift | 17 + .../STPPaymentMethod+PaymentSheet.swift | 49 + .../STPPaymentMethodParams+PaymentSheet.swift | 30 + .../Source/Categories/String+Localized.swift | 189 + .../String+StripePaymentSheet.swift | 19 + .../Source/Helpers/BoolReference.swift | 20 + .../Source/Helpers/Images.swift | 67 + .../Source/Helpers/IntentStatusPoller.swift | 101 + .../Helpers/PaymentSheetLinkAccount.swift | 555 +++ .../Source/Helpers/STPCameraView.swift | 68 + .../Source/Helpers/STPCardScanner.swift | 487 +++ .../Source/Helpers/STPImageLibrary.swift | 118 + .../Source/Helpers/STPLocalizedString.swift | 13 + .../Source/Helpers/STPStringUtils.swift | 139 + .../Helpers/StaticEphemeralKeyProvider.swift | 46 + .../Helpers/StripePaymentSheet+Exports.swift | 11 + .../StripePaymentSheetBundleLocator.swift | 19 + .../Link/ConsumerSession+LookupResponse.swift | 52 + .../Link/ConsumerSession+PublishableKey.swift | 27 + .../API Bindings/Link/ConsumerSession.swift | 244 ++ .../Link/CookieStore/LinkCookieStore.swift | 78 + .../CookieStore/LinkInMemoryCookieStore.swift | 24 + .../CookieStore/LinkSecureCookieStore.swift | 99 + .../API Bindings/Link/PaymentDetails.swift | 250 ++ .../API Bindings/Link/STPAPIClient+Link.swift | 509 +++ .../Link/VerificationSession.swift | 43 + .../STPAPIClient+PaymentSheet.swift | 235 ++ .../API Bindings/STPElementsSession.swift | 87 + .../API Bindings/VO/CardExpiryDate.swift | 77 + .../Internal/Basic UI/SeparatorLabel.swift | 131 + .../LinkFinancialConnectionsAuthManager.swift | 169 + .../Link/Components/Badge/LinkBadgeView.swift | 133 + .../NavigationBar/LinkNavigationBar.swift | 187 + .../Components/Notice/LinkNoticeView.swift | 113 + .../LinkPaymentMethodPicker-AddButton.swift | 99 + .../LinkPaymentMethodPicker-Cell.swift | 280 ++ ...kPaymentMethodPicker-CellContentView.swift | 136 + .../LinkPaymentMethodPicker-Header.swift | 194 + .../LinkPaymentMethodPicker-RadioButton.swift | 114 + .../LinkPaymentMethodPicker.swift | 312 ++ .../Link/Components/Toast/LinkToast.swift | 182 + ...inkViewController-BaseViewController.swift | 126 + ...kViewController-LoaderViewController.swift | 38 + ...wController-NewPaymentViewController.swift | 309 ++ ...kViewController-SignUpViewController.swift | 310 ++ ...thLinkViewController-SignUpViewModel.swift | 227 ++ ...ntroller-UpdatePaymentViewController.swift | 208 ++ ...ntroller-VerifyAccountViewController.swift | 76 + ...kViewController-WalletViewController.swift | 550 +++ ...thLinkViewController-WalletViewModel.swift | 291 ++ .../PayWithLinkViewController.swift | 395 ++ .../LinkInlineSignupElement.swift | 60 + ...LinkInlineSignupView-CheckboxElement.swift | 68 + .../InlineSignup/LinkInlineSignupView.swift | 228 ++ .../Link/Elements/LinkCardEditElement.swift | 223 ++ .../Link/Elements/LinkEmailElement.swift | 96 + .../Link/Extensions/Button+Link.swift | 70 + .../Link/Extensions/ConfirmButton+Link.swift | 38 + .../Link/Extensions/FormElement+Link.swift | 90 + .../Link/Extensions/Intent+Link.swift | 61 + .../PaymentSheet-Configuration+Link.swift | 17 + .../Extensions/STPAnalyticsClient+Link.swift | 64 + .../Link/Extensions/UIColor+Link.swift | 178 + .../Source/Internal/Link/LinkUI.swift | 179 + .../Link/Services/LinkAccountService.swift | 116 + .../Internal/Link/Utils/Locale+Link.swift | 35 + .../Link/Utils/OperationDebouncer.swift | 63 + .../Verification/LinkAccountContext.swift | 47 + .../Link/Verification/LinkCookieKey.swift | 13 + .../Link/Verification/LinkUtils.swift | 51 + .../LinkVerificationController.swift | 49 + .../LinkVerificationView-Header.swift | 69 + .../LinkVerificationView-LogoutView.swift | 65 + .../Verification/LinkVerificationView.swift | 269 ++ ...iewController-PresentationController.swift | 211 ++ .../LinkVerificationViewController.swift | 247 ++ .../LinkInlineSignupViewModel.swift | 239 ++ .../Views/LinkInstantDebitMandateView.swift | 119 + .../LinkKeyboardAvoidingScrollView.swift | 75 + .../Link/Views/LinkLegalTermsView.swift | 141 + .../Link/Views/LinkMoreInfoView.swift | 54 + .../AddressViewController+Configuration.swift | 166 + .../AddressViewController.swift | 457 +++ .../BottomSheet/BottomSheetPresentable.swift | 14 + .../BottomSheetPresentationAnimator.swift | 107 + .../BottomSheetPresentationController.swift | 212 ++ .../BottomSheetTransitioningDelegate.swift | 67 + .../UIViewController+BottomSheet.swift | 42 + .../CardSectionWithScannerElement.swift | 166 + .../CardSectionWithScannerView.swift | 99 + .../Elements/ConnectionsElement.swift | 26 + .../LinkEnabledPaymentMethodElement.swift | 110 + .../PaymentMethodElement.swift | 88 + .../PaymentMethodElementWrapper.swift | 115 + .../Elements/SimpleMandateElement.swift | 34 + .../TextField/TextFieldElement+Card.swift | 298 ++ .../TextField/TextFieldElement+IBAN.swift | 184 + .../PaymentSheet/Error+PaymentSheet.swift | 27 + .../Source/PaymentSheet/Intent.swift | 133 + .../PaymentSheet/IntentConfirmParams.swift | 195 + .../Source/PaymentSheet/KlarnaHelper.swift | 44 + .../Link/LinkPaymentController.swift | 222 ++ .../Link/PayWithLinkController.swift | 99 + .../Link/PaymentSheet-LinkConfirmOption.swift | 70 + .../AddPaymentMethodViewController.swift | 403 ++ .../PaymentMethodTypeCollectionView.swift | 327 ++ .../WalletHeaderView.swift | 170 + .../PaymentSheet/PaymentMethodType.swift | 518 +++ .../PaymentSheet/PaymentOption+Images.swift | 173 + .../PaymentSheet/PaymentSheet+API.swift | 623 ++++ .../PaymentSheet+DeferredAPI.swift | 157 + ...ymentSheet+PaymentMethodAvailability.swift | 192 + .../PaymentSheet/PaymentSheet+SwiftUI.swift | 425 +++ .../Source/PaymentSheet/PaymentSheet.swift | 472 +++ .../PaymentSheet/PaymentSheetAppearance.swift | 194 + .../PaymentSheetConfiguration.swift | 373 ++ .../PaymentSheet/PaymentSheetError.swift | 47 + .../PaymentSheetFlowController.swift | 500 +++ .../FormSpec/FormSpec.swift | 291 ++ .../FormSpec/FormSpecPaymentHandler.swift | 258 ++ .../FormSpec/FormSpecProvider.swift | 98 + .../PaymentSheetFormFactory+Card.swift | 96 + .../PaymentSheetFormFactory+FormSpec.swift | 236 ++ .../PaymentSheetFormFactory+UPI.swift | 56 + .../PaymentSheetFormFactory.swift | 718 ++++ .../PaymentSheetIntentConfiguration.swift | 133 + .../STPAnalyticsClient+PaymentSheet.swift | 390 ++ .../STPApplePayContext+PaymentSheet.swift | 224 ++ ...ntShippingDetailsParams+PaymentSheet.swift | 31 + .../SavedPaymentMethodCollectionView.swift | 419 +++ .../SavedPaymentOptionsViewController.swift | 429 +++ ...ethodsAddPaymentMethodViewController.swift | 152 + ...ymentMethodsCollectionViewController.swift | 418 +++ .../SavedPaymentMethodsFormFactory.swift | 48 + .../SavedPaymentMethodsSheet+API.swift | 52 + .../SavedPaymentMethodsSheet.swift | 219 ++ ...avedPaymentMethodsSheetConfiguration.swift | 107 + .../SavedPaymentMethodsSheetDelegate.swift | 10 + .../SavedPaymentMethodsSheetError.swift | 38 + .../SavedPaymentMethodsViewController.swift | 633 ++++ .../USBankAccount/BankAccountInfoView.swift | 160 + .../USBankAccountPaymentMethodElement.swift | 276 ++ .../AutoComplete/AddressSearchResult.swift | 49 + .../AutoCompleteViewController.swift | 281 ++ .../AutoComplete/String+AutoComplete.swift | 83 + .../BottomSheet3DS2ViewController.swift | 102 + .../BottomSheetViewController.swift | 331 ++ .../LoadingViewController.swift | 79 + ...entSheetFlowControllerViewController.swift | 579 +++ .../PaymentSheetViewController.swift | 654 ++++ .../PollingViewController.swift | 344 ++ .../PaymentSheet/Views/AUBECSMandate.swift | 90 + .../PaymentSheet/Views/AffirmCopyLabel.swift | 45 + .../Views/AfterpayPriceBreakdownView.swift | 176 + .../Views/Appearance+FontScaling.swift | 27 + .../PaymentSheet/Views/CardScanButton.swift | 32 + .../PaymentSheet/Views/CardScanningView.swift | 258 ++ .../PaymentSheet/Views/CircularButton.swift | 147 + .../PaymentSheet/Views/ConfirmButton.swift | 669 ++++ .../Views/ManualEntryButton.swift | 42 + .../Views/PayWithLinkButton.swift | 283 ++ .../Views/PaymentSheetUIKitAdditions.swift | 122 + .../Views/RotatingCardBrandsView.swift | 203 + .../Views/ShadowedRoundedRectangleView.swift | 89 + .../Views/SheetNavigationBar.swift | 195 + .../Views/SheetNavigationButton.swift | 48 + .../Views/SimpleMandateTextView.swift | 47 + .../PaymentSheet/Views/TestModeView.swift | 50 + .../StripePaymentSheet/StripePaymentSheet.h | 18 + .../DictionaryTests.swift | 44 + .../StripePaymentSheetTests/Info.plist | 22 + ...STPAnalyticsClient+PaymentSheetTests.swift | 25 + StripePayments/Project.swift | 13 + StripePayments/README.md | 38 + .../StripePayments.xcodeproj/project.pbxproj | 1415 +++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripePayments.xcscheme | 95 + StripePayments/StripePayments/Info.plist | 22 + .../bg-BG.lproj/Localizable.strings | 59 + .../ca-ES.lproj/Localizable.strings | 59 + .../cs-CZ.lproj/Localizable.strings | 59 + .../da.lproj/Localizable.strings | 59 + .../de.lproj/Localizable.strings | 59 + .../el-GR.lproj/Localizable.strings | 59 + .../en-GB.lproj/Localizable.strings | 59 + .../en.lproj/Localizable.strings | 90 + .../es-419.lproj/Localizable.strings | 59 + .../es.lproj/Localizable.strings | 59 + .../et-EE.lproj/Localizable.strings | 59 + .../fi.lproj/Localizable.strings | 59 + .../fil.lproj/Localizable.strings | 59 + .../fr-CA.lproj/Localizable.strings | 59 + .../fr.lproj/Localizable.strings | 59 + .../hr.lproj/Localizable.strings | 59 + .../hu.lproj/Localizable.strings | 59 + .../id.lproj/Localizable.strings | 59 + .../it.lproj/Localizable.strings | 59 + .../ja.lproj/Localizable.strings | 59 + .../ko.lproj/Localizable.strings | 59 + .../lt-LT.lproj/Localizable.strings | 59 + .../lv-LV.lproj/Localizable.strings | 59 + .../ms-MY.lproj/Localizable.strings | 59 + .../mt.lproj/Localizable.strings | 59 + .../nb.lproj/Localizable.strings | 59 + .../nl.lproj/Localizable.strings | 59 + .../nn-NO.lproj/Localizable.strings | 59 + .../pl-PL.lproj/Localizable.strings | 59 + .../pt-BR.lproj/Localizable.strings | 59 + .../pt-PT.lproj/Localizable.strings | 59 + .../ro-RO.lproj/Localizable.strings | 59 + .../ru.lproj/Localizable.strings | 59 + .../sk-SK.lproj/Localizable.strings | 59 + .../sl-SI.lproj/Localizable.strings | 59 + .../sv.lproj/Localizable.strings | 59 + .../tk.lproj/Localizable.strings | 0 .../tr.lproj/Localizable.strings | 59 + .../vi.lproj/Localizable.strings | 59 + .../zh-HK.lproj/Localizable.strings | 59 + .../zh-Hans.lproj/Localizable.strings | 59 + .../zh-Hant.lproj/Localizable.strings | 59 + .../StripeAPI+Deprecated.swift | 169 + .../StripeApplePay+Import.swift | 9 + .../StripeCore+Import.swift | 10 + .../Models/ACH/LinkAccountSession.swift | 51 + .../ACH/STPCollectBankAccountParams.swift | 39 + .../STPConfirmAlipayOptions.swift | 57 + .../STPConfirmBLIKOptions.swift | 54 + .../STPConfirmCardOptions.swift | 53 + .../STPConfirmPaymentMethodOptions.swift | 66 + .../STPConfirmUSBankAccountOptions.swift | 47 + .../STPConfirmWeChatPayOptions.swift | 60 + .../PaymentIntents/STPPaymentIntent.swift | 390 ++ .../STPPaymentIntentAction.swift | 16 + .../STPPaymentIntentActionRedirectToURL.swift | 19 + .../STPPaymentIntentEnums.swift | 135 + .../STPPaymentIntentLastPaymentError.swift | 174 + .../STPPaymentIntentParams.swift | 288 ++ .../STPPaymentIntentShippingDetails.swift | 90 + ...PPaymentIntentShippingDetailsAddress.swift | 99 + ...ntIntentShippingDetailsAddressParams.swift | 116 + ...TPPaymentIntentShippingDetailsParams.swift | 110 + .../STPPaymentIntentSourceAction.swift | 17 + ...ntIntentSourceActionAuthorizeWithURL.swift | 21 + .../PaymentMethods/STPPaymentMethod.swift | 328 ++ .../STPPaymentMethodAddress.swift | 143 + .../STPPaymentMethodBillingDetails.swift | 132 + .../STPPaymentMethodEnums.swift | 142 + .../STPPaymentMethodParams.swift | 1212 ++++++ .../Types/STPPaymentMethodAUBECSDebit.swift | 64 + .../STPPaymentMethodAUBECSDebitParams.swift | 31 + .../Types/STPPaymentMethodAffirm.swift | 44 + .../Types/STPPaymentMethodAffirmParams.swift | 23 + .../STPPaymentMethodAfterpayClearpay.swift | 43 + ...PPaymentMethodAfterpayClearpayParams.swift | 24 + .../Types/STPPaymentMethodAlipay.swift | 43 + .../Types/STPPaymentMethodAlipayParams.swift | 27 + .../Types/STPPaymentMethodBLIK.swift | 44 + .../Types/STPPaymentMethodBLIKParams.swift | 27 + .../Types/STPPaymentMethodBacsDebit.swift | 54 + .../STPPaymentMethodBacsDebitParams.swift | 33 + .../Types/STPPaymentMethodBancontact.swift | 41 + .../STPPaymentMethodBancontactParams.swift | 22 + .../Types/STPPaymentMethodBoleto.swift | 56 + .../Types/STPPaymentMethodBoletoParams.swift | 49 + .../Types/STPPaymentMethodCard.swift | 124 + .../Types/STPPaymentMethodCardChecks.swift | 102 + .../Types/STPPaymentMethodCardNetworks.swift | 53 + .../Types/STPPaymentMethodCardParams.swift | 124 + .../Types/STPPaymentMethodCardPresent.swift | 38 + .../Types/STPPaymentMethodCardWallet.swift | 111 + ...STPPaymentMethodCardWalletMasterpass.swift | 66 + ...PPaymentMethodCardWalletVisaCheckout.swift | 47 + .../Types/STPPaymentMethodCashApp.swift | 43 + .../Types/STPPaymentMethodCashAppParams.swift | 24 + .../Types/STPPaymentMethodEPS.swift | 42 + .../Types/STPPaymentMethodEPSParams.swift | 22 + .../Types/STPPaymentMethodFPX.swift | 46 + .../Types/STPPaymentMethodFPXParams.swift | 47 + .../Types/STPPaymentMethodGiropay.swift | 41 + .../Types/STPPaymentMethodGiropayParams.swift | 22 + .../Types/STPPaymentMethodGrabPay.swift | 40 + .../Types/STPPaymentMethodGrabPayParams.swift | 23 + .../Types/STPPaymentMethodKlarna.swift | 45 + .../Types/STPPaymentMethodKlarnaParams.swift | 24 + .../Types/STPPaymentMethodLink.swift | 42 + .../Types/STPPaymentMethodLinkParams.swift | 35 + .../Types/STPPaymentMethodNetBanking.swift | 50 + .../STPPaymentMethodNetBankingParams.swift | 29 + .../Types/STPPaymentMethodOXXO.swift | 40 + .../Types/STPPaymentMethodOXXOParams.swift | 25 + .../Types/STPPaymentMethodPayPal.swift | 42 + .../Types/STPPaymentMethodPayPalParams.swift | 24 + .../Types/STPPaymentMethodPrzelewy24.swift | 42 + .../STPPaymentMethodPrzelewy24Params.swift | 24 + .../Types/STPPaymentMethodSEPADebit.swift | 69 + .../STPPaymentMethodSEPADebitParams.swift | 30 + .../Types/STPPaymentMethodSofort.swift | 48 + .../Types/STPPaymentMethodSofortParams.swift | 29 + .../STPPaymentMethodThreeDSecureUsage.swift | 54 + .../Types/STPPaymentMethodUPI.swift | 50 + .../Types/STPPaymentMethodUPIParams.swift | 29 + .../Types/STPPaymentMethodUSBankAccount.swift | 213 ++ .../STPPaymentMethodUSBankAccountParams.swift | 80 + .../Types/STPPaymentMethodWeChatPay.swift | 42 + .../STPPaymentMethodWeChatPayParams.swift | 24 + .../Types/STPPaymentMethodiDEAL.swift | 51 + .../Types/STPPaymentMethodiDEALParams.swift | 30 + .../Models/STPAPIResponseDecodable.swift | 23 + .../API Bindings/Models/STPAddress.swift | 305 ++ .../API Bindings/Models/STPCardBrand.swift | 76 + .../Models/STPConnectAccountAddress.swift | 82 + .../STPConnectAccountCompanyParams.swift | 107 + .../STPConnectAccountIndividualParams.swift | 246 ++ .../Models/STPConnectAccountParams.swift | 161 + .../API Bindings/Models/STPContactField.swift | 39 + .../API Bindings/Models/STPCustomer.swift | 268 ++ .../API Bindings/Models/STPFPXBankBrand.swift | 333 ++ .../Source/API Bindings/Models/STPFile.swift | 139 + .../Models/STPFormEncodable.swift | 23 + .../Models/STPIssuingCardPin.swift | 55 + .../API Bindings/Models/STPRadarSession.swift | 38 + .../Source/API Bindings/Models/STPToken.swift | 160 + .../API Bindings/Models/STPiDEALBank.swift | 59 + .../Models/SetupIntents/STPSetupIntent.swift | 270 ++ .../STPSetupIntentConfirmParams.swift | 164 + .../SetupIntents/STPSetupIntentEnums.swift | 41 + .../STPSetupIntentLastSetupError.swift | 140 + .../Models/Shared/LinkSettings.swift | 50 + .../Models/Shared/STPIntentAction.swift | 358 ++ .../STPIntentActionAlipayHandleRedirect.swift | 126 + .../STPIntentActionBoletoDisplayDetails.swift | 68 + .../STPIntentActionCashAppRedirectToApp.swift | 65 + .../STPIntentActionOXXODisplayDetails.swift | 71 + .../Shared/STPIntentActionRedirectToURL.swift | 79 + ...PIntentActionVerifyWithMicrodeposits.swift | 88 + ...TPIntentActionWeChatPayRedirectToApp.swift | 64 + .../STPMandateCustomerAcceptanceParams.swift | 70 + .../Models/Shared/STPMandateDataParams.swift | 52 + .../Shared/STPMandateOnlineParams.swift | 68 + .../Shared/STPPaymentMethodOptions.swift | 120 + .../Models/Sources/STPSource.swift | 331 ++ .../Models/Sources/STPSourceEnums.swift | 100 + .../Models/Sources/STPSourceOwner.swift | 61 + .../Models/Sources/STPSourceParams.swift | 975 +++++ .../Models/Sources/STPSourceProtocol.swift | 18 + .../Models/Sources/STPSourceReceiver.swift | 64 + .../Models/Sources/STPSourceRedirect.swift | 113 + .../Sources/STPSourceVerification.swift | 100 + .../Models/Sources/Types/STPBankAccount.swift | 202 + .../Sources/Types/STPBankAccountParams.swift | 128 + .../Models/Sources/Types/STPCard.swift | 340 ++ .../Models/Sources/Types/STPCardParams.swift | 212 ++ .../Sources/Types/STPKlarnaLineItem.swift | 54 + .../Sources/Types/STPSourceCardDetails.swift | 135 + .../Types/STPSourceKlarnaDetails.swift | 50 + .../Types/STPSourceSEPADebitDetails.swift | 74 + .../Types/STPSourceWeChatPayDetails.swift | 44 + .../API Bindings/STPAPIClient+ApplePay.swift | 198 + .../STPAPIClient+LinkAccountSession.swift | 128 + .../API Bindings/STPAPIClient+Payments.swift | 952 +++++ .../API Bindings/STPAPIClient+Radar.swift | 52 + .../API Bindings/STPRedirectContext.swift | 602 +++ .../Enums+CustomStringConvertible.swift | 885 +++++ .../Source/Helpers/STPBINController.swift | 425 +++ .../Helpers/STPBankAccountCollector.swift | 462 +++ .../Source/Helpers/STPBlocks.swift | 122 + .../Source/Helpers/STPCardValidator.swift | 508 +++ .../Source/Helpers/STPLocalizedString.swift | 16 + .../STPPaymentConfirmation+SwiftUI.swift | 152 + .../Helpers/StripePayments+Export.swift | 10 + .../Helpers/StripePaymentsBundleLocator.swift | 19 + .../Internal/API Bindings/APIRequest.swift | 197 + .../STP3DS2AuthenticateResponse.swift | 81 + .../API Bindings/STPEmptyStripeResponse.swift | 30 + .../API Bindings/STPFormEncoder.swift | 97 + .../STPIntentActionUseStripeSDK.swift | 224 ++ .../STPInternalAPIResponseDecodable.swift | 16 + .../STPPaymentMethodListDeserializer.swift | 44 + .../API Bindings/STPSourcePoller.swift | 234 ++ .../Analytics/Analytic+Payments.swift | 59 + .../STPAnalyticsClient+Payments.swift | 350 ++ .../Internal/Categories/NSArray+Stripe.swift | 38 + .../NSDecimalNumber+Stripe_Currency.swift | 52 + .../Categories/NSDictionary+Stripe.swift | 131 + .../Internal/Categories/NSString+Stripe.swift | 73 + .../STPAPIClient+PaymentsCore.swift | 20 + .../Helpers/ConnectionsSDKAvailability.swift | 76 + .../STPPaymentHandlerActionParams.swift | 180 + .../STPAuthenticationContext.swift | 36 + .../PaymentHandler/STPPaymentHandler.swift | 2293 ++++++++++++ .../STPThreeDSButtonCustomization.swift | 126 + .../STPThreeDSCustomizationSettings.swift | 44 + .../STPThreeDSFooterCustomization.swift | 88 + .../STPThreeDSLabelCustomization.swift | 67 + ...STPThreeDSNavigationBarCustomization.swift | 101 + .../STPThreeDSSelectionCustomization.swift | 72 + .../STPThreeDSTextFieldCustomization.swift | 94 + .../STPThreeDSUICustomization.swift | 193 + .../StripePayments/StripePayments.h | 18 + StripePayments/StripePaymentsTests/Info.plist | 22 + .../STPAnalyticsClient+StripePayments.swift | 30 + StripePaymentsUI/Project.swift | 14 + StripePaymentsUI/README.md | 44 + .../project.pbxproj | 1147 ++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripePaymentsUI.xcscheme | 95 + StripePaymentsUI/StripePaymentsUI/Info.plist | 22 + .../Resources/Images/BECS/anz@3x.png | Bin 0 -> 1063 bytes .../Images/BECS/bankofmelbourne@3x.png | Bin 0 -> 2297 bytes .../Resources/Images/BECS/banksa@3x.png | Bin 0 -> 2060 bytes .../Resources/Images/BECS/bankwest@3x.png | Bin 0 -> 2310 bytes .../Resources/Images/BECS/boq@3x.png | Bin 0 -> 1727 bytes .../Resources/Images/BECS/cba@3x.png | Bin 0 -> 421 bytes .../Resources/Images/BECS/nab@3x.png | Bin 0 -> 1791 bytes .../Resources/Images/BECS/stgeorges@3x.png | Bin 0 -> 2817 bytes .../Images/BECS/suncorpmetway@3x.png | Bin 0 -> 2256 bytes .../Resources/Images/BECS/westpac@3x.png | Bin 0 -> 627 bytes .../Images/Cards/stp_card_amex@3x.png | Bin 0 -> 891 bytes .../Cards/stp_card_amex_template@3x.png | Bin 0 -> 751 bytes .../Images/Cards/stp_card_applepay@3x.png | Bin 0 -> 2185 bytes .../Cards/stp_card_cartes_bancaires@3x.png | Bin 0 -> 5160 bytes .../stp_card_cartes_bancaires_template@3x.png | Bin 0 -> 600 bytes .../Images/Cards/stp_card_cvc@3x.png | Bin 0 -> 1408 bytes .../Images/Cards/stp_card_cvc_amex@3x.png | Bin 0 -> 1526 bytes .../Images/Cards/stp_card_diners@3x.png | Bin 0 -> 913 bytes .../Cards/stp_card_diners_template@3x.png | Bin 0 -> 982 bytes .../Images/Cards/stp_card_discover@3x.png | Bin 0 -> 1048 bytes .../Cards/stp_card_discover_template@3x.png | Bin 0 -> 1329 bytes .../Images/Cards/stp_card_error@3x.png | Bin 0 -> 960 bytes .../Images/Cards/stp_card_error_amex@3x.png | Bin 0 -> 974 bytes .../Images/Cards/stp_card_jcb@3x.png | Bin 0 -> 1305 bytes .../Images/Cards/stp_card_jcb_template@3x.png | Bin 0 -> 846 bytes .../Images/Cards/stp_card_mastercard@3x.png | Bin 0 -> 974 bytes .../Cards/stp_card_mastercard_template@3x.png | Bin 0 -> 1189 bytes .../Images/Cards/stp_card_unionpay@3x.png | Bin 0 -> 1895 bytes .../Cards/stp_card_unionpay_template@3x.png | Bin 0 -> 2275 bytes .../Images/Cards/stp_card_unknown@3x.png | Bin 0 -> 338 bytes .../Images/Cards/stp_card_visa@3x.png | Bin 0 -> 987 bytes .../Cards/stp_card_visa_template@3x.png | Bin 0 -> 1432 bytes .../Resources/Images/Cardsv2/card_amex@3x.png | Bin 0 -> 876 bytes .../Cardsv2/card_cartes_bancaires@3x.png | Bin 0 -> 10115 bytes .../Images/Cardsv2/card_cvc_amex_icon@3x.png | Bin 0 -> 1204 bytes .../Cardsv2/card_cvc_amex_icon_dark@3x.png | Bin 0 -> 1138 bytes .../Cardsv2/card_cvc_amex_updated_icon@3x.png | Bin 0 -> 736 bytes .../card_cvc_amex_updated_icon_dark@3x.png | Bin 0 -> 747 bytes .../Images/Cardsv2/card_cvc_icon@3x.png | Bin 0 -> 989 bytes .../Images/Cardsv2/card_cvc_icon_dark@3x.png | Bin 0 -> 918 bytes .../Cardsv2/card_cvc_updated_icon@3x.png | Bin 0 -> 623 bytes .../Cardsv2/card_cvc_updated_icon_dark@3x.png | Bin 0 -> 636 bytes .../Images/Cardsv2/card_diners@3x.png | Bin 0 -> 1667 bytes .../Images/Cardsv2/card_discover@3x.png | Bin 0 -> 1860 bytes .../Resources/Images/Cardsv2/card_jcb@3x.png | Bin 0 -> 1533 bytes .../Images/Cardsv2/card_mastercard@3x.png | Bin 0 -> 1854 bytes .../Images/Cardsv2/card_unionpay@3x.png | Bin 0 -> 2724 bytes .../Images/Cardsv2/card_unknown@3x.png | Bin 0 -> 818 bytes .../Images/Cardsv2/card_unknown_icon@3x.png | Bin 0 -> 392 bytes .../Cardsv2/card_unknown_icon_dark@3x.png | Bin 0 -> 387 bytes .../Cardsv2/card_unknown_updated_icon@3x.png | Bin 0 -> 371 bytes .../card_unknown_updated_icon_dark@3x.png | Bin 0 -> 375 bytes .../Resources/Images/Cardsv2/card_visa@3x.png | Bin 0 -> 1925 bytes .../Resources/Images/form_error_icon@3x.png | Bin 0 -> 594 bytes .../Images/form_error_icon_dark@3x.png | Bin 0 -> 594 bytes .../Resources/Images/stp_icon_bank@3x.png | Bin 0 -> 586 bytes .../Resources/JSON/au_becs_bsb.json | 373 ++ .../bg-BG.lproj/Localizable.strings | 65 + .../ca-ES.lproj/Localizable.strings | 65 + .../cs-CZ.lproj/Localizable.strings | 65 + .../da.lproj/Localizable.strings | 65 + .../de.lproj/Localizable.strings | 65 + .../el-GR.lproj/Localizable.strings | 65 + .../en-GB.lproj/Localizable.strings | 65 + .../en.lproj/Localizable.strings | 99 + .../es-419.lproj/Localizable.strings | 65 + .../es.lproj/Localizable.strings | 65 + .../et-EE.lproj/Localizable.strings | 65 + .../fi.lproj/Localizable.strings | 65 + .../fil.lproj/Localizable.strings | 65 + .../fr-CA.lproj/Localizable.strings | 65 + .../fr.lproj/Localizable.strings | 65 + .../hr.lproj/Localizable.strings | 65 + .../hu.lproj/Localizable.strings | 65 + .../id.lproj/Localizable.strings | 65 + .../it.lproj/Localizable.strings | 65 + .../ja.lproj/Localizable.strings | 65 + .../ko.lproj/Localizable.strings | 65 + .../lt-LT.lproj/Localizable.strings | 65 + .../lv-LV.lproj/Localizable.strings | 65 + .../ms-MY.lproj/Localizable.strings | 65 + .../mt.lproj/Localizable.strings | 65 + .../nb.lproj/Localizable.strings | 65 + .../nl.lproj/Localizable.strings | 65 + .../nn-NO.lproj/Localizable.strings | 65 + .../pl-PL.lproj/Localizable.strings | 65 + .../pt-BR.lproj/Localizable.strings | 65 + .../pt-PT.lproj/Localizable.strings | 65 + .../ro-RO.lproj/Localizable.strings | 65 + .../ru.lproj/Localizable.strings | 65 + .../sk-SK.lproj/Localizable.strings | 65 + .../sl-SI.lproj/Localizable.strings | 65 + .../sv.lproj/Localizable.strings | 65 + .../tk.lproj/Localizable.strings | 0 .../tr.lproj/Localizable.strings | 65 + .../vi.lproj/Localizable.strings | 65 + .../zh-HK.lproj/Localizable.strings | 65 + .../zh-Hans.lproj/Localizable.strings | 65 + .../zh-Hant.lproj/Localizable.strings | 65 + .../Enums+CustomStringConvertible.swift | 49 + .../STPAPIClient+CustomerContext.swift | 135 + .../STPBECSDebitAccountNumberValidator.swift | 84 + .../Helpers/STPBSBNumberValidator.swift | 143 + .../Source/Helpers/STPImageLibrary.swift | 148 + .../Source/Helpers/STPLocalizedString.swift | 16 + .../Helpers/STPPhoneNumberValidator.swift | 116 + .../Helpers/STPPostalCodeValidator.swift | 296 ++ .../Source/Helpers/STPPromise.swift | 138 + .../Source/Helpers/STPStringUtils.swift | 69 + .../Source/Helpers/String+Localized.swift | 145 + .../Helpers/StripePayments+Export.swift | 11 + .../Helpers/StripePaymentsBundleLocator.swift | 19 + .../NSAttributedString+Stripe.swift | 19 + .../Internal/Categories/UIButton+Stripe.swift | 43 + .../Internal/UI/Views/CardBrandView.swift | 206 ++ .../Card/STPCardCVCInputTextField.swift | 93 + .../STPCardCVCInputTextFieldFormatter.swift | 37 + .../STPCardCVCInputTextFieldValidator.swift | 51 + .../Card/STPCardExpiryInputTextField.swift | 51 + ...STPCardExpiryInputTextFieldFormatter.swift | 84 + ...STPCardExpiryInputTextFieldValidator.swift | 81 + .../Card/STPCardNumberInputTextField.swift | 133 + ...STPCardNumberInputTextFieldFormatter.swift | 82 + ...STPCardNumberInputTextFieldValidator.swift | 85 + .../Card/STPPostalCodeInputTextField.swift | 89 + ...STPPostalCodeInputTextFieldFormatter.swift | 48 + ...STPPostalCodeInputTextFieldValidator.swift | 76 + .../Inputs/STPCountryPickerInputField.swift | 134 + .../Inputs/STPGenericInputPickerField.swift | 229 ++ .../Inputs/STPGenericInputTextField.swift | 62 + .../UI/Views/Inputs/STPInputTextField.swift | 298 ++ .../Inputs/STPInputTextFieldFormatter.swift | 62 + .../Inputs/STPInputTextFieldValidator.swift | 65 + .../STPNumericDigitInputTextFormatter.swift | 50 + .../UI/Views/STPAUBECSFormViewModel.swift | 198 + .../UI/Views/STPCardLoadingIndicator.swift | 102 + .../Internal/UI/Views/STPFormTextField.swift | 486 +++ .../Views/STPLabeledFormTextFieldView.swift | 115 + .../STPLabeledMultiFormTextFieldView.swift | 132 + .../STPPaymentCardTextFieldViewModel.swift | 246 ++ .../UI/Views/STPValidatedTextField.swift | 99 + .../UI/Views/STPViewWithSeparator.swift | 95 + .../PersistablePaymentMethodOption.swift | 104 + .../CustomerContext/UserDefaults+Stripe.swift | 29 + .../_stpspmsbeta_STPBackendAPIAdapter.swift | 98 + .../_stpspmsbeta_STPCustomerContext.swift | 439 +++ .../_stpspmsbeta_STPEphemeralKey.swift | 104 + .../_stpspmsbeta_STPEphemeralKeyManager.swift | 178 + ..._stpspmsbeta_STPEphemeralKeyProvider.swift | 74 + ...entMethodMessagingView+Configuration.swift | 86 + .../PaymentMethodMessagingView.swift | 353 ++ .../STPAUBECSDebitFormView.swift | 588 +++ .../STPCardFormView+SwiftUI.swift | 68 + .../UI Components/STPCardFormView.swift | 801 ++++ .../STPFloatingPlaceholderTextField.swift | 422 +++ .../STPFormTextFieldContainer.swift | 33 + .../Source/UI Components/STPFormView.swift | 695 ++++ .../UI Components/STPMultiFormTextField.swift | 358 ++ .../STPPaymentCardTextField+SwiftUI.swift | 68 + .../STPPaymentCardTextField.swift | 2283 ++++++++++++ .../StripePaymentsUI/StripePaymentsUI.h | 18 + .../StripePaymentsUITests/Info.plist | 22 + .../STPAnalyticsClient+PaymentsUITests.swift | 33 + StripeUICore/Project.swift | 16 + .../StripeUICore.xcodeproj/project.pbxproj | 1174 ++++++ .../contents.xcworkspacedata | 7 + .../xcschemes/StripeUICore.xcscheme | 104 + StripeUICore/StripeUICore/Info.plist | 22 + .../Resources/Images/brand_stripe@3x.png | Bin 0 -> 790 bytes .../Resources/Images/form_error_icon@3x.png | Bin 0 -> 594 bytes .../Resources/Images/icon_chevron_down@3x.png | Bin 0 -> 256 bytes .../Resources/Images/icon_clear@3x.png | Bin 0 -> 564 bytes .../Resources/JSON/au_becs_bsb.json | 123 + .../JSON/localized_address_data.json | 1197 ++++++ .../bg-BG.lproj/Localizable.strings | 121 + .../ca-ES.lproj/Localizable.strings | 121 + .../cs-CZ.lproj/Localizable.strings | 121 + .../da.lproj/Localizable.strings | 121 + .../de.lproj/Localizable.strings | 121 + .../el-GR.lproj/Localizable.strings | 121 + .../en-GB.lproj/Localizable.strings | 121 + .../en.lproj/Localizable.strings | 190 + .../es-419.lproj/Localizable.strings | 121 + .../es.lproj/Localizable.strings | 121 + .../et-EE.lproj/Localizable.strings | 121 + .../fi.lproj/Localizable.strings | 121 + .../fil.lproj/Localizable.strings | 121 + .../fr-CA.lproj/Localizable.strings | 121 + .../fr.lproj/Localizable.strings | 121 + .../hr.lproj/Localizable.strings | 121 + .../hu.lproj/Localizable.strings | 121 + .../id.lproj/Localizable.strings | 121 + .../it.lproj/Localizable.strings | 121 + .../ja.lproj/Localizable.strings | 121 + .../ko.lproj/Localizable.strings | 121 + .../lt-LT.lproj/Localizable.strings | 121 + .../lv-LV.lproj/Localizable.strings | 121 + .../ms-MY.lproj/Localizable.strings | 121 + .../mt.lproj/Localizable.strings | 121 + .../nb.lproj/Localizable.strings | 121 + .../nl.lproj/Localizable.strings | 121 + .../nn-NO.lproj/Localizable.strings | 121 + .../pl-PL.lproj/Localizable.strings | 121 + .../pt-BR.lproj/Localizable.strings | 121 + .../pt-PT.lproj/Localizable.strings | 121 + .../ro-RO.lproj/Localizable.strings | 121 + .../ru.lproj/Localizable.strings | 121 + .../sk-SK.lproj/Localizable.strings | 121 + .../sl-SI.lproj/Localizable.strings | 121 + .../sv.lproj/Localizable.strings | 121 + .../tr.lproj/Localizable.strings | 121 + .../vi.lproj/Localizable.strings | 121 + .../zh-HK.lproj/Localizable.strings | 121 + .../zh-Hans.lproj/Localizable.strings | 121 + .../zh-Hant.lproj/Localizable.strings | 121 + .../Categories/CALayer+StripeUICore.swift | 26 + .../Enums+CustomStringConvertible.swift | 7 + .../Categories/Locale+StripeUICore.swift | 40 + ...NSDirectionalEdgeInsets+StripeUICore.swift | 19 + .../UIBarButtonItem+StripeUICore.swift | 21 + .../Categories/UIButton+StripeUICore.swift | 50 + .../Categories/UIColor+StripeUICore.swift | 182 + .../Categories/UIFont+StripeUICore.swift | 31 + .../UIKeyboardType+StripeUICore.swift | 33 + ...ISpringTimingParameters+StripeUICore.swift | 22 + .../Categories/UIStackView+StripeUICore.swift | 122 + .../UITraitCollection+StripeUICore.swift | 17 + .../Categories/UIView+StripeUICore.swift | 94 + .../UIViewController+StripeUICore.swift | 53 + .../Categories/UIWindow+StripeUICore.swift | 20 + .../Source/Controls/ActivityIndicator.swift | 263 ++ .../StripeUICore/Source/Controls/Button.swift | 557 +++ .../OneTimeCodeTextField-TextStorage.swift | 217 ++ .../Controls/OneTimeCodeTextField.swift | 771 ++++ .../Elements/Checkbox/CheckboxButton.swift | 332 ++ .../Elements/Checkbox/CheckboxElement.swift | 59 + .../Source/Elements/ContainerElement.swift | 63 + .../Source/Elements/DateFieldElement.swift | 187 + .../Elements/DropdownFieldElement.swift | 193 + .../Source/Elements/Element.swift | 116 + .../Source/Elements/ElementsUI.swift | 123 + ...dressSectionElement+DummyAddressLine.swift | 68 + .../Address/AddressSectionElement.swift | 429 +++ .../Address/AddressSpec+ElementFactory.swift | 55 + .../Factories/Address/AddressSpec.swift | 147 + .../Address/AddressSpecProvider.swift | 58 + .../Factories/BSB/BSBNumberProvider.swift | 53 + .../DropdownFieldElement+AddressFactory.swift | 47 + .../IDNumberTextFieldConfiguration.swift | 142 + .../TextFieldElement+AccountFactory.swift | 82 + .../TextFieldElement+AddressFactory.swift | 147 + .../Factories/TextFieldElement+Factory.swift | 174 + .../Source/Elements/Form/FormElement.swift | 75 + .../Source/Elements/Form/FormView.swift | 51 + .../PhoneNumber/PhoneNumberElement.swift | 170 + .../PickerField/PickerFieldView.swift | 211 ++ .../PickerField/PickerTextField.swift | 39 + .../Section/SectionContainerView.swift | 226 ++ .../SectionElement+MultiElementRow.swift | 30 + .../Elements/Section/SectionElement.swift | 104 + .../Source/Elements/Section/SectionView.swift | 72 + .../Source/Elements/StaticElement.swift | 26 + .../FloatingPlaceholderTextFieldView.swift | 199 + .../TextFieldElement+Validation.swift | 87 + .../Elements/TextField/TextFieldElement.swift | 179 + .../TextFieldElementConfiguration.swift | 135 + .../TextField/TextFieldFormatter.swift | 108 + .../Elements/TextField/TextFieldView.swift | 290 ++ .../Elements/TextOrDropdownElement.swift | 47 + StripeUICore/StripeUICore/Source/Events.swift | 30 + .../Source/Helpers/CompatibleColor.swift | 15 + .../Source/Helpers/ImageMaker.swift | 71 + .../Source/Helpers/InputFormColors.swift | 45 + .../Source/Helpers/RegionCodeProvider.swift | 14 + .../Source/Helpers/STPLocalizedString.swift | 13 + .../Helpers/StackViewWithSeparator.swift | 213 ++ .../Source/Helpers/String+CountryEmoji.swift | 26 + .../Source/Helpers/String+Localized.swift | 296 ++ .../Helpers/String+RegionCodeProvider.swift | 16 + .../Helpers/StripeUICoreBundleLocator.swift | 19 + StripeUICore/StripeUICore/Source/Image.swift | 30 + .../Source/Validators/BSBNumber.swift | 43 + .../Source/Validators/PhoneNumber.swift | 436 +++ .../Validators/STPEmailAddressValidator.swift | 29 + .../Validators/STPVPANumberValidator.swift | 28 + .../Source/Views/DoneButtonToolbar.swift | 49 + .../Views/DynamicHeightContainerView.swift | 77 + .../Source/Views/DynamicImageView.swift | 56 + .../Source/Views/LinkOpeningTextView.swift | 57 + StripeUICore/StripeUICore/StripeUICore.h | 18 + StripeUICore/StripeUICoreTests/Info.plist | 22 + .../Controls/ButtonSnapshotTest.swift | 105 + .../AddressSectionElementSnapshotTest.swift | 38 + .../CheckboxButtonSnapshotTests.swift | 107 + .../DateFieldElementSnapshotTest.swift | 86 + .../DropdownFieldElementSnapshotTest.swift | 63 + .../PhoneNumberElementSnapshotTests.swift | 69 + .../Categories/Locale+StripeUICoreTests.swift | 73 + .../UIColor+StripeUICoreTests.swift | 222 ++ .../Elements/AddressSectionElementTest.swift | 233 ++ .../Elements/AddressSpecProviderTest.swift | 50 + .../Unit/Elements/BSBNumberProviderTest.swift | 76 + .../Unit/Elements/DateFieldElementTest.swift | 85 + .../Elements/DropdownFieldElementTest.swift | 75 + .../IDNumberTextFieldConfigurationTest.swift | 108 + .../Elements/PhoneNumberElementTests.swift | 177 + .../Unit/Elements/SectionElementTest.swift | 61 + .../TestFieldElement+AccountFactoryTest.swift | 62 + .../TextFieldElement+AddressFactoryTest.swift | 137 + .../Unit/Elements/TextFieldElementTest.swift | 78 + .../Elements/TextFieldFormatterTest.swift | 70 + .../Unit/Validators/BSBNumberTests.swift | 81 + .../Unit/Validators/PhoneNumberTests.swift | 179 + .../STPEmailAddressValidatorTest.swift | 26 + .../STPVPANumberValidatorTest.swift | 31 + VERSION | 1 + 2489 files changed, 273722 insertions(+), 3 deletions(-) create mode 100644 CHANGELOG.md create mode 100644 MIGRATING.md create mode 100644 Package.swift create mode 100644 Stripe/BuildConfigurations/Stripe Tests-Debug.xcconfig create mode 100644 Stripe/BuildConfigurations/Stripe Tests-Release.xcconfig create mode 100644 Stripe/BuildConfigurations/Stripe-Debug.xcconfig create mode 100644 Stripe/BuildConfigurations/Stripe-Release.xcconfig create mode 100644 Stripe/Project.swift create mode 100644 Stripe/Stripe.xcodeproj/project.pbxproj create mode 100644 Stripe/Stripe.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOS.xcscheme create mode 100644 Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOSTestHostApp.xcscheme create mode 100644 Stripe/StripeiOS/Info.plist create mode 100644 Stripe/StripeiOS/Resources/Images/Cards/stp_card_form_amex_cvc@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/Cards/stp_card_form_back@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/Cards/stp_card_form_front@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_affin_bank@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_alliance_bank@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_ambank@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bank_islam@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bank_muamalat@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bank_rakyat@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bsn@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_cimb@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_hong_leong_bank@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_hsbc@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_kfh@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_maybank2e@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_maybank2u@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_ocbc@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_public_bank@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_rhb@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_standard_chartered@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_uob@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_fpx_big_logo@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/FPX/stp_fpx_logo@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/stp_icon_add@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/stp_icon_bank@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/stp_icon_checkmark@3x.png create mode 100644 Stripe/StripeiOS/Resources/Images/stp_shipping_form@3x.png create mode 100644 Stripe/StripeiOS/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/tk.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 Stripe/StripeiOS/Source/Enums+CustomStringConvertible.swift create mode 100644 Stripe/StripeiOS/Source/PKAddPaymentPassRequest+Stripe_Error.swift create mode 100644 Stripe/StripeiOS/Source/PKPaymentAuthorizationViewController+Stripe_Blocks.swift create mode 100644 Stripe/StripeiOS/Source/STPAPIClient+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPAPIClient+PushProvisioning.swift create mode 100644 Stripe/StripeiOS/Source/STPAddCardViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPAddress+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPAddressFieldTableViewCell.swift create mode 100644 Stripe/StripeiOS/Source/STPAddressViewModel.swift create mode 100644 Stripe/StripeiOS/Source/STPAnalyticsClient+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPAnalyticsClient+Payments.swift create mode 100644 Stripe/StripeiOS/Source/STPApplePayContextDelegate.swift create mode 100644 Stripe/StripeiOS/Source/STPApplePayPaymentOption.swift create mode 100644 Stripe/StripeiOS/Source/STPBackendAPIAdapter.swift create mode 100644 Stripe/StripeiOS/Source/STPBankSelectionTableViewCell.swift create mode 100644 Stripe/StripeiOS/Source/STPBankSelectionViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPBlocks.swift create mode 100644 Stripe/StripeiOS/Source/STPCameraView.swift create mode 100644 Stripe/StripeiOS/Source/STPCard+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPCardScanner.swift create mode 100644 Stripe/StripeiOS/Source/STPCardScannerTableViewCell.swift create mode 100644 Stripe/StripeiOS/Source/STPCardValidationState.swift create mode 100644 Stripe/StripeiOS/Source/STPCoreScrollViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPCoreTableViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPCoreViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPCustomerContext.swift create mode 100644 Stripe/StripeiOS/Source/STPEphemeralKey.swift create mode 100644 Stripe/StripeiOS/Source/STPEphemeralKeyManager.swift create mode 100644 Stripe/StripeiOS/Source/STPEphemeralKeyProvider.swift create mode 100644 Stripe/StripeiOS/Source/STPFPXBankStatusResponse.swift create mode 100644 Stripe/StripeiOS/Source/STPFakeAddPaymentPassViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPImageLibrary.swift create mode 100644 Stripe/StripeiOS/Source/STPIntentActionLinkAuthenticateAccount.swift create mode 100644 Stripe/StripeiOS/Source/STPLocalizedString.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentActivityIndicatorView.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentCardTextFieldCell.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentConfiguration.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentContext.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentContextAmountModel.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentIntentParams+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentMethod+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentMethodParams+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentOption.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentOptionTableViewCell.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentOptionTuple.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentOptionsInternalViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentOptionsViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPPaymentResult.swift create mode 100644 Stripe/StripeiOS/Source/STPPinManagementService.swift create mode 100644 Stripe/StripeiOS/Source/STPPushProvisioningContext.swift create mode 100644 Stripe/StripeiOS/Source/STPPushProvisioningDetails.swift create mode 100644 Stripe/StripeiOS/Source/STPPushProvisioningDetailsParams.swift create mode 100644 Stripe/StripeiOS/Source/STPSectionHeaderView.swift create mode 100644 Stripe/StripeiOS/Source/STPShippingAddressViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPShippingMethodTableViewCell.swift create mode 100644 Stripe/StripeiOS/Source/STPShippingMethodsViewController.swift create mode 100644 Stripe/StripeiOS/Source/STPSource+BasicUI.swift create mode 100644 Stripe/StripeiOS/Source/STPTheme.swift create mode 100644 Stripe/StripeiOS/Source/STPUserInformation.swift create mode 100644 Stripe/StripeiOS/Source/String+Localized.swift create mode 100644 Stripe/StripeiOS/Source/Stripe+Exports.swift create mode 100644 Stripe/StripeiOS/Source/StripeBundleLocator.swift create mode 100644 Stripe/StripeiOS/Source/UIBarButtonItem+Stripe.swift create mode 100644 Stripe/StripeiOS/Source/UINavigationBar+Stripe_Theme.swift create mode 100644 Stripe/StripeiOS/Source/UINavigationController+Stripe_Completion.swift create mode 100644 Stripe/StripeiOS/Source/UITableViewCell+Stripe_Borders.swift create mode 100644 Stripe/StripeiOS/Source/UIToolbar+Stripe_InputAccessory.swift create mode 100644 Stripe/StripeiOS/Source/UIView+Helpers.swift create mode 100644 Stripe/StripeiOS/Source/UIView+Stripe_FirstResponder.swift create mode 100644 Stripe/StripeiOS/Source/UIView+Stripe_SafeAreaBounds.swift create mode 100644 Stripe/StripeiOS/Source/UIViewController+Stripe_KeyboardAvoiding.swift create mode 100644 Stripe/StripeiOS/Source/UIViewController+Stripe_NavigationItemProxy.swift create mode 100644 Stripe/StripeiOS/Source/UIViewController+Stripe_ParentViewController.swift create mode 100644 Stripe/StripeiOS/Stripe-umbrella.h create mode 100644 Stripe/StripeiOS/Stripe.modulemap create mode 100644 Stripe/StripeiOSAppHostedTests/Info.plist create mode 100644 Stripe/StripeiOSAppHostedTests/LinkSecureCookieStoreTests.swift create mode 100644 Stripe/StripeiOSTestHostApp/AppDelegate.swift create mode 100644 Stripe/StripeiOSTestHostApp/Info.plist create mode 100644 Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AccentColor.colorset/Contents.json create mode 100644 Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AppIcon.appiconset/Contents.json create mode 100644 Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/Contents.json create mode 100644 Stripe/StripeiOSTestHostApp/Resources/Base.lproj/LaunchScreen.storyboard create mode 100644 Stripe/StripeiOSTestHostApp/Resources/Base.lproj/Main.storyboard create mode 100644 Stripe/StripeiOSTestHostApp/ViewController.swift create mode 100644 Stripe/StripeiOSTests.xctestplan create mode 100644 Stripe/StripeiOSTests/APIRequestTest.swift create mode 100644 Stripe/StripeiOSTests/AddPaymentMethodViewControllerSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/AddressViewControllerSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/AfterpayPriceBreakdownViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/AnalyticsHelperTests.swift create mode 100644 Stripe/StripeiOSTests/AutoCompleteViewControllerSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/ButtonLinkSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/CardExpiryDateTests.swift create mode 100644 Stripe/StripeiOSTests/CircularButtonSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/ConfirmButtonSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/ConfirmButtonTests.swift create mode 100644 Stripe/StripeiOSTests/ConsumerSessionTests.swift create mode 100644 Stripe/StripeiOSTests/Error+PaymentSheetTests.swift create mode 100644 Stripe/StripeiOSTests/FBSnapshotTestCase+STPViewControllerLoading.swift create mode 100644 Stripe/StripeiOSTests/FormSpecProviderTest.swift create mode 100644 Stripe/StripeiOSTests/FraudDetectionDataTest.swift create mode 100644 Stripe/StripeiOSTests/ImageTest.swift create mode 100644 Stripe/StripeiOSTests/Info.plist create mode 100644 Stripe/StripeiOSTests/KlarnaHelperTest.swift create mode 100644 Stripe/StripeiOSTests/LinkAccountServiceTests.swift create mode 100644 Stripe/StripeiOSTests/LinkBadgeViewSnapshotTest.swift create mode 100644 Stripe/StripeiOSTests/LinkCardEditElementSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkInMemoryCookieStoreTests.swift create mode 100644 Stripe/StripeiOSTests/LinkInlineSignupElementSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkInstantDebitMandateViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkLegalTermsViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkNavigationBarSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkNoticeViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkPaymentMethodPickerSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkSignupViewModelTests.swift create mode 100644 Stripe/StripeiOSTests/LinkStubs.swift create mode 100644 Stripe/StripeiOSTests/LinkToastSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/LinkVerificationViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/MKPlacemark+PaymentSheetTests.swift create mode 100644 Stripe/StripeiOSTests/NSArray+StripeTest.swift create mode 100644 Stripe/StripeiOSTests/NSDecimalNumber+StripeTest.swift create mode 100644 Stripe/StripeiOSTests/NSDictionary+StripeTest.swift create mode 100644 Stripe/StripeiOSTests/NSLocale+STPSwizzling.h create mode 100644 Stripe/StripeiOSTests/NSLocale+STPSwizzling.m create mode 100644 Stripe/StripeiOSTests/NSString+StripeTest.swift create mode 100644 Stripe/StripeiOSTests/NSURLComponents_StripeTest.swift create mode 100644 Stripe/StripeiOSTests/OneTimeCodeTextFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/OneTimeCodeTextFieldTests.swift create mode 100644 Stripe/StripeiOSTests/OperationDebouncerTests.swift create mode 100644 Stripe/StripeiOSTests/PKPayment+StripeTest.swift create mode 100644 Stripe/StripeiOSTests/PayWithLinkButtonSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/PayWithLinkViewController-WalletViewModelTests.swift create mode 100644 Stripe/StripeiOSTests/PaymentAnalyticTest.swift create mode 100644 Stripe/StripeiOSTests/PaymentMethodMessagingViewFunctionalTest.swift create mode 100644 Stripe/StripeiOSTests/PaymentMethodMessagingViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/PaymentSheet+APITest.swift create mode 100644 Stripe/StripeiOSTests/PaymentSheetAddressTests.swift create mode 100644 Stripe/StripeiOSTests/PaymentSheetFormFactorySnapshotTest.swift create mode 100644 Stripe/StripeiOSTests/PaymentSheetFormFactoryTest.swift create mode 100644 Stripe/StripeiOSTests/PaymentSheetLinkAccountTests.swift create mode 100644 Stripe/StripeiOSTests/PaymentSheetPaymentMethodTypeTest.swift create mode 100644 Stripe/StripeiOSTests/PaymentSheetTestUtils.swift create mode 100644 Stripe/StripeiOSTests/PaymentTypeCellSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/Resources/3DSSource.json create mode 100644 Stripe/StripeiOSTests/Resources/AlipaySource.json create mode 100644 Stripe/StripeiOSTests/Resources/ApplePayPaymentMethod.json create mode 100644 Stripe/StripeiOSTests/Resources/BacsDebitPaymentMethod.json create mode 100644 Stripe/StripeiOSTests/Resources/BancontactSource.json create mode 100644 Stripe/StripeiOSTests/Resources/BankAccount.json create mode 100644 Stripe/StripeiOSTests/Resources/Card.json create mode 100644 Stripe/StripeiOSTests/Resources/CardPaymentMethod.json create mode 100644 Stripe/StripeiOSTests/Resources/CardSource.json create mode 100644 Stripe/StripeiOSTests/Resources/Customer.json create mode 100644 Stripe/StripeiOSTests/Resources/EPSSource.json create mode 100644 Stripe/StripeiOSTests/Resources/ElementsSession.json create mode 100644 Stripe/StripeiOSTests/Resources/EphemeralKey.json create mode 100644 Stripe/StripeiOSTests/Resources/FileUpload.json create mode 100644 Stripe/StripeiOSTests/Resources/GiropaySource.json create mode 100644 Stripe/StripeiOSTests/Resources/Images.xcassets/Contents.json create mode 100644 Stripe/StripeiOSTests/Resources/MockFiles/paymentIntentResponse.json create mode 100644 Stripe/StripeiOSTests/Resources/MultibancoSource.json create mode 100644 Stripe/StripeiOSTests/Resources/P24Source.json create mode 100644 Stripe/StripeiOSTests/Resources/PaymentIntent.json create mode 100644 Stripe/StripeiOSTests/Resources/SEPADebitSource.json create mode 100644 Stripe/StripeiOSTests/Resources/SetupIntent.json create mode 100644 Stripe/StripeiOSTests/Resources/SofortSource.json create mode 100644 Stripe/StripeiOSTests/Resources/WeChatPaySource.json create mode 100644 Stripe/StripeiOSTests/Resources/iDEALSource.json create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_content_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_afterpay_logo_black.png_1.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_klarna_logo_black.png_2.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testInitializingWithBadConfigurationReturnsError/get_content_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateSourceWithPayment/post_v1_tokens_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPayment/post_v1_tokens_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPaymentClassic/post_v1_tokens_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_create_payment_intent_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_3ds2_authenticate_2.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_payment_intents_pi_3KgG1XFY0qyl6XeW1TLgmwD3_confirm_1.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentMethodFunctionalTest/testCreateBacsPaymentMethod/post_v1_payment_methods_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePin/get_v1_issuing_cards_ic_token_pin_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePinWithError/get_v1_issuing_cards_ic_token_pin_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testUpdatePin/post_v1_issuing_cards_ic_token_pin_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPushProvisioningDetailsFunctionalTest/testRetrievePushProvisioningDetails/get_v1_issuing_cards_ic_1C0Xig4JYtv6MPZK91WoXa9u_push_provisioning_details_0.tail create mode 100644 Stripe/StripeiOSTests/Resources/stp_test_upload_image.jpeg create mode 100644 Stripe/StripeiOSTests/RotatingCardBrandsViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/RotatingCardBrandsViewTests.swift create mode 100644 Stripe/StripeiOSTests/STPAPIClientNetworkBridgeTest.m create mode 100644 Stripe/StripeiOSTests/STPAPIClientStubbedTest.swift create mode 100644 Stripe/StripeiOSTests/STPAPIClientTest.swift create mode 100644 Stripe/StripeiOSTests/STPAPISettingsBridgeTest.m create mode 100644 Stripe/StripeiOSTests/STPAUBECSDebitFormViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPAUBECSFormViewModelTests.swift create mode 100644 Stripe/StripeiOSTests/STPAddCardViewControllerLocalizationTests.swift create mode 100644 Stripe/StripeiOSTests/STPAddCardViewControllerTest.swift create mode 100644 Stripe/StripeiOSTests/STPAddressTests.m create mode 100644 Stripe/StripeiOSTests/STPAddressViewModelTest.swift create mode 100644 Stripe/StripeiOSTests/STPAnalyticsClientPaymentSheetTest.swift create mode 100644 Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift create mode 100644 Stripe/StripeiOSTests/STPApplePayContextFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPApplePayContextFunctionalTestExtras.swift create mode 100644 Stripe/StripeiOSTests/STPApplePayContextTest.swift create mode 100644 Stripe/StripeiOSTests/STPApplePayFunctionalTest.swift create mode 100644 Stripe/StripeiOSTests/STPApplePayPaymentOptionTest.m create mode 100644 Stripe/StripeiOSTests/STPApplePayTest.m create mode 100644 Stripe/StripeiOSTests/STPApplePayTest.swift create mode 100644 Stripe/StripeiOSTests/STPBECSDebitAccountNumberValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPBSBNumberValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPBankAccountFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPBankAccountParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPBankAccountTest.m create mode 100644 Stripe/StripeiOSTests/STPBinRangeTest.swift create mode 100644 Stripe/StripeiOSTests/STPBlocks.h create mode 100644 Stripe/StripeiOSTests/STPCardBINMetadataTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardBrandTest.m create mode 100644 Stripe/StripeiOSTests/STPCardCVCInputTextFieldFormatterTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardCVCInputTextFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardCVCInputTextFieldTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardCVCInputTextFieldValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardExpiryInputTextFieldFormatterTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardExpiryInputTextFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardExpiryInputTextFieldValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardFormViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardFormViewTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPCardNumberInputTextFieldFormatterTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardNumberInputTextFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardNumberInputTextFieldValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPCardParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPCardTest.swift create mode 100644 Stripe/StripeiOSTests/STPCardValidatorTest.swift create mode 100644 Stripe/StripeiOSTests/STPCertTest.swift create mode 100644 Stripe/StripeiOSTests/STPConfirmCardOptionsTest.m create mode 100644 Stripe/StripeiOSTests/STPConfirmPaymentMethodOptionsTest.m create mode 100644 Stripe/StripeiOSTests/STPConnectAccountAddressTest.m create mode 100644 Stripe/StripeiOSTests/STPConnectAccountFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPConnectAccountParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPCountryPickerInputFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPCustomerContextTest.swift create mode 100644 Stripe/StripeiOSTests/STPCustomerTest.m create mode 100644 Stripe/StripeiOSTests/STPE2ETest.swift create mode 100644 Stripe/StripeiOSTests/STPElementsSessionTest.swift create mode 100644 Stripe/StripeiOSTests/STPEphemeralKeyManagerTest.m create mode 100644 Stripe/StripeiOSTests/STPEphemeralKeyTest.swift create mode 100644 Stripe/StripeiOSTests/STPErrorBridgeTest.m create mode 100644 Stripe/StripeiOSTests/STPFPXBankBrandTest.m create mode 100644 Stripe/StripeiOSTests/STPFileFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPFileTest.m create mode 100644 Stripe/StripeiOSTests/STPFixtures+Swift.swift create mode 100644 Stripe/StripeiOSTests/STPFixtures.h create mode 100644 Stripe/StripeiOSTests/STPFixtures.m create mode 100644 Stripe/StripeiOSTests/STPFloatingPlaceholderTextFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPFormEncoderTest.swift create mode 100644 Stripe/StripeiOSTests/STPFormTextFieldTest.swift create mode 100644 Stripe/StripeiOSTests/STPFormViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPGenericInputPickerFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPGenericInputPickerFieldValidatorTest.swift create mode 100644 Stripe/StripeiOSTests/STPGenericInputTextFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPImageLibraryTest.swift create mode 100644 Stripe/StripeiOSTests/STPInputTextFieldFormatterTests.swift create mode 100644 Stripe/StripeiOSTests/STPInputTextFieldValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPIntentActionAlipayHandleRedirectTest.swift create mode 100644 Stripe/StripeiOSTests/STPIntentActionTest.m create mode 100644 Stripe/StripeiOSTests/STPIntentActionTypeTest.swift create mode 100644 Stripe/StripeiOSTests/STPIntentActionWeChatPayRedirectToAppTest.swift create mode 100644 Stripe/StripeiOSTests/STPIntentWithPreferencesTest.swift create mode 100644 Stripe/StripeiOSTests/STPLabeledFormTextFieldViewSnapshotTests.m create mode 100644 Stripe/StripeiOSTests/STPLabeledMultiFormTextFieldViewSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPMandateCustomerAcceptanceParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPMandateDataParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPMandateOnlineParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPMocks.h create mode 100644 Stripe/StripeiOSTests/STPMocks.m create mode 100644 Stripe/StripeiOSTests/STPNetworkStubbingTestCase.h create mode 100644 Stripe/StripeiOSTests/STPNetworkStubbingTestCase.m create mode 100644 Stripe/StripeiOSTests/STPNetworkStubbingTestCase.swift create mode 100644 Stripe/StripeiOSTests/STPNumericDigitInputTextFormatterTests.swift create mode 100644 Stripe/StripeiOSTests/STPNumericStringValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPPIIFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentCardTextFieldTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentCardTextFieldTestsSwift.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentCardTextFieldViewModelTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentConfigurationTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentContextApplePayTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentContextSnapshotTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentHandlerFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentHandlerStubbedMockedFilesTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentHandlerTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentIntentEnumsTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentIntentLastPaymentErrorTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentIntentParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentIntentTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodAddressTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodAffirmParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodAffirmTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodBacsDebitTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodBancontactParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodBancontactTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTests+Link.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodBoletoParamsTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodBoletoTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCardChecksTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCardParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCardTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCardWalletMasterpassTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCardWalletTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCardWalletVisaCheckoutTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCashAppParamsTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodCashAppTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodEPSParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodEPSTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodFPXTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodGiropayParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodGiropayTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodGrabPayParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodKlarnaParamsTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodKlarnaTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodNetBankingParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodNetBankingTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodOXXOParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodOXXOTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodOptionsTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodPayPalParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodPayPalTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24ParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24Tests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodSEPADebitTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodSofortParamsTests.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodSofortTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodThreeDSecureUsageTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodUPIParamsTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodUPITests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsStubbedTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountTest.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentMethodiDEALTest.m create mode 100644 Stripe/StripeiOSTests/STPPaymentOptionsViewControllerLocalizationTests.swift create mode 100644 Stripe/StripeiOSTests/STPPaymentOptionsViewControllerTest.swift create mode 100644 Stripe/StripeiOSTests/STPPhoneNumberValidatorTest.swift create mode 100644 Stripe/StripeiOSTests/STPPinManagementServiceFunctionalTest.swift create mode 100644 Stripe/StripeiOSTests/STPPostalCodeInputTextFieldFormatterTests.swift create mode 100644 Stripe/StripeiOSTests/STPPostalCodeInputTextFieldSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPPostalCodeInputTextFieldTests.swift create mode 100644 Stripe/StripeiOSTests/STPPostalCodeInputTextFieldValidatorTests.swift create mode 100644 Stripe/StripeiOSTests/STPPostalCodeValidatorTest.swift create mode 100644 Stripe/StripeiOSTests/STPPushProvisioningDetailsFunctionalTest.swift create mode 100644 Stripe/StripeiOSTests/STPRadarSessionFunctionalTest.swift create mode 100644 Stripe/StripeiOSTests/STPRedirectContextTest.m create mode 100644 Stripe/StripeiOSTests/STPSTPViewWithSeparatorSnapshotTests.m create mode 100644 Stripe/StripeiOSTests/STPSetupIntentConfirmParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.swift create mode 100644 Stripe/StripeiOSTests/STPSetupIntentLastSetupErrorTest.m create mode 100644 Stripe/StripeiOSTests/STPSetupIntentTest.m create mode 100644 Stripe/StripeiOSTests/STPShippingAddressViewControllerLocalizationTests.swift create mode 100644 Stripe/StripeiOSTests/STPShippingAddressViewControllerTest.swift create mode 100644 Stripe/StripeiOSTests/STPShippingMethodsViewControllerLocalizationTests.swift create mode 100644 Stripe/StripeiOSTests/STPSourceCardDetailsTest.swift create mode 100644 Stripe/StripeiOSTests/STPSourceFunctionalTest.m create mode 100644 Stripe/StripeiOSTests/STPSourceOwnerTest.m create mode 100644 Stripe/StripeiOSTests/STPSourceParamsTest.swift create mode 100644 Stripe/StripeiOSTests/STPSourceReceiverTest.m create mode 100644 Stripe/StripeiOSTests/STPSourceRedirectTest.m create mode 100644 Stripe/StripeiOSTests/STPSourceSEPADebitDetailsTest.m create mode 100644 Stripe/StripeiOSTests/STPSourceTest.m create mode 100644 Stripe/StripeiOSTests/STPSourceVerificationTest.m create mode 100644 Stripe/StripeiOSTests/STPStackViewWithSeparatorSnapshotTests.swift create mode 100644 Stripe/StripeiOSTests/STPStringUtilsTest.m create mode 100644 Stripe/StripeiOSTests/STPStringUtilsTest.swift create mode 100644 Stripe/StripeiOSTests/STPSwiftFixtures.swift create mode 100644 Stripe/StripeiOSTests/STPTestAPIClient+Swift.swift create mode 100644 Stripe/StripeiOSTests/STPTestUtils.h create mode 100644 Stripe/StripeiOSTests/STPTestUtils.m create mode 100644 Stripe/StripeiOSTests/STPTestingAPIClient.h create mode 100644 Stripe/StripeiOSTests/STPTestingAPIClient.m create mode 100644 Stripe/StripeiOSTests/STPTextFieldDelegateProxyTests.swift create mode 100644 Stripe/StripeiOSTests/STPThreeDSButtonCustomizationTest.swift create mode 100644 Stripe/StripeiOSTests/STPThreeDSFooterCustomizationTest.swift create mode 100644 Stripe/StripeiOSTests/STPThreeDSLabelCustomizationTest.swift create mode 100644 Stripe/StripeiOSTests/STPThreeDSNavigationBarCustomizationTest.swift create mode 100644 Stripe/StripeiOSTests/STPThreeDSSelectionCustomizationTest.swift create mode 100644 Stripe/StripeiOSTests/STPThreeDSTextFieldCustomizationTest.swift create mode 100644 Stripe/StripeiOSTests/STPThreeDSUICustomizationTest.swift create mode 100644 Stripe/StripeiOSTests/STPTokenTest.m create mode 100644 Stripe/StripeiOSTests/STPUIVCStripeParentViewControllerTests.m create mode 100755 Stripe/StripeiOSTests/SWHttpTrafficRecorder.h create mode 100755 Stripe/StripeiOSTests/SWHttpTrafficRecorder.m create mode 100644 Stripe/StripeiOSTests/ServerErrorMapperTest.swift create mode 100644 Stripe/StripeiOSTests/StripeErrorTest.swift create mode 100644 Stripe/StripeiOSTests/StripeTests-Prefix.pch create mode 100644 Stripe/StripeiOSTests/StripeiOS Tests-Bridging-Header.h create mode 100644 Stripe/StripeiOSTests/TextFieldElement+CardTest.swift create mode 100644 Stripe/StripeiOSTests/TextFieldElement+IBANTest.swift create mode 100644 Stripe/StripeiOSTests/UINavigationBar+StripeTest.m create mode 100644 Stripe/StripeiOSTests/UserDefaults+StripeTest.swift create mode 100644 Stripe/StripeiOSTests/WalletHeaderViewSnapshotTests.swift create mode 100644 Stripe3DS2/BuildConfigurations/Project-Debug.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Project-Release.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Project-Shared.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2-Debug.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2-Release.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2-Shared.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Debug.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Release.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Shared.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Debug.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Release.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Shared.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Debug.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Release.xcconfig create mode 100644 Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Shared.xcconfig create mode 100644 Stripe3DS2/Project.swift create mode 100644 Stripe3DS2/Stripe3DS2.xcodeproj/project.pbxproj create mode 100644 Stripe3DS2/Stripe3DS2.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2.xcscheme create mode 100644 Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2DemoUI.xcscheme create mode 100644 Stripe3DS2/Stripe3DS2/Info.plist create mode 100644 Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.h create mode 100644 Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.m create mode 100644 Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.h create mode 100644 Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.m create mode 100644 Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.h create mode 100644 Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.m create mode 100644 Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.h create mode 100644 Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.m create mode 100644 Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.h create mode 100644 Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.m create mode 100644 Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.h create mode 100644 Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.m create mode 100644 Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/amex.der create mode 100644 Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/cartes-bancaires.der create mode 100644 Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/discover.der create mode 100644 Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/ec_test.der create mode 100644 Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/mastercard.der create mode 100644 Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/ul-test.der create mode 100644 Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/visa.der create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/Chevron@3x.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/amex-logo@3x.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/cartes-bancaires-logo.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/discover-logo.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/error@3x.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/mastercard-logo@3x.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/visa-logo@3x.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/Images/visa-white-logo@3x.png create mode 100644 Stripe3DS2/Stripe3DS2/Resources/bg-BG.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/ca-ES.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/cs-CZ.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/da.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/de.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/el-GR.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/en-GB.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/en.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/es-419.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/es.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/et-EE.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/fi.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/fil.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/fr-CA.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/fr.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/hr.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/hu.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/id.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/it.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/ja.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/ko.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/lt-LT.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/lv-LV.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/ms-MY.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/mt.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/nb.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/nl.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/nn-NO.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/pl-PL.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/pt-BR.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/pt-PT.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/ro-RO.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/ru.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/sk-SK.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/sl-SI.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/sv.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/tr.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/vi.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/zh-HK.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/zh-Hans.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/Resources/zh-Hant.lproj/Localizable.strings create mode 100644 Stripe3DS2/Stripe3DS2/STDSACSNetworkingManager.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSACSNetworkingManager.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSBrandingView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSBrandingView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSBundleLocator.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSBundleLocator.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponse.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseImage.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtension.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfo.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSDeviceInformation.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDeviceInformation.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter+Private.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSDirectoryServer.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate+Internal.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair+Testing.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSException+Internal.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSIPAddress.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSIPAddress.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSImageLoader.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSImageLoader.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSLocalizedString.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSProcessingView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSProcessingView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSProgressViewController.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSProgressViewController.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSSelectionButton.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSSelectionButton.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSSpacerView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSSpacerView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSStackView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSStackView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSTextChallengeView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSTextChallengeView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion+Private.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSTransaction+Private.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSWebView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSWebView.m create mode 100644 Stripe3DS2/Stripe3DS2/STDSWhitelistView.h create mode 100644 Stripe3DS2/Stripe3DS2/STDSWhitelistView.m create mode 100644 Stripe3DS2/Stripe3DS2/Stripe3DS2-Bridging-Header.h create mode 100644 Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.h create mode 100644 Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.m create mode 100644 Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.h create mode 100644 Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.m create mode 100644 Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.h create mode 100644 Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.m create mode 100644 Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.h create mode 100644 Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.m create mode 100644 Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.h create mode 100644 Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.m create mode 100644 Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.h create mode 100644 Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSAuthenticationResponse.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSChallengeStatusReceiver.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSCustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSCustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSException.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSException.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSJSONDecodable.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSJSONEncodable.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSThreeDSProtocolVersion.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSTransaction.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSTransaction.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSUICustomization.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSUICustomization.m create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSWarning.h create mode 100644 Stripe3DS2/Stripe3DS2/include/STDSWarning.m create mode 100644 Stripe3DS2/Stripe3DS2/include/Stripe3DS2-Prefix.pch create mode 100644 Stripe3DS2/Stripe3DS2/include/Stripe3DS2.h create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Info.plist create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Resources/acs_challenge.html create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.h create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.m create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.h create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.m create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.h create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.m create mode 100644 Stripe3DS2/Stripe3DS2DemoUI/Sources/main.m create mode 100644 Stripe3DS2/Stripe3DS2DemoUITests/Info.plist create mode 100644 Stripe3DS2/Stripe3DS2DemoUITests/STDSChallengeResponseViewControllerSnapshotTests.m create mode 100644 Stripe3DS2/Stripe3DS2Resources/Info.plist create mode 100644 Stripe3DS2/Stripe3DS2Tests/Info.plist create mode 100644 Stripe3DS2/Stripe3DS2Tests/JSON/ARes.json create mode 100644 Stripe3DS2/Stripe3DS2Tests/JSON/CRes.json create mode 100644 Stripe3DS2/Stripe3DS2Tests/JSON/ErrorMessage.json create mode 100644 Stripe3DS2/Stripe3DS2Tests/NSDictionary+DecodingHelpersTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/NSString+EmptyCheckingTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSACSNetworkingManagerTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationRequestParametersTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationResponseTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSBase64URLEncodingTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSChallengeParametersTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSChallengeRequestParametersTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSChallengeResponseObjectTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSConfigParametersTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationManagerTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationParameterTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSDirectoryServerCertificateTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSEllipticCurvePointTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSEphemeralKeyPairTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSErrorMessageTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSJSONEncoderTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSJSONWebEncryptionTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSJSONWebSignatureTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSSecTypeUtilitiesTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSSynchronousLocationManagerTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.h create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSThreeDS2ServiceTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSTransactionTest.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSUICustomizationTests.m create mode 100644 Stripe3DS2/Stripe3DS2Tests/STDSWarningTests.m create mode 100644 Stripe3DS2/exported_symbols.txt create mode 100644 StripeApplePay/Project.swift create mode 100644 StripeApplePay/README.md create mode 100644 StripeApplePay/StripeApplePay.xcodeproj/project.pbxproj create mode 100644 StripeApplePay/StripeApplePay.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeApplePay/StripeApplePay.xcodeproj/xcshareddata/xcschemes/StripeApplePay.xcscheme create mode 100644 StripeApplePay/StripeApplePay/Info.plist create mode 100644 StripeApplePay/StripeApplePay/Source/ApplePayContext/STPAPIClient+ApplePay.swift create mode 100644 StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext+LegacySupport.swift create mode 100644 StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext.swift create mode 100644 StripeApplePay/StripeApplePay/Source/Blocks.swift create mode 100644 StripeApplePay/StripeApplePay/Source/Extensions/BillingDetails+ApplePay.swift create mode 100644 StripeApplePay/StripeApplePay/Source/Extensions/PKContact+Stripe.swift create mode 100644 StripeApplePay/StripeApplePay/Source/Extensions/PKPayment+Stripe.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Address.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/BillingDetails.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/CardBrand.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntent.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntentParams.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethod.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethodParams.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntent.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntentParams.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/ShippingDetails.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Token.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentIntent+API.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentMethod+API.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/SetupIntent+API.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Token+API.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+Payments.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+PaymentsAPI.swift create mode 100644 StripeApplePay/StripeApplePay/Source/PaymentsCore/Categories/STPAPIClient+PaymentsCore.swift create mode 100644 StripeApplePay/StripeApplePay/Source/StripeCore+Import.swift create mode 100644 StripeApplePay/StripeApplePay/StripeApplePay.h create mode 100644 StripeApplePay/StripeApplePayTests/Info.plist create mode 100644 StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientFunctionalTest.swift create mode 100644 StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientTest.swift create mode 100644 StripeApplePay/StripeApplePayTests/STPAnalyticsClient+ApplePayTest.swift create mode 100644 StripeApplePay/StripeApplePayTests/STPPaymentMethodFunctionalTest.swift create mode 100644 StripeApplePay/StripeApplePayTests/TelemetryInjectionTest.swift create mode 100644 StripeCameraCore/Project.swift create mode 100644 StripeCameraCore/StripeCameraCore.xcodeproj/project.pbxproj create mode 100644 StripeCameraCore/StripeCameraCore.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeCameraCore/StripeCameraCore.xcodeproj/xcshareddata/xcschemes/StripeCameraCore.xcscheme create mode 100644 StripeCameraCore/StripeCameraCore/Info.plist create mode 100644 StripeCameraCore/StripeCameraCore/Source/CameraExifMetadata.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Categories/CGRect+StripeCameraCore.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Categories/CVPixelBuffer+StripeCameraCore.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Categories/UIDeviceOrientation+StripeCameraCore.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Categories/UIImage+Buffer.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Coordinators/AppSettingsHelper.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraPermissionsManager.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraSession.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Coordinators/MockSimulatorCameraSession.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Coordinators/Torch.swift create mode 100644 StripeCameraCore/StripeCameraCore/Source/Views/CameraPreviewView.swift create mode 100644 StripeCameraCore/StripeCameraCore/StripeCameraCore.h create mode 100644 StripeCameraCore/StripeCameraCoreTestUtils/Info.plist create mode 100644 StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockAppSettingsHelper.swift create mode 100644 StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockCameraPermissionsManager.swift create mode 100644 StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockTestCameraSession.swift create mode 100644 StripeCameraCore/StripeCameraCoreTestUtils/StripeCameraCoreTestUtils.h create mode 100644 StripeCameraCore/StripeCameraCoreTests/Info.plist create mode 100644 StripeCameraCore/StripeCameraCoreTests/Unit/Categories/CGRect_StripeCameraCoreTest.swift create mode 100644 StripeCardScan/BuildConfigurations/StripeCardScan-Debug.xcconfig create mode 100644 StripeCardScan/BuildConfigurations/StripeCardScan-Release.xcconfig create mode 100644 StripeCardScan/Project.swift create mode 100644 StripeCardScan/README.md create mode 100644 StripeCardScan/StripeCardScan.xcodeproj/project.pbxproj create mode 100644 StripeCardScan/StripeCardScan.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeCardScan/StripeCardScan.xcodeproj/xcshareddata/xcschemes/StripeCardScan.xcscheme create mode 100644 StripeCardScan/StripeCardScan/Info.plist create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/analytics/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/metadata.json create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/model.espresso.net create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/model.espresso.shape create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/model.espresso.weights create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/model/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/neural_network_optionals/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/analytics/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/metadata.json create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/model.espresso.net create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/model.espresso.shape create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/model.espresso.weights create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/model/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/CompiledModels/UxModel.mlmodelc/neural_network_optionals/coremldata.bin create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/AppleOcr/AppleOcr.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CardUtils/CardNetwork.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CardUtils/CardType.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CardUtils/CreditCardUtils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CardUtils/Expiry.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/AppleCreditCardOcr.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/CreditCardOcrImplementation.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/CreditCardOcrPrediction.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/CreditCardOcrResult.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/ErrorCorrection.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/MachineLearningResult.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/MainLoopStateMachine.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/NonNameWords.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/OcrMainLoop.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/OcrObject.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/CreditCardOcr/SSDCreditCardOcr.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Extensions/Array+utils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Extensions/CGRectExtension.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Extensions/CGrect+utils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Extensions/CreditCardOcrPrediction+expiry.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Extensions/Image+utils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Extensions/UIImage+pixelBuffer.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLModels/AsyncModelLoading.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLModels/SSDOcr+Utils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLModels/SSDOcr.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLModels/UxModel+Utils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLModels/UxModel.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/ActiveStateComputation.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/AppState.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/DetectedAllBoxes.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/DetectedAllOcrBoxes.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/DetectedBox.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/DetectedSSDBox.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/DetectedSSDOcrBox.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/NMS.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/OcrDD.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/OcrDDUtils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/OcrPriorsGen.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/PostDetectionAlgorithm.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/PredictionAPI.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/PredictionResult.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/PredictionUtilOcr.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/SSDOcrDetect.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/SSDOcrOutputExtensions.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/MLRuntime/SoftNMS.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/BlurView.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/CardScanSheet.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/CornerView.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/InterfaceOrientation.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/PreviewView.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/ScanBaseViewController.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/ScanConfiguration.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/ScanEventsProtocol.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/ScanStats.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/SimpleScanViewController.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/Torch.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/UI/VideoFeed.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Utils/AppInfoUtils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Utils/AtomicPropertyWrapper.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardScan/Utils/DeviceUtils.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Api/CardImageVerificationDetailsResponse.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Api/Models/Scan Analytics/ScanStatsPayload+Common.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Api/Models/Scan Analytics/ScanStatsPayload+Tasks.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Api/Models/Scan Analytics/ScanStatsPayload.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Api/Models/VerificationFramesData.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Api/Models/VerifyFrames.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Api/STPAPIClient+CardImageVerification.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Bouncer.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/CancellationReason.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/CardImageVerificationController.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/CardImageVerificationIntent.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/CardImageVerificationSheet.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/CardImageVerificationSheetConfiguration.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/CardScanSheetError.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/ScanAnalyticsManager+Helpers.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/ScanAnalyticsManager+Managers.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/ScanAnalyticsManager+Tasks.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/ScanAnalyticsManager.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/ScannedCard.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/ScannedCardImageData+Verification.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/ScannedCardImageData.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Card Image Verification/StripeCore+Import.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/CardBase.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/CardScanFraudData.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/CardScanMisc.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/CardVerifyFraudData.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/CardVerifyStateMachine.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/FadeInAnimation.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/FrameData.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Helpers/EndToEndTestDataSource.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Helpers/STPLocalizedString.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/Helpers/String+Localized.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/PaymentCard.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/SimpleScanViewController+Verify.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/StripeCardScanBundleLocator.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/UxAnalyzer.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/UxAndOcrMainLoop.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/VerifyCardAddViewController.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/VerifyCardViewController.swift create mode 100644 StripeCardScan/StripeCardScan/Source/CardVerify/ZoomedInCGImage.swift create mode 100644 StripeCardScan/StripeCardScan/StripeCardScan.h create mode 100644 StripeCardScan/StripeCardScanTests/Helpers/CardScanMockData.swift create mode 100644 StripeCardScan/StripeCardScanTests/Helpers/Data+Sha256.swift create mode 100644 StripeCardScan/StripeCardScanTests/Helpers/ImageHelpers.swift create mode 100644 StripeCardScan/StripeCardScanTests/Helpers/ScannedCardDetails.swift create mode 100644 StripeCardScan/StripeCardScanTests/Helpers/String+Sha256.swift create mode 100644 StripeCardScan/StripeCardScanTests/Info.plist create mode 100644 StripeCardScan/StripeCardScanTests/Mock Data/JSON/CardImageVerification_CardAdd_200.json create mode 100644 StripeCardScan/StripeCardScanTests/Mock Data/JSON/CardImageVerification_CardSet_200.json create mode 100644 StripeCardScan/StripeCardScanTests/Resources/synthetic_test_image.jpg create mode 100644 StripeCardScan/StripeCardScanTests/Unit/API Bindings/STPAPIClient+CardImageVerificationTest.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/API Bindings/ScanStatsPayloadAPIBindingsTests.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/API Bindings/VerifyFramesAPIBindingsTests.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/CardImageVerificationControllerTests.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/CardImageVerificationDetailsResponseTest.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/ImageCompressionTests.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/ML Models/UxModelTests.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/ScanAnalyticsManagerTests.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/StrictModeFramesTest.swift create mode 100644 StripeCardScan/StripeCardScanTests/Unit/StringResourceTests.swift create mode 100644 StripeCore/Project.swift create mode 100644 StripeCore/StripeCore.xcodeproj/project.pbxproj create mode 100644 StripeCore/StripeCore.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeCore/StripeCore.xcodeproj/xcshareddata/xcschemes/StripeCore.xcscheme create mode 100644 StripeCore/StripeCore/Info.plist create mode 100644 StripeCore/StripeCore/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripeCore/StripeCore/Source/API Bindings/Models/EmptyResponse.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/Models/StripeFile.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/STPAPIClient+FileUpload.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/STPAPIClient.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/STPAppInfo.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/STPMultipartFormDataEncoder.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/STPMultipartFormDataPart.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/StripeAPI.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/StripeAPIConfiguration+Version.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/StripeAPIConfiguration.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/StripeError.swift create mode 100644 StripeCore/StripeCore/Source/API Bindings/StripeServiceError.swift create mode 100644 StripeCore/StripeCore/Source/Analytics/Analytic.swift create mode 100644 StripeCore/StripeCore/Source/Analytics/AnalyticLoggableError.swift create mode 100644 StripeCore/StripeCore/Source/Analytics/AnalyticsClientV2.swift create mode 100644 StripeCore/StripeCore/Source/Analytics/PluginDetector.swift create mode 100644 StripeCore/StripeCore/Source/Analytics/STPAnalyticEvent.swift create mode 100644 StripeCore/StripeCore/Source/Analytics/STPAnalyticsClient.swift create mode 100644 StripeCore/StripeCore/Source/Categories/Decimal+StripeCore.swift create mode 100644 StripeCore/StripeCore/Source/Categories/Dictionary+Stripe.swift create mode 100644 StripeCore/StripeCore/Source/Categories/Enums+CustomStringConvertible.swift create mode 100644 StripeCore/StripeCore/Source/Categories/NSArray+Stripe.swift create mode 100644 StripeCore/StripeCore/Source/Categories/NSBundle+Stripe_AppName.swift create mode 100644 StripeCore/StripeCore/Source/Categories/NSCharacterSet+StripeCore.swift create mode 100644 StripeCore/StripeCore/Source/Categories/NSError+Stripe.swift create mode 100644 StripeCore/StripeCore/Source/Categories/NSError+StripeCore.swift create mode 100644 StripeCore/StripeCore/Source/Categories/NSMutableURLRequest+Stripe.swift create mode 100644 StripeCore/StripeCore/Source/Categories/NSURLComponents+Stripe.swift create mode 100644 StripeCore/StripeCore/Source/Categories/String+StripeCore.swift create mode 100644 StripeCore/StripeCore/Source/Categories/UIImage+StripeCore.swift create mode 100644 StripeCore/StripeCore/Source/Coder/StripeCodable.swift create mode 100644 StripeCore/StripeCore/Source/Coder/StripeJSONDecoder.swift create mode 100644 StripeCore/StripeCore/Source/Coder/StripeJSONEncoder.swift create mode 100644 StripeCore/StripeCore/Source/Coder/StripeJSONShared.swift create mode 100644 StripeCore/StripeCore/Source/Coder/UnknownFields.swift create mode 100644 StripeCore/StripeCore/Source/Connections Bindings/ConnectionsSDKInterface.swift create mode 100644 StripeCore/StripeCore/Source/DownloadManager/DownloadManager.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/Async.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/BundleLocatorProtocol.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/FileDownloader.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/InstallMethod.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/PaymentsSDKVariant.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/STPDeviceUtils.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/STPDispatchFunctions.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/STPError.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/STPNumericStringValidator.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/STPURLCallbackHandler.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/ServerErrorMapper.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/StripeCoreBundleLocator.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/URLEncoder.swift create mode 100644 StripeCore/StripeCore/Source/Helpers/URLSession+Retry.swift create mode 100644 StripeCore/StripeCore/Source/Localization/STPLocalizationUtils.swift create mode 100644 StripeCore/StripeCore/Source/Localization/STPLocalizedString.swift create mode 100644 StripeCore/StripeCore/Source/Localization/String+Localized.swift create mode 100644 StripeCore/StripeCore/Source/Telemetry/FraudDetectionData.swift create mode 100644 StripeCore/StripeCore/Source/Telemetry/STPTelemetryClient.swift create mode 100644 StripeCore/StripeCore/Source/Telemetry/UserDefaults+PaymentsCore.swift create mode 100644 StripeCore/StripeCore/Source/UI/UIActivityIndicatorView+Stripe.swift create mode 100644 StripeCore/StripeCore/Source/UI/UIFont+Stripe.swift create mode 100644 StripeCore/StripeCore/StripeCore.h create mode 100644 StripeCore/StripeCoreTestUtils/APIStubbedTestCase.swift create mode 100644 StripeCore/StripeCoreTestUtils/Categories/UIImage+StripeCoreTestingUtils.swift create mode 100644 StripeCore/StripeCoreTestUtils/Categories/UIView+StripeCoreTestingUtils.swift create mode 100644 StripeCore/StripeCoreTestUtils/Info.plist create mode 100644 StripeCore/StripeCoreTestUtils/KeyPathExpectation.swift create mode 100644 StripeCore/StripeCoreTestUtils/Mock Files/File_IdentityDocument.json create mode 100644 StripeCore/StripeCoreTestUtils/Mocks/MockAnalyticsClient.swift create mode 100644 StripeCore/StripeCoreTestUtils/Mocks/MockAnalyticsClientV2.swift create mode 100644 StripeCore/StripeCoreTestUtils/Mocks/MockData.swift create mode 100644 StripeCore/StripeCoreTestUtils/STPSnapshotVerifyView.swift create mode 100644 StripeCore/StripeCoreTestUtils/StripeCoreTestUtils.h create mode 100644 StripeCore/StripeCoreTestUtils/TestConstants.swift create mode 100644 StripeCore/StripeCoreTestUtils/URLRequest+StripeTest.swift create mode 100644 StripeCore/StripeCoreTestUtils/XCTestCase+Stripe.swift create mode 100644 StripeCore/StripeCoreTests/API Bindings/STPAPIClient+EmptyResponseTest.swift create mode 100644 StripeCore/StripeCoreTests/API Bindings/STPAPIClient+ErrorResponseTest.swift create mode 100644 StripeCore/StripeCoreTests/API Bindings/StripeCodableTest.swift create mode 100644 StripeCore/StripeCoreTests/Analytics/AnalyticsClientV2Test.swift create mode 100644 StripeCore/StripeCoreTests/Analytics/Error_SerializeForLoggingTest.swift create mode 100644 StripeCore/StripeCoreTests/Analytics/STPAnalyticsClientTest.swift create mode 100644 StripeCore/StripeCoreTests/Categories/Dictionary+StripeTests.swift create mode 100644 StripeCore/StripeCoreTests/Categories/NSArray+StripeCoreTest.swift create mode 100644 StripeCore/StripeCoreTests/Categories/NSMutableURLRequest+StripeTest.swift create mode 100644 StripeCore/StripeCoreTests/Categories/UIImage+StripeCoreTests.swift create mode 100644 StripeCore/StripeCoreTests/DownloadManager/DownloadManagerTest.swift create mode 100644 StripeCore/StripeCoreTests/External/TestJSONEncoder.swift create mode 100644 StripeCore/StripeCoreTests/Helpers/URLEncoderTest.swift create mode 100644 StripeCore/StripeCoreTests/Info.plist create mode 100644 StripeCore/StripeCoreTests/Mock Files/test_image.png create mode 100644 StripeFinancialConnections/Project.swift create mode 100644 StripeFinancialConnections/README.md create mode 100644 StripeFinancialConnections/StripeFinancialConnections.xcodeproj/project.pbxproj create mode 100644 StripeFinancialConnections/StripeFinancialConnections.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeFinancialConnections/StripeFinancialConnections.xcodeproj/xcshareddata/xcschemes/StripeFinancialConnections.xcscheme create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Info.plist create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/add@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/back_arrow@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/bank@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/bank_check@2x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/brandicon_default@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/bullet@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/cancel_circle@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/check@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/chevron_down@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/close@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/ellipsis@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/generic_error@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/prepane_phone_background@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/search@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/spinner@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/stripe_logo@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/warning_circle@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Images/warning_triangle@3x.png create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/APIPollingHelper.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/APIVersion.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/FinancialConnectionsAPIClient.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/BankAccountToken.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsAccount.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsAuthSession.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsBulletPoint.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsConsent.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsDataAccessNotice.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsImage.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsInstitution.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsInstitutionSearchResultResource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsLegalDetailsNotice.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsMixedOAuthParams.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsOAuthPrepane.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsPartnerAccount.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsPaymentAccountResource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsPaymentMethodType.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsSession.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsSessionManifest.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/API Bindings/Models/FinancialConnectionsSynchronize.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Analytics/FinancialConnectionsAnalyticsClient.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Analytics/FinancialConnectionsSheetAnalytics.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/ExperimentHelper.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/FinancialConnectionsCustomManualEntryRequiredError.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/FinancialConnectionsNavigationController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/FlowRouter.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/HostController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/HostViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/LoadingView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Common/ModalPresentationWrapperViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/FinancialConnectionsSDK/FinancialConnectionsSDKImplementation.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/FinancialConnectionsSheet.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/FinancialConnectionsSheetError.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/Helpers.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/Image.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/NSAttributedString+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/STPLocalizedString.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/String+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/String+Localized.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/StripeFinancialConnectionsBundleLocator.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/UIColor+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/UIFont+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Helpers/UIViewController+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerAccountLoadErrorView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerFooterView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerHelpers.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerLabelRowView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerNoAccountEligibleErrorView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerSelectionListView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerSelectionRowView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerSelectionView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/AccountPickerViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/CheckboxView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/LinkingAccountsLoadingView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AccountPicker/RadioButtonView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AttachLinkedPaymentAccount/AccountNumberRetrievalErrorView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AttachLinkedPaymentAccount/AttachLinkedPaymentAccountDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/AttachLinkedPaymentAccount/AttachLinkedPaymentAccountViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Consent/ConsentBodyView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Consent/ConsentDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Consent/ConsentFooterView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Consent/ConsentLogoView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Consent/ConsentViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/FeaturedInstitutionGridCell.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/FeaturedInstitutionGridView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/InstitutionDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/InstitutionPickerViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/InstitutionSearchBar.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/InstitutionSearchErrorView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/InstitutionSearchFooterView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/InstitutionSearchTableView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/InstitutionPicker/InstitutionSearchTableViewCell.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryCheckView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryErrorView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryFooterView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryFormView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryTextField.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryValidator.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntry/ManualEntryViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntrySuccess/ManualEntrySuccessTransactionTableView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ManualEntrySuccess/ManualEntrySuccessViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/NativeFlowController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/NativeFlowDataManager.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/PartnerAuth/PartnerAuthDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/PartnerAuth/PartnerAuthViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/PartnerAuth/PrepaneImageView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/PartnerAuth/PrepaneView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Placeholder/PlaceholderViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ResetFlow/ResetFlowDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/ResetFlow/ResetFlowViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/AlwaysTemplateImageView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/AuthFlowHelpers.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/BulletPointLabelView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/Button+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/ClickableLabel.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/CloseConfirmationAlertHandler.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/ConsentBottomSheetModel.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/ConsentBottomSheetView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/ConsentBottomSheetViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/HitTestStackView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/HitTestView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/InstitutionIconView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/MerchantDataAccessView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/PaneLayoutView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/PaneWithCustomHeaderLayoutView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/PaneWithHeaderLayoutView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/ReusableInformationView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/SFSafariViewController+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/SpinnerIconView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/SuccessIconView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/TimeInterval+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/UIImage+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/UIImageView+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/UITableView+Extensions.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Shared/UIViewController+KeyboardAvoiding.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Success/SuccessAccountListView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Success/SuccessBodyView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Success/SuccessDataSource.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Success/SuccessFooterView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/Success/SuccessViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Native/TerminalError/TerminalErrorViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Placeholder.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/StripeCore+Import.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Web/AuthenticationSessionManager.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Web/ContinueStateView.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Web/FinancialConnectionsAccountFetcher.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Web/FinancialConnectionsSessionFetcher.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/Source/Web/FinancialConnectionsWebFlowViewController.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnections/StripeFinancialConnections.h create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/APIPollingHelperTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/AccountFetcherTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/AccountPickerHelpersTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/AuthFlowHelpersTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/EmptyFinancialConnectionsAPIClient.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/FinancialConnectionsAnalyticsTest.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/FinancialConnectionsSessionTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/FinancialConnectionsSheetTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/Info.plist create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/ManualEntryValidatorTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/MarkdownBoldAttributedStringTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/MockData/FinancialConnectionsSession_both_accounts_la.json create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/MockData/FinancialConnectionsSession_only_accounts.json create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/MockData/FinancialConnectionsSession_only_both_missing.json create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/MockData/FinancialConnectionsSession_only_la.json create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/SessionFetcherTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/SoftLinkTests.swift create mode 100644 StripeFinancialConnections/StripeFinancialConnectionsTests/StringExtensionsTests.swift create mode 100644 StripeIdentity/Project.swift create mode 100644 StripeIdentity/README.md create mode 100644 StripeIdentity/StripeIdentity.xcodeproj/project.pbxproj create mode 100644 StripeIdentity/StripeIdentity.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeIdentity/StripeIdentity.xcodeproj/xcshareddata/xcschemes/StripeIdentity.xcscheme create mode 100644 StripeIdentity/StripeIdentity/Info.plist create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_add@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_camera@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_checkmark@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_checkmark_92@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_clock@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_info@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_warning@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Images/icon_warning_92@3x.png create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/DocumentScanner+API.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/DocumentType+StripeIdentity.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/DocumentUploader+API.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/FaceScanner+API.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/IdentityAPIClient.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/DocumentType.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/TruncatedDecimal.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageFieldType.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageRequirements.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentConsentPage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentCountryNotListedPage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentDocumentCaptureModels.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentDocumentCapturePage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentDocumentSelectPage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentIndividualPage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentIndividualWelcomePage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentSelfieModels.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentSelfiePage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPage/VerificationPageStaticContentTextPage.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageData/VerificationPageData.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageData/VerificationPageDataRequirementError.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageData/VerificationPageDataRequirements.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/RequiredInternationalAddress.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageClearData.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageCollectedData.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageDataDob.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageDataDocumentFileData.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageDataFace.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageDataIdNumber.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageDataName.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/Models/VerificationPageDataUpdate/VerificationPageDataUpdate.swift create mode 100644 StripeIdentity/StripeIdentity/Source/API Bindings/SelfieUploader+API.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Analytics/IdentityAnalyticsClient.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Categories/Array+StripeIdentity.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Categories/CGImage+StripeIdentity.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Categories/MLMultiArray+StripeIdentity.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Categories/NSAttributedString+HTML.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Categories/TimeInterval+StripeIdentity.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Categories/UINavigationController+StripeIdentity.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Categories/VNBarcodeSymbology+StripeIdentity.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Elements/IdNumberElement.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Elements/IdentityElementsFactory.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Elements/IdentityTextButtonElement.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Elements/IndividualFormElement.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Enums+CustomStringConvertible.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Helpers/Image.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Helpers/STPLocalizedString.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Helpers/String+Localized.swift create mode 100644 StripeIdentity/StripeIdentity/Source/Helpers/StripeIdentityBundleLocator.swift create mode 100644 StripeIdentity/StripeIdentity/Source/IdentityVerificationSheet.swift create mode 100644 StripeIdentity/StripeIdentity/Source/IdentityVerificationSheetError.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/IdentityTopLevelDestination.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/DocumentScanner/DocumentScanner.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/DocumentScanner/DocumentScannerConfiguration.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/DocumentScanner/DocumentScannerOutput.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/FaceScanner/FaceCaptureData.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/FaceScanner/FaceScanner.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/FaceScanner/FaceScannerConfiguration.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/FaceScanner/FaceScannerOutput.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/ImageScanner.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanner/ImageScanningConcurrencyManager.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanningSession/ImageScanningSession.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageScanningSession/ImageScanningSessionDelegate.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageUploaders/DocumentUploader.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageUploaders/IdentityImageUploader.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/ImageUploaders/SelfieUploader.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/VerificationSheetController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/VerificationSheetFlowController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Coordinators/VerificationSheetFlowControllerError.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/BarcodeDetector.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/FaceDetector/FaceDetector.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/FaceDetector/FaceDetectorOutput.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/IDDetector/IDDetector.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/IDDetector/IDDetectorConstants.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/IDDetector/IDDetectorOutput.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/MLDetectorConfiguration.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/MLDetectorMetricsTracker.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/MotionBlurDetector.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Detectors/VisionBasedDetector.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/DocumentSide.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/IdentityDataCollecting.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/IdentityFlowNavigationController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/IdentityUI.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ML/Helpers/MLModelLoader.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ML/Helpers/MLModelUnexpectedOutputError.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ML/Helpers/NonMaxSuppression.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ML/IdentityMLModelLoader.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/BiometricConsentViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/CountryNotListedViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/DebugViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/DocumentCaptureViewController+Strings.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/DocumentCaptureViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/DocumentFileUploadViewController+Strings.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/DocumentFileUploadViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/DocumentTypeSelectViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/ErrorViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/IdentityFlowViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/IndividualViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/IndividualWelcomeViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/LoadingViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/SelfieCaptureViewController+Strings.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/SelfieCaptureViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/ViewControllers/SuccessViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/BottomAlignedLabel.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/CameraPreviewContainerView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/ContentCenteringScrollView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/DebugView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/DocumentCapture/AnimatedBorderView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/DocumentCapture/DocumentCaptureView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/DocumentCapture/DocumentScanningView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/DocumentCapture/InstructionalDocumentScanningView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/ErrorView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/HeaderIconView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/HeaderView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/IdentityFlowView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/IdentityHTMLView/HTMLTextView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/IdentityHTMLView/HTMLViewWithIconLabels.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/IdentityHTMLView/IconLabelHTMLView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/InstructionListView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/ListView/ListItemView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/ListView/ListView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/Selfie/SelfieCaptureView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/Selfie/SelfieScanningView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/ShadowConfiguration.swift create mode 100644 StripeIdentity/StripeIdentity/Source/NativeComponents/Views/ShadowedCorneredImageView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/StripeCore+Import.swift create mode 100644 StripeIdentity/StripeIdentity/Source/VerificationClientSecret.swift create mode 100644 StripeIdentity/StripeIdentity/Source/VerificationSheetAnalytics.swift create mode 100644 StripeIdentity/StripeIdentity/Source/WebWrapper/VerificationFlowWebView.swift create mode 100644 StripeIdentity/StripeIdentity/Source/WebWrapper/VerificationFlowWebViewController.swift create mode 100644 StripeIdentity/StripeIdentity/Source/WebWrapper/VerifyWebURLHelper.swift create mode 100644 StripeIdentity/StripeIdentity/StripeIdentity.h create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/DocumentUploaderMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/IdentityAPIClientTestMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/IdentityAnalyticsClientTestHelpers.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/IdentityMLModelLoaderMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/IdentityMockData.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/ImageScannerMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/ImageScanningConcurrencyManagerMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/MLDetectorMetricsTrackerMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/SnapshotTestMockData.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/VerificationFlowResult+Equatable.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/VerificationSheetControllerMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Helpers/VerificationSheetFlowControllerMock.swift create mode 100644 StripeIdentity/StripeIdentityTests/Info.plist create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/Mock Photos/back_drivers_license.jpg create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/Mock Photos/cgimage_stripeidentity_test.png create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/Mock Photos/front_drivers_license.jpg create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/Mock Photos/header_icon.png create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_200.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_200_submitted.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_200_testMode.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_no_selfie.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_require_live_capture.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_type_address.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_type_doc_require_address.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_type_doc_require_idNumber.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_type_doc_require_idNumber_and_address.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPage/VerificationPage_type_idNumber.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPageData/VerificationPageData_200.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPageData/VerificationPageData_no_errors.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPageData/VerificationPageData_no_errors_needback.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/VerificationPageData/VerificationPageData_submitted.json create mode 100644 StripeIdentity/StripeIdentityTests/Mock Files/mock.html create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/AnimatedBorderViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/BiometricConsentViewControllerSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/CGImage_StripeIdentitySnapshotTest.png create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/CGImage_StripeIdentitySnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/DebugViewControllerSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/DocumentScanningViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/ErrorViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/HeaderIconViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/HeaderViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/IdentityFlowViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/IdentityHTMLViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/IndividualWelcomeViewControllerSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/InstructionListViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/InstructionalDocumentScanningViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/ListItemViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/ListViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/NSAttributedString_HTMLSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/SelfieCaptureViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/SelfieScanningViewSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/SuccessViewControllerSnapshotTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Snapshot/VerificationFlowWebViewSnapshotTests.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/API Bindings/IdentityAPIClientTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/API Bindings/TruncatedDecimalTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/Categories/CGImage+StripeIdentityUnitTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/Elements/IdentityElementsFactoryTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/Elements/IndividualElementTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/Coordinators/DocumentUploaderTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/Coordinators/IdentityImageUploaderTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/Coordinators/IdentityVerificationSheetTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/Coordinators/VerificationSheetControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/Coordinators/VerificationSheetFlowControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/BiometricConsentViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/DocumentCaptureViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/DocumentFileUploadViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/DocumentTypeSelectViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/ErrorViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/IdentityFlowNavigationControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/IndividualViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/NativeComponents/ViewControllers/IndividualWelcomeViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/VerificationClientSecretTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/VerificationSheetAnalyticsTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/WebWrapper/VerificationFlowWebViewControllerTest.swift create mode 100644 StripeIdentity/StripeIdentityTests/Unit/WebWrapper/VerificationFlowWebViewTest.swift create mode 100644 StripeLinkCore/Project.swift create mode 100644 StripeLinkCore/StripeLinkCore.xcodeproj/project.pbxproj create mode 100644 StripeLinkCore/StripeLinkCore.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeLinkCore/StripeLinkCore.xcodeproj/xcshareddata/xcschemes/StripeLinkCore.xcscheme create mode 100644 StripeLinkCore/StripeLinkCore/Info.plist create mode 100644 StripeLinkCore/StripeLinkCore/Source/Placeholder.swift create mode 100644 StripeLinkCore/StripeLinkCore/StripeLinkCore.h create mode 100644 StripeLinkCore/StripeLinkCoreTests/Info.plist create mode 100644 StripeLinkCore/StripeLinkCoreTests/StripeLinkCoreTests.swift create mode 100644 StripePaymentSheet/Project.swift create mode 100644 StripePaymentSheet/README.md create mode 100644 StripePaymentSheet/StripePaymentSheet.xcodeproj/project.pbxproj create mode 100644 StripePaymentSheet/StripePaymentSheet.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripePaymentSheet/StripePaymentSheet.xcodeproj/xcshareddata/xcschemes/StripePaymentSheet.xcscheme create mode 100644 StripePaymentSheet/StripePaymentSheet/Info.plist create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_boa@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_capitalone@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_citibank@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_compass@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_default@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_morganchase@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_nfcu@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_pnc@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_stripe@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_suntrust@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_svb@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_td@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_usaa@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_usbank@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/InstantDebitIcons/bank_icon_wellsfargo@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/back_button@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/icon-pm-link@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/icon_add_bordered@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/icon_cancel@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/icon_link_error@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/icon_link_success@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/icon_menu@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/icon_menu_horizontal@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/link_carousel_logo@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/Link/link_logo@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-affirm@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-afterpay@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-aubecsdebit@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-bancontact@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-bank@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-card@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-cashapp@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-eps@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-giropay@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-ideal@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-klarna@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-p24@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-paypal@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-paypal_dark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-sepa@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentMethods/icon-pm-upi@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentSheet/icon_checkmark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentSheet/icon_chevron_left@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentSheet/icon_chevron_left_standalone@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentSheet/icon_lock@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentSheet/icon_plus@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentSheet/icon_x@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/PaymentSheet/icon_x_standalone@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/affirm_mark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/affirm_mark_dark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/afterpay_icon_info@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/afterpay_mark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/afterpay_mark_dark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/apple_pay_mark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/clearpay_mark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/clearpay_mark_dark@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Images/polling_error_icon@3x.png create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/JSON/form_specs.json create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/tk.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Analytics/AnalyticsHelper.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Analytics/STPAnalyticsClient+Address.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Analytics/STPAnalyticsClient+LUXE.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Categories/Data+SHA256.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Categories/Date+Distance.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Categories/NSAttributedString+Stripe.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Categories/STPPaymentMethod+PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Categories/STPPaymentMethodParams+PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Categories/String+Localized.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Categories/String+StripePaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/BoolReference.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/Images.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/IntentStatusPoller.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/PaymentSheetLinkAccount.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/STPCameraView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/STPCardScanner.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/STPImageLibrary.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/STPLocalizedString.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/STPStringUtils.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/StaticEphemeralKeyProvider.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/StripePaymentSheet+Exports.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Helpers/StripePaymentSheetBundleLocator.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/ConsumerSession+LookupResponse.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/ConsumerSession+PublishableKey.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/ConsumerSession.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/CookieStore/LinkCookieStore.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/CookieStore/LinkInMemoryCookieStore.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/CookieStore/LinkSecureCookieStore.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/PaymentDetails.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/STPAPIClient+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/Link/VerificationSession.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/STPAPIClient+PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/STPElementsSession.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/API Bindings/VO/CardExpiryDate.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Basic UI/SeparatorLabel.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/ACH/LinkFinancialConnectionsAuthManager.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/Badge/LinkBadgeView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/NavigationBar/LinkNavigationBar.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/Notice/LinkNoticeView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/PaymentMethodPicker/LinkPaymentMethodPicker-AddButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/PaymentMethodPicker/LinkPaymentMethodPicker-Cell.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/PaymentMethodPicker/LinkPaymentMethodPicker-CellContentView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/PaymentMethodPicker/LinkPaymentMethodPicker-Header.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/PaymentMethodPicker/LinkPaymentMethodPicker-RadioButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/PaymentMethodPicker/LinkPaymentMethodPicker.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Components/Toast/LinkToast.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-BaseViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-LoaderViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-NewPaymentViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-SignUpViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-SignUpViewModel.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-UpdatePaymentViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-VerifyAccountViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-WalletViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController-WalletViewModel.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Controllers/PayWithLinkViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Elements/InlineSignup/LinkInlineSignupElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Elements/InlineSignup/LinkInlineSignupView-CheckboxElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Elements/InlineSignup/LinkInlineSignupView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Elements/LinkCardEditElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Elements/LinkEmailElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Extensions/Button+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Extensions/ConfirmButton+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Extensions/FormElement+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Extensions/Intent+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Extensions/PaymentSheet-Configuration+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Extensions/STPAnalyticsClient+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Extensions/UIColor+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/LinkUI.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Services/LinkAccountService.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Utils/Locale+Link.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Utils/OperationDebouncer.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkAccountContext.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkCookieKey.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkUtils.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkVerificationController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkVerificationView-Header.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkVerificationView-LogoutView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkVerificationView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkVerificationViewController-PresentationController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Verification/LinkVerificationViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/ViewModels/LinkInlineSignupViewModel.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Views/LinkInstantDebitMandateView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Views/LinkKeyboardAvoidingScrollView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Views/LinkLegalTermsView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/Internal/Link/Views/LinkMoreInfoView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/AddressViewController/AddressViewController+Configuration.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/AddressViewController/AddressViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/BottomSheet/BottomSheetPresentable.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/BottomSheet/BottomSheetPresentationAnimator.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/BottomSheet/BottomSheetPresentationController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/BottomSheet/BottomSheetTransitioningDelegate.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/BottomSheet/UIViewController+BottomSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/CardSectionWithScanner/CardSectionWithScannerElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/CardSectionWithScanner/CardSectionWithScannerView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/ConnectionsElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/LinkEnabledPaymentMethodElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/PaymentMethodElement/PaymentMethodElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/PaymentMethodElement/PaymentMethodElementWrapper.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/SimpleMandateElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/TextField/TextFieldElement+Card.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Elements/TextField/TextFieldElement+IBAN.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Error+PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Intent.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/IntentConfirmParams.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/KlarnaHelper.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Link/LinkPaymentController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Link/PayWithLinkController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Link/PaymentSheet-LinkConfirmOption.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/New Payment Method Screen/AddPaymentMethodViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/New Payment Method Screen/PaymentMethodTypeCollectionView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/New Payment Method Screen/WalletHeaderView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentMethodType.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentOption+Images.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheet+API.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheet+DeferredAPI.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheet+PaymentMethodAvailability.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheet+SwiftUI.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetAppearance.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetConfiguration.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetError.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFlowController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFormFactory/FormSpec/FormSpec.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFormFactory/FormSpec/FormSpecPaymentHandler.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFormFactory/FormSpec/FormSpecProvider.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFormFactory/PaymentSheetFormFactory+Card.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFormFactory/PaymentSheetFormFactory+FormSpec.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFormFactory/PaymentSheetFormFactory+UPI.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetFormFactory/PaymentSheetFormFactory.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/PaymentSheetIntentConfiguration.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/STPAnalyticsClient+PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/STPApplePayContext+PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/STPPaymentIntentShippingDetailsParams+PaymentSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Saved Payment Method Screen/SavedPaymentMethodCollectionView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Saved Payment Method Screen/SavedPaymentOptionsViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsAddPaymentMethodViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsCollectionViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsFormFactory.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsSheet+API.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsSheet.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsSheetConfiguration.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsSheetDelegate.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsSheetError.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/SavedPaymentMethods/SavedPaymentMethodsViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/USBankAccount/BankAccountInfoView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/USBankAccount/USBankAccountPaymentMethodElement.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/AutoComplete/AddressSearchResult.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/AutoComplete/AutoCompleteViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/AutoComplete/String+AutoComplete.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/BottomSheet3DS2ViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/BottomSheetViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/LoadingViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/PaymentSheetFlowControllerViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/PaymentSheetViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/ViewControllers/PollingViewController.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/AUBECSMandate.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/AffirmCopyLabel.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/AfterpayPriceBreakdownView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/Appearance+FontScaling.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/CardScanButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/CardScanningView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/CircularButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/ConfirmButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/ManualEntryButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/PayWithLinkButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/PaymentSheetUIKitAdditions.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/RotatingCardBrandsView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/ShadowedRoundedRectangleView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/SheetNavigationBar.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/SheetNavigationButton.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/SimpleMandateTextView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/Source/PaymentSheet/Views/TestModeView.swift create mode 100644 StripePaymentSheet/StripePaymentSheet/StripePaymentSheet.h create mode 100644 StripePaymentSheet/StripePaymentSheetTests/DictionaryTests.swift create mode 100644 StripePaymentSheet/StripePaymentSheetTests/Info.plist create mode 100644 StripePaymentSheet/StripePaymentSheetTests/STPAnalyticsClient+PaymentSheetTests.swift create mode 100644 StripePayments/Project.swift create mode 100644 StripePayments/README.md create mode 100644 StripePayments/StripePayments.xcodeproj/project.pbxproj create mode 100644 StripePayments/StripePayments.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripePayments/StripePayments.xcodeproj/xcshareddata/xcschemes/StripePayments.xcscheme create mode 100644 StripePayments/StripePayments/Info.plist create mode 100644 StripePayments/StripePayments/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/tk.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripePayments/StripePayments/Source/API Bindings/Legacy Compatability/StripeAPI+Deprecated.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Legacy Compatability/StripeApplePay+Import.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Legacy Compatability/StripeCore+Import.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/ACH/LinkAccountSession.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/ACH/STPCollectBankAccountParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPConfirmAlipayOptions.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPConfirmBLIKOptions.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPConfirmCardOptions.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPConfirmPaymentMethodOptions.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPConfirmUSBankAccountOptions.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPConfirmWeChatPayOptions.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntent.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentAction.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentActionRedirectToURL.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentEnums.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentLastPaymentError.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentShippingDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentShippingDetailsAddress.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentShippingDetailsAddressParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentShippingDetailsParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentSourceAction.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentIntents/STPPaymentIntentSourceActionAuthorizeWithURL.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/STPPaymentMethod.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/STPPaymentMethodAddress.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/STPPaymentMethodBillingDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/STPPaymentMethodEnums.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/STPPaymentMethodParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAUBECSDebit.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAUBECSDebitParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAffirm.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAffirmParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAfterpayClearpay.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAfterpayClearpayParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAlipay.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodAlipayParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBLIK.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBLIKParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBacsDebit.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBacsDebitParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBancontact.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBancontactParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBoleto.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodBoletoParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCard.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCardChecks.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCardNetworks.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCardParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCardPresent.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCardWallet.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCardWalletMasterpass.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCardWalletVisaCheckout.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCashApp.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodCashAppParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodEPS.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodEPSParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodFPX.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodFPXParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodGiropay.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodGiropayParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodGrabPay.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodGrabPayParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodKlarna.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodKlarnaParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodLink.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodLinkParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodNetBanking.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodNetBankingParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodOXXO.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodOXXOParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodPayPal.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodPayPalParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodPrzelewy24.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodPrzelewy24Params.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodSEPADebit.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodSEPADebitParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodSofort.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodSofortParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodThreeDSecureUsage.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodUPI.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodUPIParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodUSBankAccount.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodUSBankAccountParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodWeChatPay.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodWeChatPayParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodiDEAL.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/PaymentMethods/Types/STPPaymentMethodiDEALParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPAPIResponseDecodable.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPAddress.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPCardBrand.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPConnectAccountAddress.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPConnectAccountCompanyParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPConnectAccountIndividualParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPConnectAccountParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPContactField.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPCustomer.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPFPXBankBrand.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPFile.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPFormEncodable.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPIssuingCardPin.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPRadarSession.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPToken.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/STPiDEALBank.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/SetupIntents/STPSetupIntent.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/SetupIntents/STPSetupIntentConfirmParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/SetupIntents/STPSetupIntentEnums.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/SetupIntents/STPSetupIntentLastSetupError.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/LinkSettings.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentAction.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentActionAlipayHandleRedirect.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentActionBoletoDisplayDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentActionCashAppRedirectToApp.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentActionOXXODisplayDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentActionRedirectToURL.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentActionVerifyWithMicrodeposits.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPIntentActionWeChatPayRedirectToApp.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPMandateCustomerAcceptanceParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPMandateDataParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPMandateOnlineParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Shared/STPPaymentMethodOptions.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSource.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSourceEnums.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSourceOwner.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSourceParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSourceProtocol.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSourceReceiver.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSourceRedirect.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/STPSourceVerification.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPBankAccount.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPBankAccountParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPCard.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPCardParams.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPKlarnaLineItem.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPSourceCardDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPSourceKlarnaDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPSourceSEPADebitDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/Models/Sources/Types/STPSourceWeChatPayDetails.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/STPAPIClient+ApplePay.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/STPAPIClient+LinkAccountSession.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/STPAPIClient+Payments.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/STPAPIClient+Radar.swift create mode 100644 StripePayments/StripePayments/Source/API Bindings/STPRedirectContext.swift create mode 100644 StripePayments/StripePayments/Source/Categories/Enums+CustomStringConvertible.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/STPBINController.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/STPBankAccountCollector.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/STPBlocks.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/STPCardValidator.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/STPLocalizedString.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/STPPaymentConfirmation+SwiftUI.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/StripePayments+Export.swift create mode 100644 StripePayments/StripePayments/Source/Helpers/StripePaymentsBundleLocator.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/APIRequest.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/STP3DS2AuthenticateResponse.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/STPEmptyStripeResponse.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/STPFormEncoder.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/STPIntentActionUseStripeSDK.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/STPInternalAPIResponseDecodable.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/STPPaymentMethodListDeserializer.swift create mode 100644 StripePayments/StripePayments/Source/Internal/API Bindings/STPSourcePoller.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Analytics/Analytic+Payments.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Analytics/STPAnalyticsClient+Payments.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Categories/NSArray+Stripe.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Categories/NSDecimalNumber+Stripe_Currency.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Categories/NSDictionary+Stripe.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Categories/NSString+Stripe.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Categories/STPAPIClient+PaymentsCore.swift create mode 100644 StripePayments/StripePayments/Source/Internal/Helpers/ConnectionsSDKAvailability.swift create mode 100644 StripePayments/StripePayments/Source/Internal/STPPaymentHandlerActionParams.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPAuthenticationContext.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPPaymentHandler.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSButtonCustomization.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSCustomizationSettings.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSFooterCustomization.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSLabelCustomization.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSNavigationBarCustomization.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSSelectionCustomization.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSTextFieldCustomization.swift create mode 100644 StripePayments/StripePayments/Source/PaymentHandler/STPThreeDSUICustomization.swift create mode 100644 StripePayments/StripePayments/StripePayments.h create mode 100644 StripePayments/StripePaymentsTests/Info.plist create mode 100644 StripePayments/StripePaymentsTests/STPAnalyticsClient+StripePayments.swift create mode 100644 StripePaymentsUI/Project.swift create mode 100644 StripePaymentsUI/README.md create mode 100644 StripePaymentsUI/StripePaymentsUI.xcodeproj/project.pbxproj create mode 100644 StripePaymentsUI/StripePaymentsUI.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripePaymentsUI/StripePaymentsUI.xcodeproj/xcshareddata/xcschemes/StripePaymentsUI.xcscheme create mode 100644 StripePaymentsUI/StripePaymentsUI/Info.plist create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/anz@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/bankofmelbourne@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/banksa@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/bankwest@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/boq@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/cba@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/nab@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/stgeorges@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/suncorpmetway@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/BECS/westpac@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_amex@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_amex_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_applepay@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_cartes_bancaires@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_cartes_bancaires_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_cvc@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_cvc_amex@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_diners@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_diners_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_discover@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_discover_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_error@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_error_amex@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_jcb@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_jcb_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_mastercard@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_mastercard_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_unionpay@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_unionpay_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_unknown@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_visa@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cards/stp_card_visa_template@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_amex@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cartes_bancaires@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_amex_icon@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_amex_icon_dark@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_amex_updated_icon@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_amex_updated_icon_dark@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_icon@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_icon_dark@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_updated_icon@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_cvc_updated_icon_dark@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_diners@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_discover@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_jcb@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_mastercard@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_unionpay@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_unknown@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_unknown_icon@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_unknown_icon_dark@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_unknown_updated_icon@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_unknown_updated_icon_dark@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/Cardsv2/card_visa@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/form_error_icon@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/form_error_icon_dark@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Images/stp_icon_bank@3x.png create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/JSON/au_becs_bsb.json create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/tk.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Categories/Enums+CustomStringConvertible.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPAPIClient+CustomerContext.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPBECSDebitAccountNumberValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPBSBNumberValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPImageLibrary.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPLocalizedString.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPPhoneNumberValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPPostalCodeValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPPromise.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/STPStringUtils.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/String+Localized.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/StripePayments+Export.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Helpers/StripePaymentsBundleLocator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/Categories/NSAttributedString+Stripe.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/Categories/UIButton+Stripe.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/CardBrandView.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardCVCInputTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardCVCInputTextFieldFormatter.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardCVCInputTextFieldValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardExpiryInputTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardExpiryInputTextFieldFormatter.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardExpiryInputTextFieldValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardNumberInputTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardNumberInputTextFieldFormatter.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPCardNumberInputTextFieldValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPPostalCodeInputTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPPostalCodeInputTextFieldFormatter.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/Card/STPPostalCodeInputTextFieldValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/STPCountryPickerInputField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/STPGenericInputPickerField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/STPGenericInputTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/STPInputTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/STPInputTextFieldFormatter.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/STPInputTextFieldValidator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/Inputs/STPNumericDigitInputTextFormatter.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPAUBECSFormViewModel.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPCardLoadingIndicator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPFormTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPLabeledFormTextFieldView.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPLabeledMultiFormTextFieldView.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPPaymentCardTextFieldViewModel.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPValidatedTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/Internal/UI/Views/STPViewWithSeparator.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/CustomerContext/PersistablePaymentMethodOption.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/CustomerContext/UserDefaults+Stripe.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/CustomerContext/_stpspmsbeta_STPBackendAPIAdapter.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/CustomerContext/_stpspmsbeta_STPCustomerContext.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/CustomerContext/_stpspmsbeta_STPEphemeralKey.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/CustomerContext/_stpspmsbeta_STPEphemeralKeyManager.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/CustomerContext/_stpspmsbeta_STPEphemeralKeyProvider.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/PaymentMethodMessagingView+Configuration.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/PaymentMethodMessagingView.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPAUBECSDebitFormView.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPCardFormView+SwiftUI.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPCardFormView.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPFloatingPlaceholderTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPFormTextFieldContainer.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPFormView.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPMultiFormTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPPaymentCardTextField+SwiftUI.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/Source/UI Components/STPPaymentCardTextField.swift create mode 100644 StripePaymentsUI/StripePaymentsUI/StripePaymentsUI.h create mode 100644 StripePaymentsUI/StripePaymentsUITests/Info.plist create mode 100644 StripePaymentsUI/StripePaymentsUITests/STPAnalyticsClient+PaymentsUITests.swift create mode 100644 StripeUICore/Project.swift create mode 100644 StripeUICore/StripeUICore.xcodeproj/project.pbxproj create mode 100644 StripeUICore/StripeUICore.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 StripeUICore/StripeUICore.xcodeproj/xcshareddata/xcschemes/StripeUICore.xcscheme create mode 100644 StripeUICore/StripeUICore/Info.plist create mode 100644 StripeUICore/StripeUICore/Resources/Images/brand_stripe@3x.png create mode 100644 StripeUICore/StripeUICore/Resources/Images/form_error_icon@3x.png create mode 100644 StripeUICore/StripeUICore/Resources/Images/icon_chevron_down@3x.png create mode 100644 StripeUICore/StripeUICore/Resources/Images/icon_clear@3x.png create mode 100644 StripeUICore/StripeUICore/Resources/JSON/au_becs_bsb.json create mode 100644 StripeUICore/StripeUICore/Resources/JSON/localized_address_data.json create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/bg-BG.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/ca-ES.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/cs-CZ.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/da.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/de.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/el-GR.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/en-GB.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/en.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/es-419.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/es.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/et-EE.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/fi.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/fil.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/fr-CA.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/fr.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/hr.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/hu.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/id.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/it.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/ja.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/ko.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/lt-LT.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/lv-LV.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/ms-MY.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/mt.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/nb.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/nl.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/nn-NO.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/pl-PL.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/pt-BR.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/pt-PT.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/ro-RO.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/ru.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/sk-SK.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/sl-SI.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/sv.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/tr.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/vi.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/zh-HK.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/zh-Hans.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Resources/Localizations/zh-Hant.lproj/Localizable.strings create mode 100644 StripeUICore/StripeUICore/Source/Categories/CALayer+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/Enums+CustomStringConvertible.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/Locale+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/NSDirectionalEdgeInsets+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIBarButtonItem+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIButton+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIColor+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIFont+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIKeyboardType+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UISpringTimingParameters+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIStackView+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UITraitCollection+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIView+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIViewController+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Categories/UIWindow+StripeUICore.swift create mode 100644 StripeUICore/StripeUICore/Source/Controls/ActivityIndicator.swift create mode 100644 StripeUICore/StripeUICore/Source/Controls/Button.swift create mode 100644 StripeUICore/StripeUICore/Source/Controls/OneTimeCodeTextField-TextStorage.swift create mode 100644 StripeUICore/StripeUICore/Source/Controls/OneTimeCodeTextField.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Checkbox/CheckboxButton.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Checkbox/CheckboxElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/ContainerElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/DateFieldElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/DropdownFieldElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Element.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/ElementsUI.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/Address/AddressSectionElement+DummyAddressLine.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/Address/AddressSectionElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/Address/AddressSpec+ElementFactory.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/Address/AddressSpec.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/Address/AddressSpecProvider.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/BSB/BSBNumberProvider.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/DropdownFieldElement+AddressFactory.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/IDNumberTextFieldConfiguration.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/TextFieldElement+AccountFactory.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/TextFieldElement+AddressFactory.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Factories/TextFieldElement+Factory.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Form/FormElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Form/FormView.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/PhoneNumber/PhoneNumberElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/PickerField/PickerFieldView.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/PickerField/PickerTextField.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Section/SectionContainerView.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Section/SectionElement+MultiElementRow.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Section/SectionElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/Section/SectionView.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/StaticElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/TextField/FloatingPlaceholderTextFieldView.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/TextField/TextFieldElement+Validation.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/TextField/TextFieldElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/TextField/TextFieldElementConfiguration.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/TextField/TextFieldFormatter.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/TextField/TextFieldView.swift create mode 100644 StripeUICore/StripeUICore/Source/Elements/TextOrDropdownElement.swift create mode 100644 StripeUICore/StripeUICore/Source/Events.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/CompatibleColor.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/ImageMaker.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/InputFormColors.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/RegionCodeProvider.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/STPLocalizedString.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/StackViewWithSeparator.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/String+CountryEmoji.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/String+Localized.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/String+RegionCodeProvider.swift create mode 100644 StripeUICore/StripeUICore/Source/Helpers/StripeUICoreBundleLocator.swift create mode 100644 StripeUICore/StripeUICore/Source/Image.swift create mode 100644 StripeUICore/StripeUICore/Source/Validators/BSBNumber.swift create mode 100644 StripeUICore/StripeUICore/Source/Validators/PhoneNumber.swift create mode 100644 StripeUICore/StripeUICore/Source/Validators/STPEmailAddressValidator.swift create mode 100644 StripeUICore/StripeUICore/Source/Validators/STPVPANumberValidator.swift create mode 100644 StripeUICore/StripeUICore/Source/Views/DoneButtonToolbar.swift create mode 100644 StripeUICore/StripeUICore/Source/Views/DynamicHeightContainerView.swift create mode 100644 StripeUICore/StripeUICore/Source/Views/DynamicImageView.swift create mode 100644 StripeUICore/StripeUICore/Source/Views/LinkOpeningTextView.swift create mode 100644 StripeUICore/StripeUICore/StripeUICore.h create mode 100644 StripeUICore/StripeUICoreTests/Info.plist create mode 100644 StripeUICore/StripeUICoreTests/Snapshot/Controls/ButtonSnapshotTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Snapshot/Elements/AddressSectionElementSnapshotTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Snapshot/Elements/CheckboxButtonSnapshotTests.swift create mode 100644 StripeUICore/StripeUICoreTests/Snapshot/Elements/DateFieldElementSnapshotTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Snapshot/Elements/DropdownFieldElementSnapshotTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Snapshot/Elements/PhoneNumberElementSnapshotTests.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Categories/Locale+StripeUICoreTests.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Categories/UIColor+StripeUICoreTests.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/AddressSectionElementTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/AddressSpecProviderTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/BSBNumberProviderTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/DateFieldElementTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/DropdownFieldElementTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/IDNumberTextFieldConfigurationTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/PhoneNumberElementTests.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/SectionElementTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/TestFieldElement+AccountFactoryTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/TextFieldElement+AddressFactoryTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/TextFieldElementTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Elements/TextFieldFormatterTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Validators/BSBNumberTests.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Validators/PhoneNumberTests.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Validators/STPEmailAddressValidatorTest.swift create mode 100644 StripeUICore/StripeUICoreTests/Unit/Validators/STPVPANumberValidatorTest.swift create mode 100644 VERSION diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 00000000..faab755c --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,1028 @@ +## 23.7.0 2023-04-24 + +### PaymentSheet +* [Fixed] Fixed disabled text color, using a lower opacity version of the original color instead of the previous `.tertiaryLabel`. + +### Identity +* [Added] Added test mode for the SDK. + +## 23.6.2 2023-04-20 + +### Payments +* [Fixed] Fixed UnionPay cards appearing as invalid in some cases. + +### PaymentSheet +* [Fixed] Fixed a bug that prevents users from using SEPA Debit w/ PaymentIntents or SetupIntents and Paypal in PaymentIntent+setup_future_usage or SetupIntent. + +## 23.6.1 2023-04-17 +### All +* Xcode 13 is [no longer supported by Apple](https://developer.apple.com/news/upcoming-requirements/). Please upgrade to Xcode 14.1 or later. +### PaymentSheet +* [Fixed] Visual bug of the delete icon when deleting saved payment methods reported in [#2461](https://github.com/stripe/stripe-ios/issues/2461). + +## 23.6.0 2023-03-27 +### PaymentSheet +* [Added] Added `billingDetailsCollectionConfiguration` to configure how you want to collect billing details. See the docs [here](https://stripe.com/docs/payments/accept-a-payment?platform=ios&ui=payment-sheet#billing-details-collection). + +## 23.5.1 2023-03-20 +### Payments +* [Fixed] Fixed amounts in COP being formatted incorrectly. +* [Fixed] Fixed BLIK payment bindings not handling next actions correctly. +* [Changed] Removed usage of `UIDevice.currentDevice.name`. + +### Identity +* [Added] Added a retake photo button on selfie scanning screen. + +## 23.5.0 2023-03-13 +### Payments +* [Added] API bindings support for Cash App Pay. See the docs [here](https://stripe.com/docs/payments/cash-app-pay/accept-a-payment?platform=mobile). +* [Added] Added `STPCardValidator.possibleBrands(forCard:completion:)`, which returns the list of available networks for a card. + +### PaymentSheet +* [Added] Support for Cash App Pay in PaymentSheet. + +## 23.4.2 2023-03-06 +### Identity +* [Added] ID/Address verification. + +## 23.4.1 2023-02-27 +### PaymentSheet +* [Added] Debug logging to help identify why specific payment methods are not showing up in PaymentSheet. + +### Basic Integration +* [Fixed] Race condition reported in #2302 + +## 23.4.0 2023-02-21 +### PaymentSheet +* [Added] Adds support for setting up PayPal using a SetupIntent or a PaymentIntent w/ setup_future_usage=off_session. Note: PayPal is in beta. + +## 23.3.4 2023-02-13 +### Financial Connections +* [Changed] Polished Financial Connections UI. + +## 23.3.3 2023-01-30 +### Payments +* [Changed] Updated image asset for AFFIN bank. + +### Financial Connections +* [Fixed] Double encoding of GET parameters. + +## 23.3.2 2023-01-09 +* [Changed] Using [Tuist](https://tuist.io) to generate Xcode projects. From now on, only release versions of the SDK will include Xcode project files, in case you want to build a non release revision from source, you can follow [these instructions](https://docs.tuist.io/tutorial/get-started) to generate the project files. For Carthage users, this also means that you will only be able to depend on release versions. + +### PaymentSheet +* [Added] `PaymentSheetError` now conforms to `CustomDebugStringConvertible` and has a more useful description when no payment method types are available. +* [Changed] Customers can now re-enter the autocomplete flow of `AddressViewController` by tapping an icon in the line 1 text field. + +## 23.3.1 2022-12-12 +* [Fixed] Fixed a bug where 3 decimal place currencies were not being formatted properly. + +### PaymentSheet +* [Fixed] Fixed an issue that caused animations of the card logos in the Card input field to glitch. +* [Fixed] Fixed a layout issue in the "Save my info" checkbox. + +### CardScan +* [Fixed] Fixed UX model loading from the wrong bundle. [#2078](https://github.com/stripe/stripe-ios/issues/2078) (Thanks [nickm01](https://github.com/nickm01)) + +## 23.3.0 2022-12-05 +### PaymentSheet +* [Added] Added logos of accepted card brands on Card input field. +* [Fixed] Fixed erroneously displaying the card scan button when card scanning is not available. + +### Financial Connections +* [Changed] FinancialConnectionsSheet methods now require to be called from non-extensions. +* [Changed] BankAccountToken.bankAccount was changed to an optional. + +## 23.2.0 2022-11-14 +### PaymentSheet +* [Added] Added `AddressViewController`, a customizable view controller that collects local and international addresses for your customers. See https://stripe.com/docs/elements/address-element?platform=ios. +* [Added] Added `PaymentSheet.Configuration.allowsPaymentMethodsRequiringShippingAddress`. Previously, to allow payment methods that require a shipping address (e.g. Afterpay and Affirm) in PaymentSheet, you attached a shipping address to the PaymentIntent before initializing PaymentSheet. Now, you can instead set this property to `true` and set `PaymentSheet.Configuration.shippingDetails` to a closure that returns your customers' shipping address. The shipping address will be attached to the PaymentIntent when the customer completes the checkout. +* [Fixed] Fixed user facing error messages for card related errors. +* [Fixed] Fixed `setup_future_usage` value being set when there's no customer. + +## 23.1.1 2022-11-07 +### Payments +* [Fixed] Fixed an issue with linking the StripePayments SDK in certain configurations. + +## 23.1.0 2022-10-31 +### CardScan +* [Added] Added a README.md for the `CardScanSheet` integration. + +### PaymentSheet +* [Added] Added parameters to customize the primary button and Apple Pay button labels. They can be found under `PaymentSheet.Configuration.primaryButtonLabel` and `PaymentSheet.ApplePayConfiguration.buttonType` respectively. + +## 23.0.0 2022-10-24 +### Payments +* [Changed] Reduced the size of the SDK by splitting the `Stripe` module into `StripePaymentSheet`, `StripePayments`, and `StripePaymentsUI`. Some manual changes may be required. Migration instructions are available at [https://stripe.com/docs/mobile/ios/sdk-23-migration](https://stripe.com/docs/mobile/ios/sdk-23-migration). + +|Module|Description|Compressed|Uncompressed| +|------|-----------|----------|------------| +|StripePaymentSheet|Stripe's [prebuilt payment UI](https://stripe.com/docs/payments/accept-a-payment?platform=ios&ui=payment-sheet).|2.7MB|6.3MB| +|Stripe|Contains all the below frameworks, plus [Issuing](https://stripe.com/docs/issuing/cards/digital-wallets?platform=iOS) and [Basic Integration](/docs/mobile/ios/basic).|2.3MB|5.1MB| +|StripeApplePay|[Apple Pay support](/docs/apple-pay), including `STPApplePayContext`.|0.4MB|1.0MB| +|StripePayments|Bindings for the Stripe Payments API.|1.0MB|2.6MB| +|StripePaymentsUI|Bindings for the Stripe Payments API, [STPPaymentCardTextField](https://stripe.com/docs/payments/accept-a-payment?platform=ios&ui=custom), STPCardFormView, and other UI elements.|1.7MB|3.9MB| + +* [Changed] The minimum iOS version is now 13.0. If you'd like to deploy for iOS 12.0, please use Stripe SDK 22.8.4. +* [Changed] STPPaymentCardTextField's `cardParams` parameter has been deprecated in favor of `paymentMethodParams`, making it easier to include the postal code from the card field. If you need to access the `STPPaymentMethodCardParams`, use `.paymentMethodParams.card`. + +### PaymentSheet +* [Fixed] Fixed a validation issue where cards expiring at the end of the current month were incorrectly treated as expired. +* [Fixed] Fixed a visual bug in iOS 16 where advancing between text fields would momentarily dismiss the keyboard. + +## 22.8.4 2022-10-12 +### PaymentSheet +* [Fixed] Use `.formSheet` modal presentation in Mac Catalyst. [#2023](https://github.com/stripe/stripe-ios/issues/2023) (Thanks [sergiocampama](https://github.com/sergiocampama)!) + +## 22.8.3 2022-10-03 +### CardScan +* [Fixed] [Garbled privacy link text in Card Scan UI](https://github.com/stripe/stripe-ios/issues/2015) + +## 22.8.2 2022-09-19 +### Identity +* [Changed] Support uploading single side documents. +* [Fixed] Fixed Xcode 14 support. +### Financial Connections +* [Fixed] Fixes an issue of returning canceled result from FinancialConnections if user taps cancel on the manual entry success screen. +### CardScan +* [Added] Added a new parameter to CardScanSheet.present() to specify if the presentation should be done animated or not. Defaults to true. +* [Changed] Changed card scan ML model loading to be async. +* [Changed] Changed minimum deployment target for card scan to iOS 13. + +## 22.8.1 2022-09-12 +### PaymentSheet +* [Fixed] Fixed potential crash when using Link in Mac Catalyst. +* [Fixed] Fixed Right-to-Left (RTL) layout issues. + +### Apple Pay +* [Fixed] Fixed an issue where `applePayContext:willCompleteWithResult:authorizationResult:handler:` may not be called in Objective-C implementations of `STPApplePayContextDelegate`. + +## 22.8.0 2022-09-06 +### PaymentSheet +* [Changed] Renamed `PaymentSheet.reset()` to `PaymentSheet.resetCustomer()`. See `MIGRATING.md` for more info. +* [Added] You can now set closures in `PaymentSheet.ApplePayConfiguration.customHandlers` to configure the PKPaymentRequest and PKPaymentAuthorizationResult during a transaction. This enables you to build support for [Merchant Tokens](https://developer.apple.com/documentation/passkit/pkpaymentrequest/3916053-recurringpaymentrequest) and [Order Tracking](https://developer.apple.com/documentation/passkit/pkpaymentorderdetails) in iOS 16. + +### Apple Pay +* [Added] You can now implement the `applePayContext(_:willCompleteWithResult:handler:)` function in your `ApplePayContextDelegate` to configure the PKPaymentAuthorizationResult during a transaction. This enables you to build support for [Order Tracking](https://developer.apple.com/documentation/passkit/pkpaymentorderdetails) in iOS 16. + +## 22.7.1 2022-08-31 +* [Fixed] Fixed Mac Catalyst support in Xcode 14. [#2001](https://github.com/stripe/stripe-ios/issues/2001) + +### PaymentSheet +* [Fixed] PaymentSheet now uses configuration.apiClient for Apple Pay instead of always using STPAPIClient.shared. +* [Fixed] Fixed a layout issue with PaymentSheet in landscape. + +## 22.7.0 2022-08-15 +### PaymentSheet +* [Fixed] Fixed a layout issue on iPad. +* [Changed] Improved Link support in custom flow (`PaymentSheet.FlowController`). + +## 22.6.0 2022-07-05 +### PaymentSheet +* [Added] PaymentSheet now supports Link payment method. +* [Changed] Change behavior of Afterpay/Clearpay: Charge in 3 for GB, FR, and ES + +### STPCardFormView +* [Changed] Postal code is no longer collected for billing addresses in Japan. + +### Identity +* [Added] The ability to capture Selfie images in the native component flow. +* [Fixed] Fixed an issue where the welcome and confirmation screens were not correctly decoding non-ascii characters. +* [Fixed] Fixed an issue where, if a manually uploaded document could not be decoded on the server, there was no way to select a new image to upload. +* [Fixed] Fixed an issue where the IdentityVerificationSheet completion block was called early when manually uploading a document image instead of using auto-capture. + +## 22.5.1 2022-06-21 +* [Fixed] Fixed an issue with `STPPaymentHandler` where returning an app redirect could cause a crash. + +## 22.5.0 2022-06-13 +### PaymentSheet +* [Added] You can now use `PaymentSheet.ApplePayConfiguration.paymentSummaryItems` to directly configure the payment summary items displayed in the Apple Pay sheet. This is useful for recurring payments. + +## 22.4.0 2022-05-23 +### PaymentSheet +* [Added] The ability to customize the appearance of the PaymentSheet using `PaymentSheet.Appearance`. +* [Added] Support for collecting payments from customers in 54 additional countries within PaymentSheet. Most of these countries are located in Africa and the Middle East. +* [Added] `affirm` and `AUBECSDebit` payment methods are now available in PaymentSheet + +## 22.3.2 2022-05-18 +### CardScan +* [Added] Added privacy text to the CardImageVerification Sheet UI + +## 22.3.1 2022-05-16 +* [Fixed] Fixed an issue where ApplePayContext failed to parse an API response if the funding source was unknown. +* [Fixed] Fixed an issue where PaymentIntent confirmation could fail when the user closes the challenge window immediately after successfully completing a challenge + +### Identity +* [Fixed] Fixed an issue where the verification flow would get stuck in a document upload loop when verifying with a passport and uploading an image manually. + +## 22.3.0 2022-05-03 + +### PaymentSheet +* [Added] `us_bank_account` PaymentMethod is now available in payment sheet + +## 22.2.0 2022-04-25 + +### Connections +* [Changed] `StripeConnections` SDK has been renamed to `StripeFinancialConnections`. See `MIGRATING.md` for more info. + +### PaymentSheet +* [Fixed] Fixed an issue where `source_cancel` API requests were being made for non-3DS payment method types. +* [Fixed] Fixed an issue where certain error messages were not being localized. +* [Added] `us_bank_account` PaymentMethod is now available in PaymentSheet. + +### Identity +* [Fixed] Minor UI fixes when using `IdentityVerificationSheet` with native components +* [Changed] Improvements to native component `IdentityVerificationSheet` document detection + +## 22.1.1 2022-04-11 + +### Identity +* [Fixed] Fixes VerificationClientSecret (Thanks [Masataka-n](https://github.com/Masataka-n)!) + +## 22.1.0 2022-04-04 +* [Changed] Localization improvements. +### Identity +* [Added] `IdentityVerificationSheet` can now be used with native iOS components. + +## 22.0.0 2022-03-28 +* [Changed] The minimum iOS version is now 12.0. If you'd like to deploy for iOS 11.0, please use Stripe SDK 21.12.0. +* [Added] `us_bank_account` PaymentMethod is now available for ACH Direct Debit payments, including APIs to collect customer bank information (requires `StripeConnections`) and verify microdeposits. +* [Added] `StripeConnections` SDK can be optionally included to support ACH Direct Debit payments. + +### PaymentSheet +* [Changed] PaymentSheet now uses light and dark mode agnostic icons for payment method types. +* [Changed] Link payment method (private beta) UX improvements. + +### Identity +* [Changed] `IdentityVerificationSheet` now has an availability requirement of iOS 14.3 on its initializer instead of the `present` method. + +## 21.13.0 2022-03-15 +* [Changed] Binary framework distribution now requires Xcode 13. Carthage users using Xcode 12 need to add the `--no-use-binaries` flag. + +### PaymentSheet +* [Fixed] Fixed potential crash when using PaymentSheet custom flow with SwiftUI. +* [Fixed] Fixed being unable to cancel native 3DS2 in PaymentSheet. +* [Fixed] The payment method icons will now use the correct colors when PaymentSheet is configured with `alwaysLight` or `alwaysDark`. +* [Fixed] A race condition when setting the `primaryButtonColor` on `PaymentSheet.Configuration`. +* [Added] PaymentSheet now supports Link (private beta). + +### CardScan +* [Added] The `CardImageVerificationSheet` initializer can now take an additional `Configuration` object. + +## 21.12.0 2022-02-14 +* [Added] We now offer a 1MB Apple Pay SDK module intended for use in an App Clip. Visit [our App Clips docs](https://stripe.com/docs/apple-pay#app-clips) for details. +* `Stripe` now requires `StripeApplePay`. See `MIGRATING.md` for more info. +* [Added] Added a convenience initializer to create an STPCardParams from an STPPaymentMethodParams. + +### PaymentSheet +* [Changed] The "save this card" checkbox in PaymentSheet is now unchecked by default in non-US countries. +* [Fixed] Fixes issue that could cause symbol name collisions when using Objective-C +* [Fixed] Fixes potential crash when using PaymentSheet with SwiftUI + +## 21.11.1 2022-01-10 +* Fixes a build warning in SPM caused by an invalid Package.swift file. + +## 21.11.0 2022-01-04 +* [Changed] The maximum `identity_document` file upload size has been increased, improving the quality of compressed images. See https://stripe.com/docs/file-upload +* [Fixed] The maximum `dispute_evidence` file upload size has been decreased to match server requirements, preventing the server from rejecting uploads that exceeded 5MB. See https://stripe.com/docs/file-upload +* [Added] PaymentSheet now supports Afterpay / Clearpay, EPS, Giropay, Klarna, Paypal (private beta), and P24. + +## 21.10.0 2021-12-14 +* Added API bindings for Klarna +* `StripeIdentity` now requires `StripeCameraCore`. See `MIGRATING.md` for more info. +* Releasing `StripeCardScan` Beta iOS SDK +* Fixes a bug where the text field would cause a crash when typing a space (U+0020) followed by pressing the backspace key on iPad. [#1907](https://github.com/stripe/stripe-ios/issues/1907) (Thanks [buhikon](https://github.com/buhikon)!) + +## 21.9.1 2021-12-02 +* Fixes a build warning caused by a duplicate NSURLComponents+Stripe.swift file. + +## 21.9.0 2021-10-18 +### PaymentSheet +This release adds several new features to PaymentSheet, our drop-in UI integration: + +#### More supported payment methods +The list of supported payment methods depends on your integration. +If you’re using a PaymentIntent, we support: +- Card +- SEPA Debit, bancontact, iDEAL, sofort + +If you’re using a PaymentIntent with `setup_future_usage` or a SetupIntent, we support: +- Card +- Apple/GooglePay + +Note: To enable SEPA Debit and sofort, set `PaymentSheet.configuration.allowsDelayedPaymentMethods` to `true` on the client. +These payment methods can't guarantee you will receive funds from your customer at the end of the checkout because they take time to settle. Don't enable these if your business requires immediate payment (e.g., an on-demand service). See https://stripe.com/payments/payment-methods-guide + +#### Pre-fill billing details +PaymentSheet collects billing details like name and email for certain payment methods. Pre-fill these fields to save customers time by setting `PaymentSheet.Configuration.defaultBillingDetails`. + +#### Save payment methods on payment +> This is currently only available for cards + Apple/Google Pay. + +PaymentSheet supports PaymentIntents with `setup_future_usage` set. This property tells us to save the payment method for future use (e.g., taking initial payment of a recurring subscription). +When set, PaymentSheet hides the 'Save this card for future use' checkbox and always saves. + +#### SetupIntent support +> This is currently only available for cards + Apple/Google Pay. + +Initialize PaymentSheet with a SetupIntent to set up cards for future use without charging. + +#### Smart payment method ordering +When a customer is adding a new payment method, PaymentSheet uses information like the customers region to show the most relevant payment methods first. + +#### Other changes +* Postal code collection for cards is now limited to US, CA, UK +* Fixed SwiftUI memory leaks [Issue #1881](https://github.com/stripe/stripe-ios/issues/1881) +* Added "hint" for error messages +* Adds many new localizations. The SDK now localizes in the following languages: bg-BG,ca-ES,cs-CZ,da,de,el-GR,en-GB,es-419,es,et-EE,fi,fil,fr-CA,fr,hr,hu,id,it,ja,ko,lt-LT,lv-LV,ms-MY,mt,nb,nl,nn-NO,pl-PL,pt-BR,pt-PT,ro-RO,ru,sk-SK,sl-SI,sv,tk,tr,vi,zh-Hans,zh-Hant,zh-HK +* `Stripe` and `StripeIdentity` now require `StripeUICore`. See `MIGRATING.md` for more info. + +## 21.8.1 2021-08-10 +* Fixes an issue with image loading when using Swift Package Manager. +* Temporarily disabled WeChat Pay support in PaymentMethods. +* The `Stripe` module now requires `StripeCore`. See `MIGRATING.md` for more info. + +## 21.8.0 2021-08-04 +* Fixes broken card scanning links. (Thanks [ricsantos](https://github.com/ricsantos)) +* Fixes accessibilityLabel for postal code field. (Thanks [romanilchyshyndepop](https://github.com/romanilchyshyndepop)) +* Improves compile time by 30% [#1846](https://github.com/stripe/stripe-ios/pull/1846) (Thanks [JonathanDowning](https://github.com/JonathanDowning)!) +* Releasing `StripeIdentity` iOS SDK for use with [Stripe Identity](https://stripe.com/identity). + +## 21.7.0 2021-07-07 +* Fixes an issue with `additionaDocument` field typo [#1833](https://github.com/stripe/stripe-ios/issues/1833) +* Adds support for WeChat Pay to PaymentMethods +* Weak-links SwiftUI [#1828](https://github.com/stripe/stripe-ios/issues/1828) +* Adds 3DS2 support for Cartes Bancaires +* Fixes an issue with camera rotation during card scanning on iPad +* Fixes an issue where PaymentSheet could cause conflicts when included in an app that also includes PanModal [#1818](https://github.com/stripe/stripe-ios/issues/1818) +* Fixes an issue with building on Xcode 13 [#1822](https://github.com/stripe/stripe-ios/issues/1822) +* Fixes an issue where overriding STPPaymentCardTextField's `brandImage()` func had no effect [#1827](https://github.com/stripe/stripe-ios/issues/1827) +* Fixes documentation typo. (Thanks [iAugux](https://github.com/iAugux)) + +## 21.6.0 2021-05-27 +* Adds `STPCardFormView`, a UI component that collects card details +* Adds 'STPRadarSession'. Note this requires additional Stripe permissions to use. + +## 21.5.1 2021-05-07 +* Fixes the `PaymentSheet` API not being public. +* Fixes an issue with missing headers. (Thanks [jctrouble](https://github.com/jctrouble)!) + +## 21.5.0 2021-05-06 +* Adds the `PaymentSheet`(https://stripe.dev/stripe-ios/docs/Classes/PaymentSheet.html) API, a prebuilt payment UI. +* Fixes Mac Catalyst support in Xcode 12.5 [#1797](https://github.com/stripe/stripe-ios/issues/1797) +* Fixes `STPPaymentCardTextField` not being open [#1768](https://github.com/stripe/stripe-ios/issues/1797) + +## 21.4.0 2021-04-08 +* Fixed warnings in Xcode 12.5. [#1772](https://github.com/stripe/stripe-ios/issues/1772) +* Fixes a layout issue when confirming payments in SwiftUI. [#1761](https://github.com/stripe/stripe-ios/issues/1761) (Thanks [mvarie](https://github.com/mvarie)!) +* Fixes a potential race condition when finalizing 3DS2 confirmations. +* Fixes an issue where a 3DS2 transaction could result in an incorrect error message when the card number is incorrect. [#1778](https://github.com/stripe/stripe-ios/issues/1778) +* Fixes an issue where `STPPaymentHandler.shared().handleNextAction` sometimes didn't return a `handleActionError`. [#1769](https://github.com/stripe/stripe-ios/issues/1769) +* Fixes a layout issue when confirming payments in SwiftUI. [#1761](https://github.com/stripe/stripe-ios/issues/1761) (Thanks [mvarie](https://github.com/mvarie)!) +* Fixes an issue with opening URLs on Mac Catalyst +* Fixes an issue where OXXO next action is mistaken for a cancel in STPPaymentHandler +* SetupIntents for iDEAL, Bancontact, EPS, and Sofort will now send the required mandate information. +* Adds support for BLIK. +* Adds `decline_code` information to STPError. [#1755](https://github.com/stripe/stripe-ios/issues/1755) +* Adds support for SetupIntents to STPApplePayContext +* Allows STPPaymentCardTextField to be subclassed. [#1768](https://github.com/stripe/stripe-ios/issues/1768) + +## 21.3.1 2021-03-25 +* Adds support for Maestro in Apple Pay on iOS 12 or later. + +## 21.3.0 2021-02-18 +* Adds support for SwiftUI in custom integration using the `STPPaymentCardTextField.Representable` View and the `.paymentConfirmationSheet()` ViewModifier. See `IntegrationTester` for usage examples. +* Removes the UIViewController requirement from STPApplePayContext, allowing it to be used in SwiftUI. +* Fixes an issue where `STPPaymentOptionsViewController` could fail to register a card. [#1758](https://github.com/stripe/stripe-ios/issues/1758) +* Fixes an issue where some UnionPay test cards were marked as invalid. [#1759](https://github.com/stripe/stripe-ios/issues/1759) +* Updates tests to run on Carthage 0.37 with .xcframeworks. + + +## 21.2.1 2021-01-29 +* Fixed an issue where a payment card text field could resize incorrectly on smaller devices or with certain languages. [#1600](https://github.com/stripe/stripe-ios/issues/1600) +* Fixed an issue where the SDK could always return English strings in certain situations. [#1677](https://github.com/stripe/stripe-ios/pull/1677) (Thanks [glaures-ioki](https://github.com/glaures-ioki)!) +* Fixed an issue where an STPTheme had no effect on the navigation bar. [#1753](https://github.com/stripe/stripe-ios/pull/1753) (Thanks [@rbenna](https://github.com/rbenna)!) +* Fixed handling of nil region codes. [#1752](https://github.com/stripe/stripe-ios/issues/1752) +* Fixed an issue preventing card scanning from being disabled. [#1751](https://github.com/stripe/stripe-ios/issues/1751) +* Fixed an issue with enabling card scanning in an app with a localized Info.plist.[#1745](https://github.com/stripe/stripe-ios/issues/1745) +* Added a missing additionalDocument parameter to STPConnectAccountIndividualVerification. +* Added support for Afterpay/Clearpay. + +## 21.2.0 2021-01-06 +* Stripe3DS2 is now open source software under the MIT License. +* Fixed various issues with bundling Stripe3DS2 in Cocoapods and Swift Package Manager. All binary dependencies have been removed. +* Fixed an infinite loop during layout on small screen sizes. [#1731](https://github.com/stripe/stripe-ios/issues/1731) +* Fixed issues with missing image assets when using Cocoapods. [#1655](https://github.com/stripe/stripe-ios/issues/1655) [#1722](https://github.com/stripe/stripe-ios/issues/1722) +* Fixed an issue which resulted in unnecessary queries to the BIN information service. +* Adds the ability to `attach` and `detach` PaymentMethod IDs to/from a CustomerContext. [#1729](https://github.com/stripe/stripe-ios/issues/1729) +* Adds support for NetBanking. + +## 21.1.0 2020-12-07 +* Fixes a crash during manual confirmation of a 3DS2 payment. [#1725](https://github.com/stripe/stripe-ios/issues/1725) +* Fixes an issue that could cause some image assets to be missing in certain configurations. [#1722](https://github.com/stripe/stripe-ios/issues/1722) +* Fixes an issue with confirming Alipay transactions. +* Re-exposes `cardNumber` parameter in `STPPaymentCardTextField`. +* Adds support for UPI. + +## 21.0.1 2020-11-19 +* Fixes an issue with some initializers not being exposed publicly following the [conversion to Swift](https://stripe.com/docs/mobile/ios/sdk-21-migration). +* Updates GrabPay integration to support synchronous updates. + +## 21.0.0 2020-11-18 +* The SDK is now written in Swift, and some manual changes are required. Migration instructions are available at [https://stripe.com/docs/mobile/ios/sdk-21-migration](https://stripe.com/docs/mobile/ios/sdk-21-migration). +* Adds full support for Apple silicon. +* Xcode 12.2 is now required. + +## 20.1.1 2020-10-23 +* Fixes an issue when using Cocoapods 1.10 and Xcode 12. [#1683](https://github.com/stripe/stripe-ios/pull/1683) +* Fixes a warning when using Swift Package Manager. [#1675](https://github.com/stripe/stripe-ios/pull/1675) + +## 20.1.0 2020-10-15 +* Adds support for OXXO. [#1592](https://github.com/stripe/stripe-ios/pull/1592) +* Applies a workaround for various bugs in Swift Package Manager. [#1671](https://github.com/stripe/stripe-ios/pull/1671) Please see [#1673](https://github.com/stripe/stripe-ios/issues/1673) for additional notes when using Xcode 12.0. +* Card scanning now works when the device's orientation is unknown. [#1659](https://github.com/stripe/stripe-ios/issues/1659) +* The expiration date field's Simplified Chinese localization has been corrected. (Thanks [cythb](https://github.com/cythb)!) [#1654](https://github.com/stripe/stripe-ios/pull/1654) + +## 20.0.0 2020-09-14 +* [Card scanning](https://github.com/stripe/stripe-ios#card-scanning) is now built into STPAddCardViewController. Card.io support has been removed. [#1629](https://github.com/stripe/stripe-ios/pull/1629) +* Shrunk the SDK from 1.3MB when compressed & thinned to 0.7MB, allowing for easier App Clips integration. [#1643](https://github.com/stripe/stripe-ios/pull/1643) +* Swift Package Manager, Apple Silicon, and Catalyst are now fully supported on Xcode 12. [#1644](https://github.com/stripe/stripe-ios/pull/1644) +* Adds support for 19-digit cards. [#1608](https://github.com/stripe/stripe-ios/pull/1608) +* Adds GrabPay and Sofort as PaymentMethod. [#1627](https://github.com/stripe/stripe-ios/pull/1627) +* Drops support for iOS 10. [#1643](https://github.com/stripe/stripe-ios/pull/1643) + +## 19.4.0 2020-08-13 +* `pkPaymentErrorForStripeError` no longer returns PKPaymentUnknownErrors. Instead, it returns the original NSError back, resulting in dismissal of the Apple Pay sheet. This means ApplePayContext dismisses the Apple Pay sheet for all errors that aren't specifically PKPaymentError types. +* `metadata` fields are no longer populated on retrieved Stripe API objects and must be fetched on your server using your secret key. If this is causing issues with your deployed app versions please reach out to [Stripe Support](https://support.stripe.com/?contact=true). These fields have been marked as deprecated and will be removed in a future SDK version. + +## 19.3.0 2020-05-28 +* Adds giropay PaymentMethod bindings [#1569](https://github.com/stripe/stripe-ios/pull/1569) +* Adds Przelewy24 (P24) PaymentMethod bindings [#1556](https://github.com/stripe/stripe-ios/pull/1556) +* Adds Bancontact PaymentMethod bindings [#1565](https://github.com/stripe/stripe-ios/pull/1565) +* Adds EPS PaymentMethod bindings [#1578](https://github.com/stripe/stripe-ios/pull/1578) +* Replaces es-AR localization with es-419 for full Latin American Spanish support and updates multiple localizations [#1549](https://github.com/stripe/stripe-ios/pull/1549) [#1570](https://github.com/stripe/stripe-ios/pull/1570) +* Fixes missing custom number placeholder in `STPPaymentCardTextField` [#1576](https://github.com/stripe/stripe-ios/pull/1576) +* Adds tabbing on external keyboard support to `STPAUBECSFormView` and correctly types it as a `UIView` instead of `UIControl` [#1580](https://github.com/stripe/stripe-ios/pull/1580) + +## 19.2.0 2020-05-01 +* Adds ability to attach shipping details when confirming PaymentIntents [#1558](https://github.com/stripe/stripe-ios/pull/1558) +* `STPApplePayContext` now provides shipping details in the `applePayContext:didCreatePaymentMethod:paymentInformation:completion:` delegate method and automatically attaches shipping details to PaymentIntents (unless manual confirmation)[#1561](https://github.com/stripe/stripe-ios/pull/1561) +* Adds support for the BECS Direct Debit payment method for Stripe users in Australia [#1547](https://github.com/stripe/stripe-ios/pull/1547) + +## 19.1.1 2020-04-28 +* Add advancedFraudSignalsEnabled property [#1560](https://github.com/stripe/stripe-ios/pull/1560) + +## 19.1.0 2020-04-15 +* Relaxes need for dob for full name connect account (`STPConnectAccountIndividualParams`). [#1539](https://github.com/stripe/stripe-ios/pull/1539) +* Adds Chinese (Traditional) and Chinese (Hong Kong) localizations [#1536](https://github.com/stripe/stripe-ios/pull/1536) +* Adds `STPApplePayContext`, a helper class for Apple Pay. [#1499](https://github.com/stripe/stripe-ios/pull/1499) +* Improves accessibility [#1513](https://github.com/stripe/stripe-ios/pull/1513), [#1504](https://github.com/stripe/stripe-ios/pull/1504) +* Adds support for the Bacs Direct Debit payment method [#1487](https://github.com/stripe/stripe-ios/pull/1487) +* Adds support for 16 digit Diners Club cards [#1498](https://github.com/stripe/stripe-ios/pull/1498) + +## 19.0.1 2020-03-24 +* Fixes an issue building with Xcode 11.4 [#1526](https://github.com/stripe/stripe-ios/pull/1526) + +## 19.0.0 2020-02-12 +* Deprecates the `STPAPIClient` `initWithConfiguration:` method. Set the `configuration` property on the `STPAPIClient` instance instead. [#1474](https://github.com/stripe/stripe-ios/pull/1474) +* Deprecates `publishableKey` and `stripeAccount` properties of `STPPaymentConfiguration`. See [MIGRATING.md](https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md) for more details. [#1474](https://github.com/stripe/stripe-ios/pull/1474) +* Adds explicit STPAPIClient properties on all SDK components that make API requests. These default to `[STPAPIClient sharedClient]`. This is a breaking change for some users of `stripeAccount`. See [MIGRATING.md](https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md) for more details. [#1469](https://github.com/stripe/stripe-ios/pull/1469) +* The user's postal code is now collected by default in countries that support postal codes. We always recommend collecting a postal code to increase card acceptance rates and reduce fraud. See [MIGRATING.md](https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md) for more details. [#1479](https://github.com/stripe/stripe-ios/pull/1479) + +## 18.4.0 2020-01-15 +* Adds support for Klarna Pay on Sources API [#1444](https://github.com/stripe/stripe-ios/pull/1444) +* Compresses images using `pngcrush` to reduce SDK size [#1471](https://github.com/stripe/stripe-ios/pull/1471) +* Adds support for CVC recollection in PaymentIntent confirm [#1473](https://github.com/stripe/stripe-ios/pull/1473) +* Fixes a race condition when setting `defaultPaymentMethod` on `STPPaymentOptionsViewController` [#1476](https://github.com/stripe/stripe-ios/pull/1476) + +## 18.3.0 2019-12-3 +* STPAddCardViewControllerDelegate methods previously removed in v16.0.0 are now marked as deprecated, to help migrating users [#1439](https://github.com/stripe/stripe-ios/pull/1439) +* Fixes an issue where canceling 3DS authentication could leave PaymentIntents in an inaccurate `requires_action` state [#1443](https://github.com/stripe/stripe-ios/pull/1443) +* Fixes text color for large titles [#1446](https://github.com/stripe/stripe-ios/pull/1446) +* Re-adds support for pre-selecting the last selected payment method in STPPaymentContext and STPPaymentOptionsViewController. [#1445](https://github.com/stripe/stripe-ios/pull/1445) +* Fix crash when adding/removing postal code cells [#1450](https://github.com/stripe/stripe-ios/pull/1450) + +## 18.2.0 2019-10-31 +* Adds support for creating tokens with the last 4 digits of an SSN [#1432](https://github.com/stripe/stripe-ios/pull/1432) +* Renames Standard Integration to Basic Integration + +## 18.1.0 2019-10-29 +* Adds localizations for English (Great Britain), Korean, Russian, and Turkish [#1373](https://github.com/stripe/stripe-ios/pull/1373) +* Adds support for SEPA Debit as a PaymentMethod [#1415](https://github.com/stripe/stripe-ios/pull/1415) +* Adds support for custom SEPA Debit Mandate params with PaymentMethod [#1420](https://github.com/stripe/stripe-ios/pull/1420) +* Improves postal code UI for users with mismatched regions [#1302](https://github.com/stripe/stripe-ios/issues/1302) +* Fixes a potential crash when presenting the add card view controller [#1426](https://github.com/stripe/stripe-ios/issues/1426) +* Adds offline status checking to FPX payment flows [#1422](https://github.com/stripe/stripe-ios/pull/1422) +* Adds support for push provisions for Issuing users [#1396](https://github.com/stripe/stripe-ios/pull/1396) + +## 18.0.0 2019-10-04 +* Adds support for building on macOS 10.15 with Catalyst. Use the .xcframework file attached to the release in GitHub. Cocoapods support is coming soon. [#1364](https://github.com/stripe/stripe-ios/issues/1364) +* Errors from the Payment Intents API are now localized by default. See [MIGRATING.md](https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md) for details. +* Adds support for FPX in Standard Integration. [#1390](https://github.com/stripe/stripe-ios/pull/1390) +* Simplified Apple Pay integration when using 3DS2. [#1386](https://github.com/stripe/stripe-ios/pull/1386) +* Improved autocomplete behavior for some STPPaymentHandler blocks. [#1403](https://github.com/stripe/stripe-ios/pull/1403) +* Fixed spurious `keyboardWillAppear` messages triggered by STPPaymentTextCard. [#1393](https://github.com/stripe/stripe-ios/pull/1393) +* Fixed an issue with non-numeric placeholders in STPPaymentTextCard. [#1394](https://github.com/stripe/stripe-ios/pull/1394) +* Dropped support for iOS 9. Please continue to use 17.0.2 if you need to support iOS 9. + +## 17.0.2 2019-09-24 +* Fixes an error that could prevent a 3D Secure 2 challenge dialog from appearing in certain situations. +* Improved VoiceOver support. [#1384](https://github.com/stripe/stripe-ios/pull/1384) +* Updated Apple Pay and Mastercard branding. [#1374](https://github.com/stripe/stripe-ios/pull/1374) +* Updated the Standard Integration example app to use automatic confirmation. [#1363](https://github.com/stripe/stripe-ios/pull/1363) +* Added support for collecting email addresses and phone numbers from Apple Pay. [#1372](https://github.com/stripe/stripe-ios/pull/1372) +* Introduced support for FPX payments. (Invite-only Beta) [#1375](https://github.com/stripe/stripe-ios/pull/1375) + +## 17.0.1 2019-09-09 +* Cancellation during the 3DS2 flow will no longer cause an unexpected error. [#1353](https://github.com/stripe/stripe-ios/pull/1353) +* Large Title UIViewControllers will no longer have a transparent background in iOS 13. [#1362](https://github.com/stripe/stripe-ios/pull/1362) +* Adds an `availableCountries` option to STPPaymentConfiguration, allowing one to limit the list of countries in the address entry view. [#1327](https://github.com/stripe/stripe-ios/pull/1327) +* Fixes a crash when using card.io. [#1357](https://github.com/stripe/stripe-ios/pull/1357) +* Fixes an issue with birthdates when creating a Connect account. [#1361](https://github.com/stripe/stripe-ios/pull/1361) +* Updates example code to Swift 5. [#1354](https://github.com/stripe/stripe-ios/pull/1354) +* The default value of `[STPTheme translucentNavigationBar]` is now `YES`. [#1367](https://github.com/stripe/stripe-ios/pull/1367) + +## 17.0.0 2019-09-04 +* Adds support for iOS 13, including Dark Mode and minor bug fixes. [#1307](https://github.com/stripe/stripe-ios/pull/1307) +* Updates API version from 2015-10-12 to 2019-05-16 [#1254](https://github.com/stripe/stripe-ios/pull/1254) + * Adds `STPSourceRedirectStatusNotRequired` to `STPSourceRedirectStatus`. Previously, optional redirects were marked as `STPSourceRedirectStatusSucceeded`. + * Adds `STPSourceCard3DSecureStatusRecommended` to `STPSourceCard3DSecureStatus`. + * Removes `STPLegalEntityParams`. Initialize an `STPConnectAccountParams` with an `individual` or `company` dictionary instead. See https://stripe.com/docs/api/tokens/create_account#create_account_token-account +* Changes the `STPPaymentContextDelegate paymentContext:didCreatePaymentResult:completion:` completion block type to `STPPaymentStatusBlock`, to let you inform the context that the user canceled. +* Adds initial support for WeChat Pay. [#1326](https://github.com/stripe/stripe-ios/pull/1326) +* The user's billing address will now be included when creating a PaymentIntent from an Apple Pay token. [#1334](https://github.com/stripe/stripe-ios/pull/1334) + + +## 16.0.7 2019-08-23 +* Fixes STPThreeDSUICustomization not initializing defaults correctly. [#1303](https://github.com/stripe/stripe-ios/pull/1303) +* Fixes STPPaymentHandler treating post-authentication errors as authentication errors [#1291](https://github.com/stripe/stripe-ios/pull/1291) +* Removes preferredStatusBarStyle from STPThreeDSUICustomization, see STPThreeDSNavigationBarCustomization.barStyle instead [#1308](https://github.com/stripe/stripe-ios/pull/1308) + +## 16.0.6 2019-08-13 +* Adds a method to STPAuthenticationContext allowing you to configure the SFSafariViewController presented for web-based authentication. +* Adds STPAddress initializer that takes STPPaymentMethodBillingDetails. [#1278](https://github.com/stripe/stripe-ios/pull/1278) +* Adds convenience method to populate STPUserInformation with STPPaymentMethodBillingDetails. [#1278](https://github.com/stripe/stripe-ios/pull/1278) +* STPShippingAddressViewController prefills billing address for PaymentMethods too now, not just Card. [#1278](https://github.com/stripe/stripe-ios/pull/1278) +* Update libStripe3DS2.a to avoid a conflict with Firebase. [#1293](https://github.com/stripe/stripe-ios/issues/1293) + +## 16.0.5 2019-08-09 +* Fixed an compatibility issue when building with certain Cocoapods configurations. [#1288](https://github.com/stripe/stripe-ios/issues/1288) + +## 16.0.4 2019-08-08 +* Improved compatibility with other OpenSSL-using libraries. [#1265](https://github.com/stripe/stripe-ios/issues/1265) +* Fixed compatibility with Xcode 10.1. [#1273](https://github.com/stripe/stripe-ios/issues/1273) +* Fixed an issue where STPPaymentContext could be left in a bad state when cancelled. [#1284](https://github.com/stripe/stripe-ios/pull/1284) + +## 16.0.3 2019-08-01 +* Changes to code obfuscation, resolving an issue with App Store review [#1269](https://github.com/stripe/stripe-ios/pull/1269) +* Adds Apple Pay support to STPPaymentHandler [#1264](https://github.com/stripe/stripe-ios/pull/1264) + +## 16.0.2 2019-07-29 +* Adds API to let users set a default payment option for Standard Integration [#1252](https://github.com/stripe/stripe-ios/pull/1252) +* Removes querying the Advertising Identifier (IDFA). +* Adds customizable UIStatusBarStyle to STDSUICustomization. + +## 16.0.1 2019-07-25 +* Migrates Stripe3DS2.framework to libStripe3DS2.a, resolving an issue with App Store validation. [#1246](https://github.com/stripe/stripe-ios/pull/1246) +* Fixes a crash in STPPaymentHandler. [#1244](https://github.com/stripe/stripe-ios/pull/1244) + +## 16.0.0 2019-07-18 +* Migrates STPPaymentCardTextField.cardParams property type from STPCardParams to STPPaymentMethodCardParams +* STPAddCardViewController: + * Migrates addCardViewController:didCreateSource:completion: and addCardViewController:didCreateToken:completion: to addCardViewController:didCreatePaymentMethod:completion + * Removes managedAccountCurrency property - there’s no equivalent parameter necessary for PaymentMethods. +* STPPaymentOptionViewController now shows, adds, removes PaymentMethods instead of Source/Tokens. +* STPCustomerContext, STPBackendAPIAdapter: + * Removes selectDefaultCustomerSource:completion: - Users must explicitly select their Payment Method of choice. + * Migrates detachSourceFromCustomer:completion:, attachSourceToCustomer:completion to detachPaymentMethodFromCustomer:completion:, attachPaymentMethodToCustomer:completion: + * Adds listPaymentMethodsForCustomerWithCompletion: - the Customer object doesn’t contain attached Payment Methods; you must fetch it from the Payment Methods API. +* STPPaymentContext now uses the new Payment Method APIs listed above instead of Source/Token, and returns the reworked STPPaymentResult containing a PaymentMethod. +* Migrates STPPaymentResult.source to paymentMethod of type STPPaymentMethod +* Deprecates STPPaymentIntentAction* types, replaced by STPIntentAction*. [#1208](https://github.com/stripe/stripe-ios/pull/1208) + * Deprecates `STPPaymentIntentAction`, replaced by `STPIntentAction` + * Deprecates `STPPaymentIntentActionType`, replaced by `STPIntentActionType` + * Deprecates `STPPaymentIntentActionRedirectToURL`, replaced by `STPIntentActionTypeRedirectToURL` +* Adds support for SetupIntents. See https://stripe.com/docs/payments/cards/saving-cards#saving-card-without-payment +* Adds support for 3DS2 authentication. See https://stripe.com/docs/mobile/ios/authentication + +## 15.0.1 2019-04-16 +* Adds configurable support for JCB (Apple Pay). [#1158](https://github.com/stripe/stripe-ios/pull/1158) +* Updates sample apps to use `PaymentIntents` and `PaymentMethods` where available. [#1159](https://github.com/stripe/stripe-ios/pull/1159) +* Changes `STPPaymentMethodCardParams` `expMonth` and `expYear` property types to `NSNumber *` to fix a bug using Apple Pay. [#1161](https://github.com/stripe/stripe-ios/pull/1161) + +## 15.0.0 2019-3-19 +* Renames all former references to 'PaymentMethod' to 'PaymentOption'. See [MIGRATING.md](/MIGRATING.md) for more details. [#1139](https://github.com/stripe/stripe-ios/pull/1139) + * Renames `STPPaymentMethod` to `STPPaymentOption` + * Renames `STPPaymentMethodType` to `STPPaymentOptionType` + * Renames `STPApplePaymentMethod` to `STPApplePayPaymentOption` + * Renames `STPPaymentMethodTuple` to `STPPaymentOptionTuple` + * Renames `STPPaymentMethodsViewController` to `STPPaymentOptionsViewController` + * Renames all properties, methods, comments referencing 'PaymentMethod' to 'PaymentOption' +* Rewrites `STPaymentMethod` and `STPPaymentMethodType` to match the [Stripe API](https://stripe.com/docs/api/payment_methods/object). [#1140](https://github.com/stripe/stripe-ios/pull/1140). +* Adds `[STPAPI createPaymentMethodWithParams:completion:]`, which creates a PaymentMethod. [#1141](https://github.com/stripe/stripe-ios/pull/1141) +* Adds `paymentMethodParams` and `paymentMethodId` to `STPPaymentIntentParams`. You can now confirm a PaymentIntent with a PaymentMethod. [#1142](https://github.com/stripe/stripe-ios/pull/1142) +* Adds `paymentMethodTypes` to `STPPaymentIntent`. +* Deprecates several Source-named properties, based on changes to the [Stripe API](https://stripe.com/docs/upgrades#2019-02-11). [#1146](https://github.com/stripe/stripe-ios/pull/1146) + * Deprecates `STPPaymentIntentParams.saveSourceToCustomer`, replaced by `savePaymentMethod` + * Deprecates `STPPaymentIntentsStatusRequiresSource`, replaced by `STPPaymentIntentsStatusRequiresPaymentMethod` + * Deprecates `STPPaymentIntentsStatusRequiresSourceAction`, replaced by `STPPaymentIntentsStatusRequiresAction` + * Deprecates `STPPaymentIntentSourceAction`, replaced by `STPPaymentIntentAction` + * Deprecates `STPPaymentSourceActionAuthorizeWithURL`, replaced by `STPPaymentActionRedirectToURL` + * Deprecates `STPPaymentIntent.nextSourceAction`, replaced by `nextAction` +* Added new localizations for the following languages [#1050](https://github.com/stripe/stripe-ios/pull/1050) + * Danish + * Spanish (Argentina/Latin America) + * French (Canada) + * Norwegian + * Portuguese (Brazil) + * Portuguese (Portugal) + * Swedish +* Deprecates `STPEphemeralKeyProvider`, replaced by `STPCustomerEphemeralKeyProvider`. We now allow for ephemeral keys that are not customer [#1131](https://github.com/stripe/stripe-ios/pull/1131) +* Adds CVC image for Amex cards [#1046](https://github.com/stripe/stripe-ios/pull/1046) +* Fixed `STPPaymentCardTextField.nextFirstResponderField` to never return nil [#1059](https://github.com/stripe/stripe-ios/pull/1059) +* Improves return key functionality for `STPPaymentCardTextField`, `STPAddCardViewController` [#1059](https://github.com/stripe/stripe-ios/pull/1059) +* Add postal code support for Saudi Arabia [#1127](https://github.com/stripe/stripe-ios/pull/1127) +* CVC field updates validity if card number/brand change [#1128](https://github.com/stripe/stripe-ios/pull/1128) + +## 14.0.0 2018-11-14 +* Changes `STPPaymentCardTextField`, which now copies the `cardParams` property. See [MIGRATING.md](/MIGRATING.md) for more details. [#1031](https://github.com/stripe/stripe-ios/pull/1031) +* Renames `STPPaymentIntentParams.returnUrl` to `STPPaymentIntentParams.returnURL`. [#1037](https://github.com/stripe/stripe-ios/pull/1037) +* Removes `STPPaymentIntent.returnUrl` and adds `STPPaymentIntent.nextSourceAction`, based on changes to the [Stripe API](https://stripe.com/docs/upgrades#2018-11-08). [#1038](https://github.com/stripe/stripe-ios/pull/1038) +* Adds `STPVerificationParams.document_back` property. [#1017](https://github.com/stripe/stripe-ios/pull/1017) +* Fixes bug in `STPPaymentMethodsViewController` where selected payment method changes back if it wasn't dismissed in the `didFinish` delegate method. [#1020](https://github.com/stripe/stripe-ios/pull/1020) + +## 13.2.0 2018-08-14 +* Adds `STPPaymentMethod` protocol implementation for `STPSource`. You can now call `image`/`templatedImage`/`label` on a source. [#976](https://github.com/stripe/stripe-ios/pull/976) +* Fixes crash in `STPAddCardViewController` with some prefilled billing addresses [#1004](https://github.com/stripe/stripe-ios/pull/1004) +* Fixes `STPPaymentCardTextField` layout issues on small screens [#1009](https://github.com/stripe/stripe-ios/pull/1009) +* Fixes hidden text fields in `STPPaymentCardTextField` from being read by VoiceOver [#1012](https://github.com/stripe/stripe-ios/pull/1012) +* Updates example app to add client-side metadata `charge_request_id` to requests to `example-ios-backend` [#1008](https://github.com/stripe/stripe-ios/pull/1008) + +## 13.1.0 2018-07-13 +* Adds `STPPaymentIntent` to support PaymentIntents. [#985](https://github.com/stripe/stripe-ios/pull/985), [#986](https://github.com/stripe/stripe-ios/pull/986), [#987](https://github.com/stripe/stripe-ios/pull/987), [#988](https://github.com/stripe/stripe-ios/pull/988) +* Reduce `NSURLSession` memory footprint. [#969](https://github.com/stripe/stripe-ios/pull/969) +* Fixes invalid JSON error when deleting `Card` from a `Customer`. [#992](https://github.com/stripe/stripe-ios/pull/992) + +## 13.0.3 2018-06-11 +* Fixes payment method label overlapping the checkmark, for Amex on small devices [#952](https://github.com/stripe/stripe-ios/pull/952) +* Adds EPS and Multibanco support to `STPSourceParams` [#961](https://github.com/stripe/stripe-ios/pull/961) +* Adds `STPBillingAddressFieldsName` option to `STPBillingAddressFields` [#964](https://github.com/stripe/stripe-ios/pull/964) +* Fixes crash in `STPColorUtils.perceivedBrightnessForColor` [#954](https://github.com/stripe/stripe-ios/pull/954) +* Applies recommended project changes for Xcode 9.4 [#963](https://github.com/stripe/stripe-ios/pull/963) +* Fixes `[Stripe handleStripeURLCallbackWithURL:url]` incorrectly returning `NO` [#962](https://github.com/stripe/stripe-ios/pull/962) + +## 13.0.2 2018-05-24 +* Makes iDEAL `name` parameter optional, also accepts empty string as `nil` [#940](https://github.com/stripe/stripe-ios/pull/940) +* Adjusts scroll view content offset behavior when focusing on a text field [#943](https://github.com/stripe/stripe-ios/pull/943) + +## 13.0.1 2018-05-17 +* Fixes an issue in `STPRedirectContext` causing some redirecting sources to fail in live mode due to prematurely dismissing the `SFSafariViewController` during the initial redirects. [#937](https://github.com/stripe/stripe-ios/pull/937) + +## 13.0.0 2018-04-26 +* Removes Bitcoin source support. See MIGRATING.md. [#931](https://github.com/stripe/stripe-ios/pull/931) +* Adds Masterpass support to `STPSourceParams` [#928](https://github.com/stripe/stripe-ios/pull/928) +* Adds community submitted Norwegian (nb) translation. Thank @Nailer! +* Fixes example app usage of localization files (they were not able to be tested in Finnish and Norwegian before) +* Silences STPAddress deprecation warnings we ignore to stay compatible with older iOS versions +* Fixes "Card IO" link in full SDK reference [#913](https://github.com/stripe/stripe-ios/pull/913) + +## 12.1.2 2018-03-16 +* Updated the "62..." credit card number BIN range to show a UnionPay icon + +## 12.1.1 2018-02-22 +* Fix issue with apple pay token creation in PaymentContext, introduced by 12.1.0. [#899](https://github.com/stripe/stripe-ios/pull/899) +* Now matches clang static analyzer settings with Cocoapods, so you won't see any more analyzer issues. [#897](https://github.com/stripe/stripe-ios/pull/897) + +## 12.1.0 2018-02-05 +* Adds `createCardSources` to `STPPaymentConfiguration`. If you enable this option, when your user adds a card in the SDK's UI, a card source will be created and attached to their Stripe Customer. If this option is disabled (the default), a card token is created. For more information on card sources, see https://stripe.com/docs/sources/cards + +## 12.0.1 2018-01-31 +* Adding Visa Checkout support to `STPSourceParams` [#889](https://github.com/stripe/stripe-ios/pull/889) + +## 12.0.0 2018-01-16 +* Minimum supported iOS version is now 9.0. + * If you need to support iOS 8, the last supported version is [11.5.0](https://github.com/stripe/stripe-ios/releases/tag/v11.5.0) +* Minimum supported Xcode version is now 9.0 +* `AddressBook` framework support has been removed. +* `STPRedirectContext` will no longer retain itself for the duration of the redirect, you must explicitly maintain a reference to it yourself. [#846](https://github.com/stripe/stripe-ios/pull/846) +* `STPPaymentConfiguration.requiredShippingAddress` now is a set of `STPContactField` objects instead of a `PKAddressField` bitmask. [#848](https://github.com/stripe/stripe-ios/pull/848) +* See MIGRATING.md for more information on any of the previously mentioned breaking API changes. +* Pre-built view controllers now layout properly on iPhone X in landscape orientation, respecting `safeAreaInsets`. [#854](https://github.com/stripe/stripe-ios/pull/854) +* Fixes a bug in `STPAddCardViewController` that prevented users in countries without postal codes from adding a card when `requiredBillingFields = .Zip`. [#853](https://github.com/stripe/stripe-ios/pull/853) +* Fixes a bug in `STPPaymentCardTextField`. When completely filled out, it ignored calls to `becomeFirstResponder`. [#855](https://github.com/stripe/stripe-ios/pull/855) +* `STPPaymentContext` now has a `largeTitleDisplayMode` property, which you can use to control the title display mode in the navigation bar of our pre-built view controllers. [#849](https://github.com/stripe/stripe-ios/pull/849) +* Fixes a bug where `STPPaymentContext`'s `retryLoading` method would not re-retrieve the customer object, even after calling `STPCustomerContext`'s `clearCachedCustomer` method. [#863](https://github.com/stripe/stripe-ios/pull/863) +* `STPPaymentContext`'s `retryLoading` method will now always attempt to retrieve a new customer object, regardless of whether a cached customer object is available. Previously, this method was only intended for recovery from a loading error; if a customer had already been retrieved, `retryLoading` would do nothing. [#863](https://github.com/stripe/stripe-ios/pull/863) +* `STPCustomerContext` has a new property: `includeApplePaySources`. It is turned off by default. [#864](https://github.com/stripe/stripe-ios/pull/864) +* Adds `UITextContentType` support. This turns on QuickType suggestions for the name, email, and address fields; and uses a better keyboard for Payment Card fields. [#870](https://github.com/stripe/stripe-ios/pull/870) +* Fixes a bug that prevented redirects to the 3D Secure authentication flow when it was optional. [#878](https://github.com/stripe/stripe-ios/pull/878) +* `STPPaymentConfiguration` now has a `stripeAccount` property, which can be used to make API requests on behalf of a Connected account. [#875](https://github.com/stripe/stripe-ios/pull/875) +* Adds `- [STPAPIClient createTokenWithConnectAccount:completion:]`, which creates Tokens for Connect Accounts: (optionally) accepting the Terms of Service, and sending information about the legal entity. [#876](https://github.com/stripe/stripe-ios/pull/876) +* Fixes an iOS 11 bug in `STPPaymentCardTextField` that blocked tapping on the number field while editing the expiration or CVC on narrow devices (4" screens). [#883](https://github.com/stripe/stripe-ios/pull/883) + +## 11.5.0 2017-11-09 +* Adds a new helper method to `STPSourceParams` for creating reusable Alipay sources. [#811](https://github.com/stripe/stripe-ios/pull/811) +* Silences spurious availability warnings when using Xcode9 [#823](https://github.com/stripe/stripe-ios/pull/823) +* Auto capitalizes currency code when using `paymentRequestWithMerchantIdentifier ` to improve compatibility with iOS 11 `PKPaymentAuthorizationViewController` [#829](https://github.com/stripe/stripe-ios/pull/829) +* Fixes a bug in `STPRedirectContext` which caused `SFSafariViewController`-based redirects to incorrectly dismiss when switching apps. [#833](https://github.com/stripe/stripe-ios/pull/833) +* Fixes a bug that incorrectly offered users the option to "Use Billing Address" on the shipping address screen when there was no existing billing address to fill in. [#834](https://github.com/stripe/stripe-ios/pull/834) + +## 11.4.0 2017-10-20 +* Restores `[STPCard brandFromString:]` method which was marked as deprecated in a recent version [#801](https://github.com/stripe/stripe-ios/pull/801) +* Adds `[STPBankAccount metadata]` and `[STPCard metadata]` read-only accessors and improves annotation for `[STPSource metadata]` [#808](https://github.com/stripe/stripe-ios/pull/808) +* Un-deprecates `STPBackendAPIAdapter` and all associated methods. [#813](https://github.com/stripe/stripe-ios/pull/813) +* The `STPBackendAPIAdapter` protocol now includes two optional methods, `detachSourceFromCustomer` and `updateCustomerWithShipping`. If you've implemented a class conforming to `STPBackendAPIAdapter`, you may add implementations of these methods to support deleting cards from a customer and saving shipping info to a customer. [#813](https://github.com/stripe/stripe-ios/pull/813) +* Adds the ability to set custom footers on view controllers managed by the SDK. [#792](https://github.com/stripe/stripe-ios/pull/792) +* `STPPaymentMethodsViewController` will now display saved card sources in addition to saved card tokens. [#810](https://github.com/stripe/stripe-ios/pull/810) +* Fixes a bug where certain requests would return a generic failed to parse response error instead of the actual API error. [#809](https://github.com/stripe/stripe-ios/pull/809) + +## 11.3.0 2017-09-13 +* Adds support for creating `STPSourceParams` for P24 source [#779](https://github.com/stripe/stripe-ios/pull/779) +* Adds support for native app-to-app Alipay redirects [#783](https://github.com/stripe/stripe-ios/pull/783) +* Fixes crash when `paymentContext.hostViewController` is set to a `UINavigationController` [#786](https://github.com/stripe/stripe-ios/pull/786) +* Improves support and compatibility with iOS 11 + * Explicitly disable code coverage generation for compatibility with Carthage in Xcode 9 [#795](https://github.com/stripe/stripe-ios/pull/795) + * Restore use of native "Back" buttons [#789](https://github.com/stripe/stripe-ios/pull/789) +* Changes and fixes methods on `STPCard`, `STPCardParams`, `STPBankAccount`, and `STPBankAccountParams` to bring card objects more in line with the rest of the API. See MIGRATING for further details. + * `STPCard` and `STPCardParams` [#760](https://github.com/stripe/stripe-ios/pull/760) + * `STPBankAccount` and `STPBankAccountParams` [#761](https://github.com/stripe/stripe-ios/pull/761) +* Adds nullability annotations to `STPPaymentMethod` protocol [#753](https://github.com/stripe/stripe-ios/pull/753) +* Improves the `[STPAPIResponseDecodable allResponseFields]` by removing all instances of `[NSNull null]` including ones that are nested. See MIGRATING.md. [#747](https://github.com/stripe/stripe-ios/pull/747) + +## 11.2.0 2017-07-27 +* Adds an option to allow users to delete payment methods from the `STPPaymentMethodsViewController`. Enabled by default but can disabled using the `canDeletePaymentMethods` property of `STPPaymentConfiguration`. + * Screenshots: https://user-images.githubusercontent.com/28276156/28131357-7a353474-66ee-11e7-846c-b38277d111fd.png +* Adds a postal code field to `STPPaymentCardTextField`, configurable with `postalCodeEntryEnabled` and `postalCodePlaceholder`. Disabled by default. +* `STPCustomer`'s `shippingAddress` property is now correctly annotated as nullable. +* Removed `STPCheckoutUnknownError`, `STPCheckoutTooManyAttemptsError`, and `STPCustomerContextMissingKeyProviderError`. These errors will no longer occur. + +## 11.1.0 2017-07-12 +* Adds stripeAccount property to `STPAPIClient`, set this to perform API requests on behalf of a connected account +* Fixes the `routingNumber` property of `STPBankAccount` so that it is populated when the information is available +* Adds iOS Objective-C Style Guide + +## 11.0.0 2017-06-27 +* We've greatly simplified the integration for `STPPaymentContext`. See MIGRATING.md. +* As part of this new integration, we've added a new class, `STPCustomerContext`, which will automatically prefetch your customer and cache it for a brief interval. We recommend initializing your `STPCustomerContext` before your user enters your checkout flow so their payment methods are loaded in advance. If in addition to using `STPPaymentContext`, you create a separate `STPPaymentMethodsViewController` to let your customer manage their payment methods outside of your checkout flow, you can use the same instance of `STPCustomerContext` for both. +* We've added a `shippingAddress` property to `STPUserInformation`, which you can use to pre-fill your user's shipping information. +* `STPPaymentContext` will now save your user's shipping information to their Stripe customer object. Shipping information will automatically be pre-filled from the customer object for subsequent checkouts. +* Fixes nullability annotation for `[STPFile stringFromPurpose:]`. See MIGRATING.md. +* Adds description implementations to all public models, for easier logging and debugging. +* The card autofill via SMS feature of `STPPaymentContext` has been removed. See MIGRATING.md. + +## 10.2.0 2017-06-19 +* We've added a `paymentCountry` property to `STPPaymentContext`. This affects the countryCode of Apple Pay payments, and defaults to "US". You should set this to the country your Stripe account is in. +* `paymentRequestWithMerchantIdentifier:` has been deprecated. See MIGRATING.md +* If the card.io framework is present in your app, `STPPaymentContext` and `STPAddCardViewController` will show a "scan card" button. +* `STPAddCardViewController` will now attempt to auto-fill the users city and state from their entered Zip code (United States only) +* Polling for source object updates is deprecated. Check https://stripe.com/docs for the latest best practices on how to integrate with the sources API using webhooks. +* Fixes a crash in `STPCustomerDeserializer` when both data and error are nil. +* `paymentMethodsViewController:didSelectPaymentMethod:` is now optional. +* Updates the example apps to use Alamofire. + +## 10.1.0 2017-05-05 +* Adds STPRedirectContext, a helper class for handling redirect sources. +* STPAPIClient now supports tokenizing a PII number and uploading images. +* Updates STPPaymentCardTextField's icons to match Elements on the web. When the card number is invalid, the field will now display an error icon. +* The alignment of the new brand icons has changed to match the new CVC and error icons. If you use these icons via `STPImageLibrary`, you may need to adjust your layout. +* STPPaymentCardTextField's isValid property is now KVO-observable. +* When creating STPSourceParams for a SEPA debit source, address fields are now optional. +* `STPPaymentMethodsViewControllerDelegate` now has a separate `paymentMethodsViewControllerDidCancel:` callback, differentiating from successful method selections. You should make sure to also dismiss the view controller in that callback +* Because collecting some basic data on tokenization helps us detect fraud, we've removed the ability to disable analytics collection using `[Stripe disableAnalytics]`. + +## 10.0.1 2017-03-16 +* Fixes a bug where card sources didn't include the card owner's name. +* Fixes an issue where STPPaymentMethodsViewController didn't reload after adding a new payment method. + +## 10.0.0 2017-03-06 +* Adds support for creating, retrieving, and polling Sources. You can enable any payment methods available to you in the Dashboard. + * https://stripe.com/docs/mobile/ios/sources + * https://dashboard.stripe.com/account/payments/settings +* Updates the Objective-C example app to include example integrations using several different payment methods. +* Updates `STPCustomer` to include `STPSource` objects in its `sources` array if a customer has attached sources. +* Removes methods deprecated in Version 6.0. +* Fixes property declarations missing strong/nullable identifiers. + +## 9.4.0 2017-02-03 +* Adds button to billing/shipping entry screens to fill address information from the other one. +* Fixes and unifies view controller behavior around theming and nav bars. +* Adds month validity check to `validationStateForExpirationYear` +* Changes some Apple Pay images to better conform to official guidelines. +* Changes STPPaymentCardTextField's card number placeholder to "4242..." +* Updates STPPaymentCardTextField's CVC placeholder so that it changes to "CVV" for Amex cards + +## 9.3.0 2017-01-05 +* Fixes a regression introduced in v9.0.0 in which color in STPTheme is used as the background color for UINavigationBar + * Note: This will cause navigation bar theming to work properly as described in the Stripe iOS docs, but you may need to audit your custom theme settings if you based them on the actual behavior of 9.0-9.2 +* If the navigation bar has a theme different than the view controller's theme, STP view controllers will use the bar's theme to style it's UIBarButtonItems +* Adds a fallback to using main bundle for localized strings lookup if locale is set to a language the SDK doesn't support +* Adds method to get a string of a card brand from `STPCardBrand` +* Updated description of how to run tests in README +* Fixes crash when user cancels payment before STPBackendAPIAdapter methods finish +* Fixes bug where country picker wouldn't update when first selected. + + +## 9.2.0 2016-11-14 +* Moves FBSnapshotTestCase dependency to Cartfile.private. No changes if you are not using Carthage. +* Adds prebuilt UI for collecting shipping information. + +## 9.1.0 2016-11-01 +* Adds localized strings for 7 languages: de, es, fr, it, ja, nl, zh-Hans. +* Slight redesign to card/billing address entry screen. +* Improved internationalization for State/Province/County address field. +* Adds new Mastercard 2-series BIN ranges. +* Fixes an issue where callbacks may be run on the wrong thread. +* Fixes UIAppearance compatibility in STPPaymentCardTextField. +* Fixes a crash when changing application language via an Xcode scheme. + +## 9.0.0 2016-10-04 +* Change minimum requirements to iOS 8 and Xcode 8 +* Adds "app extension API only" support. +* Updates Swift example app to Swift 3 +* Various fixes to ObjC example app + +## 8.0.7 2016-09-15 +* Add ability to set currency for managed accounts when adding card +* Fix broken links for Privacy Policy/Terms of Service for Remember Me feature +* Sort countries in picker alphabetically by name instead of ISO code +* Make "County" field optional on billing address screen. +* PKPayment-related methods are now annotated as available in iOS8+ only +* Optimized speed of input sanitation methods (thanks @kballard!) + +## 8.0.6 2016-09-01 +* Improved internationalization on billing address forms + * Users in countries that don't use postal codes will no longer see that field. + * The country field is now auto filled in with the phone's region + * Changing the selected country will now live update other fields on the form (such as State/County or Zip/Postal Code). +* Fixed an issue where certain Cocoapods configurations could result in Stripe resource files being used in place of other frameworks' or the app's resources. +* Fixed an issue where when using Apple Pay, STPPaymentContext would fire two `didFinishWithStatus` messages. +* Fixed the `deviceSupportsApplePay` method to also check for Discover cards. +* Removed keys from Stripe.bundle's Info.plist that were causing iTunes Connect to sometimes error on app submission. + +## 8.0.5 2016-08-26 +* You can now optionally use an array of PKPaymentSummaryItems to set your payment amount, if you would like more control over how Apple Pay forms are rendered. +* Updated credit card and Apple Pay icons. +* Fixed some images not being included in the resources bundle target. +* Non-US locales now have an alphanumeric keyboard for postal code entry. +* Modals now use UIModalPresentationStyleFormSheet. +* Added more accessibility labels. +* STPPaymentCardTextField now conforms to UIKeyInput (thanks @theill). + +## 8.0.4 2016-08-01 +* Fixed an issue with Apple Pay payments not using the correct currency. +* View controllers now update their status bar and scroll view indicator styles based on their theme. +* SMS code screen now offers to paste copied codes. + +## 8.0.3 2016-07-25 +* Fixed an issue with some Cocoapods installations + +## 8.0.2 2016-07-09 +* Fixed an issue with custom theming of Stripe UI + +## 8.0.1 2016-07-06 +* Fixed error handling in STPAddCardViewController + +## 8.0.0 2016-06-30 +* Added prebuilt UI for collecting and managing card information. + +## 7.0.2 2016-05-24 +* Fixed an issue with validating certain Visa cards. + +## 7.0.1 2016-04-29 +* Added Discover support for Apple Pay +* Add the now-required `accountHolderName` and `accountHolderType` properties to STPBankAccountParams +* We now record performance metrics for the /v1/tokens API - to disable this behavior, call [Stripe disableAnalytics]. +* You can now demo the SDK more easily by running `pod try stripe`. +* This release also removes the deprecated Checkout functionality from the SDK. + +## 6.2.0 2016-02-05 +* Added an `additionalAPIParameters` field to STPCardParams and STPBankAccountParams for sending additional values to the API - useful for beta features. Similarly, added an `allResponseFields` property to STPToken, STPCard, and STPBankAccount for accessing fields in the response that are not yet reflected in those classes' @properties. + +## 6.1.0 2016-01-21 +* Renamed card on STPPaymentCardTextField to cardParams. +* You can now set an STPPaymentCardTextField's contents programmatically by setting cardParams to an STPCardParams object. +* Added delegate methods for responding to didBeginEditing events in STPPaymentCardTextField. +* Added a UIImage category for accessing our card icon images +* Fixed deprecation warnings for deployment targets >= iOS 9.0 + +## 6.0.0 2015-10-19 +* Splits logic in STPCard into 2 classes - STPCard and STPCardParams. STPCardParams is for making requests to the Stripe API, while STPCard represents the response (you'll almost certainly want just to replace any usage of STPCard in your app with STPCardParams). This also applies to STPBankAccount and the newly-created STPBankAccountParams. +* Version 6.0.1 fixes a minor Cocoapods issue. + +## 5.1.0 2015-08-17 +* Adds STPPaymentCardTextField, a new version of github.com/stripe/PaymentKit featuring many bugfixes. It's useful if you need a pre-built credit card entry form. +* Adds the currency param to STPCard for those using managed accounts & debit card payouts. +* Versions 5.1.1 and 5.1.2 fix minor issues with CocoaPods installation +* Version 5.1.3 contains bug fixes for STPPaymentCardTextField. +* Version 5.1.4 improves compatibility with iOS 9. + +## 5.0.0 2015-08-06 +* Fix an issue with Carthage installation +* Fix an issue with CocoaPods frameworks +* Deprecate native Stripe Checkout + +## 4.0.1 2015-05-06 +* Fix a compiler warning +* Versions 4.0.1 and 4.0.2 fix minor issues with CocoaPods and Carthage installation. + +## 4.0.0 2015-05-06 +* Remove STPPaymentPresenter +* Support for latest ApplePayStubs +* Add nullability annotations to improve Swift support (note: this now requires Swift 1.2) +* Bug fixes + +## 3.1.0 2015-01-19 +* Add support for native Stripe Checkout, as well as STPPaymentPresenter for automatically using Checkout as a fallback for Apple Pay +* Add OSX support, including Checkout +* Add framework targets and Carthage support +* It's safe to remove the STRIPE_ENABLE_APPLEPAY compiler flag after this release. + +## 3.0.0 2015-01-05 +* Migrate code into STPAPIClient +* Add 'brand' and 'funding' properties to STPCard + +## 2.2.2 2014-11-17 +* Add bank account tokenization methods + +## 2.2.1 2014-10-27 +* Add billing address fields to our Apple Pay API +* Various bug fixes and code improvements + +## 2.2.0 2014-10-08 +* Move Apple Pay testing functionality into a separate project, ApplePayStubs. For more info, see github.com/stripe/ApplePayStubs. +* Improve the provided example app + +## 2.1.0 2014-10-07 +* Remove token retrieval API method +* Refactor functional tests to use new XCTestCase functionality + +## 2.0.3 2014-09-24 +* Group ApplePay code in a CocoaPods subspec + +## 2.0.2 2014-09-24 +* Move ApplePay code behind a compiler flag to avoid warnings from Apple when accidentally including it + +## 2.0.1 2014-09-18 +* Fix some small bugs related to ApplePay and iOS8 + +## 2.0 2014-09-09 +* Add support for native payments via Pay + +## 1.2 2014-08-21 +* Removed PaymentKit as a dependency. If you'd like to use it, you may still do so by including it separately. +* Removed STPView. PaymentKit provides a near-identical version of this functionality if you need to migrate. +* Improve example project +* Various code fixes + +## 1.1.4 2014-05-22 +* Fixed an issue where tokenization requests would fail under iOS 6 due to SSL certificate verification + +## 1.1.3 2014-05-12 +* Send some basic version and device details with requests for debugging. +* Added -description to STPToken +* Fixed some minor code nits +* Modernized code + +## 1.1.2 2014-04-21 +* Added test suite for SSL certificate expiry/revocation +* You can now set STPView's delegate from Interface Builder + +## 1.1.1 2014-04-14 +* API methods now verify the server's SSL certificate against a preset blacklist. +* Fixed some bugs with SSL verification. +* Note: This version now requires the `Security` framework. You will need to add this to your app if you're not using CocoaPods. + +## 1.0.4 2014-03-24 + +* Upgraded tests from OCUnit to XCTest +* Fixed an issue with the SenTestingKit dependency +* Removed some dead code + +## 1.0.3 2014-03-21 + +* Fixed: Some example files had target memberships set for StripeiOS and iOSTest. +* Fixed: The example publishable key was expired. +* Fixed: Podspec did not pass linting. +* Some fixes for 64-bit. +* Many improvements to the README. +* Fixed example under iOS 7 +* Some source code cleaning and modernization. + +## 1.0.2 2013-09-09 + +* Add exceptions for null successHandler and errorHandler. +* Added the ability to POST the created token to a URL. +* Made STPCard properties nonatomic. +* Moved PaymentKit to be a submodule; added to Podfile as a dependency. +* Fixed some warnings caught by the static analyzer (thanks to jcjimenez!) + +## 1.0.1 2012-11-16 + +* Add CocoaPods support +* Change directory structure of bindings to make it easier to install + +## 1.0.0 2012-11-16 + +* Initial release + +Special thanks to: Todd Heasley, jcjimenez. diff --git a/MIGRATING.md b/MIGRATING.md new file mode 100644 index 00000000..5d9648e6 --- /dev/null +++ b/MIGRATING.md @@ -0,0 +1,308 @@ +## Migration Guides +### Migrating from versions < 23.0.0 +* The `Stripe` module is now split between `StripePaymentSheet`, `StripePayments`, and `StripePaymentsUI`. Some manual changes may be required. Migration instructions are available at [https://stripe.com/docs/mobile/ios/sdk-23-migration](https://stripe.com/docs/mobile/ios/sdk-23-migration). +* [Changed] If you use PaymentSheet, you must now `import StripePaymentSheet`. PaymentSheet users no longer need to import the `Stripe` module. +* [Changed] The minimum iOS version is now 13.0. If you'd like to deploy for iOS 12.0, please use Stripe SDK 22.8.4. +* [Changed] STPPaymentCardTextField's `cardParams` parameter has been deprecated in favor of `paymentMethodParams`, making it easier to include the postal code from the card field. If you need to access the `STPPaymentMethodCardParams`, use `.paymentMethodParams.card`. + * Note that `.paymentMethodParams` returns a copy, so `paymentMethodParams.card` should not be set directly. If you need to set the card information, set `.paymentMethodParams` to a new STPPaymentMethodParams: +``` +cardField.paymentMethodParams = STPPaymentMethodParams(card: card, billingDetails: nil, metadata: nil) +``` + +### Migrating from versions < 22.8.0 +* `PaymentSheet.reset()` has been renamed to `PaymentSheet.resetCustomer()`. If calling the former method, follow the warning in Xcode and apply the suggested fix-it. + +### Migrating from versions < 22.2.0 +* `StripeConnections` SDK has been renamed to `StripeFinancialConnections`. If you included `StripeConnections` to support ACH Direct Debit payments, you will need to rename the dependency to `StripeFinancialConnections`. If you are manually installing `StripeConnections`, you will need to remove the old `StripeConnections.xcframework` and include the new `StripeFinancialConnections.xcframework`, which can be found in the [release assets](https://github.com/stripe/stripe-ios/releases/tag/22.2.0) for version 22.2.0 of the SDK. + +### Migrating from versions < 22.0.0 +* The minimum iOS version is now 12.0. If you'd like to deploy for iOS 11.0, please use Stripe SDK 21.12.0. +* `IdentityVerificationSheet` now has an availability requirement of iOS 14.3 on its initializer instead of the `present` method. If your app supports iOS versions < 14.3, you will need to add an availability check for iOS 14.3 before initializing the sheet. + +### Migrating from versions < 21.12.0 +* `Stripe` now requires `StripeApplePay`. If you are manually installing `Stripe`, you will need to include `StripeApplePay.xcframework`, which can be found in the [release assets](https://github.com/stripe/stripe-ios/releases/tag/21.12.0) for version 21.12.0 of the SDK. If you are using CocoaPods or Swift Package Manager, these dependencies will be imported automatically. + +### Migrating from versions < 21.10.0 +* `StripeIdentity` now requires `StripeCameraCore`. If you are manually installing `StripeIdentity`, you will need to include `StripeCameraCore.xcframework`, which can be found in the [release assets](https://github.com/stripe/stripe-ios/releases/tag/21.10.0) for version 21.10.0 of the SDK. If you are using CocoaPods or Swift Package Manager, these dependencies will be imported automatically. + +### Migrating from versions < 21.9.0 +* `Stripe` and `StripeIdentity` now require `StripeUICore`. If you are manually installing `Stripe` or `StripeIdentity`, you will need to include `StripeUICore.xcframework`, which can be found in the [release assets](https://github.com/stripe/stripe-ios/releases/tag/21.9.0) for version 21.9.0 of the SDK. If you are using CocoaPods or Swift Package Manager, these dependencies will be imported automatically. + +### Migrating from versions < 21.8.1 +* The `Stripe` module now requires `StripeCore`. If you are manually installing the Stripe SDK, you will need to include `StripeCore.xcframework`, which can be found in the [release assets](https://github.com/stripe/stripe-ios/releases/tag/21.8.1) for version 21.8.1 of the SDK. If you are using CocoaPods or Swift Package Manager, these dependencies will be imported automatically. + +### Migrating from versions < 21.4.0 +* STPPaymentHandler now presents its SFSafariViewController using the `.overFullScreen` presentation style by default. To select a different style, implement the `STPAuthenticationContext.configureSafariViewController(_:)` function in your `STPAuthenticationContext`. + +### Migrating from versions < 21.2.0 +* Stripe3DS2 is now a separate component for Carthage users. You must embed both Stripe.xcframework and Stripe3DS2.xcframework in your app. + +### Migrating from versions < 21.0.0 +* The SDK is now written in Swift, and some manual changes are required. Migration instructions are available at [https://stripe.com/docs/mobile/ios/sdk-21-migration](https://stripe.com/docs/mobile/ios/sdk-21-migration). + +### Migrating from versions < 20.1.0 +* Swift Package Manager users may need to remove and re-add Stripe from the `Frameworks, Libraries, and Embedded Content` section of your target's settings after updating. +* Swift Package Manager users with Xcode 12.0 may need to use a [workaround](https://github.com/stripe/stripe-ios/issues/1673) for a code signing issue. This is fixed in Xcode 12.2. + +### Migrating from versions < 20.0.0 +* The minimum iOS version is now 11.0. If you'd like to deploy for iOS 10.0, please use Stripe SDK 19.4.0. +* Card.io is no longer supported. To enable our built-in [card scanning](https://github.com/stripe/stripe-ios#card-scanning) beta, set the `cardScanningEnabled` flag on STPPaymentConfiguration. +* Catalyst support is out of beta, and now requires Swift Package Manager with Xcode 12 or Cocoapods 1.10. + +### Migrating from versions < 19.4.0 +* `metadata` fields are no longer populated on retrieved Stripe API objects and must be fetched on your server using your secret key. If this is causing issues with your deployed app versions please reach out to [Stripe Support](https://support.stripe.com/?contact=true). These fields have been marked as deprecated and will be removed in a future SDK version. + +### Migrating from versions < 19.3.0 +* `STPAUBECSFormView` now inherits from `UIView` instead of `UIControl` + +### Migrating from versions < 19.2.0 +* The `STPApplePayContext` 'applePayContext:didCreatePaymentMethod:completion:` delegate method now includes paymentInformation: 'applePayContext:didCreatePaymentMethod:paymentInformation:completion:`. + + +### Migrating from versions < 19.0.0 +#### `publishableKey` and `stripeAccount` changes +* Deprecates `publishableKey` and `stripeAccount` properties of `STPPaymentConfiguration`. + * If you used `STPPaymentConfiguration.sharedConfiguration` to set `publishableKey` and/or `stripeAccount`, use `STPAPIClient.sharedClient` instead. + * If you passed a STPPaymentConfiguration instance to an SDK component, you should instead create an STPAPIClient, set publishableKey on it, and set the SDK component's APIClient property. +* The SDK now uses `STPAPIClient.sharedClient` to make API requests by default. + +This changes the behavior of the following classes, which previously used API client instances configured from `STPPaymentConfiguration.shared`: `STPCustomerContext`, `STPPaymentOptionsViewController`, `STPAddCardViewController`, `STPPaymentContext`, `STPPinManagementService`, `STPPushProvisioningContext`. + +You are affected by this change if: + +1. You use `stripeAccount` to work with your Connected accounts +2. You use one of the above affected classes +3. You set different `stripeAccount` values on `STPPaymentConfiguration` and `STPAPIClient`, i.e. `STPPaymentConfiguration.shared.stripeAccount != STPAPIClient.shared.stripeAccount` + +If all three of the above conditions are true, you must update your integration! The SDK used to send `STPPaymentConfiguration.shared.stripeAccount`, and will now send `STPAPIClient.shared.stripeAccount`. + +For example, if you are a Connect user who stores Payment Methods on your platform, but clones PaymentMethods to a connected account and creates direct charges on that connected account i.e. if: + +1. You never set `STPPaymentConfiguration.shared.stripeAccount` +2. You set `STPAPIClient.shared.stripeAccount` + +We recommend you do the following: + +``` + // By default, you don't want the SDK to pass stripeAccount + STPAPIClient.shared().publishableKey = "pk_platform" + STPAPIClient.shared().stripeAccount = nil + + // You do want the SDK to pass stripeAccount when it makes payments directly on your connected account, so + // you create a separate APIClient instance... + let connectedAccountAPIClient = STPAPIClient(publishableKey: "pk_platform") + + // ...set stripeAccount on it... + connectedAccountAPIClient.stripeAccount = "your connected account's id" + + // ...and either set the relevant SDK components' apiClient property to your custom APIClient instance: + STPPaymentHandler.shared().apiClient = connectedAccountAPIClient // e.g. if you are using PaymentIntents + + // ...or use it directly to make API requests with `stripeAccount` set: + connectedAccountAPIClient.createToken(withCard:...) // e.g. if you are using Tokens + Charges +``` +#### Postal code changes +* The user's postal code is now collected by default in countries that support postal codes. We always recommend collecting a postal code to increase card acceptance rates and reduce fraud. To disable this behavior: + * For STPPaymentContext and other full-screen UI, set your `STPPaymentConfiguration`'s `.requiredBillingAddressFields` to `STPBillingAddressFieldsNone`. + * For a PKPaymentRequest, set `.requiredBillingContactFields` to an empty set. If your app supports iOS 10, also set `.requiredBillingAddressFields` to `PKAddressFieldNone`. + * For STPPaymentCardView, set `.postalCodeEntryEnabled` to `NO`. +* Users may now enter spaces, dashes, and uppercase letters into the postal code field in situations where the user has not explicitly selected a country. This allows users with non-US addreses to enter their postal code. +* `STPBillingAddressFieldsZip` has been renamed to `STPBillingAddressFieldsPostalCode`. +#### Localization changes +* All [Stripe Error messages](https://stripe.com/docs/api/errors#errors-message) are now localized + based on the device locale. + + For example, when retrieving a SetupIntent with a nonexistent `id` + when the device locale is set to `Locale.JAPAN`, the error message will now be localized. + ``` + // before - English + "No such setupintent: seti_invalid123" + + // after - Japanese + "そのような setupintent はありません : seti_invalid123" + ``` + +### Migrating from versions < 18.0.0 +* Some error messages from the Payment Intents API are now localized to the user's display language. If your application's logic depends on specific `message` strings from the Stripe API, please use the error [`code`](https://stripe.com/docs/error-codes) instead. +* `STPPaymentResult` may contain a `paymentMethodParams` instead of a `paymentMethod` when using single-use payment methods such as FPX. Because of this, `STPPaymentResult.paymentMethod` is now nullable. Instead of setting the `paymentMethodId` manually on your `paymentIntentParams`, you may now call `paymentIntentParams.configure(with result: STPPaymentResult)`: +``` +// 17.0.0 +paymentIntentParams.paymentMethodId = paymentResult.paymentMethod.stripeId + +// 18.0.0 +paymentIntentParams.configure(with: paymentResult) +``` +* `STPPaymentOptionTypeAll` has been renamed to `STPPaymentOptionTypeDefault`. This option will not include FPX or future optional payment methods. +* The minimum iOS version is now 10.0. If you'd like to deploy for iOS 9.0, please use Stripe SDK 17.0.2. + +### Migrating from versions < 17.0.0 +* The API version has been updated from 2015-10-12 to 2019-05-16. CHANGELOG.md has details on the changes made, which includes breaking changes for `STPConnectAccountParams` users. Your backend Stripe API version should be sufficiently decoupled from the SDK's so that keeping their versions in sync is not required, and no further action is required to migrate to this version of the SDK. +* For STPPaymentContext users: the completion block type in `paymentContext:didCreatePaymentResult:completion:` has changed to `STPPaymentStatusBlock`, to let you inform the context that the user has cancelled. + +### Migrating from versions < 16.0.0 +* The following have been migrated from Source/Token to PaymentMethod. If you have integrated with any of these things, you must also migrate to PaymentMethod and the Payment Intent API. See https://stripe.com/docs/payments/payment-intents/migration. See CHANGELOG.md for more details. + * UI components + * STPPaymentCardTextField + * STPAddCardViewController + * STPPaymentOptionsViewController + * PaymentContext + * STPPaymentContext + * STPCustomerContext + * STPBackendAPIAdapter + * STPPaymentResult + * Standard Integration example project +* `STPPaymentIntentAction*` types have been renamed to `STPIntentAction*`. Xcode should offer a deprecation warning & fix-it to help you migrate. +* `STPPaymentHandler` supports 3DS2 authentication, and is recommended instead of `STPRedirectContext`. See https://stripe.com/docs/mobile/ios/authentication + +### Migrating from versions < 15.0.0 +* "PaymentMethod" has a new meaning: https://stripe.com/docs/api/payment_methods/object. All things referring to "PaymentMethod" have been renamed to "PaymentOption" (see CHANGELOG.md for the full list). `STPPaymentMethod` and `STPPaymentMethodType` have been rewritten to match this new API object. +* PaymentMethod succeeds Source as the recommended way to charge customers. In this vein, several 'Source'-named things have been deprecated, and replaced with 'PaymentMethod' equivalents. For example, `STPPaymentIntentsStatusRequiresSource` is replaced by `STPPaymentIntentsStatusRequiresPaymentMethod` (see CHANGELOG.md for the full list). Following the deprecation warnings & fix-its will be enough to migrate your code - they've simply been renamed, and will continue to work for Source-based flows. + +### Migrating from versions < 14.0.0 +* `STPPaymentCardTextField` now copies the `STPCardParams` object when setting/getting the `cardParams` property, instead of sharing the object with the caller. + * Changes to the `STPCardParams` object after setting `cardParams` no longer mutate the object held by the `STPPaymentCardTextField` + * Changes to the object returned by `STPPaymentCardTextField.cardParams` no longer mutate the object held by the `STPPaymentCardTextField` + * This is a breaking change for code like: `paymentCardTextField.cardParams.name = @"Jane Doe";` +* `STPPaymentIntentParams.returnUrl` has been renamed to `STPPaymentIntentParams.returnURL`. Xcode should offer a deprecation warning & fix-it to help you migrate. +* `STPPaymentIntent.returnUrl` has been removed, because it's no longer a property of the PaymentIntent. When the PaymentIntent status is `.requiresSourceAction`, and the `nextSourceAction.type` is `.authorizeWithURL`, you can find the return URL at `nextSourceAction.authorizeWithURL.returnURL`. + +### Migrating from versions < 13.1.0 + * The SDK now supports PaymentIntents with `STPPaymentIntent`, which use `STPRedirectContext` in the same way that `STPSource` does + * `STPRedirectContextCompletionBlock` has been renamed to `STPRedirectContextSourceCompletionBlock`. It has the same signature, and Xcode should offer a deprecation warning & fix-it to help you migrate. + +### Migrating from versions < 13.0.0 +* Remove Bitcoin source support because Stripe no longer processes Bitcoin payments: https://stripe.com/blog/ending-bitcoin-support + * Sources can no longer have a "STPSourceTypeBitcoin" source type. These sources will now be interpreted as "STPSourceTypeUnknown". + * You can no longer `createBitcoinParams`. Please use a different payment method. + +### Migrating from versions < 12.0.0 +* The SDK now requires iOS 9+ and Xcode version 9+. If you need to support iOS 8 or Xcode 8, the last supported version is [11.5.0](https://github.com/stripe/stripe-ios/releases/tag/v11.5.0) +* `STPPaymentConfiguration.requiredShippingAddress` now is a set of `STPContactField` objects instead of a `PKAddressField` bitmask. + * Most of the previous `PKAddressField` constants have matching `STPContactField` constants. To convert your code, switch to passing in a set of the matching constants + * Example: `(PKAddressField)(PKAddressFieldName|PKAddressFieldPostalAddress)` becomes `[NSSet setwithArray:@[STPContactFieldName, STPContactFieldPostalAddress]]`) + * Anywhere you were using `PKAddressFieldNone` you can now simply pass in `nil` + * If you were using `PKAddressFieldAll`, you must switch to manually listing all the fields that you want. + * The new constants also correspond to and work similarly to Apple's new `PKContactField` values. +* `AddressBook` framework support has been removed. If you were using AddressBook related functionality, you must switch over to using the `Contacts` framework. +* `STPRedirectContext` will no longer retain itself for the duration of the redirect. If you were relying on this functionality, you must change your code to explicitly maintain a reference to it. + +### Migrating from versions < 11.4.0 +* The `STPBackendAPIAdapter` protocol and all associated methods are no longer deprecated. We still recommend using `STPCustomerContext` to update a Stripe customer object on your behalf, rather than using your own implementation of `STPBackendAPIAdapter`. + +### Migrating from versions < 11.3.0 +* Changes to `STPCard`, `STPCardParams`, `STPBankAccount`, and `STPBankAccountParams` + * `STPCard` no longer subclasses from `STPCardParams`. You must now specifically create `STPCardParams` objects to create new tokens. + * `STPBankAccount` no longer subclasses from `STPBankAccountParams`. + * You can no longer directly create `STPCard` objects, you should only use ones that have been decoded from Stripe API responses via `STPAPIClient`. + * All `STPCard` and `STPBankAccount` properties have been made readonly. + * Broken out individual address properties on `STPCard` and `STPCardParams` have been deprecated in favor of the grouped `address` property. +* The value of `[STPAPIResponseDecodable allResponseFields]` is now completely (deeply) filtered to not contain any instances of `[NSNull null]`. Previously, only `[NSNull null]` one level deep (shallow) were removed. + +### Migrating from versions < 11.2.0 +* `STPCustomer`'s `shippingAddress` property is now correctly annotated as nullable. Its type is an optional (`STPAddress?`) in Swift. + +### Migrating from versions < 11.0.0 +- We've greatly simplified the integration for `STPPaymentContext`. In order to migrate to the new `STPPaymentContext` integration using ephemeral keys, you'll need to: + 1. On your backend, add a new endpoint that creates an ephemeral key for the Stripe customer associated with your user, and returns its raw JSON. Note that you should _not_ remove the 3 endpoints you added for your initial PaymentContext integration until you're ready to drop support for previous versions of your app. + 2. In your app, make your API client class conform to `STPEphemeralKeyProvider` by adding a method that requests an ephemeral key from the endpoint you added in (1). + 3. In your app, remove any references to `STPBackendAPIAdapter`. Your API client class will no longer need to conform to `STPBackendAPIAdapter`, and you can delete the `retrieveCustomer`, `attachSourceToCustomer`, and `selectDefaultCustomerSource` methods. + 4. Instead of using the initializers for `STPPaymentContext` or `STPPaymentMethodsViewController` that take an `STPBackendAPIAdapter` parameter, you should use the new initializers that take an `STPCustomerContext` parameter. You'll need to set up your instance of `STPCustomerContext` using the key provider you set up in (2). +- For a more detailed overview of the new integration, you can refer to our tutorial at https://stripe.com/docs/mobile/ios/standard +- `[STPFile stringFromPurpose:]` now returns `nil` for `STPFilePurposeUnknown`. Will return a non-nil value for all other `STPFilePurpose`. +- We've removed the `email` and `phone` properties in `STPUserInformation`. You can pre-fill this information in the shipping form using the new `shippingAddress` property. +- The SMS card fill feature has been removed from `STPPaymentContext`, as well as the associated `smsAutofillDisabled` configuration option (ie it will now always behave as if it is disabled). + +### Migrating from versions < 10.2.0 +- `paymentRequestWithMerchantIdentifier:` has been deprecated. You should instead use `paymentRequestWithMerchantIdentifier:country:currency:`. Apple Pay is now available in many countries and currencies, and you should use the appropriate values for your business. +- We've added a `paymentCountry` property to `STPPaymentContext`. This affects the countryCode of Apple Pay payments, and defaults to "US". You should set this to the country your Stripe account is in. +- Polling for source object updates is deprecated. Check https://stripe.com/docs for the latest best practices on how to integrate with the sources API using webhooks. +- `paymentMethodsViewController:didSelectPaymentMethod:` is now optional. If you have an empty implementation of this method, you can remove it. + +### Migrating from versions < 10.1.0 + +- STPPaymentMethodsViewControllerDelegate now has a separate `paymentMethodsViewControllerDidCancel:` callback, differentiating from successful method selections. You should make sure to also dismiss the view controller in that callback. + +### Migrating from versions < 10.0 + +- Methods deprecated in Version 6.0 have now been removed. +- The `STPSource` protocol has been renamed `STPSourceProtocol`. +- `STPSource` is now a model object representing a source from the Stripe API. https://stripe.com/docs/sources +- `STPCustomer` will now include `STPSource` objects in its `sources` array if a customer has attached sources. +- `STPErrorCode` and `STPCardErrorCode` are now first class Swift enums (before, their types were `Int` and `String`, respectively) + +### Migrating from versions < 9.0 + +Version 9.0 drops support for iOS 7.x and Xcode 7.x. If you need to support iOS or Xcode versions below 8.0, the last compatible Stripe SDK release is version 8.0.7. + +### Migrating from versions < 6.0 + +6.0 moves most of the contents of `STPCard` into a new class, `STPCardParams`, which represents a request to the Stripe API. `STPCard` now only refers to responses from the Stripe API. Most apps should be able to simply replace all usage of `STPCard` with `STPCardParams` - you should only use `STPCard` if you're dealing with an API response, e.g. a card attached to an `STPToken`. This renaming has been done in a way that will avoid breaking changes, although using `STPCard`s to make requests to the Stripe API will produce deprecation warnings. + +### Migrating from versions < 5.0 + +5.0 deprecates our native Stripe Checkout adapters. If you were using these, we recommend building your own credit card form instead. If you need help with this, please contact support@stripe.com. + +### Migrating from versions < 3.0 + +Before version 3.0, most token-creation methods were class methods on the `Stripe` class. These are now all instance methods on the `STPAPIClient` class. Where previously you might write +```objective-c +[Stripe createTokenWithCard:card publishableKey:myPublishableKey completion:completion]; +``` +you would now instead write +```objective-c +STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:myPublishableKey]; +[client createTokenWithCard:card completion:completion]; +``` +This version also made several helper classes, including `STPAPIConnection` and `STPUtils`, private. You should remove any references to them from your code (most apps shouldn't have any). + +## Migrating from versions < 1.2 + +Versions of Stripe-iOS prior to 1.2 included a class called `STPView`, which provided a pre-built credit card form. This functionality has been moved from Stripe-iOS to PaymentKit, a separate project. If you were using `STPView` prior to version 1.2, migrating is simple: + +1. Add PaymentKit to your project, as explained on its [project page](https://github.com/stripe/PaymentKit). +2. Replace any references to `STPView` with a `PTKView` instead. Similarly, any classes that implement `STPViewDelegate` should now instead implement the equivalent `PTKViewDelegate` methods. Note that unlike `STPView`, `PTKView` does not take a Stripe API key in its constructor. +3. To submit the credit card details from your `PTKView` instance, where you would previously call `createToken` on your `STPView`, replace that with the following code (assuming `self.paymentView` is your `PTKView` instance): + +```objective-c +if (![self.paymentView isValid]) { + return; +} +STPCard *card = [[STPCard alloc] init]; +card.number = self.paymentView.card.number; +card.expMonth = self.paymentView.card.expMonth; +card.expYear = self.paymentView.card.expYear; +card.cvc = self.paymentView.card.cvc; +STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:publishableKey]; +[client createTokenWithCard:card completion:^(STPToken *token, NSError *error) { + if (error) { + // handle the error as you did previously + } else { + // submit the token to your payment backend as you did previously + } +}]; +``` + +## Misc. notes + +### Handling errors + +See [StripeError.h](https://github.com/stripe/stripe-ios/blob/master/Stripe/PublicHeaders/Stripe/StripeError.h) for a list of error codes that may be returned from the Stripe API. + +### Validating STPCards + +You have a few options for handling validation of credit card data on the client, depending on what your application does. Client-side validation of credit card data is not required since our API will correctly reject invalid card information, but can be useful to validate information as soon as a user enters it, or simply to save a network request. + +The simplest thing you can do is to populate an `STPCard` object and, before sending the request, call `- (BOOL)validateCardReturningError:` on the card. This validates the entire card object, but is not useful for validating card properties one at a time. + +To validate `STPCard` properties individually, you should use the following: + +```objective-c + - (BOOL)validateNumber:error: + - (BOOL)validateCvc:error: + - (BOOL)validateExpMonth:error: + - (BOOL)validateExpYear:error: +``` + +These methods follow the validation method convention used by [key-value validation](http://developer.apple.com/library/mac/#documentation/cocoa/conceptual/KeyValueCoding/Articles/Validation.html). So, you can use these methods by invoking them directly, or by calling `[card validateValue:forKey:error]` for a property on the `STPCard` object. + +When using these validation methods, you will want to set the property on your card object when a property does validate before validating the next property. This allows the methods to use existing properties on the card correctly to validate a new property. For example, validating `5` for the `expMonth` property will return YES if no `expYear` is set. But if `expYear` is set and you try to set `expMonth` to 5 and the combination of `expMonth` and `expYear` is in the past, `5` will not validate. The order in which you call the validate methods does not matter for this though. diff --git a/Package.swift b/Package.swift new file mode 100644 index 00000000..05c4e18c --- /dev/null +++ b/Package.swift @@ -0,0 +1,155 @@ +// swift-tools-version:5.5 +import PackageDescription + +let package = Package( + name: "Stripe", + defaultLocalization: "en", + platforms: [ + .iOS(.v13) + ], + products: [ + .library( + name: "Stripe", + targets: ["Stripe"] + ), + .library( + name: "StripePayments", + targets: ["StripePayments"] + ), + .library( + name: "StripePaymentsUI", + targets: ["StripePaymentsUI"] + ), + .library( + name: "StripePaymentSheet", + targets: ["StripePaymentSheet"] + ), + .library( + name: "StripeApplePay", + targets: ["StripeApplePay"] + ), + .library( + name: "StripeIdentity", + targets: ["StripeIdentity"] + ), + .library( + name: "StripeCardScan", + targets: ["StripeCardScan"] + ), + .library( + name: "StripeFinancialConnections", + targets: ["StripeFinancialConnections"] + ) + ], + targets: [ + .target( + name: "Stripe", + dependencies: ["Stripe3DS2", "StripeCore", "StripeApplePay", "StripeUICore", "StripePayments", "StripePaymentsUI"], + path: "Stripe/StripeiOS", + exclude: ["Info.plist"], + resources: [ + .process("Resources/Images") + ] + ), + .target( + name: "Stripe3DS2", + path: "Stripe3DS2/Stripe3DS2", + exclude: ["Info.plist", "Resources/CertificateFiles", "include/Stripe3DS2-Prefix.pch"], + resources: [ + .process("Resources") + ], + cSettings: [ + .headerSearchPath(".") + ] + ), + .target( + name: "StripeCameraCore", + dependencies: ["StripeCore"], + path: "StripeCameraCore/StripeCameraCore", + exclude: ["Info.plist"] + ), + .target( + name: "StripeCore", + path: "StripeCore/StripeCore", + exclude: ["Info.plist"] + ), + .target( + name: "StripeApplePay", + dependencies: ["StripeCore"], + path: "StripeApplePay/StripeApplePay", + exclude: ["Info.plist"] + ), + .target( + name: "StripeIdentity", + dependencies: ["StripeCore", "StripeUICore", "StripeCameraCore"], + path: "StripeIdentity/StripeIdentity", + exclude: ["Info.plist"], + resources: [ + .process("Resources/Images") + ] + ), + .target( + name: "StripeCardScan", + dependencies: ["StripeCore"], + path: "StripeCardScan/StripeCardScan", + exclude: ["Info.plist"], + resources: [ + .process("Resources/CompiledModels") + ] + ), + .target( + name: "StripeUICore", + dependencies: ["StripeCore"], + path: "StripeUICore/StripeUICore", + exclude: ["Info.plist"], + resources: [ + .process("Resources/Images"), + .process("Resources/JSON") + ] + ), + .target( + name: "StripePayments", + dependencies: ["StripeCore", "Stripe3DS2"], + path: "StripePayments/StripePayments", + exclude: ["Info.plist"], + resources: [ + .process("Resources") + ] + ), + .target( + name: "StripePaymentsUI", + dependencies: ["StripeCore", "Stripe3DS2", "StripePayments", "StripeUICore"], + path: "StripePaymentsUI/StripePaymentsUI", + exclude: ["Info.plist"], + resources: [ + .process("Resources/Images"), + .process("Resources/JSON") + ] + ), + .target( + name: "StripePaymentSheet", + dependencies: ["StripePaymentsUI", "StripeApplePay", "StripePayments", "StripeCore", "StripeUICore"], + path: "StripePaymentSheet/StripePaymentSheet", + exclude: ["Info.plist"], + resources: [ + .process("Resources/Images"), + .process("Resources/JSON") + ] + ), + .target( + name: "StripeFinancialConnections", + dependencies: ["StripeCore", "StripeUICore"], + path: "StripeFinancialConnections/StripeFinancialConnections", + exclude: ["Info.plist"], + resources: [ + .process("Resources/Images"), + ] + ), + .target( + name: "StripeLinkCore", + //dependencies: ["StripeCore"], + path: "StripeLinkCore/StripeLinkCore", + exclude: ["Info.plist"] + ) + ] +) diff --git a/README.md b/README.md index 4fea3738..a1ad313a 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,154 @@ -# stripe-ios-spm -This is a lightweight Swift Package Manager mirror for the [Stripe iOS SDK](https://github.com/stripe/stripe-ios). It offers [tagged source releases](https://github.com/stripe/stripe-ios-spm/releases) for each version of the SDK. +# Stripe iOS SDK -Please file issues on the [`stripe-ios` issues page](https://github.com/stripe/stripe-ios/issues). +[![CocoaPods](https://img.shields.io/cocoapods/v/Stripe.svg?style=flat)](http://cocoapods.org/?q=author%3Astripe%20name%3Astripe) +[![Carthage compatible](https://img.shields.io/badge/Carthage-compatible-4BC51D.svg?style=flat)](https://github.com/Carthage/Carthage) +[![Tuist badge](https://img.shields.io/badge/Powered%20by-Tuist-blue)](https://tuist.io) +[![License](https://img.shields.io/cocoapods/l/Stripe.svg?style=flat)](https://github.com/stripe/stripe-ios/blob/master/LICENSE) +[![Platform](https://img.shields.io/cocoapods/p/Stripe.svg?style=flat)](https://github.com/stripe/stripe-ios#) + +The Stripe iOS SDK makes it quick and easy to build an excellent payment experience in your iOS app. We provide powerful and customizable UI screens and elements that can be used out-of-the-box to collect your users' payment details. We also expose the low-level APIs that power those UIs so that you can build fully custom experiences. + +Get started with our [📚 integration guides](https://stripe.com/docs/payments/accept-a-payment?platform=ios) and [example projects](#examples), or [📘 browse the SDK reference](https://stripe.dev/stripe-ios/docs/index.html). + +Learn about our [Stripe Identity iOS SDK](StripeIdentity/README.md) to verify the identity of your users on iOS. + +> Updating to a newer version of the SDK? See our [migration guide](https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md) and [changelog](https://github.com/stripe/stripe-ios/blob/master/CHANGELOG.md). + +Table of contents +================= + + + * [Features](#features) + * [Releases](#releases) + * [Requirements](#requirements) + * [Getting started](#getting-started) + * [Integration](#integration) + * [Examples](#examples) + * [Building from source](#building-from-source) + * [Card scanning](#card-scanning) + * [Contributing](#contributing) + * [Migrating](#migrating-from-older-versions) + * [Licenses](#licenses) + + + +## Features + +**Simplified security**: We make it simple for you to collect sensitive data such as credit card numbers and remain [PCI compliant](https://stripe.com/docs/security#pci-dss-guidelines). This means the sensitive data is sent directly to Stripe instead of passing through your server. For more information, see our [integration security guide](https://stripe.com/docs/security). + +**Apple Pay**: [StripeApplePay](StripeApplePay/README.md) provides a [seamless integration with Apple Pay](https://stripe.com/docs/apple-pay). + +**SCA-ready**: The SDK automatically performs native [3D Secure authentication](https://stripe.com/docs/payments/3d-secure) if needed to comply with [Strong Customer Authentication](https://stripe.com/docs/strong-customer-authentication) regulation in Europe. + +**Native UI**: We provide native screens and elements to collect payment details. For example, [PaymentSheet](https://stripe.com/docs/payments/accept-a-payment?platform=ios) is a prebuilt UI that combines all the steps required to pay - collecting payment details, billing details, and confirming the payment - into a single sheet that displays on top of your app. + +PaymentSheet + +**Stripe API**: [StripePayments](StripePayments/README.md) provides [low-level APIs](https://stripe.dev/stripe-ios/docs/Classes/STPAPIClient.html) that correspond to objects and methods in the Stripe API. You can build your own entirely custom UI on top of this layer, while still taking advantage of utilities like [STPCardValidator](https://stripe.dev/stripe-ios/docs/Classes/STPCardValidator.html) to validate your user’s input. + +**Card scanning**: We support card scanning on iOS 13 and higher. See our [Card scanning](#card-scanning) section. + +**App Clips**: The `StripeApplePay` module provides a [lightweight SDK for offering Apple Pay in an App Clip](https://stripe.com/docs/apple-pay#app-clips). + +**Localized**: We support the following localizations: Bulgarian, Catalan, Chinese (Hong Kong), Chinese (Simplified), Chinese (Traditional), Croatian, Czech, Danish, Dutch, English (US), English (United Kingdom), Estonian, Filipino, Finnish, French, French (Canada), German, Greek, Hungarian, Indonesian, Italian, Japanese, Korean, Latvian, Lithuanian, Malay, Maltese, Norwegian Bokmål, Norwegian Nynorsk (Norway), Polish, Portuguese, Portuguese (Brazil), Romanian, Russian, Slovak, Slovenian, Spanish, Spanish (Latin America), Swedish, Turkish, Thai and Vietnamese. + +#### Recommended usage + +If you're selling digital products or services that will be consumed within your app, (e.g. subscriptions, in-game currencies, game levels, access to premium content, or unlocking a full version), you must use Apple's in-app purchase APIs. See the [App Store review guidelines](https://developer.apple.com/app-store/review/guidelines/#payments) for more information. For all other scenarios you can use this SDK to process payments via Stripe. + +#### Privacy + +The Stripe iOS SDK collects data to help us improve our products and prevent fraud. This data is never used for advertising and is not rented, sold, or given to advertisers. Our full privacy policy is available at [https://stripe.com/privacy](https://stripe.com/privacy). + +For help with Apple's App Privacy Details form in App Store Connect, visit [Stripe iOS SDK Privacy Details](https://support.stripe.com/questions/stripe-ios-sdk-privacy-details). + +## Modules +|Module|Description|Compressed|Uncompressed| +|------|-----------|----------|------------| +|StripePaymentSheet|Stripe's [prebuilt payment UI](https://stripe.com/docs/payments/accept-a-payment?platform=ios&ui=payment-sheet).|2.7MB|6.3MB| +|Stripe|Contains all the below frameworks, plus [Issuing](https://stripe.com/docs/issuing/cards/digital-wallets?platform=iOS) and [Basic Integration](/docs/mobile/ios/basic).|2.3MB|5.1MB| +|StripeApplePay|[Apple Pay support](/docs/apple-pay), including `STPApplePayContext`.|0.4MB|1.0MB| +|StripePayments|Bindings for the Stripe Payments API.|1.0MB|2.6MB| +|StripePaymentsUI|Bindings for the Stripe Payments API, [STPPaymentCardTextField](https://stripe.com/docs/payments/accept-a-payment?platform=ios&ui=custom), STPCardFormView, and other UI elements.|1.7MB|3.9MB| + +## Releases + +We support Cocoapods, Carthage, and Swift Package Manager. + +If you link the library manually, use a version from our [releases](https://github.com/stripe/stripe-ios/releases) page and make sure to embed all of the required frameworks. + +For the `Stripe` module, link the following frameworks: +- `Stripe.xcframework` +- `Stripe3DS2.xcframework` +- `StripeApplePay.xcframework` +- `StripePayments.xcframework` +- `StripePaymentsUI.xcframework` +- `StripeCore.xcframework` +- `StripeUICore.xcframework` + +For other modules, follow the instructions below: +- [StripePaymentSheet](StripePaymentSheet/README.md#manual-linking) +- [StripePayments](StripePayments/README.md#manual-linking) +- [StripePaymentsUI](StripePaymentsUI/README.md#manual-linking) +- [StripeApplePay](StripeApplePay/README.md#manual-linking) +- [StripeIdentity](StripeIdentity/README.md#manual-linking) + +If you're reading this on GitHub.com, please make sure you are looking at the [tagged version](https://github.com/stripe/stripe-ios/tags) that corresponds to the release you have installed. Otherwise, the instructions and example code may be mismatched with your copy. + +## Requirements + +The Stripe iOS SDK requires Xcode 14.1 or later and is compatible with apps targeting iOS 13 or above. We support Catalyst on macOS 10.16 or later. + +For iOS 12 support, please use [v22.8.4](https://github.com/stripe/stripe-ios/tree/v22.8.4). For iOS 11 support, please use [v21.13.0](https://github.com/stripe/stripe-ios/tree/v21.13.0). For iOS 10, please use [v19.4.0](https://github.com/stripe/stripe-ios/tree/v19.4.0). If you need to support iOS 9, use [v17.0.2](https://github.com/stripe/stripe-ios/tree/v17.0.2). + +## Getting started + +### Integration + +Get started with our [📚 integration guides](https://stripe.com/docs/payments/accept-a-payment?platform=ios) and [example projects](/Example), or [📘 browse the SDK reference](https://stripe.dev/stripe-ios/docs/index.html) for fine-grained documentation of all the classes and methods in the SDK. + +### Examples + +- [Prebuilt UI](Example/PaymentSheet%20Example) (Recommended) + - This example demonstrates how to build a payment flow using [`PaymentSheet`](https://stripe.com/docs/payments/accept-a-payment?platform=ios), an embeddable native UI component that lets you accept [10+ payment methods](https://stripe.com/docs/payments/payment-methods/integration-options#payment-method-product-support) with a single integration. + +- [Non-Card Payment Examples](Example/Non-Card%20Payment%20Examples) + - This example demonstrates how to manually accept various payment methods using the Stripe API. + +### Building from source + +We use [Tuist](https://tuist.io) to generate Xcode projects, and all Xcode related files have been removed from the master branch of the repository. Note that project files are still available on tagged releases. + +If you want to build from the master branch you need to follow these steps: + +- Clone the repository and `cd` into its directory. +- Install Tuist by running `curl -Ls https://install.tuist.io | bash` +- Run `tuist generate`, optionally pass the `-n` option if you don't want to open Xcode automatically. + +You can build any of the generated targets as you normally would. + +For more information about Tuist, visit https://tuist.io. + +## Card scanning + +[PaymentSheet](https://stripe.com/docs/payments/accept-a-payment?platform=ios) offers built-in card scanning. To enable card scanning, you'll need to set `NSCameraUsageDescription` in your application's plist, and provide a reason for accessing the camera (e.g. "To scan cards"). Card scanning is supported on devices with iOS 13 or higher. + +You can demo this feature in our [PaymentSheet example app](Example/PaymentSheet%20Example). When you run the example app on a device, you'll see a "Scan Card" button when adding a new card. + +## Contributing + +We welcome contributions of any kind including new features, bug fixes, and documentation improvements. Please first open an issue describing what you want to build if it is a major change so that we can discuss how to move forward. Otherwise, go ahead and open a pull request for minor changes such as typo fixes and one liners. + +### Running tests + +1. Install Carthage 0.37 or later (if you have homebrew installed, `brew install carthage`) +2. From the root of the repo, run `bundle install && bundle exec fastlane stripeios_tests`. This will install the test dependencies and run the tests. +3. Once you have run this once, you can also run the tests in Xcode from the `StripeiOS` target in `Stripe.xcworkspace`. Make sure to use the iPhone 12 mini, iOS 16.1 simulator so the snapshot tests will pass. + +## Migrating from older versions + +See [MIGRATING.md](https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md) + +## Licenses + +- [Stripe iOS SDK License](LICENSE) diff --git a/Stripe/BuildConfigurations/Stripe Tests-Debug.xcconfig b/Stripe/BuildConfigurations/Stripe Tests-Debug.xcconfig new file mode 100644 index 00000000..6d153e27 --- /dev/null +++ b/Stripe/BuildConfigurations/Stripe Tests-Debug.xcconfig @@ -0,0 +1,8 @@ +// +// Stripe Tests-Debug.xcconfig +// + +#include "../../BuildConfigurations/StripeiOS Tests-Debug.xcconfig" + +SWIFT_OBJC_BRIDGING_HEADER = StripeiOSTests/StripeiOS Tests-Bridging-Header.h +GCC_PREFIX_HEADER = StripeiOSTests/StripeTests-Prefix.pch diff --git a/Stripe/BuildConfigurations/Stripe Tests-Release.xcconfig b/Stripe/BuildConfigurations/Stripe Tests-Release.xcconfig new file mode 100644 index 00000000..b72a89b2 --- /dev/null +++ b/Stripe/BuildConfigurations/Stripe Tests-Release.xcconfig @@ -0,0 +1,8 @@ +// +// Stripe Tests-Release.xcconfig +// + +#include "../../BuildConfigurations/StripeiOS Tests-Release.xcconfig" + +SWIFT_OBJC_BRIDGING_HEADER = StripeiOSTests/StripeiOS Tests-Bridging-Header.h +GCC_PREFIX_HEADER = StripeiOSTests/StripeTests-Prefix.pch diff --git a/Stripe/BuildConfigurations/Stripe-Debug.xcconfig b/Stripe/BuildConfigurations/Stripe-Debug.xcconfig new file mode 100644 index 00000000..c5831ee6 --- /dev/null +++ b/Stripe/BuildConfigurations/Stripe-Debug.xcconfig @@ -0,0 +1,7 @@ +// +// Stripe-Debug.xcconfig +// + +#include "../../BuildConfigurations/StripeiOS-Debug.xcconfig" + +MODULEMAP_FILE = StripeiOS/Stripe.modulemap diff --git a/Stripe/BuildConfigurations/Stripe-Release.xcconfig b/Stripe/BuildConfigurations/Stripe-Release.xcconfig new file mode 100644 index 00000000..662db163 --- /dev/null +++ b/Stripe/BuildConfigurations/Stripe-Release.xcconfig @@ -0,0 +1,7 @@ +// +// Stripe-Release.xcconfig +// + +#include "../../BuildConfigurations/StripeiOS-Release.xcconfig" + +MODULEMAP_FILE = StripeiOS/Stripe.modulemap diff --git a/Stripe/Project.swift b/Stripe/Project.swift new file mode 100644 index 00000000..10203bb6 --- /dev/null +++ b/Stripe/Project.swift @@ -0,0 +1,145 @@ +import ProjectDescription +import ProjectDescriptionHelpers + +let project = Project( + name: "Stripe", + options: .options( + automaticSchemesOptions: .disabled, + disableBundleAccessors: true, + disableSynthesizedResourceAccessors: true + ), + packages: [ + .remote( + url: "https://github.com/uber/ios-snapshot-test-case", + requirement: .upToNextMajor(from: "8.0.0") + ), + .remote( + url: "https://github.com/eurias-stripe/OHHTTPStubs", + requirement: .branch("master") + ), + .remote(url: "https://github.com/erikdoe/ocmock", requirement: .branch("master")), + ], + settings: .settings( + configurations: [ + .debug( + name: "Debug", + xcconfig: "//BuildConfigurations/Project-Debug.xcconfig" + ), + .release( + name: "Release", + xcconfig: "//BuildConfigurations/Project-Release.xcconfig" + ), + ], + defaultSettings: .none + ), + targets: [ + Target( + name: "StripeiOS", + platform: .iOS, + product: .framework, + productName: "Stripe", + bundleId: "com.stripe.stripe-ios", + infoPlist: "StripeiOS/Info.plist", + sources: [ + "StripeiOS/Source/**/*.swift", + "StripeiOS/*.docc", + ], + resources: "StripeiOS/Resources/**", + headers: .headers( + public: "StripeiOS/Stripe-umbrella.h" + ), + dependencies: [ + .project(target: "Stripe3DS2", path: "//Stripe3DS2"), + .project(target: "StripeCore", path: "//StripeCore"), + .project(target: "StripeUICore", path: "//StripeUICore"), + .project(target: "StripeApplePay", path: "//StripeApplePay"), + .project(target: "StripePayments", path: "//StripePayments"), + .project(target: "StripePaymentsUI", path: "//StripePaymentsUI"), + ], + settings: .stripeTargetSettings( + baseXcconfigFilePath: "BuildConfigurations/Stripe" + ) + ), + Target( + name: "StripeiOSTests", + platform: .iOS, + product: .unitTests, + productName: "StripeiOS_Tests", + bundleId: "com.stripe.StripeiOSTests", + infoPlist: "StripeiOSTests/Info.plist", + sources: [ + "StripeiOSTests/**/*.swift", + "StripeiOSTests/**/*.m", + ], + resources: .init(resources: [ + "StripeiOSTests/Resources/*.*", + "StripeiOSTests/Resources/Images.xcassets", + .folderReference(path: "StripeiOSTests/Resources/recorded_network_traffic"), + .folderReference(path: "StripeiOSTests/Resources/MockFiles"), + ]), + headers: .headers( + project: "StripeiOSTests/*.h" + ), + dependencies: [ + .xctest, + .target(name: "StripeiOS"), + .package(product: "OHHTTPStubs"), + .package(product: "OHHTTPStubsSwift"), + .package(product: "OCMock"), + .package(product: "iOSSnapshotTestCase"), + .project(target: "StripeCoreTestUtils", path: "//StripeCore"), + .project(target: "StripePayments", path: "//StripePayments"), + .project(target: "StripePaymentsUI", path: "//StripePaymentsUI"), + .project(target: "StripePaymentSheet", path: "//StripePaymentSheet"), + ], + settings: .stripeTargetSettings( + baseXcconfigFilePath: "BuildConfigurations/Stripe Tests" + ), + additionalFiles: [ + "StripeiOSTests.xctestplan" + ] + ), + Target( + name: "StripeiOSTestHostApp", + platform: .iOS, + product: .app, + bundleId: "com.stripe.StripeiOSTestHostApp", + infoPlist: "StripeiOSTestHostApp/Info.plist", + sources: "StripeiOSTestHostApp/*.swift", + resources: "StripeiOSTestHostApp/Resources/**", + settings: .stripeTargetSettings( + baseXcconfigFilePath: "//BuildConfigurations/StripeiOS Tests" + ) + ), + Target( + name: "StripeiOSAppHostedTests", + platform: .iOS, + product: .unitTests, + bundleId: "com.stripe.StripeiOSAppHostedTests", + infoPlist: "StripeiOSAppHostedTests/Info.plist", + sources: "StripeiOSAppHostedTests/*.swift", + dependencies: [ + .xctest, + .target(name: "StripeiOS"), + .target(name: "StripeiOSTestHostApp"), + .project(target: "StripePaymentSheet", path: "//StripePaymentSheet"), + ], + settings: .stripeTargetSettings( + baseXcconfigFilePath: "//BuildConfigurations/StripeiOS Tests" + ) + ), + ], + schemes: [ + Scheme( + name: "StripeiOS", + buildAction: .buildAction(targets: ["StripeiOS"]), + testAction: .testPlans(["StripeiOSTests.xctestplan"]) + ), + Scheme( + name: "StripeiOSTestHostApp", + buildAction: .buildAction(targets: ["StripeiOS"]), + testAction: .targets(["StripeiOSAppHostedTests"]), + runAction: .runAction(executable: "StripeiOSTestHostApp") + ), + ] +) diff --git a/Stripe/Stripe.xcodeproj/project.pbxproj b/Stripe/Stripe.xcodeproj/project.pbxproj new file mode 100644 index 00000000..bb80496d --- /dev/null +++ b/Stripe/Stripe.xcodeproj/project.pbxproj @@ -0,0 +1,2581 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 52; + objects = { + +/* Begin PBXBuildFile section */ + 013F991AB34E38BDBA6E4521 /* STPFormViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F44327A2B2C9483F52EE343B /* STPFormViewSnapshotTests.swift */; }; + 0185AC6B123CD73E877D4FCE /* STPPaymentMethodCashAppParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0ABB2CA7E96BE249CE8C0566 /* STPPaymentMethodCashAppParamsTests.swift */; }; + 01F2C8715D740FDE683FAECF /* stp_card_form_amex_cvc@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 0227131370F51B0533585BEB /* stp_card_form_amex_cvc@3x.png */; }; + 03A2D8DD31042AC83AD2EFC5 /* AlipaySource.json in Resources */ = {isa = PBXBuildFile; fileRef = A32EE0B1FC16687797E7C9DA /* AlipaySource.json */; }; + 03E60F9EF24C975AF90E2447 /* StripePaymentsUI.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = E136A967522048B313E3C62F /* StripePaymentsUI.framework */; }; + 044B7BECFBDB1F6C8CA08514 /* STPSetupIntentConfirmParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6CC0B1FC92A573AAEA4F4E94 /* STPSetupIntentConfirmParamsTest.swift */; }; + 06FB8C17B3AA4957FD0F3CD2 /* STPPaymentMethodFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 45BBD120FDBA434BC7E64435 /* STPPaymentMethodFunctionalTest.m */; }; + 07A5CDBFDF2340BAD99D6EB3 /* STPCardFormViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1D983E089196152DA1C69469 /* STPCardFormViewSnapshotTests.swift */; }; + 07BF3CF1656AF5F5A0678873 /* STPPhoneNumberValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 924E878428D15506711CA628 /* STPPhoneNumberValidatorTest.swift */; }; + 08111F4AD3CA0755420E05F7 /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 147D2DC1FFDFC99269039377 /* LaunchScreen.storyboard */; }; + 088B79E49A52A57B74B23F6E /* STPPaymentMethodCardWalletVisaCheckoutTest.m in Sources */ = {isa = PBXBuildFile; fileRef = B5BA1ABFF1739E4051FAE6CB /* STPPaymentMethodCardWalletVisaCheckoutTest.m */; }; + 08ED7A4EB7E64FDAED2C2D39 /* STPShippingAddressViewControllerTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 789D0B49B0788794739E3DD4 /* STPShippingAddressViewControllerTest.swift */; }; + 093FE3D65978E3DB6B79AE05 /* UIToolbar+Stripe_InputAccessory.swift in Sources */ = {isa = PBXBuildFile; fileRef = E2307F2C5E53540D4ACAA1F6 /* UIToolbar+Stripe_InputAccessory.swift */; }; + 0A181790DAF17BD039F01B15 /* STPPaymentMethodAfterpayClearpayParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = CF8E26B31C0F2B256763BDDB /* STPPaymentMethodAfterpayClearpayParamsTest.m */; }; + 0B9C0E9A7A750607413C9E53 /* STPFakeAddPaymentPassViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0669B4CA326CE74D125C789C /* STPFakeAddPaymentPassViewController.swift */; }; + 0CBBE909CA773D7D45B9AD4C /* STPAnalyticsClientPaymentSheetTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0241DB84973B21393BEC703E /* STPAnalyticsClientPaymentSheetTest.swift */; }; + 0D4211C0767F66A35914FE07 /* 3DSSource.json in Resources */ = {isa = PBXBuildFile; fileRef = A8F5797C145751AD55858B80 /* 3DSSource.json */; }; + 0DFA17378D894C70D72C9F62 /* Error+PaymentSheetTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8CE85C770AEEDBE4AEC93EAA /* Error+PaymentSheetTests.swift */; }; + 0F0F35439565AA0D284A6A70 /* STPElementsSessionTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 63BDA70DB74745C5F457CF88 /* STPElementsSessionTest.swift */; }; + 0FA3C1494BA57884B5DE3B20 /* Stripe.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4259421D2CD26E37B96F97B2 /* Stripe.framework */; }; + 10342D659764A88A695EF38B /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = DFA7A75BA785EBBE4C05DAA3 /* Images.xcassets */; }; + 124D43C1A633922B1DA3E1E7 /* STPShippingAddressViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 71711FC8E2FB66E52A5FDD9A /* STPShippingAddressViewController.swift */; }; + 15772D8E06236054D11A1034 /* P24Source.json in Resources */ = {isa = PBXBuildFile; fileRef = A9B50EE8ABC2506A6397C056 /* P24Source.json */; }; + 162C101E57D66F0051164C4A /* Stripe.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 4259421D2CD26E37B96F97B2 /* Stripe.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + 172D96526023A80534D54CC0 /* STPBankSelectionViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 81352A0CBE46A59E6B1A712E /* STPBankSelectionViewController.swift */; }; + 17BD7C0391F3182E32A63D6B /* STPAUBECSDebitFormViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = BCBEA9E4823F08C1F5057B5A /* STPAUBECSDebitFormViewSnapshotTests.swift */; }; + 181B8C34A3FFCF2DBEF0086E /* PaymentSheetFormFactoryTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = EEEC1EA82490933B169915C4 /* PaymentSheetFormFactoryTest.swift */; }; + 187966C4EDE4FEE1FC6B9F24 /* stp_bank_fpx_hong_leong_bank@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 7F90C21B77335533395C4321 /* stp_bank_fpx_hong_leong_bank@3x.png */; }; + 194154708E1A9E013DCE2C72 /* STPPaymentHandlerStubbedMockedFilesTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = EF48EC440E1ED5D6BAA567FF /* STPPaymentHandlerStubbedMockedFilesTests.swift */; }; + 1A058C42C4703458CA1CA522 /* STPCardNumberInputTextFieldValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = D7C4773C2D193BEDF1CBB530 /* STPCardNumberInputTextFieldValidatorTests.swift */; }; + 1AB740655742E5D00E905A4B /* iDEALSource.json in Resources */ = {isa = PBXBuildFile; fileRef = F24B1E06C600CA30A97F5AEA /* iDEALSource.json */; }; + 1BC4044802EE7D3E2643DC84 /* STPPaymentIntentEnumsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 46C31CAD0FA74B58BA2B8530 /* STPPaymentIntentEnumsTest.swift */; }; + 1CCFC43F7FCD273E2100D321 /* STPPaymentMethodBancontactTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F4E5416F6AE8BED88980D6F8 /* STPPaymentMethodBancontactTests.swift */; }; + 1E8D8E2494062262A332879C /* STPCardValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E1644AA33E81233EF33022BA /* STPCardValidatorTest.swift */; }; + 1F417D0874CC86F4C9AB2790 /* STPIntentWithPreferencesTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6C8FE751333F580004BD72BA /* STPIntentWithPreferencesTest.swift */; }; + 1F432D0B37949217E4299A20 /* STPPaymentOptionsInternalViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = D5C4A4CC7D2E9B5AB3EC3B79 /* STPPaymentOptionsInternalViewController.swift */; }; + 2197EAE8DE995628312490BE /* WeChatPaySource.json in Resources */ = {isa = PBXBuildFile; fileRef = 0A726D87E8C916E8FEA0C780 /* WeChatPaySource.json */; }; + 2199D89054CACD1658CDC2F1 /* STPPaymentMethodNetBankingParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = E4442901582C822007C3BB67 /* STPPaymentMethodNetBankingParamsTest.m */; }; + 222F49985A1817D8D41D8B56 /* STPCardParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 20A74D4FA51D15AF12B58767 /* STPCardParamsTest.m */; }; + 225140E0BD9C0630116DDE4A /* STPPaymentMethodUSBankAccountTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3B112FFF3FCA82094281493F /* STPPaymentMethodUSBankAccountTest.swift */; }; + 229E25F8DFCC55CA9EDD15AB /* LinkInMemoryCookieStoreTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = CEB30E3846E24FD57A44764A /* LinkInMemoryCookieStoreTests.swift */; }; + 22BE2ABB29F77362FF16D945 /* Localizable.strings in Resources */ = {isa = PBXBuildFile; fileRef = C2427C1CDFA85BFC6570F1E9 /* Localizable.strings */; }; + 234C71F480318E9062075924 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9BBCE3A905041A709E8F279A /* AppDelegate.swift */; }; + 23D1246A5DAB5333650F104F /* STPSectionHeaderView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 96E1FED5CE5974C9C1162E93 /* STPSectionHeaderView.swift */; }; + 246920234EE8382FB4E56516 /* STPCardFormViewTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF5E08A1651D9DFE502DA021 /* STPCardFormViewTests.swift */; }; + 258E3F6C9DBB2B510CCEC525 /* STPPaymentContextSnapshotTests.m in Sources */ = {isa = PBXBuildFile; fileRef = D106EFD1BF019522BE7BEBBE /* STPPaymentContextSnapshotTests.m */; }; + 25E98D9074E582E91A10F5FF /* STPPaymentMethodPayPalParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = FE1929CD1B3D3DCC2401E2F8 /* STPPaymentMethodPayPalParamsTests.m */; }; + 26D73AA93A48A1112FECD86D /* AddPaymentMethodViewControllerSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5430F6704CFF793CBFAAF549 /* AddPaymentMethodViewControllerSnapshotTests.swift */; }; + 279D2BA91198E18730626CE6 /* STPUserInformation.swift in Sources */ = {isa = PBXBuildFile; fileRef = B1217AD643A9E8F88B60F645 /* STPUserInformation.swift */; }; + 27F1783CBFEC06BFD6C114F6 /* STPPaymentMethodKlarnaTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4FD94FF270165D699DA89B24 /* STPPaymentMethodKlarnaTests.swift */; }; + 284BCD4B0744CDD91F7D8B15 /* STPPaymentMethodSofortParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 3871630C99B378C67B6E9F7C /* STPPaymentMethodSofortParamsTests.m */; }; + 29428CDB658E6F504402D844 /* STPPaymentMethodBillingDetailsTests+Link.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7AC0A18C441FCA394BEF6A3D /* STPPaymentMethodBillingDetailsTests+Link.swift */; }; + 2990B7B1513ACA58065A1D0B /* STPApplePayPaymentOptionTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 0320B47C1DD836F7A7144B58 /* STPApplePayPaymentOptionTest.m */; }; + 2A528B7B2579E5F977797822 /* STPPaymentHandlerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FC30E6129279F14506219E98 /* STPPaymentHandlerTests.swift */; }; + 2AC91F23CF3949ADC60D27F7 /* STPThreeDSTextFieldCustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = A583966A33DCDCF04322A592 /* STPThreeDSTextFieldCustomizationTest.swift */; }; + 2AE9ABA774B430E174279FEA /* stp_test_upload_image.jpeg in Resources */ = {isa = PBXBuildFile; fileRef = FD3398E2352CEA0264F20AEA /* stp_test_upload_image.jpeg */; }; + 2BA38A013A538479C3518424 /* stp_fpx_logo@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = B2C7ED5A39AB3BB2248CEACA /* stp_fpx_logo@3x.png */; }; + 2BD45625F6F665B60C6CAD30 /* STPAddressViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 28A50AFA4603E488FF3D82D0 /* STPAddressViewModel.swift */; }; + 2C7991FDF7B374E0E65E253F /* STPPaymentOptionsViewControllerTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = A2922A32A754CFC9AB8B48AE /* STPPaymentOptionsViewControllerTest.swift */; }; + 2CD7968DA48F7129E16EA0CB /* OHHTTPStubsSwift in Frameworks */ = {isa = PBXBuildFile; productRef = 911CA85A1610303FA0AF0643 /* OHHTTPStubsSwift */; }; + 2CE1FB3A85DE7A74D0A80901 /* CardSource.json in Resources */ = {isa = PBXBuildFile; fileRef = 582C9C6F076A0FF1780C9769 /* CardSource.json */; }; + 2E35B0FB60FCBE7608080642 /* STPPushProvisioningContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6887F19BB9804BF45FD703FF /* STPPushProvisioningContext.swift */; }; + 2EB68A59660A4D1E14799DA4 /* OHHTTPStubs in Frameworks */ = {isa = PBXBuildFile; productRef = 62887B4538E4E41E735685E1 /* OHHTTPStubs */; }; + 2F0FC4E67BE577AD66CD1475 /* StripePaymentSheet.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = DDC55CC034022DFAC9366E2E /* StripePaymentSheet.framework */; }; + 2F18A1903244E144C7802E09 /* STPCardValidationState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6A9E7B637A8747431B38FD1D /* STPCardValidationState.swift */; }; + 2F2923C176FFD66CC5F96A2D /* stp_bank_fpx_rhb@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 152E45642578F76AAEB1CF61 /* stp_bank_fpx_rhb@3x.png */; }; + 2F6814A67EA18BFD306A9E5D /* STPUIVCStripeParentViewControllerTests.m in Sources */ = {isa = PBXBuildFile; fileRef = F22EE02CE12F3EBFACDAE9A4 /* STPUIVCStripeParentViewControllerTests.m */; }; + 2F866A52BA39ED5D32BCA9DD /* LinkPaymentMethodPickerSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3024CAAF54280B0357014D39 /* LinkPaymentMethodPickerSnapshotTests.swift */; }; + 2F9FA9CBCA3C0CE52FAC9B6B /* ConfirmButtonSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 806124200E77795DCFC8418E /* ConfirmButtonSnapshotTests.swift */; }; + 2FFA7C2D1C7337FDB4C608A5 /* STPCardScannerTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0B91C4D5B93FF71C61B140F1 /* STPCardScannerTableViewCell.swift */; }; + 307FD6A103EF7AF3CE451598 /* STPPaymentResult.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7D964D9E01627B419B4BD23C /* STPPaymentResult.swift */; }; + 30D48C62B2FA6B28EC23A5BB /* Stripe-umbrella.h in Headers */ = {isa = PBXBuildFile; fileRef = A8598727045C6268B57A5FC7 /* Stripe-umbrella.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 314B144D765DB6CD8ABE8B7D /* SetupIntent.json in Resources */ = {isa = PBXBuildFile; fileRef = 0364E74C0FB269E93D18966D /* SetupIntent.json */; }; + 315713352C770DA3ED9CBDCD /* Enums+CustomStringConvertible.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8F7E3CE2105E4A39032CD919 /* Enums+CustomStringConvertible.swift */; }; + 317275D3FB6DC708BD905040 /* NSLocale+STPSwizzling.m in Sources */ = {isa = PBXBuildFile; fileRef = 86D961D8952114BE479E1EB7 /* NSLocale+STPSwizzling.m */; }; + 3172C789DF2CE133ECA359D7 /* STPPushProvisioningDetailsFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1A8A6B88797870BC71CCB3AF /* STPPushProvisioningDetailsFunctionalTest.swift */; }; + 31B49323CC357B478431B716 /* stp_fpx_big_logo@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = CEAF8A072B0BF86AD02655B0 /* stp_fpx_big_logo@3x.png */; }; + 32874C6147344A9CB2EF4DAD /* Stripe3DS2.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 33FDC634FD5D79E824240DDC /* Stripe3DS2.framework */; }; + 331924F0801287BAD413FDCB /* STPMandateCustomerAcceptanceParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C713F58BC61A962C720AE0AE /* STPMandateCustomerAcceptanceParamsTest.swift */; }; + 33742D8595DCB200A2507B0A /* STPSourceFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 20D90EE8CC6606CAE9B06C63 /* STPSourceFunctionalTest.m */; }; + 34F3ABB5BF9D3645D3B6DA80 /* AddressViewControllerSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = A9F06DC32147A8ABCD709E79 /* AddressViewControllerSnapshotTests.swift */; }; + 35048D07C8323A96161F63C5 /* STPPaymentMethodCardWalletMasterpassTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 1443F134C35E15AB9A1EA560 /* STPPaymentMethodCardWalletMasterpassTest.m */; }; + 3573889F65E85478DE770A49 /* STPPaymentMethodOXXOTests.m in Sources */ = {isa = PBXBuildFile; fileRef = EAD0042713B4C2FEB8916EDC /* STPPaymentMethodOXXOTests.m */; }; + 35C1CF73701EECC7DB6AB722 /* FormSpecProviderTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7C04AFC9CDE50D09D38A3232 /* FormSpecProviderTest.swift */; }; + 360EEE8B706D2A4A49666F7A /* StripePayments.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 22D1C6EB5826E2D7C80B6CF3 /* StripePayments.framework */; }; + 369C7A7A0DA46C58B02DBA00 /* stp_bank_fpx_hsbc@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 05A3E36498CBA9E5FB8323C9 /* stp_bank_fpx_hsbc@3x.png */; }; + 37E9160706C9EEEFEF133617 /* STPPaymentIntentFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = CDDEAB86BE4711841D426F3B /* STPPaymentIntentFunctionalTest.swift */; }; + 385CAC4D2FF119D2E925916B /* STPPostalCodeInputTextFieldFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 917154477796779ECFA1334A /* STPPostalCodeInputTextFieldFormatterTests.swift */; }; + 3930ECBEE003772C1245D25B /* UIViewController+Stripe_ParentViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5804C2B9C0704E386B3D25A4 /* UIViewController+Stripe_ParentViewController.swift */; }; + 395E5AE58C4B5B4EE511B182 /* Customer.json in Resources */ = {isa = PBXBuildFile; fileRef = 4928F057B093B6DB5D76D32D /* Customer.json */; }; + 3AAE488F2461A46143B3A687 /* STPPaymentConfigurationTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 79ABE6A14AF9D14103050876 /* STPPaymentConfigurationTest.m */; }; + 3AD22E0BD44B02D968C6569A /* STPImageLibraryTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = F546088BA4F763334CFD3D34 /* STPImageLibraryTest.swift */; }; + 3B237145902E3DB07E747E32 /* TextFieldElement+IBANTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 51BD2CE41E4F0CF648F44E4A /* TextFieldElement+IBANTest.swift */; }; + 3B6EF1B8C8C6A37D220AF282 /* STPPaymentMethodFPXTest.m in Sources */ = {isa = PBXBuildFile; fileRef = BF8A27B4CE855392E3E3F735 /* STPPaymentMethodFPXTest.m */; }; + 3C1A7B9810B038177FF1CF52 /* ViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1F29C15B47C7CB0941CD4C9E /* ViewController.swift */; }; + 3C1E9069CD03ED9981D7F3E2 /* ConfirmButtonTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = CFFE40AD9D875709F643D2E5 /* ConfirmButtonTests.swift */; }; + 3CE88568CB9648D6F1503B88 /* STPSource+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = C3B75875C55D2C2723DC5090 /* STPSource+BasicUI.swift */; }; + 3D0C7EA60D59DFC66F0E02FE /* STPRedirectContextTest.m in Sources */ = {isa = PBXBuildFile; fileRef = E3810757AF83E98307634814 /* STPRedirectContextTest.m */; }; + 3DC523AEEA01D68424C8B41B /* STPStringUtilsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B527265D186B66150D2787E /* STPStringUtilsTest.m */; }; + 3DE8D85607C0F78C67E35DF4 /* STPSetupIntentTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 3BCC4BB345F320EEECE0437C /* STPSetupIntentTest.m */; }; + 3EB3745F556EA12AB27A8545 /* APIRequestTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 85AAB72218409F85FE29E69E /* APIRequestTest.swift */; }; + 3EFC86756F23D8AE6708ECE2 /* ButtonLinkSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 86673B13C7762666C774F124 /* ButtonLinkSnapshotTests.swift */; }; + 3F4356512B60B90BFCDA7E01 /* stp_bank_fpx_bank_islam@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 87DB9F06FDC13DD3BFEAA27F /* stp_bank_fpx_bank_islam@3x.png */; }; + 3FA556CF8B11E2486F505161 /* UIViewController+Stripe_NavigationItemProxy.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6F017A08C7E633FB4297D274 /* UIViewController+Stripe_NavigationItemProxy.swift */; }; + 4048B897F3AD627A0BF36D62 /* STPPaymentMethodThreeDSecureUsageTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 8B5AC148E7D8B2EBAEDCB364 /* STPPaymentMethodThreeDSecureUsageTest.m */; }; + 41A4FD6DA3754A61880111D4 /* STPPIIFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = AFB9F0B171497353AADD6544 /* STPPIIFunctionalTest.m */; }; + 420F8CAB4FAD6D9AF4AF25C0 /* StripePaymentSheet.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = DDC55CC034022DFAC9366E2E /* StripePaymentSheet.framework */; }; + 42395EF962DB8AD6A094630B /* StripePaymentSheet.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = DDC55CC034022DFAC9366E2E /* StripePaymentSheet.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + 429DBA641E926EBC2D049FE7 /* STPCustomerContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = BB8FCDBC63A79CD1571A2DFB /* STPCustomerContext.swift */; }; + 42B74740CEF8F04A3F34971A /* stp_bank_fpx_uob@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 8EC1D194936D061E3255BCF4 /* stp_bank_fpx_uob@3x.png */; }; + 43FFF2881D4EFA7B57A60E09 /* STPPaymentMethodCardTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8BD02D8298877F10F2EF2A9D /* STPPaymentMethodCardTest.swift */; }; + 44672917D3AC4B83F9EC3BC3 /* UIView+Stripe_SafeAreaBounds.swift in Sources */ = {isa = PBXBuildFile; fileRef = 86798C95A778362EF815B4C6 /* UIView+Stripe_SafeAreaBounds.swift */; }; + 446A108C8EB6C338A1D774F8 /* STPPaymentConfiguration.swift in Sources */ = {isa = PBXBuildFile; fileRef = 18FCB69CD3B8C3DAB216A5F0 /* STPPaymentConfiguration.swift */; }; + 447C19BDB2CF5445045F81F7 /* STPPaymentContextAmountModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = A1C876DC7F3E31D7189506A8 /* STPPaymentContextAmountModel.swift */; }; + 44A68EC2EE889D75474422F9 /* PayWithLinkViewController-WalletViewModelTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83707086116ED364BE3C1F2E /* PayWithLinkViewController-WalletViewModelTests.swift */; }; + 450FAE41FB4538462D05F2E4 /* LinkSignupViewModelTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = E7C7D85A7FAAFDF4F59BA85E /* LinkSignupViewModelTests.swift */; }; + 45A2E2904C48F19F4D3AD1BF /* FileUpload.json in Resources */ = {isa = PBXBuildFile; fileRef = 3F31C891BDDD871F48F6E60B /* FileUpload.json */; }; + 45FA9B8CC2D18E29BE81CF8F /* STPIntentActionAlipayHandleRedirectTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = AD06AED0AF8A9A7FB4A2E66F /* STPIntentActionAlipayHandleRedirectTest.swift */; }; + 47E93085491222CB4C7FA9D4 /* STPSTPViewWithSeparatorSnapshotTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 9290B75648F2D23764FA71E9 /* STPSTPViewWithSeparatorSnapshotTests.m */; }; + 4935C8B3ECFBAD947E694934 /* STPIntentActionTypeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = D2F205F920E971DEA59E3C31 /* STPIntentActionTypeTest.swift */; }; + 4993037E5386D0AF87B24871 /* STPPaymentMethodAffirmParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = D87817A1D3D213AA4ADF6A4C /* STPPaymentMethodAffirmParamsTest.swift */; }; + 49A77DA7B76E498C9F23D428 /* STPAPIClientNetworkBridgeTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 51F49F2466BBBE64799209B5 /* STPAPIClientNetworkBridgeTest.m */; }; + 4A61DC36F10B9C9C24345613 /* STPRadarSessionFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2D878F923A1F69B58D6B2812 /* STPRadarSessionFunctionalTest.swift */; }; + 4AAA2CD5AEF1F913395B3B95 /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = CF13BAEF86594C9CABD4F42A /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift */; }; + 4ABF7BEF11E1F4715DCFF446 /* STPAddressTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 81B23BF8256AA75F9903AC0C /* STPAddressTests.m */; }; + 4B0917FC15BF56D0100E0ED1 /* STPGenericInputTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 58A53F005EA8FDDAA66126BA /* STPGenericInputTextFieldSnapshotTests.swift */; }; + 4C3B161481D11385352B06D4 /* STPCustomerContextTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2CC4B06AB5C02FF54091E5A8 /* STPCustomerContextTest.swift */; }; + 4DA70CAD042B5968D833B5BF /* SWHttpTrafficRecorder.m in Sources */ = {isa = PBXBuildFile; fileRef = 40DC59F285B3AB0F60B33443 /* SWHttpTrafficRecorder.m */; }; + 4E09E54E7FEC35C49C59A379 /* STPPushProvisioningDetailsParams.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2B28B8A547CD846277ECD578 /* STPPushProvisioningDetailsParams.swift */; }; + 4E31B1864DA407598FB1BBC6 /* STPPostalCodeInputTextFieldTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6618739767139C25C05B3631 /* STPPostalCodeInputTextFieldTests.swift */; }; + 4ED44ACF24949F516867235C /* STPPaymentOptionTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6215A9BF343775B1BD0F62AF /* STPPaymentOptionTableViewCell.swift */; }; + 4EFF8B46B12DA4D9AAB22523 /* STPPaymentCardTextFieldCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 01B42BE6FB5EC1F708875AB8 /* STPPaymentCardTextFieldCell.swift */; }; + 4F315E738D87C06682C4C504 /* LinkInstantDebitMandateViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6F29142B51DC99E2258B4F60 /* LinkInstantDebitMandateViewSnapshotTests.swift */; }; + 4FB67F10A0B7106A8142B842 /* STPEphemeralKeyManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 588C260880FFC584A00A89F5 /* STPEphemeralKeyManager.swift */; }; + 50D084E50661D174297FC30B /* STPCardFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 94042E0B406F6AA45593D154 /* STPCardFunctionalTest.m */; }; + 50FB2476C611B94E5D283836 /* STPPaymentMethodiDEALTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 415554EBF13241D12094103B /* STPPaymentMethodiDEALTest.m */; }; + 51044B947A7FDB99451466D8 /* STPGenericInputPickerFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 284C67269D2606DA147AE01D /* STPGenericInputPickerFieldSnapshotTests.swift */; }; + 5170651536332C4842E9D009 /* STPPaymentMethodBoletoTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4E371E9B3B2E343FE954531C /* STPPaymentMethodBoletoTests.swift */; }; + 51D515315F02D4C03BA12366 /* UserDefaults+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3B0E131538728BC4802627B1 /* UserDefaults+StripeTest.swift */; }; + 5212C7875C07F9BF16AFD98D /* STPAPIClient+PushProvisioning.swift in Sources */ = {isa = PBXBuildFile; fileRef = 807FF966F1DE05F3496B817B /* STPAPIClient+PushProvisioning.swift */; }; + 5224B2FFB1786C8AF3095248 /* GiropaySource.json in Resources */ = {isa = PBXBuildFile; fileRef = 682296D57E9486BBCC6ED7F5 /* GiropaySource.json */; }; + 53853E75323918758A1A3B35 /* STPEphemeralKeyManagerTest.m in Sources */ = {isa = PBXBuildFile; fileRef = F0D010B03BB8B290806267A0 /* STPEphemeralKeyManagerTest.m */; }; + 53D0248D7927B0E62B8C967B /* PaymentIntent.json in Resources */ = {isa = PBXBuildFile; fileRef = 68778692A4A89E311FC9DEAE /* PaymentIntent.json */; }; + 542610492B38FEB652C6823E /* String+Localized.swift in Sources */ = {isa = PBXBuildFile; fileRef = A5398E1156E0BFEBBF56FD2F /* String+Localized.swift */; }; + 542B23E3AC288AF2F4E5D6C8 /* STPNetworkStubbingTestCase.m in Sources */ = {isa = PBXBuildFile; fileRef = 9475002E030E98735A0FD2EC /* STPNetworkStubbingTestCase.m */; }; + 54331380F5AC68846DBE94D5 /* UITableViewCell+Stripe_Borders.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5476BD87E0480A93958F0328 /* UITableViewCell+Stripe_Borders.swift */; }; + 5660A88BD792A7A844084F16 /* STPLabeledFormTextFieldViewSnapshotTests.m in Sources */ = {isa = PBXBuildFile; fileRef = CF5E437205850C2039BBEBCB /* STPLabeledFormTextFieldViewSnapshotTests.m */; }; + 58240AA66FAE55131268E4A0 /* STPAddressFieldTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 72903593DC432D01720DC9D9 /* STPAddressFieldTableViewCell.swift */; }; + 583DE9869C885BA02E0A071E /* STPAUBECSFormViewModelTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4AAE5EE11611F9F7762B64C6 /* STPAUBECSFormViewModelTests.swift */; }; + 58A8B3F57FE98C22D8F90C77 /* STPThreeDSButtonCustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = B6F3B966470A530E0DC53F8C /* STPThreeDSButtonCustomizationTest.swift */; }; + 590DB84AC15709E3C6F1FC3B /* STPThreeDSNavigationBarCustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 51408DE266D0345784ADD4FA /* STPThreeDSNavigationBarCustomizationTest.swift */; }; + 5910FCB9822259D5EC7E4051 /* AutoCompleteViewControllerSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = EFD2F6A5A046A620BAB75B41 /* AutoCompleteViewControllerSnapshotTests.swift */; }; + 59D4B49B6C7EEF121EE266E4 /* STPTestUtils.m in Sources */ = {isa = PBXBuildFile; fileRef = 9F8F79CBFDC930A18998D7B1 /* STPTestUtils.m */; }; + 5A6CF4BEE7E5B3587217C848 /* PaymentSheetLinkAccountTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = E9A962F163EC016697A82124 /* PaymentSheetLinkAccountTests.swift */; }; + 5A75EF4349F6B444B6D56552 /* stp_bank_fpx_affin_bank@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 2A588999859430CC959C6F4B /* stp_bank_fpx_affin_bank@3x.png */; }; + 5AC2B7D5912DBD0FF7DB2349 /* BacsDebitPaymentMethod.json in Resources */ = {isa = PBXBuildFile; fileRef = 2EADE1EE40283DEF52B0B5D4 /* BacsDebitPaymentMethod.json */; }; + 5B6F1BF973FC2D8DD6127B7F /* StripePaymentsUI.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = E136A967522048B313E3C62F /* StripePaymentsUI.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + 5C51167CC14F653E7117BA61 /* OCMock in Frameworks */ = {isa = PBXBuildFile; productRef = E804AA8C4156CC85FFD9595F /* OCMock */; }; + 5C5E1CE53D89DE8F0B867115 /* STPPaymentMethodSofortTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = E8063E8073A32E0B081A1DFA /* STPPaymentMethodSofortTests.swift */; }; + 5D1A9F97F79DAA3F46C82A28 /* LinkAccountServiceTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = BC7B7ED388B6C6AE57D3D627 /* LinkAccountServiceTests.swift */; }; + 5D6B52EB4D7258129F134D07 /* STPImageLibrary.swift in Sources */ = {isa = PBXBuildFile; fileRef = D93C23F55BEADF9BC74DFBDB /* STPImageLibrary.swift */; }; + 5DD402F5E453D4A2194A346B /* PaymentSheetAddressTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = D61146D58FD219AE39E71D7D /* PaymentSheetAddressTests.swift */; }; + 5E5EE69D140F6FEDA5F0A346 /* STPAPIClientTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = B8DD70E5ED8E9DE8E9752C9E /* STPAPIClientTest.swift */; }; + 5ECED204FD22CFEA3A806767 /* STPPaymentOptionTuple.swift in Sources */ = {isa = PBXBuildFile; fileRef = EDD30E5DB8DB3AA3567F5C20 /* STPPaymentOptionTuple.swift */; }; + 605EFBDD21426FD30581563F /* STPAnalyticsClient+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 12DBB3F72AEFB52DE27C27ED /* STPAnalyticsClient+BasicUI.swift */; }; + 609C2C8F10AFAA2711639CD0 /* NSArray+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C17799DC7FA54E758EED31A6 /* NSArray+StripeTest.swift */; }; + 609E4D384B75F6A111DC0E27 /* STPPaymentActivityIndicatorView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1FFBAA4B44967B157A4F4E91 /* STPPaymentActivityIndicatorView.swift */; }; + 621CBBB2E116055B85E860B8 /* STPPaymentMethodAfterpayClearpayTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 26D0BC36C2610307A01C52CB /* STPPaymentMethodAfterpayClearpayTest.m */; }; + 62B91808A088C4F9FDB62C53 /* STPEphemeralKey.swift in Sources */ = {isa = PBXBuildFile; fileRef = 890660C21E3666CE7B82695B /* STPEphemeralKey.swift */; }; + 62D65741A6D49C7EBFB7CB61 /* stp_card_form_back@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 2A0E991FB88CBE8C9135DFE0 /* stp_card_form_back@3x.png */; }; + 6326FD82A0E6AFB10C773B03 /* LinkNavigationBarSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 157826E456AC3ABFE8000B81 /* LinkNavigationBarSnapshotTests.swift */; }; + 64801CF2D2CAC9008C17D154 /* LinkSecureCookieStoreTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 10BAD33BEC6C2894F9266902 /* LinkSecureCookieStoreTests.swift */; }; + 64A685F033DC45A99FB3E300 /* STPPaymentMethodBacsDebitTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 71C175861E0F8882BCF85F78 /* STPPaymentMethodBacsDebitTest.m */; }; + 64D168991F7FF6CB9A223F54 /* STPSourceTest.m in Sources */ = {isa = PBXBuildFile; fileRef = B7B5B626BCED2938C256BF49 /* STPSourceTest.m */; }; + 6509C8C171F9A76E14F59050 /* ElementsSession.json in Resources */ = {isa = PBXBuildFile; fileRef = CFE6A87CDA08F216E1DFE4CD /* ElementsSession.json */; }; + 659E5859BD4F4F5BACB6F3C1 /* Card.json in Resources */ = {isa = PBXBuildFile; fileRef = C861CC96A54A24BDC1303C1B /* Card.json */; }; + 65E6B1133364DA6854F570A2 /* STPSourceSEPADebitDetailsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 342398F7CB61693AAADE3F26 /* STPSourceSEPADebitDetailsTest.m */; }; + 66065B1D65D7D5502D4E2F2B /* STPCard+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3E5FB20B2BEFC00D54FDD87D /* STPCard+BasicUI.swift */; }; + 66B7EF2DC1CBF813707C767C /* STPBSBNumberValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F3C732C25FD961631BD44FDD /* STPBSBNumberValidatorTests.swift */; }; + 66C38EA9CFB1ED4DF2F974BF /* STPPaymentOptionsViewControllerLocalizationTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6C7094644EE056260D6F3B67 /* STPPaymentOptionsViewControllerLocalizationTests.swift */; }; + 66ED28E18385B2644F4EF3CC /* EPSSource.json in Resources */ = {isa = PBXBuildFile; fileRef = 1020F76D0722D685A97EF202 /* EPSSource.json */; }; + 672B820564FFBAFFAD93B27E /* STPPaymentIntentFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = A8CF4EBC85BDC771DF2213C0 /* STPPaymentIntentFunctionalTest.m */; }; + 67E18B3DA020900C83AE29B9 /* LinkInlineSignupElementSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83A5A0074F69DE467F94ABFB /* LinkInlineSignupElementSnapshotTests.swift */; }; + 68318DB86DFCD19505FC47BA /* NSURLComponents_StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E1CD20E00EAD41091B71ABD5 /* NSURLComponents_StripeTest.swift */; }; + 6964CE3F9E07A53AA2954E8E /* LinkStubs.swift in Sources */ = {isa = PBXBuildFile; fileRef = 52FA1AFD632CCE18C4315FC6 /* LinkStubs.swift */; }; + 69AC1EDE2A3C03B1D980CA54 /* STPPaymentOptionsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = A9A1BB31C7B514984231125B /* STPPaymentOptionsViewController.swift */; }; + 6BF6ECC4A4E61E2FFC3EA20B /* STPAPIClient+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1E8AFAE24610EC983727F860 /* STPAPIClient+BasicUI.swift */; }; + 6D578C4501AD509C88010ABB /* STPPaymentMethodAUBECSDebitParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 8A7FBDBADC1EBC62FFC4A366 /* STPPaymentMethodAUBECSDebitParamsTests.m */; }; + 6E7AD3CCC966A7F34922B172 /* NSDictionary+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = D07275F94914B7E7937D24FE /* NSDictionary+StripeTest.swift */; }; + 6EF3F611E6EA3CB479D62450 /* AfterpayPriceBreakdownViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3C742844915B96CFD25BFFF9 /* AfterpayPriceBreakdownViewSnapshotTests.swift */; }; + 6F4FBB4F10B5DB2CF8BB3460 /* STPBinRangeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3FC0560A312147C37CFE6CF9 /* STPBinRangeTest.swift */; }; + 6F9525063D76A9F86A10CCBF /* STPApplePayContextFunctionalTestExtras.swift in Sources */ = {isa = PBXBuildFile; fileRef = A1272F2E05A0E294DD9ECA26 /* STPApplePayContextFunctionalTestExtras.swift */; }; + 6FCA954C32AB351F902BA876 /* STPPostalCodeValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = FDF7394DDD552EDE996EAD8E /* STPPostalCodeValidatorTest.swift */; }; + 701C464523173C6809544935 /* STPThreeDSUICustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3ED44491EB0AC72B1B1A773C /* STPThreeDSUICustomizationTest.swift */; }; + 71116C2D5831E271E12DB059 /* ServerErrorMapperTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 20879436DCFB1F03BE1608B3 /* ServerErrorMapperTest.swift */; }; + 72CB8A91146F3EC17CEFA235 /* STPSetupIntentFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 60E21B0DB0D3B307040EA88E /* STPSetupIntentFunctionalTest.m */; }; + 73AFE2A8839EFAB8330F6CF0 /* STPPaymentIntentTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE8998EAF997A78759E49B5 /* STPPaymentIntentTest.swift */; }; + 7435E6BB6971012A9B0DB52E /* STPErrorBridgeTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 9EAE9A2AE65771403CE57C11 /* STPErrorBridgeTest.m */; }; + 7589E37795D21AB818B0C333 /* STPAnalyticsClient+Payments.swift in Sources */ = {isa = PBXBuildFile; fileRef = BD89580A3E41D7167C30B287 /* STPAnalyticsClient+Payments.swift */; }; + 75F81259F0DEE1ED09FDECBA /* ApplePayPaymentMethod.json in Resources */ = {isa = PBXBuildFile; fileRef = 848AAE69CBF96A057F734365 /* ApplePayPaymentMethod.json */; }; + 7797B149DF5D5CA201089BC2 /* stp_bank_fpx_bsn@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = BCFA6BB95613ED3F668A0DC9 /* stp_bank_fpx_bsn@3x.png */; }; + 77ED42569FEC0EC5AF83538A /* BankAccount.json in Resources */ = {isa = PBXBuildFile; fileRef = 62169E37BD48681A914FA4CB /* BankAccount.json */; }; + 781EC0163AC001C6A66045B6 /* STPMandateDataParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = B7F7AA0B7B86BA5BB2FE92CE /* STPMandateDataParamsTest.swift */; }; + 7844BB705AEB002965EF82B0 /* STPPaymentMethodKlarnaParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47E12A0CBFA259A032F7AF0C /* STPPaymentMethodKlarnaParamsTests.swift */; }; + 78B70C2EE8334F0FA91439CA /* Stripe.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4259421D2CD26E37B96F97B2 /* Stripe.framework */; }; + 78F39965489F75A82584F7E9 /* STPPaymentMethodUPIParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = EAB51503D64F3464BFBC18CB /* STPPaymentMethodUPIParamsTest.m */; }; + 795F3783D62AB8E2A00DCD05 /* ConsumerSessionTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = E6312182B5BCAB940D216650 /* ConsumerSessionTests.swift */; }; + 7A1F658DC9D1494153AF215E /* STPPaymentMethodOXXOParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 5F23996EC0E06D47B20316AC /* STPPaymentMethodOXXOParamsTests.m */; }; + 7B9C0D039EA9EF593AEC682D /* STPShippingMethodTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 98544B08552407D41D398C68 /* STPShippingMethodTableViewCell.swift */; }; + 7BC98BE168781C5B3EC8A8DB /* STPPaymentMethod+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 35C1E9B0EE03825DABF6471A /* STPPaymentMethod+BasicUI.swift */; }; + 7BEC4847DDD51C9C36C758E9 /* stp_bank_fpx_public_bank@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = CAC5FD19B4843C454E6881DB /* stp_bank_fpx_public_bank@3x.png */; }; + 7D251ABF1EBF65ACA8A4BDD4 /* STPPaymentMethodUSBankAccountParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74FDEF9F687C63BADFB96480 /* STPPaymentMethodUSBankAccountParamsTest.swift */; }; + 7EAA7334372DBC38DF8FA0AA /* STPPinManagementService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3EBB07171F6FDCE6E20C454A /* STPPinManagementService.swift */; }; + 7EB99B1286C38DD944D0D9DC /* MultibancoSource.json in Resources */ = {isa = PBXBuildFile; fileRef = 63A16B33FCF0B351D4A60247 /* MultibancoSource.json */; }; + 7F235CD649F6E97E4E7DD180 /* UIView+Stripe_FirstResponder.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1B76DF0FE363F59BF0940A8B /* UIView+Stripe_FirstResponder.swift */; }; + 7F9D08AC5A448C7693162D7D /* STPShippingMethodsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 21E4B84223DDA131544DBBA7 /* STPShippingMethodsViewController.swift */; }; + 812682EA323986B8F698FF3C /* STPPaymentMethodParams+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53C5AB22D6328E85A6DDF663 /* STPPaymentMethodParams+BasicUI.swift */; }; + 825CBE20F190E96EFA95B35A /* STPAPISettingsBridgeTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 4A57C346FA98A057AF31C8B1 /* STPAPISettingsBridgeTest.m */; }; + 829D43B6705D125FEC9926DA /* STPPaymentContextApplePayTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3C995125252BED1EEC018B9D /* STPPaymentContextApplePayTest.swift */; }; + 82B4D3FC63C069728459BABD /* STPPaymentCardTextFieldTest.m in Sources */ = {isa = PBXBuildFile; fileRef = B7439CC6C0FB060EA312FF6D /* STPPaymentCardTextFieldTest.m */; }; + 8378F2A4B0796819BB1C6C54 /* STPPaymentMethodCardParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1671EC46C713D51013AD7D8B /* STPPaymentMethodCardParamsTest.swift */; }; + 83FE814F7C472EB31DD9D28F /* stp_bank_fpx_bank_rakyat@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 34D8B80455B75885088E9DAC /* stp_bank_fpx_bank_rakyat@3x.png */; }; + 8520A27C204A068C43592024 /* StripeApplePay.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 52F8AEC50D4623F80F04A533 /* StripeApplePay.framework */; }; + 8532FEBF4F2E0EB282D466CE /* STPGenericInputPickerFieldValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2D63B73C5773432CA134D1FC /* STPGenericInputPickerFieldValidatorTest.swift */; }; + 86BAF121184D71F5F4FFAD7B /* STPPaymentCardTextFieldTestsSwift.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1F16C36797D978E72E612100 /* STPPaymentCardTextFieldTestsSwift.swift */; }; + 87061C07E2B17DD8B7052B72 /* SEPADebitSource.json in Resources */ = {isa = PBXBuildFile; fileRef = 252FD94C68761E893E53D5F9 /* SEPADebitSource.json */; }; + 87AACDD643A998FFDD505D22 /* KlarnaHelperTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8D866DCC403994A0D3CFB1D7 /* KlarnaHelperTest.swift */; }; + 89C927D276A90357F06ECE3C /* CardPaymentMethod.json in Resources */ = {isa = PBXBuildFile; fileRef = 344C5B4D8789FEFA4CFC0E43 /* CardPaymentMethod.json */; }; + 89E246E74FACFAB5EFBA6980 /* STPCustomerTest.m in Sources */ = {isa = PBXBuildFile; fileRef = FABC6FF8D581998A773A8656 /* STPCustomerTest.m */; }; + 8A3B74851E988FC00BE172E1 /* STPFixtures.m in Sources */ = {isa = PBXBuildFile; fileRef = 17DDEC7D4C1D5FF6488D71D5 /* STPFixtures.m */; }; + 8B80FB6FC88D411A90E9D487 /* WalletHeaderViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0DB03E83746FE78361831546 /* WalletHeaderViewSnapshotTests.swift */; }; + 8C977F8D224A7360AE8E15A7 /* STPPaymentMethodBoletoParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = A1C67AC5D415615E9F27D3E3 /* STPPaymentMethodBoletoParamsTests.swift */; }; + 8CE31C2917AC0B9084C90650 /* stp_card_form_front@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 5C0462C8B2776183A2B03D51 /* stp_card_form_front@3x.png */; }; + 8DB0749467DC61349532FB7B /* STPPaymentMethodPrzelewy24ParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = CFB4BBE0C4760DB457DE9A57 /* STPPaymentMethodPrzelewy24ParamsTests.m */; }; + 8DCE81502850557222978D6A /* PaymentSheetFormFactorySnapshotTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 83760BB83F6391E0CF66234D /* PaymentSheetFormFactorySnapshotTest.swift */; }; + 8E423294AB602BF25DB11D8E /* OperationDebouncerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = A61763BA2CCA86F9B8FD4F1F /* OperationDebouncerTests.swift */; }; + 8EC1820299152F8565D30A40 /* STPCardCVCInputTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = D1859673CAD068B345F5DD7D /* STPCardCVCInputTextFieldSnapshotTests.swift */; }; + 8F0326E98C74EB62E34B9FEA /* STPIntentActionWeChatPayRedirectToAppTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4CA5D11C977A95B8E936E907 /* STPIntentActionWeChatPayRedirectToAppTest.swift */; }; + 903FFB756C6ED520BE38EF6F /* STPInputTextFieldValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9FEE395C4DD0E0112AF3720C /* STPInputTextFieldValidatorTests.swift */; }; + 91558F51B87C72E745244958 /* STPPostalCodeInputTextFieldValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = EA9975553E669AF69F3CE437 /* STPPostalCodeInputTextFieldValidatorTests.swift */; }; + 91A839DEDA7D1EAF6FC66BE0 /* STPFormEncoderTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 30964128998473CAA9F2DD7E /* STPFormEncoderTest.swift */; }; + 91F90088C3F1102E3C041014 /* LinkToastSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5BB7453E4E1AD36D604A00E8 /* LinkToastSnapshotTests.swift */; }; + 9252E3E10BD2FEC4CDD05959 /* PaymentSheetPaymentMethodTypeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = A1CF1C43EEE33FC0AF31A2BB /* PaymentSheetPaymentMethodTypeTest.swift */; }; + 9291A08CCB34504FCA4B7481 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 005650A59D692F820EF20F5F /* XCTest.framework */; }; + 9299ECF19D79770F54AC4732 /* TextFieldElement+CardTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5CE0ABE46E1C27A6AD8D8BD9 /* TextFieldElement+CardTest.swift */; }; + 9363F8F389C04C19B37D0F0A /* StripePaymentsUI.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = E136A967522048B313E3C62F /* StripePaymentsUI.framework */; }; + 9482B2A9A13CA7F5F8C79780 /* STPConfirmPaymentMethodOptionsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 0B989830717894A60B74E830 /* STPConfirmPaymentMethodOptionsTest.m */; }; + 951344464ACF84F0F6D43D10 /* OneTimeCodeTextFieldTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 98F9CB667BC68767DFB5FACD /* OneTimeCodeTextFieldTests.swift */; }; + 96098727EFA6A72087A35A52 /* STPFormTextFieldTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2E1862744F23286D1FB9D4AE /* STPFormTextFieldTest.swift */; }; + 9668F3E0DC7FAC4725B8C446 /* PaymentSheetTestUtils.swift in Sources */ = {isa = PBXBuildFile; fileRef = CDDB4BC3C16AB7218E6FF1FA /* PaymentSheetTestUtils.swift */; }; + 9708289AA2B48C0828F21FA1 /* PaymentSheet+APITest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9D73D989C9D488D820029FA1 /* PaymentSheet+APITest.swift */; }; + 97F2F817247B1CB07F1E6600 /* STPFileFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 547F890FCD6D6E8B0774E919 /* STPFileFunctionalTest.m */; }; + 98E2332DE7F54E970BE5EEF7 /* UIBarButtonItem+Stripe.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3B3000668A75E095B514241F /* UIBarButtonItem+Stripe.swift */; }; + 98EE8326C1D133E1C998114F /* STPLocalizedString.swift in Sources */ = {isa = PBXBuildFile; fileRef = AFF957F38AABE5F748C38C0B /* STPLocalizedString.swift */; }; + 9A24970C5FB6D3F7314AE550 /* STPAPIClientStubbedTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E4194E605BB5F31E9CBB8F96 /* STPAPIClientStubbedTest.swift */; }; + 9B149DA42FB38C3542E0CB4B /* STPApplePayFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3C77C7BC4BA57EC296CF2F1C /* STPApplePayFunctionalTest.swift */; }; + 9B1AC278FDCDABF26C5E468C /* STPPostalCodeInputTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 090EF7D598B8DE779C275395 /* STPPostalCodeInputTextFieldSnapshotTests.swift */; }; + 9BD0FF43A32AAAA30A5973A5 /* STPConnectAccountFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = D45CA902199FABE1254E8D1F /* STPConnectAccountFunctionalTest.m */; }; + 9C13E8A017A4E23BCCDE618B /* UINavigationController+Stripe_Completion.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2C078573F46762353664AC92 /* UINavigationController+Stripe_Completion.swift */; }; + 9D464A252FBD0D4E2A0A7398 /* STPCountryPickerInputFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8D49257A97E71A475A9F6E08 /* STPCountryPickerInputFieldSnapshotTests.swift */; }; + 9D8354BDB04CEC5D1EFCF54F /* STPSwiftFixtures.swift in Sources */ = {isa = PBXBuildFile; fileRef = E315168EF07F52B733EA77F8 /* STPSwiftFixtures.swift */; }; + 9D9692DFC4F06F8C70145000 /* UINavigationBar+Stripe_Theme.swift in Sources */ = {isa = PBXBuildFile; fileRef = A7E369CEC9F5B3758F78E88F /* UINavigationBar+Stripe_Theme.swift */; }; + 9FD92B3ADEBEC96660B70409 /* STPMocks.m in Sources */ = {isa = PBXBuildFile; fileRef = 88AEABC15CCBB9EA393C175F /* STPMocks.m */; }; + 9FFC5B74959F202EDF277DF8 /* STPConfirmCardOptionsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 444A6B234A4BD3CB2C1C977D /* STPConfirmCardOptionsTest.m */; }; + A08C2F0E7F642515B1D263ED /* STPPaymentMethodTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 148C1D7D1BBBC6B74894A869 /* STPPaymentMethodTest.swift */; }; + A0AA0B8AEF5B429858D71F6B /* STPBlocks.swift in Sources */ = {isa = PBXBuildFile; fileRef = E3B42EBAC0DC7ED0D9200DB7 /* STPBlocks.swift */; }; + A0C3E888938414D185B71CF7 /* LinkBadgeViewSnapshotTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 72B1E4D9DA48CED66CD96308 /* LinkBadgeViewSnapshotTest.swift */; }; + A22D548084E7DE1FE5ABE8E7 /* STPLabeledMultiFormTextFieldViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7F68637E75142DCD46710796 /* STPLabeledMultiFormTextFieldViewSnapshotTests.swift */; }; + A66C279957B6AC8F72DE05C7 /* STPCardExpiryInputTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0C75157665428685C7A4FD20 /* STPCardExpiryInputTextFieldSnapshotTests.swift */; }; + A7244AE4E4D7E5BC66303F62 /* STPBankAccountParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 9F304A4028D563EBD2A523C8 /* STPBankAccountParamsTest.m */; }; + A77C5769B20D7884FC8FC4FB /* STPNumericDigitInputTextFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6223E57D3A198F956A37ED89 /* STPNumericDigitInputTextFormatterTests.swift */; }; + A77EC5CE65161573062E9F98 /* STPShippingAddressViewControllerLocalizationTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = DE3D0F21FB3B3E8BCA15FF7C /* STPShippingAddressViewControllerLocalizationTests.swift */; }; + A781FB0F586B26655FAEC3C0 /* STPCertTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1D51B04D83D4FEF7F90DF16A /* STPCertTest.swift */; }; + A7A1D3C0D75DCD7217D297FF /* STPShippingMethodsViewControllerLocalizationTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = C7F54E1E8FFA1505D24538A6 /* STPShippingMethodsViewControllerLocalizationTests.swift */; }; + A8B0DB753CAA2223C8BED099 /* StripeErrorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = DC3AD586DDED620B9E68F461 /* StripeErrorTest.swift */; }; + A9A77D698551322BF3A067BC /* BancontactSource.json in Resources */ = {isa = PBXBuildFile; fileRef = 31F32B2852B1683819BD02F3 /* BancontactSource.json */; }; + AB3BD6A5E8660EC6D2C299BD /* LinkNoticeViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 382A01724CE8659A5210073F /* LinkNoticeViewSnapshotTests.swift */; }; + AC35943F1EAD50E9D5D509B3 /* STPCardExpiryInputTextFieldFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 40BB87E28719FE0C6B946BB5 /* STPCardExpiryInputTextFieldFormatterTests.swift */; }; + AC7C127B11A60222465F4696 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 005650A59D692F820EF20F5F /* XCTest.framework */; }; + ACE783C5E133EB9962908BA8 /* STPPaymentMethodGrabPayParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 03CEA8792C992D865ECC46AA /* STPPaymentMethodGrabPayParamsTest.m */; }; + ACF6CFE0F8B88FDBBB16968C /* FraudDetectionDataTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 45FF7A07CFC3B9B7AD6B49EE /* FraudDetectionDataTest.swift */; }; + AD731C41CE95233A733E0289 /* STPTestingAPIClient.m in Sources */ = {isa = PBXBuildFile; fileRef = C1AAF76C3F4850217C28E362 /* STPTestingAPIClient.m */; }; + AD9B9F3FF697D4A3892E86F2 /* PaymentAnalyticTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C8F8FCC84601E4ADC6B7F3CE /* PaymentAnalyticTest.swift */; }; + AE5C68FE305F63791B59CDD1 /* STPSourceRedirectTest.m in Sources */ = {isa = PBXBuildFile; fileRef = B6799CC0D9C5FED26EE3494F /* STPSourceRedirectTest.m */; }; + AE747ADA2841AA06F32558D8 /* STPSourceCardDetailsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 63114D0EAAE2606732DF5AA0 /* STPSourceCardDetailsTest.swift */; }; + AE90123C3DCF6C56329ABD72 /* STPIntentActionTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 9CE65B72986B6B404B036183 /* STPIntentActionTest.m */; }; + AE94F473534AE66F984D0254 /* STPPaymentMethodParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 7B4DEF058786BD9D3D5EE2FB /* STPPaymentMethodParamsTest.m */; }; + AE9AB2E4E82CC2AFF8B2DC96 /* STPApplePayTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 47A1AA1C80AE910FCFF58996 /* STPApplePayTest.m */; }; + AF0FE00E669705885EAF6F20 /* stp_icon_bank@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = A1003B17EB955A29BC77BF01 /* stp_icon_bank@3x.png */; }; + AF18D569B296BFC1EB5A7338 /* ImageTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = AAF368BCD5990EE5DC17D299 /* ImageTest.swift */; }; + AF23CB4EF17E87007CFC3E96 /* STPFPXBankStatusResponse.swift in Sources */ = {isa = PBXBuildFile; fileRef = BA08DCDD421CE92ECB61EF5C /* STPFPXBankStatusResponse.swift */; }; + AF44725558E654548FED2A2B /* STPPaymentMethodUPITests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 940209E5D30E86E856016906 /* STPPaymentMethodUPITests.swift */; }; + B00F7FC372E376C6B2170D37 /* STPPaymentContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = C750C2C4AB33BC232D1592BA /* STPPaymentContext.swift */; }; + B1B3E2766CB2B4AA48D130F6 /* stp_bank_fpx_maybank2u@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = A982DCED06CCAEBE2C8835F3 /* stp_bank_fpx_maybank2u@3x.png */; }; + B1BF689B91D538BDCA4C8578 /* STPCoreViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 02FC9ED423D40C88D5A24441 /* STPCoreViewController.swift */; }; + B3C3DCC5BD2BE09DEC0906F0 /* STPFileTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 0841697DF701B4A4637E67D9 /* STPFileTest.m */; }; + B4719234E4BBDAD260E31373 /* STPPaymentCardTextFieldViewModelTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E6DEE912364C9F4B51B374D0 /* STPPaymentCardTextFieldViewModelTest.swift */; }; + B5A0B3CF303CD41FF73D4310 /* STPTestAPIClient+Swift.swift in Sources */ = {isa = PBXBuildFile; fileRef = E3AC82FE96A69220B300FBE6 /* STPTestAPIClient+Swift.swift */; }; + B6656829DEC006DBEED2AA0E /* STPEphemeralKeyTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C5BEAA15B53AC5662A33D0E1 /* STPEphemeralKeyTest.swift */; }; + B6784B7F4B9B04617C0EE510 /* PKAddPaymentPassRequest+Stripe_Error.swift in Sources */ = {isa = PBXBuildFile; fileRef = 273EE407039913F0B644172B /* PKAddPaymentPassRequest+Stripe_Error.swift */; }; + B6B89E3F7DE0811BD5CB9D31 /* STPAddCardViewControllerLocalizationTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 78BA15D1F0844B1DA71A5348 /* STPAddCardViewControllerLocalizationTests.swift */; }; + B7519705686EAA1BC4F0BC5A /* STPPaymentMethodPayPalTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 02DEC25F0E12F9F4396E87B2 /* STPPaymentMethodPayPalTests.m */; }; + B795A5EB8FDECA1060A9655C /* iOSSnapshotTestCase in Frameworks */ = {isa = PBXBuildFile; productRef = C55551F29B99CF6D6DD9EE2F /* iOSSnapshotTestCase */; }; + B7CE5D774F8BA17274995BA2 /* STPBankAccountFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6152335A2DCCBB5490B39BB0 /* STPBankAccountFunctionalTest.m */; }; + B82859A4444B9F735720F232 /* STPMandateOnlineParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB58AC5E2E0A68221260FD44 /* STPMandateOnlineParamsTest.swift */; }; + B8385576DC25BDEEB92D812F /* STPEphemeralKeyProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = 485E747DA1F72F091986787B /* STPEphemeralKeyProvider.swift */; }; + B8ED1F697519A6FCD3D79431 /* STPPaymentMethodGiropayTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 583DB466066B47C0F716E474 /* STPPaymentMethodGiropayTests.swift */; }; + B917BF282C84507292112B9D /* STPCardBINMetadataTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3E1C5E08678292561255B1C5 /* STPCardBINMetadataTests.swift */; }; + B92C22D02FDA5BC7FCDF8EEF /* PaymentMethodMessagingViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F4F116895CA31148DB6CE0DF /* PaymentMethodMessagingViewSnapshotTests.swift */; }; + B98D71ED9ACC2E1B47372F53 /* NSDecimalNumber+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 20552E792B8E7BA15821AB5D /* NSDecimalNumber+StripeTest.swift */; }; + B9EE4AC9206FD77A2FB8C702 /* STPFPXBankBrandTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 966BA151AA0E428D1DD8A97C /* STPFPXBankBrandTest.m */; }; + BAEE96DF60B42EE2A559DC9F /* LinkVerificationViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 24AE1B8096363F6C51A5AF7E /* LinkVerificationViewSnapshotTests.swift */; }; + BAFD06E994739E1C38DFFBBC /* STPCardScanner.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69BD038947E8E2376A0D240B /* STPCardScanner.swift */; }; + BB46077C256C26418420F240 /* STPAddCardViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = FA793904C7B2D3AA0A4D5EFB /* STPAddCardViewController.swift */; }; + BC6912C0DE15008C8D8C303C /* STPFloatingPlaceholderTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F7385193226663A5B79E69ED /* STPFloatingPlaceholderTextFieldSnapshotTests.swift */; }; + BC694A1642DC30D530B60635 /* RotatingCardBrandsViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FDA32D0C9E8A7A69F4899EDC /* RotatingCardBrandsViewSnapshotTests.swift */; }; + BF4A92064926FD7B0E3E92F7 /* StripePayments.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 22D1C6EB5826E2D7C80B6CF3 /* StripePayments.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + BF4ED4828114E2E89A3D4AB7 /* StripeBundleLocator.swift in Sources */ = {isa = PBXBuildFile; fileRef = B4D31B0D7BD9F97AF3BB61E6 /* StripeBundleLocator.swift */; }; + C035D82D7096E3005858848C /* StripeCore.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 43B4E4B85C598D7A9AFCB4D4 /* StripeCore.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + C0688E067AE4FFDFFDDC03BB /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift in Sources */ = {isa = PBXBuildFile; fileRef = E5D9F97ABC88302478220267 /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift */; }; + C1E52E633CCF8D18AC804C97 /* stp_bank_fpx_bank_muamalat@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = A7B538B1223FBD7F409A4B29 /* stp_bank_fpx_bank_muamalat@3x.png */; }; + C1E70FD29BBE36D76A7E6929 /* CardExpiryDateTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 153071C69A0BEE033E035DCF /* CardExpiryDateTests.swift */; }; + C314A5C55064C51C2B999E6B /* STPBackendAPIAdapter.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9B782E1D974A4C131E60E2BD /* STPBackendAPIAdapter.swift */; }; + C32D7ACEBC852CBC295BBEF2 /* STPSetupIntentFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 49AA313E068FB99CEAA5F7D3 /* STPSetupIntentFunctionalTest.swift */; }; + C35CF837D67AE8DB7CBDAD98 /* STPThreeDSLabelCustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5FDD4956E0A04D33F0856F31 /* STPThreeDSLabelCustomizationTest.swift */; }; + C3C01719939575E1519EEC2C /* STPCardBrandTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 4866C83CD6596D781A053636 /* STPCardBrandTest.m */; }; + C4C1295E7DA618DFB944A534 /* NSString+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 63F5F35DB97D8A176FB6ED24 /* NSString+StripeTest.swift */; }; + C4DC3F4FA93A3BAF6EE782A0 /* PKPayment+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = B4D12508C2F1056A7EAFEC86 /* PKPayment+StripeTest.swift */; }; + C57BDA835AED735321906977 /* STPCardExpiryInputTextFieldValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4418164D75002AE6A0273176 /* STPCardExpiryInputTextFieldValidatorTests.swift */; }; + C5D295FE9988CA80ABA57801 /* RotatingCardBrandsViewTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 12632E6710DE8861CAF1BAA4 /* RotatingCardBrandsViewTests.swift */; }; + C5EAA26F7A4C71DCB08015A5 /* stp_bank_fpx_cimb@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 1FD3467084C9B5F7B4C15364 /* stp_bank_fpx_cimb@3x.png */; }; + C71B55CAA87D9B20B68E0608 /* PaymentMethodMessagingViewFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = D8099EC47BB1527E2B9A2446 /* PaymentMethodMessagingViewFunctionalTest.swift */; }; + C7EB8FB325BF491FDE25FE66 /* STPPaymentMethodEPSTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0C44B4366D6C4FD4B11662C8 /* STPPaymentMethodEPSTests.swift */; }; + C83D461C72BE7E4A309C8452 /* STPSetupIntentLastSetupErrorTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6C0F5E7908FE3C7606DEAAD5 /* STPSetupIntentLastSetupErrorTest.m */; }; + C861BB9EAAD04949E338D7FF /* PaymentTypeCellSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B9A4A2B0FB9F8C743BBED48 /* PaymentTypeCellSnapshotTests.swift */; }; + C87E17137F8E2CBA52B43A98 /* stp_bank_fpx_maybank2e@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = DA581592426609EDBE9B1CF4 /* stp_bank_fpx_maybank2e@3x.png */; }; + C8A6CA6352B7C8FEE3D91476 /* UIView+Helpers.swift in Sources */ = {isa = PBXBuildFile; fileRef = F0293B82D7078EE11F9B5639 /* UIView+Helpers.swift */; }; + C9E66A22494C02050AE34A9B /* FBSnapshotTestCase+STPViewControllerLoading.swift in Sources */ = {isa = PBXBuildFile; fileRef = 180CF848E3ABF0236C494D8B /* FBSnapshotTestCase+STPViewControllerLoading.swift */; }; + CA189278AD606BEAC62D545F /* STPPaymentIntentParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = CF902DC49DD90860BD0E5E80 /* STPPaymentIntentParamsTest.swift */; }; + CA4F392070740C56FE2BB461 /* STPStringUtilsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = EB6AE83989B0596F0C111E13 /* STPStringUtilsTest.swift */; }; + CB5AADE45B7B7A40514C054B /* StripeApplePay.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 52F8AEC50D4623F80F04A533 /* StripeApplePay.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + CBAF9C6F87F746F17495ADC2 /* STPPaymentMethodCashAppTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7DDE50CBC86AD77084C877B6 /* STPPaymentMethodCashAppTests.swift */; }; + CBCA59D39B30D869B4FDC04B /* STPE2ETest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1C1548BA518F7AC2A9ECF9D5 /* STPE2ETest.swift */; }; + CC072EBAD035AA54A2AD3ABC /* UIViewController+Stripe_KeyboardAvoiding.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2F08757CA6F6B2DA65C14E0A /* UIViewController+Stripe_KeyboardAvoiding.swift */; }; + CDBF169718F4882A946BF22D /* stp_icon_add@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = EC5115C4E72B99EE743139F1 /* stp_icon_add@3x.png */; }; + CEE483EB7B06B3C607BC755C /* UINavigationBar+StripeTest.m in Sources */ = {isa = PBXBuildFile; fileRef = E68F6B90F3BC61A49570FAF4 /* UINavigationBar+StripeTest.m */; }; + CEF318C74D2E44C78EF85306 /* STPBankSelectionTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 458F8576215E0F8ECE1D74CE /* STPBankSelectionTableViewCell.swift */; }; + CF2E17AC77EB08393B8A3F98 /* STPCardNumberInputTextFieldFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = D3803D0DED98501AA26B2EAC /* STPCardNumberInputTextFieldFormatterTests.swift */; }; + CF6AB51AA76503F2EDD42BED /* STPPaymentMethodGiropayParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 2AE3A8CA366854F49C28BD93 /* STPPaymentMethodGiropayParamsTests.m */; }; + CFDD431A9A8A82BAA11AE5BF /* Stripe3DS2.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 33FDC634FD5D79E824240DDC /* Stripe3DS2.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + D0342D50F9AC319919D93D59 /* STPBECSDebitAccountNumberValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 23997D61DF41CA84BFC33080 /* STPBECSDebitAccountNumberValidatorTests.swift */; }; + D0C81317E0AA8EB0370B1BA1 /* LinkLegalTermsViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 835CB781FBC19773ACC20676 /* LinkLegalTermsViewSnapshotTests.swift */; }; + D15160C0F0763078DBB434E4 /* STPCardTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1D575C31524E596E9C1A8E9B /* STPCardTest.swift */; }; + D151C8724925DCBA4BA4F46A /* StripeCoreTestUtils.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 512A0E7C246D5F044245E069 /* StripeCoreTestUtils.framework */; }; + D2869246B446B8B31F1CD368 /* STPIntentActionLinkAuthenticateAccount.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9EB24EC81CE2C8D1C863B044 /* STPIntentActionLinkAuthenticateAccount.swift */; }; + D2C062CE4E54094B1AC33E78 /* STPCardCVCInputTextFieldValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = D42F83F785EAF24F5DC7ED1A /* STPCardCVCInputTextFieldValidatorTests.swift */; }; + D2F19E21D5EC275A3CF23F7C /* STPTokenTest.m in Sources */ = {isa = PBXBuildFile; fileRef = A14673FA54CB8B85CF8D71C3 /* STPTokenTest.m */; }; + D375ADBD1F4B48380D5347D1 /* CircularButtonSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 46AC0B5EC7433E081825D31B /* CircularButtonSnapshotTests.swift */; }; + D4362E8B9ACB60F1DCDEDBCF /* STPPaymentMethodSEPADebitTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 8F4D19337AD2936DA69EAC4A /* STPPaymentMethodSEPADebitTest.m */; }; + D567569568C0D8F2D7B179B3 /* STPPaymentMethodAUBECSDebitTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 54426CBF6F77ABEFBDFDA8C4 /* STPPaymentMethodAUBECSDebitTests.swift */; }; + D597AF40C2DFEE3321566DD0 /* STPPaymentMethodEPSParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 28B8ADA1C10FED0733B4FAD3 /* STPPaymentMethodEPSParamsTests.m */; }; + D5BD5B1657C9D0FD7F6593F3 /* stp_shipping_form@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 21BC7D5A0DDD653571B76168 /* stp_shipping_form@3x.png */; }; + D5EECD0F9CD1DBC927E3E4E2 /* STPPaymentMethodCardWalletTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 0DF76154CA9CACA9CDB0BA94 /* STPPaymentMethodCardWalletTest.m */; }; + D73B7A0C24EDCA415FFBBB18 /* StripeUICore.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = D794C5E6396B4A19DC4F6921 /* StripeUICore.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + D7956073A8FD3785193E0577 /* STPStackViewWithSeparatorSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 51E62BB62EA9B782778CA880 /* STPStackViewWithSeparatorSnapshotTests.swift */; }; + D7C555B36C282B99E22B8D45 /* STPInputTextFieldFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = A22E5B87755C1F05C3DB438C /* STPInputTextFieldFormatterTests.swift */; }; + D7D24DCC9402153965AF7F1B /* STPPaymentMethodPrzelewy24Tests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F6558C62376C2397030BD4A6 /* STPPaymentMethodPrzelewy24Tests.swift */; }; + D83F76F584BC345CFBA71CF8 /* OneTimeCodeTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9D85FA7B714BDD8D1FD83B75 /* OneTimeCodeTextFieldSnapshotTests.swift */; }; + D8BECFB70834CC42BA6706D8 /* STPSourceParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1F6CB4B8FAD14B4D70A63595 /* STPSourceParamsTest.swift */; }; + D8C7D8B5749708833F89024B /* STPPaymentMethodAddressTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 1DECFBCFDD8FB9362D2FB4E1 /* STPPaymentMethodAddressTest.m */; }; + D8D9DF85FE4BF1AB14DBEB95 /* stp_bank_fpx_kfh@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 43398AFBACB8BA1DB29A437C /* stp_bank_fpx_kfh@3x.png */; }; + DC57D2DC40C6BA0C9CF7EC92 /* PayWithLinkButtonSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = AF342CBC167F9CAB5B49CC32 /* PayWithLinkButtonSnapshotTests.swift */; }; + DCF615643A22D0A7B739547C /* Stripe+Exports.swift in Sources */ = {isa = PBXBuildFile; fileRef = B70DF0B659009041F485EE0F /* Stripe+Exports.swift */; }; + DD16FC7ABCA7817794ECC407 /* STPThreeDSSelectionCustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C23D612FD5AD7772E1B30DCC /* STPThreeDSSelectionCustomizationTest.swift */; }; + DD8E2B99BAE917F83258DC35 /* STPPaymentMethodOptionsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1E2638F7AA0906914117C2D5 /* STPPaymentMethodOptionsTest.swift */; }; + DDBF5AAE607C698618DDE865 /* STPCameraView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 61F8308B7250B642D19827D8 /* STPCameraView.swift */; }; + DE23FEF74E860620A334FDF5 /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 884C01B087B4D820395BD374 /* Main.storyboard */; }; + DEC21E8B6DD667AD876ABA0E /* STPPaymentMethodBillingDetailsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = A00F4D8DB80EBD753E9AC961 /* STPPaymentMethodBillingDetailsTest.m */; }; + DF73457BF349BC962A6AC502 /* STPCoreScrollViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2D1525AF65BDEF691F8BCBE8 /* STPCoreScrollViewController.swift */; }; + DF7C5EA613A5B8F5D1431622 /* STPConnectAccountParamsTest.m in Sources */ = {isa = PBXBuildFile; fileRef = DF470B57E61F36BD1C0A1224 /* STPConnectAccountParamsTest.m */; }; + DF85F5EC6E16CAD21491891A /* AnalyticsHelperTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 61AF6E95FE0DD913204CAB32 /* AnalyticsHelperTests.swift */; }; + E1EA6387717F8ACD3637AA78 /* STPBankAccountTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6FA025CE30EB3DA1C4321505 /* STPBankAccountTest.m */; }; + E2FECEC25C1C69E4969B0E2A /* STPSourceOwnerTest.m in Sources */ = {isa = PBXBuildFile; fileRef = D1681629DCEA0765515D3D49 /* STPSourceOwnerTest.m */; }; + E32AB1FF0BFE8979650FDF15 /* STPSourceReceiverTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 620CEE3CCA05E49698C4DE35 /* STPSourceReceiverTest.m */; }; + E3F1BAD22CC6E90B761B0502 /* STPTextFieldDelegateProxyTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2A8A2CD759D465290066EF65 /* STPTextFieldDelegateProxyTests.swift */; }; + E4BA3AF73442897BCA3A6962 /* recorded_network_traffic in Resources */ = {isa = PBXBuildFile; fileRef = FCA64B2C21FCFAC433EA3781 /* recorded_network_traffic */; }; + E611EAD0DBDBAE1AD3442CDC /* stp_bank_fpx_ocbc@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 7EB4A04665E4B0FBE6BA24F6 /* stp_bank_fpx_ocbc@3x.png */; }; + E63B5BAF6B5645C979BFBA71 /* STPAddress+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 498C3FB07CFD532779C755D3 /* STPAddress+BasicUI.swift */; }; + E6F428CFAD64979A8874B00B /* STPAnalyticsClientPaymentsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E7ACB4FAFAD33296DE34D036 /* STPAnalyticsClientPaymentsTest.swift */; }; + E7071A87146B640C2DEE6B80 /* stp_bank_fpx_alliance_bank@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 1F6237EC7D3A866814A7DC88 /* stp_bank_fpx_alliance_bank@3x.png */; }; + E7DE88AF9BDC4EB9E6DE73F1 /* EphemeralKey.json in Resources */ = {isa = PBXBuildFile; fileRef = F8CEB050CB2231F1459824EC /* EphemeralKey.json */; }; + E97168F37D769524B58461B6 /* StripeCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 43B4E4B85C598D7A9AFCB4D4 /* StripeCore.framework */; }; + EA34719659CB9F1A269FECC7 /* StripeUICore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = D794C5E6396B4A19DC4F6921 /* StripeUICore.framework */; }; + EA571ECEFDDF10AF87CE2B74 /* STPThreeDSFooterCustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 483B243268646AE65B06E98C /* STPThreeDSFooterCustomizationTest.swift */; }; + EA7FEC518AA07BA59405A5E3 /* STPPaymentIntentParams+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6F01150CD0255164FE2CF3A4 /* STPPaymentIntentParams+BasicUI.swift */; }; + EBD436689635CC28A24DECD4 /* STPPinManagementServiceFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 336CC555B845DED30208D39D /* STPPinManagementServiceFunctionalTest.swift */; }; + EE4E023ABD6C611BA3EDDD17 /* STPPaymentMethodCardChecksTest.m in Sources */ = {isa = PBXBuildFile; fileRef = E0A27E1D77216CE0267C981C /* STPPaymentMethodCardChecksTest.m */; }; + EEA502DF8809B8FD0D00785E /* StripePayments.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 22D1C6EB5826E2D7C80B6CF3 /* StripePayments.framework */; }; + EEB5E5E9C4E06B148A91C7BD /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 6955B3A3353F8442E4FBBBF6 /* Assets.xcassets */; }; + EEBA9A95E8057A06E5E7C103 /* STPCardNumberInputTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FD289E1EA9F0CE1C848AC0BB /* STPCardNumberInputTextFieldSnapshotTests.swift */; }; + EEC27B2210C2353DDDDEEAD1 /* SofortSource.json in Resources */ = {isa = PBXBuildFile; fileRef = E412BB731BC34BB1A7EC3B8D /* SofortSource.json */; }; + EEFFE199D9769FF449BFD7FF /* STPCoreTableViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = B78C72B0DB434EC7F700FDE0 /* STPCoreTableViewController.swift */; }; + F10FC337254A34ED8F13E341 /* STPPaymentOption.swift in Sources */ = {isa = PBXBuildFile; fileRef = 30B694A39D54886392AA5DE3 /* STPPaymentOption.swift */; }; + F228C86E3F2172FE6A349243 /* stp_icon_checkmark@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = A95FC67D6E0C11B6FE40DC92 /* stp_icon_checkmark@3x.png */; }; + F2655328479314A9C8718DE4 /* STPApplePayContextTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5F7AB40A5A10C2D267323ABE /* STPApplePayContextTest.swift */; }; + F30F6DA35482988D8EC9FCEB /* STPApplePayContextFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = E3D73CE47141F7D1E44283D2 /* STPApplePayContextFunctionalTest.m */; }; + F35E090A607EB5F86FFC3D31 /* STPCardCVCInputTextFieldTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94A7104C1C470515616E4D2B /* STPCardCVCInputTextFieldTests.swift */; }; + F3F2B3317A63067ECA26F3E2 /* stp_bank_fpx_standard_chartered@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 40E3C9AF9FA4576DAC027275 /* stp_bank_fpx_standard_chartered@3x.png */; }; + F481DAE25F9957D23F529CF7 /* STPNetworkStubbingTestCase.swift in Sources */ = {isa = PBXBuildFile; fileRef = B0D081A26B59DE51FDDD1798 /* STPNetworkStubbingTestCase.swift */; }; + F49D9C4030829D13A6EB45BE /* MockFiles in Resources */ = {isa = PBXBuildFile; fileRef = FE2DED6ABA7407C17C1391B6 /* MockFiles */; }; + F530ECEDE5FFDD9B4321CCD4 /* STPPaymentMethodBancontactParamsTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 130483A5043ACC4D60E0B0D6 /* STPPaymentMethodBancontactParamsTests.m */; }; + F53E04785DB804EA5C2AAC18 /* STPAddressViewModelTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = D3D2DB08C335695B705F544C /* STPAddressViewModelTest.swift */; }; + F5C771A7C98E78F99E13DBA1 /* stp_bank_fpx_ambank@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 89CA73D6080D5B68F6465BC1 /* stp_bank_fpx_ambank@3x.png */; }; + F5CC4F320D09A06F0B21ABE6 /* STPPaymentMethodAffirmTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1F0DF2ED9232A7CC51F5FCB1 /* STPPaymentMethodAffirmTests.swift */; }; + F729E784CFFC1F79EF5F2ABE /* STPPaymentIntentLastPaymentErrorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6C7B8DACB0A7294BC235E3BC /* STPPaymentIntentLastPaymentErrorTest.swift */; }; + F835CEC935464FF32726A0A0 /* STPTheme.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8E8CA4361964E1BA400EFC89 /* STPTheme.swift */; }; + F86F2DF6E46EFABE23AD5D27 /* STPApplePayContextDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = E452877E5D11120B1E28A6E7 /* STPApplePayContextDelegate.swift */; }; + F975CE029DF30419B8DB0D8F /* STPNumericStringValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4AA36705DED9164663A98B6A /* STPNumericStringValidatorTests.swift */; }; + FAD79E127796E97B3C1693FD /* STPSourceVerificationTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 5377FE4A168F45625923AF37 /* STPSourceVerificationTest.m */; }; + FB34906C9215D0E03850064B /* STPAddCardViewControllerTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8704ABFA91A5226847F4A69A /* STPAddCardViewControllerTest.swift */; }; + FBBA3B39598BBECB664C5E7F /* STPApplePayTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = CFC4BC1AB047ED88C4D13C89 /* STPApplePayTest.swift */; }; + FBBDAEA5DF5CB3A138D82E8A /* LinkCardEditElementSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7F9659E92E94316FE4955156 /* LinkCardEditElementSnapshotTests.swift */; }; + FC166455478EAF51F7C34E68 /* STPApplePayPaymentOption.swift in Sources */ = {isa = PBXBuildFile; fileRef = 03ACDC7EEC28D1FE50008F65 /* STPApplePayPaymentOption.swift */; }; + FDD1858CAEFCEBB22BEC9BBC /* MKPlacemark+PaymentSheetTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9DCCA0E8A02B4F4B23837FB4 /* MKPlacemark+PaymentSheetTests.swift */; }; + FDDA6674A050209FF8CB7E30 /* STPFixtures+Swift.swift in Sources */ = {isa = PBXBuildFile; fileRef = CFDBAD933B23E26CBA816018 /* STPFixtures+Swift.swift */; }; + FE6647242714D9BEA1EBC055 /* STPCardCVCInputTextFieldFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6E1F6514E7530C2A3478B2F5 /* STPCardCVCInputTextFieldFormatterTests.swift */; }; + FEE74744B657F86873EA2F3D /* STPPushProvisioningDetails.swift in Sources */ = {isa = PBXBuildFile; fileRef = 89E5DA3029F141B5111A5B2C /* STPPushProvisioningDetails.swift */; }; + FEF2E0DAC862FF42B814AFCA /* STPPaymentHandlerFunctionalTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 683F7735569D22CBEC9CA2E6 /* STPPaymentHandlerFunctionalTest.m */; }; + FF00A8AACAA255F89D9034A0 /* STPConnectAccountAddressTest.m in Sources */ = {isa = PBXBuildFile; fileRef = D82418EB84598A2CEE2DCB28 /* STPConnectAccountAddressTest.m */; }; + FF0F9BA6FE4B88297A434EA7 /* STPPaymentMethodNetBankingTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 739934737B9A09775CD278C9 /* STPPaymentMethodNetBankingTests.swift */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 32221E5BA07FB5AA4EBFE81C /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = E63832AA5BB4225708B7C838 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 1628E8B14F1F7C63CF8C9962; + remoteInfo = StripeiOSTestHostApp; + }; + 8F25D3DFD6D65DAE4B581911 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = E63832AA5BB4225708B7C838 /* Project object */; + proxyType = 1; + remoteGlobalIDString = ADF894AA8F6022D9BED17346; + remoteInfo = StripeiOS; + }; + D90CE98566BBDA0E7340E1D7 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = E63832AA5BB4225708B7C838 /* Project object */; + proxyType = 1; + remoteGlobalIDString = ADF894AA8F6022D9BED17346; + remoteInfo = StripeiOS; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 16FB342935F756BD2EA7CE4C /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + CFDD431A9A8A82BAA11AE5BF /* Stripe3DS2.framework in Embed Frameworks */, + CB5AADE45B7B7A40514C054B /* StripeApplePay.framework in Embed Frameworks */, + C035D82D7096E3005858848C /* StripeCore.framework in Embed Frameworks */, + 42395EF962DB8AD6A094630B /* StripePaymentSheet.framework in Embed Frameworks */, + BF4A92064926FD7B0E3E92F7 /* StripePayments.framework in Embed Frameworks */, + 5B6F1BF973FC2D8DD6127B7F /* StripePaymentsUI.framework in Embed Frameworks */, + D73B7A0C24EDCA415FFBBB18 /* StripeUICore.framework in Embed Frameworks */, + 162C101E57D66F0051164C4A /* Stripe.framework in Embed Frameworks */, + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + 5789BE2CD297329ED86678C0 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + 9F30C6D40956861F588C2CD0 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + CAAA5D4C8E87896995960E8C /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 002634603200AABECC9686B1 /* pt-BR */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pt-BR"; path = "pt-BR.lproj/Localizable.strings"; sourceTree = ""; }; + 005650A59D692F820EF20F5F /* XCTest.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = XCTest.framework; path = Platforms/iPhoneOS.platform/Developer/Library/Frameworks/XCTest.framework; sourceTree = DEVELOPER_DIR; }; + 01B42BE6FB5EC1F708875AB8 /* STPPaymentCardTextFieldCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentCardTextFieldCell.swift; sourceTree = ""; }; + 0227131370F51B0533585BEB /* stp_card_form_amex_cvc@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_card_form_amex_cvc@3x.png"; sourceTree = ""; }; + 0241DB84973B21393BEC703E /* STPAnalyticsClientPaymentSheetTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAnalyticsClientPaymentSheetTest.swift; sourceTree = ""; }; + 02DEC25F0E12F9F4396E87B2 /* STPPaymentMethodPayPalTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodPayPalTests.m; sourceTree = ""; }; + 02FC9ED423D40C88D5A24441 /* STPCoreViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCoreViewController.swift; sourceTree = ""; }; + 0320B47C1DD836F7A7144B58 /* STPApplePayPaymentOptionTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPApplePayPaymentOptionTest.m; sourceTree = ""; }; + 0364E74C0FB269E93D18966D /* SetupIntent.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = SetupIntent.json; sourceTree = ""; }; + 03ACDC7EEC28D1FE50008F65 /* STPApplePayPaymentOption.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayPaymentOption.swift; sourceTree = ""; }; + 03CEA8792C992D865ECC46AA /* STPPaymentMethodGrabPayParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodGrabPayParamsTest.m; sourceTree = ""; }; + 04838ACE779F5CC949C276CB /* fr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fr; path = fr.lproj/Localizable.strings; sourceTree = ""; }; + 04FE0C74090AE8A871CCE5EC /* mt */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = mt; path = mt.lproj/Localizable.strings; sourceTree = ""; }; + 05A3E36498CBA9E5FB8323C9 /* stp_bank_fpx_hsbc@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_hsbc@3x.png"; sourceTree = ""; }; + 064CFCA8FCEA9E4BAB3547D0 /* ja */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ja; path = ja.lproj/Localizable.strings; sourceTree = ""; }; + 0669B4CA326CE74D125C789C /* STPFakeAddPaymentPassViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFakeAddPaymentPassViewController.swift; sourceTree = ""; }; + 0841697DF701B4A4637E67D9 /* STPFileTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPFileTest.m; sourceTree = ""; }; + 084B1B9FCCF4AB727B4ECFB2 /* id */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = id; path = id.lproj/Localizable.strings; sourceTree = ""; }; + 090EF7D598B8DE779C275395 /* STPPostalCodeInputTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPostalCodeInputTextFieldSnapshotTests.swift; sourceTree = ""; }; + 0A16326394D71637A2CF68C3 /* fil */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fil; path = fil.lproj/Localizable.strings; sourceTree = ""; }; + 0A726D87E8C916E8FEA0C780 /* WeChatPaySource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = WeChatPaySource.json; sourceTree = ""; }; + 0ABB2CA7E96BE249CE8C0566 /* STPPaymentMethodCashAppParamsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodCashAppParamsTests.swift; sourceTree = ""; }; + 0B91C4D5B93FF71C61B140F1 /* STPCardScannerTableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardScannerTableViewCell.swift; sourceTree = ""; }; + 0B989830717894A60B74E830 /* STPConfirmPaymentMethodOptionsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPConfirmPaymentMethodOptionsTest.m; sourceTree = ""; }; + 0C44B4366D6C4FD4B11662C8 /* STPPaymentMethodEPSTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodEPSTests.swift; sourceTree = ""; }; + 0C75157665428685C7A4FD20 /* STPCardExpiryInputTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardExpiryInputTextFieldSnapshotTests.swift; sourceTree = ""; }; + 0C9D6F99E303A17A91101723 /* pl-PL */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pl-PL"; path = "pl-PL.lproj/Localizable.strings"; sourceTree = ""; }; + 0DB03E83746FE78361831546 /* WalletHeaderViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WalletHeaderViewSnapshotTests.swift; sourceTree = ""; }; + 0DF76154CA9CACA9CDB0BA94 /* STPPaymentMethodCardWalletTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodCardWalletTest.m; sourceTree = ""; }; + 0E10C4D64477638398251FFB /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 0F735744F27D46F005BB5D67 /* sk-SK */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "sk-SK"; path = "sk-SK.lproj/Localizable.strings"; sourceTree = ""; }; + 1007571188950D7FBF745A4E /* ko */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ko; path = ko.lproj/Localizable.strings; sourceTree = ""; }; + 1020F76D0722D685A97EF202 /* EPSSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = EPSSource.json; sourceTree = ""; }; + 10BAD33BEC6C2894F9266902 /* LinkSecureCookieStoreTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkSecureCookieStoreTests.swift; sourceTree = ""; }; + 121B7EDCCD0957C9A444A8E3 /* ru */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ru; path = ru.lproj/Localizable.strings; sourceTree = ""; }; + 12632E6710DE8861CAF1BAA4 /* RotatingCardBrandsViewTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RotatingCardBrandsViewTests.swift; sourceTree = ""; }; + 12D757B36030685C401A6990 /* et-EE */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "et-EE"; path = "et-EE.lproj/Localizable.strings"; sourceTree = ""; }; + 12DBB3F72AEFB52DE27C27ED /* STPAnalyticsClient+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAnalyticsClient+BasicUI.swift"; sourceTree = ""; }; + 130483A5043ACC4D60E0B0D6 /* STPPaymentMethodBancontactParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodBancontactParamsTests.m; sourceTree = ""; }; + 131B856C6C1053E5FACD44D1 /* STPNetworkStubbingTestCase.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STPNetworkStubbingTestCase.h; sourceTree = ""; }; + 13DDBEA7D444A8AC14E0F1C8 /* lv-LV */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "lv-LV"; path = "lv-LV.lproj/Localizable.strings"; sourceTree = ""; }; + 1443F134C35E15AB9A1EA560 /* STPPaymentMethodCardWalletMasterpassTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodCardWalletMasterpassTest.m; sourceTree = ""; }; + 148C1D7D1BBBC6B74894A869 /* STPPaymentMethodTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodTest.swift; sourceTree = ""; }; + 152E45642578F76AAEB1CF61 /* stp_bank_fpx_rhb@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_rhb@3x.png"; sourceTree = ""; }; + 153071C69A0BEE033E035DCF /* CardExpiryDateTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardExpiryDateTests.swift; sourceTree = ""; }; + 157826E456AC3ABFE8000B81 /* LinkNavigationBarSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkNavigationBarSnapshotTests.swift; sourceTree = ""; }; + 1671EC46C713D51013AD7D8B /* STPPaymentMethodCardParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodCardParamsTest.swift; sourceTree = ""; }; + 17DDEC7D4C1D5FF6488D71D5 /* STPFixtures.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPFixtures.m; sourceTree = ""; }; + 180CF848E3ABF0236C494D8B /* FBSnapshotTestCase+STPViewControllerLoading.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "FBSnapshotTestCase+STPViewControllerLoading.swift"; sourceTree = ""; }; + 18FCB69CD3B8C3DAB216A5F0 /* STPPaymentConfiguration.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentConfiguration.swift; sourceTree = ""; }; + 1A8A6B88797870BC71CCB3AF /* STPPushProvisioningDetailsFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPushProvisioningDetailsFunctionalTest.swift; sourceTree = ""; }; + 1B76DF0FE363F59BF0940A8B /* UIView+Stripe_FirstResponder.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIView+Stripe_FirstResponder.swift"; sourceTree = ""; }; + 1C1548BA518F7AC2A9ECF9D5 /* STPE2ETest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPE2ETest.swift; sourceTree = ""; }; + 1D23EB567F573612E0794B3A /* Stripe Tests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe Tests-Debug.xcconfig"; sourceTree = ""; }; + 1D51B04D83D4FEF7F90DF16A /* STPCertTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCertTest.swift; sourceTree = ""; }; + 1D575C31524E596E9C1A8E9B /* STPCardTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardTest.swift; sourceTree = ""; }; + 1D983E089196152DA1C69469 /* STPCardFormViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardFormViewSnapshotTests.swift; sourceTree = ""; }; + 1DD6897858F46976A946394E /* StripeiOSAppHostedTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = StripeiOSAppHostedTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 1DECFBCFDD8FB9362D2FB4E1 /* STPPaymentMethodAddressTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodAddressTest.m; sourceTree = ""; }; + 1E2638F7AA0906914117C2D5 /* STPPaymentMethodOptionsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodOptionsTest.swift; sourceTree = ""; }; + 1E8AFAE24610EC983727F860 /* STPAPIClient+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAPIClient+BasicUI.swift"; sourceTree = ""; }; + 1F0DF2ED9232A7CC51F5FCB1 /* STPPaymentMethodAffirmTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodAffirmTests.swift; sourceTree = ""; }; + 1F16C36797D978E72E612100 /* STPPaymentCardTextFieldTestsSwift.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentCardTextFieldTestsSwift.swift; sourceTree = ""; }; + 1F29C15B47C7CB0941CD4C9E /* ViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ViewController.swift; sourceTree = ""; }; + 1F6237EC7D3A866814A7DC88 /* stp_bank_fpx_alliance_bank@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_alliance_bank@3x.png"; sourceTree = ""; }; + 1F6CB4B8FAD14B4D70A63595 /* STPSourceParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSourceParamsTest.swift; sourceTree = ""; }; + 1FD3467084C9B5F7B4C15364 /* stp_bank_fpx_cimb@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_cimb@3x.png"; sourceTree = ""; }; + 1FFBAA4B44967B157A4F4E91 /* STPPaymentActivityIndicatorView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentActivityIndicatorView.swift; sourceTree = ""; }; + 20552E792B8E7BA15821AB5D /* NSDecimalNumber+StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "NSDecimalNumber+StripeTest.swift"; sourceTree = ""; }; + 20879436DCFB1F03BE1608B3 /* ServerErrorMapperTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ServerErrorMapperTest.swift; sourceTree = ""; }; + 20A74D4FA51D15AF12B58767 /* STPCardParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPCardParamsTest.m; sourceTree = ""; }; + 20D90EE8CC6606CAE9B06C63 /* STPSourceFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSourceFunctionalTest.m; sourceTree = ""; }; + 21BC7D5A0DDD653571B76168 /* stp_shipping_form@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_shipping_form@3x.png"; sourceTree = ""; }; + 21E4B84223DDA131544DBBA7 /* STPShippingMethodsViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingMethodsViewController.swift; sourceTree = ""; }; + 22D1C6EB5826E2D7C80B6CF3 /* StripePayments.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripePayments.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 23997D61DF41CA84BFC33080 /* STPBECSDebitAccountNumberValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBECSDebitAccountNumberValidatorTests.swift; sourceTree = ""; }; + 24AE1B8096363F6C51A5AF7E /* LinkVerificationViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkVerificationViewSnapshotTests.swift; sourceTree = ""; }; + 252FD94C68761E893E53D5F9 /* SEPADebitSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = SEPADebitSource.json; sourceTree = ""; }; + 2560B4EEE60D40FB31B8552F /* de */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = de; path = de.lproj/Localizable.strings; sourceTree = ""; }; + 26D0BC36C2610307A01C52CB /* STPPaymentMethodAfterpayClearpayTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodAfterpayClearpayTest.m; sourceTree = ""; }; + 26E70469F4032553B4BB62DA /* ca-ES */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ca-ES"; path = "ca-ES.lproj/Localizable.strings"; sourceTree = ""; }; + 273EE407039913F0B644172B /* PKAddPaymentPassRequest+Stripe_Error.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKAddPaymentPassRequest+Stripe_Error.swift"; sourceTree = ""; }; + 27A4D83C0E7D4AE0CCD38B89 /* da */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = da; path = da.lproj/Localizable.strings; sourceTree = ""; }; + 284C67269D2606DA147AE01D /* STPGenericInputPickerFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPGenericInputPickerFieldSnapshotTests.swift; sourceTree = ""; }; + 28A50AFA4603E488FF3D82D0 /* STPAddressViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddressViewModel.swift; sourceTree = ""; }; + 28B8ADA1C10FED0733B4FAD3 /* STPPaymentMethodEPSParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodEPSParamsTests.m; sourceTree = ""; }; + 294CD46E24BB2743042872D7 /* StripeiOSTestHostApp.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = StripeiOSTestHostApp.app; sourceTree = BUILT_PRODUCTS_DIR; }; + 2A0E991FB88CBE8C9135DFE0 /* stp_card_form_back@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_card_form_back@3x.png"; sourceTree = ""; }; + 2A588999859430CC959C6F4B /* stp_bank_fpx_affin_bank@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_affin_bank@3x.png"; sourceTree = ""; }; + 2A8A2CD759D465290066EF65 /* STPTextFieldDelegateProxyTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPTextFieldDelegateProxyTests.swift; sourceTree = ""; }; + 2AE3A8CA366854F49C28BD93 /* STPPaymentMethodGiropayParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodGiropayParamsTests.m; sourceTree = ""; }; + 2B28B8A547CD846277ECD578 /* STPPushProvisioningDetailsParams.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPushProvisioningDetailsParams.swift; sourceTree = ""; }; + 2C078573F46762353664AC92 /* UINavigationController+Stripe_Completion.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UINavigationController+Stripe_Completion.swift"; sourceTree = ""; }; + 2CC4B06AB5C02FF54091E5A8 /* STPCustomerContextTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCustomerContextTest.swift; sourceTree = ""; }; + 2D1525AF65BDEF691F8BCBE8 /* STPCoreScrollViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCoreScrollViewController.swift; sourceTree = ""; }; + 2D63B73C5773432CA134D1FC /* STPGenericInputPickerFieldValidatorTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPGenericInputPickerFieldValidatorTest.swift; sourceTree = ""; }; + 2D878F923A1F69B58D6B2812 /* STPRadarSessionFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPRadarSessionFunctionalTest.swift; sourceTree = ""; }; + 2E1862744F23286D1FB9D4AE /* STPFormTextFieldTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFormTextFieldTest.swift; sourceTree = ""; }; + 2EADE1EE40283DEF52B0B5D4 /* BacsDebitPaymentMethod.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = BacsDebitPaymentMethod.json; sourceTree = ""; }; + 2F08757CA6F6B2DA65C14E0A /* UIViewController+Stripe_KeyboardAvoiding.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIViewController+Stripe_KeyboardAvoiding.swift"; sourceTree = ""; }; + 3024CAAF54280B0357014D39 /* LinkPaymentMethodPickerSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkPaymentMethodPickerSnapshotTests.swift; sourceTree = ""; }; + 30964128998473CAA9F2DD7E /* STPFormEncoderTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFormEncoderTest.swift; sourceTree = ""; }; + 30B694A39D54886392AA5DE3 /* STPPaymentOption.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOption.swift; sourceTree = ""; }; + 313263D9DC0629C5EE279FEB /* tr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = tr; path = tr.lproj/Localizable.strings; sourceTree = ""; }; + 31F32B2852B1683819BD02F3 /* BancontactSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = BancontactSource.json; sourceTree = ""; }; + 336CC555B845DED30208D39D /* STPPinManagementServiceFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPinManagementServiceFunctionalTest.swift; sourceTree = ""; }; + 33FDC634FD5D79E824240DDC /* Stripe3DS2.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Stripe3DS2.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 342398F7CB61693AAADE3F26 /* STPSourceSEPADebitDetailsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSourceSEPADebitDetailsTest.m; sourceTree = ""; }; + 344C5B4D8789FEFA4CFC0E43 /* CardPaymentMethod.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = CardPaymentMethod.json; sourceTree = ""; }; + 34D8B80455B75885088E9DAC /* stp_bank_fpx_bank_rakyat@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_bank_rakyat@3x.png"; sourceTree = ""; }; + 35ABE696542469B79A9D52E6 /* tk */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = tk; path = tk.lproj/Localizable.strings; sourceTree = ""; }; + 35C1E9B0EE03825DABF6471A /* STPPaymentMethod+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPPaymentMethod+BasicUI.swift"; sourceTree = ""; }; + 382A01724CE8659A5210073F /* LinkNoticeViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkNoticeViewSnapshotTests.swift; sourceTree = ""; }; + 3871630C99B378C67B6E9F7C /* STPPaymentMethodSofortParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodSofortParamsTests.m; sourceTree = ""; }; + 3B0E131538728BC4802627B1 /* UserDefaults+StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UserDefaults+StripeTest.swift"; sourceTree = ""; }; + 3B112FFF3FCA82094281493F /* STPPaymentMethodUSBankAccountTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodUSBankAccountTest.swift; sourceTree = ""; }; + 3B3000668A75E095B514241F /* UIBarButtonItem+Stripe.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIBarButtonItem+Stripe.swift"; sourceTree = ""; }; + 3BCC4BB345F320EEECE0437C /* STPSetupIntentTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSetupIntentTest.m; sourceTree = ""; }; + 3C742844915B96CFD25BFFF9 /* AfterpayPriceBreakdownViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AfterpayPriceBreakdownViewSnapshotTests.swift; sourceTree = ""; }; + 3C77C7BC4BA57EC296CF2F1C /* STPApplePayFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayFunctionalTest.swift; sourceTree = ""; }; + 3C995125252BED1EEC018B9D /* STPPaymentContextApplePayTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentContextApplePayTest.swift; sourceTree = ""; }; + 3E1C5E08678292561255B1C5 /* STPCardBINMetadataTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardBINMetadataTests.swift; sourceTree = ""; }; + 3E5FB20B2BEFC00D54FDD87D /* STPCard+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPCard+BasicUI.swift"; sourceTree = ""; }; + 3EBB07171F6FDCE6E20C454A /* STPPinManagementService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPinManagementService.swift; sourceTree = ""; }; + 3ED44491EB0AC72B1B1A773C /* STPThreeDSUICustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSUICustomizationTest.swift; sourceTree = ""; }; + 3F31C891BDDD871F48F6E60B /* FileUpload.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = FileUpload.json; sourceTree = ""; }; + 3FC0560A312147C37CFE6CF9 /* STPBinRangeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBinRangeTest.swift; sourceTree = ""; }; + 4077600B9B3C1ABFF38383BE /* vi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = vi; path = vi.lproj/Localizable.strings; sourceTree = ""; }; + 40BB87E28719FE0C6B946BB5 /* STPCardExpiryInputTextFieldFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardExpiryInputTextFieldFormatterTests.swift; sourceTree = ""; }; + 40DC59F285B3AB0F60B33443 /* SWHttpTrafficRecorder.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = SWHttpTrafficRecorder.m; sourceTree = ""; }; + 40E3C9AF9FA4576DAC027275 /* stp_bank_fpx_standard_chartered@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_standard_chartered@3x.png"; sourceTree = ""; }; + 415554EBF13241D12094103B /* STPPaymentMethodiDEALTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodiDEALTest.m; sourceTree = ""; }; + 4259421D2CD26E37B96F97B2 /* Stripe.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Stripe.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 43398AFBACB8BA1DB29A437C /* stp_bank_fpx_kfh@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_kfh@3x.png"; sourceTree = ""; }; + 43ADFC4EF612D7C4A46E81B9 /* lt-LT */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "lt-LT"; path = "lt-LT.lproj/Localizable.strings"; sourceTree = ""; }; + 43B4E4B85C598D7A9AFCB4D4 /* StripeCore.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCore.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 4418164D75002AE6A0273176 /* STPCardExpiryInputTextFieldValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardExpiryInputTextFieldValidatorTests.swift; sourceTree = ""; }; + 444A6B234A4BD3CB2C1C977D /* STPConfirmCardOptionsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPConfirmCardOptionsTest.m; sourceTree = ""; }; + 458F8576215E0F8ECE1D74CE /* STPBankSelectionTableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBankSelectionTableViewCell.swift; sourceTree = ""; }; + 45BBD120FDBA434BC7E64435 /* STPPaymentMethodFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodFunctionalTest.m; sourceTree = ""; }; + 45FF7A07CFC3B9B7AD6B49EE /* FraudDetectionDataTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FraudDetectionDataTest.swift; sourceTree = ""; }; + 46AC0B5EC7433E081825D31B /* CircularButtonSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CircularButtonSnapshotTests.swift; sourceTree = ""; }; + 46C31CAD0FA74B58BA2B8530 /* STPPaymentIntentEnumsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentIntentEnumsTest.swift; sourceTree = ""; }; + 47A1AA1C80AE910FCFF58996 /* STPApplePayTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPApplePayTest.m; sourceTree = ""; }; + 47E12A0CBFA259A032F7AF0C /* STPPaymentMethodKlarnaParamsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodKlarnaParamsTests.swift; sourceTree = ""; }; + 482693A3D9A527B0E671F757 /* nb */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = nb; path = nb.lproj/Localizable.strings; sourceTree = ""; }; + 483B243268646AE65B06E98C /* STPThreeDSFooterCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSFooterCustomizationTest.swift; sourceTree = ""; }; + 485E747DA1F72F091986787B /* STPEphemeralKeyProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPEphemeralKeyProvider.swift; sourceTree = ""; }; + 4866C83CD6596D781A053636 /* STPCardBrandTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPCardBrandTest.m; sourceTree = ""; }; + 4928F057B093B6DB5D76D32D /* Customer.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = Customer.json; sourceTree = ""; }; + 498C3FB07CFD532779C755D3 /* STPAddress+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAddress+BasicUI.swift"; sourceTree = ""; }; + 49AA313E068FB99CEAA5F7D3 /* STPSetupIntentFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSetupIntentFunctionalTest.swift; sourceTree = ""; }; + 4A57C346FA98A057AF31C8B1 /* STPAPISettingsBridgeTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPAPISettingsBridgeTest.m; sourceTree = ""; }; + 4AA36705DED9164663A98B6A /* STPNumericStringValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPNumericStringValidatorTests.swift; sourceTree = ""; }; + 4AAE5EE11611F9F7762B64C6 /* STPAUBECSFormViewModelTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAUBECSFormViewModelTests.swift; sourceTree = ""; }; + 4CA5D11C977A95B8E936E907 /* STPIntentActionWeChatPayRedirectToAppTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPIntentActionWeChatPayRedirectToAppTest.swift; sourceTree = ""; }; + 4DFDFC019C67AFF7C0F7630B /* zh-Hant */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-Hant"; path = "zh-Hant.lproj/Localizable.strings"; sourceTree = ""; }; + 4E29B46F2C940E0A21734E09 /* StripeiOSTests.xctestplan */ = {isa = PBXFileReference; path = StripeiOSTests.xctestplan; sourceTree = ""; }; + 4E371E9B3B2E343FE954531C /* STPPaymentMethodBoletoTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodBoletoTests.swift; sourceTree = ""; }; + 4FD94FF270165D699DA89B24 /* STPPaymentMethodKlarnaTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodKlarnaTests.swift; sourceTree = ""; }; + 4FFA8B446217CDE678D7287F /* STPBlocks.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STPBlocks.h; sourceTree = ""; }; + 512A0E7C246D5F044245E069 /* StripeCoreTestUtils.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCoreTestUtils.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 51408DE266D0345784ADD4FA /* STPThreeDSNavigationBarCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSNavigationBarCustomizationTest.swift; sourceTree = ""; }; + 51BD2CE41E4F0CF648F44E4A /* TextFieldElement+IBANTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "TextFieldElement+IBANTest.swift"; sourceTree = ""; }; + 51E62BB62EA9B782778CA880 /* STPStackViewWithSeparatorSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPStackViewWithSeparatorSnapshotTests.swift; sourceTree = ""; }; + 51F49F2466BBBE64799209B5 /* STPAPIClientNetworkBridgeTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPAPIClientNetworkBridgeTest.m; sourceTree = ""; }; + 52F8AEC50D4623F80F04A533 /* StripeApplePay.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeApplePay.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 52FA1AFD632CCE18C4315FC6 /* LinkStubs.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkStubs.swift; sourceTree = ""; }; + 5377FE4A168F45625923AF37 /* STPSourceVerificationTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSourceVerificationTest.m; sourceTree = ""; }; + 53C5AB22D6328E85A6DDF663 /* STPPaymentMethodParams+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPPaymentMethodParams+BasicUI.swift"; sourceTree = ""; }; + 5430F6704CFF793CBFAAF549 /* AddPaymentMethodViewControllerSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AddPaymentMethodViewControllerSnapshotTests.swift; sourceTree = ""; }; + 54426CBF6F77ABEFBDFDA8C4 /* STPPaymentMethodAUBECSDebitTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodAUBECSDebitTests.swift; sourceTree = ""; }; + 5476BD87E0480A93958F0328 /* UITableViewCell+Stripe_Borders.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UITableViewCell+Stripe_Borders.swift"; sourceTree = ""; }; + 547F890FCD6D6E8B0774E919 /* STPFileFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPFileFunctionalTest.m; sourceTree = ""; }; + 57852C506CC1A2B0F978D98B /* SWHttpTrafficRecorder.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = SWHttpTrafficRecorder.h; sourceTree = ""; }; + 5804C2B9C0704E386B3D25A4 /* UIViewController+Stripe_ParentViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIViewController+Stripe_ParentViewController.swift"; sourceTree = ""; }; + 58158E8A117299834246100F /* sv */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = sv; path = sv.lproj/Localizable.strings; sourceTree = ""; }; + 582C9C6F076A0FF1780C9769 /* CardSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = CardSource.json; sourceTree = ""; }; + 583DB466066B47C0F716E474 /* STPPaymentMethodGiropayTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodGiropayTests.swift; sourceTree = ""; }; + 588C260880FFC584A00A89F5 /* STPEphemeralKeyManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPEphemeralKeyManager.swift; sourceTree = ""; }; + 58A53F005EA8FDDAA66126BA /* STPGenericInputTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPGenericInputTextFieldSnapshotTests.swift; sourceTree = ""; }; + 5BB7453E4E1AD36D604A00E8 /* LinkToastSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkToastSnapshotTests.swift; sourceTree = ""; }; + 5C0462C8B2776183A2B03D51 /* stp_card_form_front@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_card_form_front@3x.png"; sourceTree = ""; }; + 5CE0ABE46E1C27A6AD8D8BD9 /* TextFieldElement+CardTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "TextFieldElement+CardTest.swift"; sourceTree = ""; }; + 5E4EA6394497D1BD57ED0032 /* Stripe Tests-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe Tests-Release.xcconfig"; sourceTree = ""; }; + 5F23996EC0E06D47B20316AC /* STPPaymentMethodOXXOParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodOXXOParamsTests.m; sourceTree = ""; }; + 5F7AB40A5A10C2D267323ABE /* STPApplePayContextTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayContextTest.swift; sourceTree = ""; }; + 5FDD4956E0A04D33F0856F31 /* STPThreeDSLabelCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSLabelCustomizationTest.swift; sourceTree = ""; }; + 60E21B0DB0D3B307040EA88E /* STPSetupIntentFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSetupIntentFunctionalTest.m; sourceTree = ""; }; + 6152335A2DCCBB5490B39BB0 /* STPBankAccountFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPBankAccountFunctionalTest.m; sourceTree = ""; }; + 618DE183886175AF23C4E668 /* sl-SI */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "sl-SI"; path = "sl-SI.lproj/Localizable.strings"; sourceTree = ""; }; + 61AF6E95FE0DD913204CAB32 /* AnalyticsHelperTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnalyticsHelperTests.swift; sourceTree = ""; }; + 61F8308B7250B642D19827D8 /* STPCameraView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCameraView.swift; sourceTree = ""; }; + 620CEE3CCA05E49698C4DE35 /* STPSourceReceiverTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSourceReceiverTest.m; sourceTree = ""; }; + 6215A9BF343775B1BD0F62AF /* STPPaymentOptionTableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionTableViewCell.swift; sourceTree = ""; }; + 62169E37BD48681A914FA4CB /* BankAccount.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = BankAccount.json; sourceTree = ""; }; + 6223E57D3A198F956A37ED89 /* STPNumericDigitInputTextFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPNumericDigitInputTextFormatterTests.swift; sourceTree = ""; }; + 63114D0EAAE2606732DF5AA0 /* STPSourceCardDetailsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSourceCardDetailsTest.swift; sourceTree = ""; }; + 63A16B33FCF0B351D4A60247 /* MultibancoSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = MultibancoSource.json; sourceTree = ""; }; + 63BDA70DB74745C5F457CF88 /* STPElementsSessionTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPElementsSessionTest.swift; sourceTree = ""; }; + 63F5F35DB97D8A176FB6ED24 /* NSString+StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "NSString+StripeTest.swift"; sourceTree = ""; }; + 65DCC9BDA647E58D3882C698 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 6618739767139C25C05B3631 /* STPPostalCodeInputTextFieldTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPostalCodeInputTextFieldTests.swift; sourceTree = ""; }; + 682296D57E9486BBCC6ED7F5 /* GiropaySource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = GiropaySource.json; sourceTree = ""; }; + 683F7735569D22CBEC9CA2E6 /* STPPaymentHandlerFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentHandlerFunctionalTest.m; sourceTree = ""; }; + 68778692A4A89E311FC9DEAE /* PaymentIntent.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = PaymentIntent.json; sourceTree = ""; }; + 6887F19BB9804BF45FD703FF /* STPPushProvisioningContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPushProvisioningContext.swift; sourceTree = ""; }; + 6955B3A3353F8442E4FBBBF6 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; + 69BD038947E8E2376A0D240B /* STPCardScanner.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardScanner.swift; sourceTree = ""; }; + 6A9E7B637A8747431B38FD1D /* STPCardValidationState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardValidationState.swift; sourceTree = ""; }; + 6B527265D186B66150D2787E /* STPStringUtilsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPStringUtilsTest.m; sourceTree = ""; }; + 6C0F5E7908FE3C7606DEAAD5 /* STPSetupIntentLastSetupErrorTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSetupIntentLastSetupErrorTest.m; sourceTree = ""; }; + 6C7094644EE056260D6F3B67 /* STPPaymentOptionsViewControllerLocalizationTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsViewControllerLocalizationTests.swift; sourceTree = ""; }; + 6C7B8DACB0A7294BC235E3BC /* STPPaymentIntentLastPaymentErrorTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentIntentLastPaymentErrorTest.swift; sourceTree = ""; }; + 6C8FE751333F580004BD72BA /* STPIntentWithPreferencesTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPIntentWithPreferencesTest.swift; sourceTree = ""; }; + 6CC0B1FC92A573AAEA4F4E94 /* STPSetupIntentConfirmParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSetupIntentConfirmParamsTest.swift; sourceTree = ""; }; + 6E1F6514E7530C2A3478B2F5 /* STPCardCVCInputTextFieldFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardCVCInputTextFieldFormatterTests.swift; sourceTree = ""; }; + 6F01150CD0255164FE2CF3A4 /* STPPaymentIntentParams+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPPaymentIntentParams+BasicUI.swift"; sourceTree = ""; }; + 6F017A08C7E633FB4297D274 /* UIViewController+Stripe_NavigationItemProxy.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIViewController+Stripe_NavigationItemProxy.swift"; sourceTree = ""; }; + 6F29142B51DC99E2258B4F60 /* LinkInstantDebitMandateViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkInstantDebitMandateViewSnapshotTests.swift; sourceTree = ""; }; + 6FA025CE30EB3DA1C4321505 /* STPBankAccountTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPBankAccountTest.m; sourceTree = ""; }; + 71711FC8E2FB66E52A5FDD9A /* STPShippingAddressViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingAddressViewController.swift; sourceTree = ""; }; + 71C175861E0F8882BCF85F78 /* STPPaymentMethodBacsDebitTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodBacsDebitTest.m; sourceTree = ""; }; + 725845FAFFAE11297BCDFD29 /* NSLocale+STPSwizzling.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSLocale+STPSwizzling.h"; sourceTree = ""; }; + 72903593DC432D01720DC9D9 /* STPAddressFieldTableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddressFieldTableViewCell.swift; sourceTree = ""; }; + 72AC063833D29BC984B008A8 /* STPTestUtils.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STPTestUtils.h; sourceTree = ""; }; + 72B1E4D9DA48CED66CD96308 /* LinkBadgeViewSnapshotTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkBadgeViewSnapshotTest.swift; sourceTree = ""; }; + 739934737B9A09775CD278C9 /* STPPaymentMethodNetBankingTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodNetBankingTests.swift; sourceTree = ""; }; + 74FDEF9F687C63BADFB96480 /* STPPaymentMethodUSBankAccountParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodUSBankAccountParamsTest.swift; sourceTree = ""; }; + 789D0B49B0788794739E3DD4 /* STPShippingAddressViewControllerTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingAddressViewControllerTest.swift; sourceTree = ""; }; + 78BA15D1F0844B1DA71A5348 /* STPAddCardViewControllerLocalizationTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddCardViewControllerLocalizationTests.swift; sourceTree = ""; }; + 79ABE6A14AF9D14103050876 /* STPPaymentConfigurationTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentConfigurationTest.m; sourceTree = ""; }; + 7AC0A18C441FCA394BEF6A3D /* STPPaymentMethodBillingDetailsTests+Link.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPPaymentMethodBillingDetailsTests+Link.swift"; sourceTree = ""; }; + 7B10B1A15063FEDBA4A59953 /* en */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = en; path = en.lproj/Localizable.strings; sourceTree = ""; }; + 7B4DEF058786BD9D3D5EE2FB /* STPPaymentMethodParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodParamsTest.m; sourceTree = ""; }; + 7B8ADF2EA2D5C4C90BCDDDD5 /* bg-BG */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "bg-BG"; path = "bg-BG.lproj/Localizable.strings"; sourceTree = ""; }; + 7B9A4A2B0FB9F8C743BBED48 /* PaymentTypeCellSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentTypeCellSnapshotTests.swift; sourceTree = ""; }; + 7C04AFC9CDE50D09D38A3232 /* FormSpecProviderTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FormSpecProviderTest.swift; sourceTree = ""; }; + 7CAE7444CEFE1D1EFB888996 /* hr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = hr; path = hr.lproj/Localizable.strings; sourceTree = ""; }; + 7D964D9E01627B419B4BD23C /* STPPaymentResult.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentResult.swift; sourceTree = ""; }; + 7DDE50CBC86AD77084C877B6 /* STPPaymentMethodCashAppTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodCashAppTests.swift; sourceTree = ""; }; + 7EB4A04665E4B0FBE6BA24F6 /* stp_bank_fpx_ocbc@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_ocbc@3x.png"; sourceTree = ""; }; + 7F68637E75142DCD46710796 /* STPLabeledMultiFormTextFieldViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPLabeledMultiFormTextFieldViewSnapshotTests.swift; sourceTree = ""; }; + 7F90C21B77335533395C4321 /* stp_bank_fpx_hong_leong_bank@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_hong_leong_bank@3x.png"; sourceTree = ""; }; + 7F9659E92E94316FE4955156 /* LinkCardEditElementSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkCardEditElementSnapshotTests.swift; sourceTree = ""; }; + 806124200E77795DCFC8418E /* ConfirmButtonSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ConfirmButtonSnapshotTests.swift; sourceTree = ""; }; + 807FF966F1DE05F3496B817B /* STPAPIClient+PushProvisioning.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAPIClient+PushProvisioning.swift"; sourceTree = ""; }; + 81352A0CBE46A59E6B1A712E /* STPBankSelectionViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBankSelectionViewController.swift; sourceTree = ""; }; + 81B23BF8256AA75F9903AC0C /* STPAddressTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPAddressTests.m; sourceTree = ""; }; + 835CB781FBC19773ACC20676 /* LinkLegalTermsViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkLegalTermsViewSnapshotTests.swift; sourceTree = ""; }; + 83707086116ED364BE3C1F2E /* PayWithLinkViewController-WalletViewModelTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PayWithLinkViewController-WalletViewModelTests.swift"; sourceTree = ""; }; + 83760BB83F6391E0CF66234D /* PaymentSheetFormFactorySnapshotTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentSheetFormFactorySnapshotTest.swift; sourceTree = ""; }; + 83A5A0074F69DE467F94ABFB /* LinkInlineSignupElementSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkInlineSignupElementSnapshotTests.swift; sourceTree = ""; }; + 848AAE69CBF96A057F734365 /* ApplePayPaymentMethod.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = ApplePayPaymentMethod.json; sourceTree = ""; }; + 85AAB72218409F85FE29E69E /* APIRequestTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = APIRequestTest.swift; sourceTree = ""; }; + 86673B13C7762666C774F124 /* ButtonLinkSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ButtonLinkSnapshotTests.swift; sourceTree = ""; }; + 86798C95A778362EF815B4C6 /* UIView+Stripe_SafeAreaBounds.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIView+Stripe_SafeAreaBounds.swift"; sourceTree = ""; }; + 86D961D8952114BE479E1EB7 /* NSLocale+STPSwizzling.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSLocale+STPSwizzling.m"; sourceTree = ""; }; + 8704ABFA91A5226847F4A69A /* STPAddCardViewControllerTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddCardViewControllerTest.swift; sourceTree = ""; }; + 87DB9F06FDC13DD3BFEAA27F /* stp_bank_fpx_bank_islam@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_bank_islam@3x.png"; sourceTree = ""; }; + 88AEABC15CCBB9EA393C175F /* STPMocks.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPMocks.m; sourceTree = ""; }; + 890660C21E3666CE7B82695B /* STPEphemeralKey.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPEphemeralKey.swift; sourceTree = ""; }; + 89CA73D6080D5B68F6465BC1 /* stp_bank_fpx_ambank@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_ambank@3x.png"; sourceTree = ""; }; + 89E5DA3029F141B5111A5B2C /* STPPushProvisioningDetails.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPushProvisioningDetails.swift; sourceTree = ""; }; + 8A7FBDBADC1EBC62FFC4A366 /* STPPaymentMethodAUBECSDebitParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodAUBECSDebitParamsTests.m; sourceTree = ""; }; + 8B5AC148E7D8B2EBAEDCB364 /* STPPaymentMethodThreeDSecureUsageTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodThreeDSecureUsageTest.m; sourceTree = ""; }; + 8BD02D8298877F10F2EF2A9D /* STPPaymentMethodCardTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodCardTest.swift; sourceTree = ""; }; + 8CE85C770AEEDBE4AEC93EAA /* Error+PaymentSheetTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Error+PaymentSheetTests.swift"; sourceTree = ""; }; + 8D49257A97E71A475A9F6E08 /* STPCountryPickerInputFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCountryPickerInputFieldSnapshotTests.swift; sourceTree = ""; }; + 8D866DCC403994A0D3CFB1D7 /* KlarnaHelperTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = KlarnaHelperTest.swift; sourceTree = ""; }; + 8E8CA4361964E1BA400EFC89 /* STPTheme.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPTheme.swift; sourceTree = ""; }; + 8EC1D194936D061E3255BCF4 /* stp_bank_fpx_uob@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_uob@3x.png"; sourceTree = ""; }; + 8ED737CB1253C3C5704B6C05 /* cs-CZ */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "cs-CZ"; path = "cs-CZ.lproj/Localizable.strings"; sourceTree = ""; }; + 8F4D19337AD2936DA69EAC4A /* STPPaymentMethodSEPADebitTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodSEPADebitTest.m; sourceTree = ""; }; + 8F7E3CE2105E4A39032CD919 /* Enums+CustomStringConvertible.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Enums+CustomStringConvertible.swift"; sourceTree = ""; }; + 90D0900C5FDBB7952BCF2C3A /* en-GB */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "en-GB"; path = "en-GB.lproj/Localizable.strings"; sourceTree = ""; }; + 917154477796779ECFA1334A /* STPPostalCodeInputTextFieldFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPostalCodeInputTextFieldFormatterTests.swift; sourceTree = ""; }; + 924E878428D15506711CA628 /* STPPhoneNumberValidatorTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPhoneNumberValidatorTest.swift; sourceTree = ""; }; + 9290B75648F2D23764FA71E9 /* STPSTPViewWithSeparatorSnapshotTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSTPViewWithSeparatorSnapshotTests.m; sourceTree = ""; }; + 940209E5D30E86E856016906 /* STPPaymentMethodUPITests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodUPITests.swift; sourceTree = ""; }; + 94042E0B406F6AA45593D154 /* STPCardFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPCardFunctionalTest.m; sourceTree = ""; }; + 9475002E030E98735A0FD2EC /* STPNetworkStubbingTestCase.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPNetworkStubbingTestCase.m; sourceTree = ""; }; + 94A7104C1C470515616E4D2B /* STPCardCVCInputTextFieldTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardCVCInputTextFieldTests.swift; sourceTree = ""; }; + 966BA151AA0E428D1DD8A97C /* STPFPXBankBrandTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPFPXBankBrandTest.m; sourceTree = ""; }; + 969E196AB597EEF68C38103E /* Project-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Release.xcconfig"; sourceTree = ""; }; + 96E1FED5CE5974C9C1162E93 /* STPSectionHeaderView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSectionHeaderView.swift; sourceTree = ""; }; + 98544B08552407D41D398C68 /* STPShippingMethodTableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingMethodTableViewCell.swift; sourceTree = ""; }; + 98F9CB667BC68767DFB5FACD /* OneTimeCodeTextFieldTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OneTimeCodeTextFieldTests.swift; sourceTree = ""; }; + 9B782E1D974A4C131E60E2BD /* STPBackendAPIAdapter.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBackendAPIAdapter.swift; sourceTree = ""; }; + 9BBCE3A905041A709E8F279A /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; + 9CE65B72986B6B404B036183 /* STPIntentActionTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPIntentActionTest.m; sourceTree = ""; }; + 9D3BBCE8C46A38D0E20DBF4E /* ms-MY */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ms-MY"; path = "ms-MY.lproj/Localizable.strings"; sourceTree = ""; }; + 9D73D989C9D488D820029FA1 /* PaymentSheet+APITest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PaymentSheet+APITest.swift"; sourceTree = ""; }; + 9D85FA7B714BDD8D1FD83B75 /* OneTimeCodeTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OneTimeCodeTextFieldSnapshotTests.swift; sourceTree = ""; }; + 9DCCA0E8A02B4F4B23837FB4 /* MKPlacemark+PaymentSheetTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "MKPlacemark+PaymentSheetTests.swift"; sourceTree = ""; }; + 9EAE9A2AE65771403CE57C11 /* STPErrorBridgeTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPErrorBridgeTest.m; sourceTree = ""; }; + 9EB24EC81CE2C8D1C863B044 /* STPIntentActionLinkAuthenticateAccount.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPIntentActionLinkAuthenticateAccount.swift; sourceTree = ""; }; + 9F304A4028D563EBD2A523C8 /* STPBankAccountParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPBankAccountParamsTest.m; sourceTree = ""; }; + 9F8F79CBFDC930A18998D7B1 /* STPTestUtils.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPTestUtils.m; sourceTree = ""; }; + 9FEE395C4DD0E0112AF3720C /* STPInputTextFieldValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPInputTextFieldValidatorTests.swift; sourceTree = ""; }; + A00F4D8DB80EBD753E9AC961 /* STPPaymentMethodBillingDetailsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodBillingDetailsTest.m; sourceTree = ""; }; + A1003B17EB955A29BC77BF01 /* stp_icon_bank@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_icon_bank@3x.png"; sourceTree = ""; }; + A1272F2E05A0E294DD9ECA26 /* STPApplePayContextFunctionalTestExtras.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayContextFunctionalTestExtras.swift; sourceTree = ""; }; + A14673FA54CB8B85CF8D71C3 /* STPTokenTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPTokenTest.m; sourceTree = ""; }; + A1C67AC5D415615E9F27D3E3 /* STPPaymentMethodBoletoParamsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodBoletoParamsTests.swift; sourceTree = ""; }; + A1C876DC7F3E31D7189506A8 /* STPPaymentContextAmountModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentContextAmountModel.swift; sourceTree = ""; }; + A1CF1C43EEE33FC0AF31A2BB /* PaymentSheetPaymentMethodTypeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentSheetPaymentMethodTypeTest.swift; sourceTree = ""; }; + A22E5B87755C1F05C3DB438C /* STPInputTextFieldFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPInputTextFieldFormatterTests.swift; sourceTree = ""; }; + A2922A32A754CFC9AB8B48AE /* STPPaymentOptionsViewControllerTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsViewControllerTest.swift; sourceTree = ""; }; + A32EE0B1FC16687797E7C9DA /* AlipaySource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = AlipaySource.json; sourceTree = ""; }; + A5398E1156E0BFEBBF56FD2F /* String+Localized.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "String+Localized.swift"; sourceTree = ""; }; + A583966A33DCDCF04322A592 /* STPThreeDSTextFieldCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSTextFieldCustomizationTest.swift; sourceTree = ""; }; + A61763BA2CCA86F9B8FD4F1F /* OperationDebouncerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OperationDebouncerTests.swift; sourceTree = ""; }; + A6F6634AD12771A9BB100DD3 /* es */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = es; path = es.lproj/Localizable.strings; sourceTree = ""; }; + A7B538B1223FBD7F409A4B29 /* stp_bank_fpx_bank_muamalat@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_bank_muamalat@3x.png"; sourceTree = ""; }; + A7E369CEC9F5B3758F78E88F /* UINavigationBar+Stripe_Theme.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UINavigationBar+Stripe_Theme.swift"; sourceTree = ""; }; + A8598727045C6268B57A5FC7 /* Stripe-umbrella.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Stripe-umbrella.h"; sourceTree = ""; }; + A8CF4EBC85BDC771DF2213C0 /* STPPaymentIntentFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentIntentFunctionalTest.m; sourceTree = ""; }; + A8F5797C145751AD55858B80 /* 3DSSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = 3DSSource.json; sourceTree = ""; }; + A95FC67D6E0C11B6FE40DC92 /* stp_icon_checkmark@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_icon_checkmark@3x.png"; sourceTree = ""; }; + A982DCED06CCAEBE2C8835F3 /* stp_bank_fpx_maybank2u@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_maybank2u@3x.png"; sourceTree = ""; }; + A9A1BB31C7B514984231125B /* STPPaymentOptionsViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsViewController.swift; sourceTree = ""; }; + A9B50EE8ABC2506A6397C056 /* P24Source.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = P24Source.json; sourceTree = ""; }; + A9F06DC32147A8ABCD709E79 /* AddressViewControllerSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AddressViewControllerSnapshotTests.swift; sourceTree = ""; }; + AA0FCB6E05CDF74CC1C42042 /* STPTestingAPIClient.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STPTestingAPIClient.h; sourceTree = ""; }; + AA3CA5B4B91838CC7ED4D5EB /* ro-RO */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ro-RO"; path = "ro-RO.lproj/Localizable.strings"; sourceTree = ""; }; + AAF368BCD5990EE5DC17D299 /* ImageTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageTest.swift; sourceTree = ""; }; + AB1A92C77AC61335184BBDBC /* es-419 */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "es-419"; path = "es-419.lproj/Localizable.strings"; sourceTree = ""; }; + ABAABB969BFB7102D5AA5598 /* it */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = it; path = it.lproj/Localizable.strings; sourceTree = ""; }; + ACE8998EAF997A78759E49B5 /* STPPaymentIntentTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentIntentTest.swift; sourceTree = ""; }; + AD06AED0AF8A9A7FB4A2E66F /* STPIntentActionAlipayHandleRedirectTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPIntentActionAlipayHandleRedirectTest.swift; sourceTree = ""; }; + AD8BED2A6066514B51693172 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; + AF342CBC167F9CAB5B49CC32 /* PayWithLinkButtonSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PayWithLinkButtonSnapshotTests.swift; sourceTree = ""; }; + AFB9F0B171497353AADD6544 /* STPPIIFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPIIFunctionalTest.m; sourceTree = ""; }; + AFF957F38AABE5F748C38C0B /* STPLocalizedString.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPLocalizedString.swift; sourceTree = ""; }; + B0D081A26B59DE51FDDD1798 /* STPNetworkStubbingTestCase.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPNetworkStubbingTestCase.swift; sourceTree = ""; }; + B1217AD643A9E8F88B60F645 /* STPUserInformation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPUserInformation.swift; sourceTree = ""; }; + B2C7ED5A39AB3BB2248CEACA /* stp_fpx_logo@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_fpx_logo@3x.png"; sourceTree = ""; }; + B334078D1C3E629BEB498BAB /* nn-NO */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "nn-NO"; path = "nn-NO.lproj/Localizable.strings"; sourceTree = ""; }; + B3E0745D13EB19BAA24F3BA3 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + B407FE2D39775902A95B1118 /* StripeiOS Tests-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "StripeiOS Tests-Bridging-Header.h"; sourceTree = ""; }; + B4D12508C2F1056A7EAFEC86 /* PKPayment+StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKPayment+StripeTest.swift"; sourceTree = ""; }; + B4D31B0D7BD9F97AF3BB61E6 /* StripeBundleLocator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StripeBundleLocator.swift; sourceTree = ""; }; + B5BA1ABFF1739E4051FAE6CB /* STPPaymentMethodCardWalletVisaCheckoutTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodCardWalletVisaCheckoutTest.m; sourceTree = ""; }; + B6799CC0D9C5FED26EE3494F /* STPSourceRedirectTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSourceRedirectTest.m; sourceTree = ""; }; + B6F3B966470A530E0DC53F8C /* STPThreeDSButtonCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSButtonCustomizationTest.swift; sourceTree = ""; }; + B70DF0B659009041F485EE0F /* Stripe+Exports.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Stripe+Exports.swift"; sourceTree = ""; }; + B7439CC6C0FB060EA312FF6D /* STPPaymentCardTextFieldTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentCardTextFieldTest.m; sourceTree = ""; }; + B78C72B0DB434EC7F700FDE0 /* STPCoreTableViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCoreTableViewController.swift; sourceTree = ""; }; + B7B5B626BCED2938C256BF49 /* STPSourceTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSourceTest.m; sourceTree = ""; }; + B7F7AA0B7B86BA5BB2FE92CE /* STPMandateDataParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPMandateDataParamsTest.swift; sourceTree = ""; }; + B8DD70E5ED8E9DE8E9752C9E /* STPAPIClientTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAPIClientTest.swift; sourceTree = ""; }; + BA08DCDD421CE92ECB61EF5C /* STPFPXBankStatusResponse.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFPXBankStatusResponse.swift; sourceTree = ""; }; + BAFD938F3A149A56CC99FE96 /* fi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fi; path = fi.lproj/Localizable.strings; sourceTree = ""; }; + BB8FCDBC63A79CD1571A2DFB /* STPCustomerContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCustomerContext.swift; sourceTree = ""; }; + BC7B7ED388B6C6AE57D3D627 /* LinkAccountServiceTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkAccountServiceTests.swift; sourceTree = ""; }; + BCBEA9E4823F08C1F5057B5A /* STPAUBECSDebitFormViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAUBECSDebitFormViewSnapshotTests.swift; sourceTree = ""; }; + BCFA6BB95613ED3F668A0DC9 /* stp_bank_fpx_bsn@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_bsn@3x.png"; sourceTree = ""; }; + BD89580A3E41D7167C30B287 /* STPAnalyticsClient+Payments.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAnalyticsClient+Payments.swift"; sourceTree = ""; }; + BF8A27B4CE855392E3E3F735 /* STPPaymentMethodFPXTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodFPXTest.m; sourceTree = ""; }; + BFB4A210A30D1D4F3D3100E5 /* hu */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = hu; path = hu.lproj/Localizable.strings; sourceTree = ""; }; + C17799DC7FA54E758EED31A6 /* NSArray+StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "NSArray+StripeTest.swift"; sourceTree = ""; }; + C1AAF76C3F4850217C28E362 /* STPTestingAPIClient.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPTestingAPIClient.m; sourceTree = ""; }; + C23D612FD5AD7772E1B30DCC /* STPThreeDSSelectionCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSSelectionCustomizationTest.swift; sourceTree = ""; }; + C3B75875C55D2C2723DC5090 /* STPSource+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPSource+BasicUI.swift"; sourceTree = ""; }; + C45852D37E323E65C47348B0 /* pt-PT */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pt-PT"; path = "pt-PT.lproj/Localizable.strings"; sourceTree = ""; }; + C5923A4DD3CD39CB64B8A8C9 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + C5BEAA15B53AC5662A33D0E1 /* STPEphemeralKeyTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPEphemeralKeyTest.swift; sourceTree = ""; }; + C713F58BC61A962C720AE0AE /* STPMandateCustomerAcceptanceParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPMandateCustomerAcceptanceParamsTest.swift; sourceTree = ""; }; + C750C2C4AB33BC232D1592BA /* STPPaymentContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentContext.swift; sourceTree = ""; }; + C7F54E1E8FFA1505D24538A6 /* STPShippingMethodsViewControllerLocalizationTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingMethodsViewControllerLocalizationTests.swift; sourceTree = ""; }; + C861CC96A54A24BDC1303C1B /* Card.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = Card.json; sourceTree = ""; }; + C8F8FCC84601E4ADC6B7F3CE /* PaymentAnalyticTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentAnalyticTest.swift; sourceTree = ""; }; + C980D24DDC884FECCE39139F /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = ""; }; + CAC5FD19B4843C454E6881DB /* stp_bank_fpx_public_bank@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_public_bank@3x.png"; sourceTree = ""; }; + CD5AC2BFBC8141F98C00CF9F /* StripeiOS_Tests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = StripeiOS_Tests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + CDDB4BC3C16AB7218E6FF1FA /* PaymentSheetTestUtils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentSheetTestUtils.swift; sourceTree = ""; }; + CDDEAB86BE4711841D426F3B /* STPPaymentIntentFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentIntentFunctionalTest.swift; sourceTree = ""; }; + CEAF8A072B0BF86AD02655B0 /* stp_fpx_big_logo@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_fpx_big_logo@3x.png"; sourceTree = ""; }; + CEB30E3846E24FD57A44764A /* LinkInMemoryCookieStoreTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkInMemoryCookieStoreTests.swift; sourceTree = ""; }; + CF13BAEF86594C9CABD4F42A /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodUSBankAccountParamsStubbedTest.swift; sourceTree = ""; }; + CF5E437205850C2039BBEBCB /* STPLabeledFormTextFieldViewSnapshotTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPLabeledFormTextFieldViewSnapshotTests.m; sourceTree = ""; }; + CF8E26B31C0F2B256763BDDB /* STPPaymentMethodAfterpayClearpayParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodAfterpayClearpayParamsTest.m; sourceTree = ""; }; + CF902DC49DD90860BD0E5E80 /* STPPaymentIntentParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentIntentParamsTest.swift; sourceTree = ""; }; + CFB4BBE0C4760DB457DE9A57 /* STPPaymentMethodPrzelewy24ParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodPrzelewy24ParamsTests.m; sourceTree = ""; }; + CFC4BC1AB047ED88C4D13C89 /* STPApplePayTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayTest.swift; sourceTree = ""; }; + CFDBAD933B23E26CBA816018 /* STPFixtures+Swift.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPFixtures+Swift.swift"; sourceTree = ""; }; + CFE6A87CDA08F216E1DFE4CD /* ElementsSession.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = ElementsSession.json; sourceTree = ""; }; + CFFE40AD9D875709F643D2E5 /* ConfirmButtonTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ConfirmButtonTests.swift; sourceTree = ""; }; + D07275F94914B7E7937D24FE /* NSDictionary+StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "NSDictionary+StripeTest.swift"; sourceTree = ""; }; + D106EFD1BF019522BE7BEBBE /* STPPaymentContextSnapshotTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentContextSnapshotTests.m; sourceTree = ""; }; + D1681629DCEA0765515D3D49 /* STPSourceOwnerTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPSourceOwnerTest.m; sourceTree = ""; }; + D1859673CAD068B345F5DD7D /* STPCardCVCInputTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardCVCInputTextFieldSnapshotTests.swift; sourceTree = ""; }; + D2F205F920E971DEA59E3C31 /* STPIntentActionTypeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPIntentActionTypeTest.swift; sourceTree = ""; }; + D3667F97B7665F699D6704BE /* nl */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = nl; path = nl.lproj/Localizable.strings; sourceTree = ""; }; + D3803D0DED98501AA26B2EAC /* STPCardNumberInputTextFieldFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardNumberInputTextFieldFormatterTests.swift; sourceTree = ""; }; + D3B00A0DC03407F2E67C3B4E /* STPFixtures.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STPFixtures.h; sourceTree = ""; }; + D3D2DB08C335695B705F544C /* STPAddressViewModelTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddressViewModelTest.swift; sourceTree = ""; }; + D42F83F785EAF24F5DC7ED1A /* STPCardCVCInputTextFieldValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardCVCInputTextFieldValidatorTests.swift; sourceTree = ""; }; + D45CA902199FABE1254E8D1F /* STPConnectAccountFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPConnectAccountFunctionalTest.m; sourceTree = ""; }; + D5C4A4CC7D2E9B5AB3EC3B79 /* STPPaymentOptionsInternalViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsInternalViewController.swift; sourceTree = ""; }; + D609FFD051FC01BF566665A0 /* StripeiOS Tests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Debug.xcconfig"; sourceTree = ""; }; + D61146D58FD219AE39E71D7D /* PaymentSheetAddressTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentSheetAddressTests.swift; sourceTree = ""; }; + D794C5E6396B4A19DC4F6921 /* StripeUICore.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeUICore.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + D7C4773C2D193BEDF1CBB530 /* STPCardNumberInputTextFieldValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardNumberInputTextFieldValidatorTests.swift; sourceTree = ""; }; + D8099EC47BB1527E2B9A2446 /* PaymentMethodMessagingViewFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentMethodMessagingViewFunctionalTest.swift; sourceTree = ""; }; + D82418EB84598A2CEE2DCB28 /* STPConnectAccountAddressTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPConnectAccountAddressTest.m; sourceTree = ""; }; + D87817A1D3D213AA4ADF6A4C /* STPPaymentMethodAffirmParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodAffirmParamsTest.swift; sourceTree = ""; }; + D93C23F55BEADF9BC74DFBDB /* STPImageLibrary.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPImageLibrary.swift; sourceTree = ""; }; + DA581592426609EDBE9B1CF4 /* stp_bank_fpx_maybank2e@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_bank_fpx_maybank2e@3x.png"; sourceTree = ""; }; + DA82BB67D434E76B9ABA4CEC /* Stripe-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe-Release.xcconfig"; sourceTree = ""; }; + DB58AC5E2E0A68221260FD44 /* STPMandateOnlineParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPMandateOnlineParamsTest.swift; sourceTree = ""; }; + DC3AD586DDED620B9E68F461 /* StripeErrorTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StripeErrorTest.swift; sourceTree = ""; }; + DDC55CC034022DFAC9366E2E /* StripePaymentSheet.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripePaymentSheet.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + DE24B3BAD7E890661CCA817D /* el-GR */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "el-GR"; path = "el-GR.lproj/Localizable.strings"; sourceTree = ""; }; + DE3D0F21FB3B3E8BCA15FF7C /* STPShippingAddressViewControllerLocalizationTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingAddressViewControllerLocalizationTests.swift; sourceTree = ""; }; + DF470B57E61F36BD1C0A1224 /* STPConnectAccountParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPConnectAccountParamsTest.m; sourceTree = ""; }; + DFA4E34E459EA612E065DE64 /* fr-CA */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "fr-CA"; path = "fr-CA.lproj/Localizable.strings"; sourceTree = ""; }; + DFA7A75BA785EBBE4C05DAA3 /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Images.xcassets; sourceTree = ""; }; + E0A27E1D77216CE0267C981C /* STPPaymentMethodCardChecksTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodCardChecksTest.m; sourceTree = ""; }; + E136A967522048B313E3C62F /* StripePaymentsUI.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripePaymentsUI.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + E1644AA33E81233EF33022BA /* STPCardValidatorTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardValidatorTest.swift; sourceTree = ""; }; + E1A2173E0891B7138687D544 /* Stripe-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe-Debug.xcconfig"; sourceTree = ""; }; + E1CD20E00EAD41091B71ABD5 /* NSURLComponents_StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NSURLComponents_StripeTest.swift; sourceTree = ""; }; + E2307F2C5E53540D4ACAA1F6 /* UIToolbar+Stripe_InputAccessory.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIToolbar+Stripe_InputAccessory.swift"; sourceTree = ""; }; + E315168EF07F52B733EA77F8 /* STPSwiftFixtures.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSwiftFixtures.swift; sourceTree = ""; }; + E3810757AF83E98307634814 /* STPRedirectContextTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPRedirectContextTest.m; sourceTree = ""; }; + E3AC82FE96A69220B300FBE6 /* STPTestAPIClient+Swift.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPTestAPIClient+Swift.swift"; sourceTree = ""; }; + E3B42EBAC0DC7ED0D9200DB7 /* STPBlocks.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBlocks.swift; sourceTree = ""; }; + E3C8833E7EBA7A1EBF349D19 /* StripeiOS Tests-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Release.xcconfig"; sourceTree = ""; }; + E3D73CE47141F7D1E44283D2 /* STPApplePayContextFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPApplePayContextFunctionalTest.m; sourceTree = ""; }; + E412BB731BC34BB1A7EC3B8D /* SofortSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = SofortSource.json; sourceTree = ""; }; + E4194E605BB5F31E9CBB8F96 /* STPAPIClientStubbedTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAPIClientStubbedTest.swift; sourceTree = ""; }; + E4442901582C822007C3BB67 /* STPPaymentMethodNetBankingParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodNetBankingParamsTest.m; sourceTree = ""; }; + E452877E5D11120B1E28A6E7 /* STPApplePayContextDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayContextDelegate.swift; sourceTree = ""; }; + E5D9F97ABC88302478220267 /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKPaymentAuthorizationViewController+Stripe_Blocks.swift"; sourceTree = ""; }; + E6312182B5BCAB940D216650 /* ConsumerSessionTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ConsumerSessionTests.swift; sourceTree = ""; }; + E68F6B90F3BC61A49570FAF4 /* UINavigationBar+StripeTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UINavigationBar+StripeTest.m"; sourceTree = ""; }; + E6DEE912364C9F4B51B374D0 /* STPPaymentCardTextFieldViewModelTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentCardTextFieldViewModelTest.swift; sourceTree = ""; }; + E7ACB4FAFAD33296DE34D036 /* STPAnalyticsClientPaymentsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAnalyticsClientPaymentsTest.swift; sourceTree = ""; }; + E7C7D85A7FAAFDF4F59BA85E /* LinkSignupViewModelTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkSignupViewModelTests.swift; sourceTree = ""; }; + E8063E8073A32E0B081A1DFA /* STPPaymentMethodSofortTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodSofortTests.swift; sourceTree = ""; }; + E9A962F163EC016697A82124 /* PaymentSheetLinkAccountTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentSheetLinkAccountTests.swift; sourceTree = ""; }; + EA20E0E29EDC1F61ADA226A2 /* zh-HK */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-HK"; path = "zh-HK.lproj/Localizable.strings"; sourceTree = ""; }; + EA9975553E669AF69F3CE437 /* STPPostalCodeInputTextFieldValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPostalCodeInputTextFieldValidatorTests.swift; sourceTree = ""; }; + EAB51503D64F3464BFBC18CB /* STPPaymentMethodUPIParamsTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodUPIParamsTest.m; sourceTree = ""; }; + EAD0042713B4C2FEB8916EDC /* STPPaymentMethodOXXOTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodOXXOTests.m; sourceTree = ""; }; + EB6AE83989B0596F0C111E13 /* STPStringUtilsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPStringUtilsTest.swift; sourceTree = ""; }; + EB71A4A2762CF864DB198BCF /* Project-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Debug.xcconfig"; sourceTree = ""; }; + EC5115C4E72B99EE743139F1 /* stp_icon_add@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "stp_icon_add@3x.png"; sourceTree = ""; }; + EDD30E5DB8DB3AA3567F5C20 /* STPPaymentOptionTuple.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionTuple.swift; sourceTree = ""; }; + EEEC1EA82490933B169915C4 /* PaymentSheetFormFactoryTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentSheetFormFactoryTest.swift; sourceTree = ""; }; + EF48EC440E1ED5D6BAA567FF /* STPPaymentHandlerStubbedMockedFilesTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentHandlerStubbedMockedFilesTests.swift; sourceTree = ""; }; + EFD2F6A5A046A620BAB75B41 /* AutoCompleteViewControllerSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AutoCompleteViewControllerSnapshotTests.swift; sourceTree = ""; }; + F0293B82D7078EE11F9B5639 /* UIView+Helpers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIView+Helpers.swift"; sourceTree = ""; }; + F0D010B03BB8B290806267A0 /* STPEphemeralKeyManagerTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPEphemeralKeyManagerTest.m; sourceTree = ""; }; + F13E3DE09A463B4501733B87 /* zh-Hans */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-Hans"; path = "zh-Hans.lproj/Localizable.strings"; sourceTree = ""; }; + F153420EA142B5CA76E89A04 /* STPMocks.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STPMocks.h; sourceTree = ""; }; + F22EE02CE12F3EBFACDAE9A4 /* STPUIVCStripeParentViewControllerTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPUIVCStripeParentViewControllerTests.m; sourceTree = ""; }; + F24B1E06C600CA30A97F5AEA /* iDEALSource.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = iDEALSource.json; sourceTree = ""; }; + F3C732C25FD961631BD44FDD /* STPBSBNumberValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBSBNumberValidatorTests.swift; sourceTree = ""; }; + F44327A2B2C9483F52EE343B /* STPFormViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFormViewSnapshotTests.swift; sourceTree = ""; }; + F4E5416F6AE8BED88980D6F8 /* STPPaymentMethodBancontactTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodBancontactTests.swift; sourceTree = ""; }; + F4F116895CA31148DB6CE0DF /* PaymentMethodMessagingViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentMethodMessagingViewSnapshotTests.swift; sourceTree = ""; }; + F546088BA4F763334CFD3D34 /* STPImageLibraryTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPImageLibraryTest.swift; sourceTree = ""; }; + F6558C62376C2397030BD4A6 /* STPPaymentMethodPrzelewy24Tests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodPrzelewy24Tests.swift; sourceTree = ""; }; + F7385193226663A5B79E69ED /* STPFloatingPlaceholderTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFloatingPlaceholderTextFieldSnapshotTests.swift; sourceTree = ""; }; + F8CEB050CB2231F1459824EC /* EphemeralKey.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = EphemeralKey.json; sourceTree = ""; }; + FA793904C7B2D3AA0A4D5EFB /* STPAddCardViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddCardViewController.swift; sourceTree = ""; }; + FABC6FF8D581998A773A8656 /* STPCustomerTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPCustomerTest.m; sourceTree = ""; }; + FC30E6129279F14506219E98 /* STPPaymentHandlerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentHandlerTests.swift; sourceTree = ""; }; + FCA64B2C21FCFAC433EA3781 /* recorded_network_traffic */ = {isa = PBXFileReference; path = recorded_network_traffic; sourceTree = ""; }; + FD289E1EA9F0CE1C848AC0BB /* STPCardNumberInputTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardNumberInputTextFieldSnapshotTests.swift; sourceTree = ""; }; + FD3398E2352CEA0264F20AEA /* stp_test_upload_image.jpeg */ = {isa = PBXFileReference; lastKnownFileType = image.jpeg; path = stp_test_upload_image.jpeg; sourceTree = ""; }; + FDA32D0C9E8A7A69F4899EDC /* RotatingCardBrandsViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RotatingCardBrandsViewSnapshotTests.swift; sourceTree = ""; }; + FDF7394DDD552EDE996EAD8E /* STPPostalCodeValidatorTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPostalCodeValidatorTest.swift; sourceTree = ""; }; + FE1929CD1B3D3DCC2401E2F8 /* STPPaymentMethodPayPalParamsTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentMethodPayPalParamsTests.m; sourceTree = ""; }; + FE2DED6ABA7407C17C1391B6 /* MockFiles */ = {isa = PBXFileReference; path = MockFiles; sourceTree = ""; }; + FF5E08A1651D9DFE502DA021 /* STPCardFormViewTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardFormViewTests.swift; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 1587AD7E52759FBC80315765 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 32874C6147344A9CB2EF4DAD /* Stripe3DS2.framework in Frameworks */, + 8520A27C204A068C43592024 /* StripeApplePay.framework in Frameworks */, + E97168F37D769524B58461B6 /* StripeCore.framework in Frameworks */, + 360EEE8B706D2A4A49666F7A /* StripePayments.framework in Frameworks */, + 9363F8F389C04C19B37D0F0A /* StripePaymentsUI.framework in Frameworks */, + EA34719659CB9F1A269FECC7 /* StripeUICore.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 6ACBDD66E08F7CEE34A8FFFA /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + AC7C127B11A60222465F4696 /* XCTest.framework in Frameworks */, + D151C8724925DCBA4BA4F46A /* StripeCoreTestUtils.framework in Frameworks */, + 420F8CAB4FAD6D9AF4AF25C0 /* StripePaymentSheet.framework in Frameworks */, + EEA502DF8809B8FD0D00785E /* StripePayments.framework in Frameworks */, + 03E60F9EF24C975AF90E2447 /* StripePaymentsUI.framework in Frameworks */, + 0FA3C1494BA57884B5DE3B20 /* Stripe.framework in Frameworks */, + 2EB68A59660A4D1E14799DA4 /* OHHTTPStubs in Frameworks */, + 2CD7968DA48F7129E16EA0CB /* OHHTTPStubsSwift in Frameworks */, + 5C51167CC14F653E7117BA61 /* OCMock in Frameworks */, + B795A5EB8FDECA1060A9655C /* iOSSnapshotTestCase in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 7129AA5AD64208D3B1A76AAE /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 9291A08CCB34504FCA4B7481 /* XCTest.framework in Frameworks */, + 2F0FC4E67BE577AD66CD1475 /* StripePaymentSheet.framework in Frameworks */, + 78B70C2EE8334F0FA91439CA /* Stripe.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + C04854A8658964DDD0A15E1C /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 00AA0CCB7109D2B82EF8EEA0 /* Project */ = { + isa = PBXGroup; + children = ( + D0F67D3A4FA2B67D2AB0A49F /* BuildConfigurations */, + 97BE66955C28DB78DA12F5F6 /* BuildConfigurations */, + B035E857851EAF160C88DC2B /* StripeiOS */, + 9C700DBA02FFEC23B92EF107 /* StripeiOSAppHostedTests */, + CFD249F84390F2310542962A /* StripeiOSTestHostApp */, + EB228B1D404A48918221D9D6 /* StripeiOSTests */, + 4E29B46F2C940E0A21734E09 /* StripeiOSTests.xctestplan */, + ); + name = Project; + sourceTree = ""; + }; + 0360377163257AE558A2CDAA /* FPX */ = { + isa = PBXGroup; + children = ( + 2A588999859430CC959C6F4B /* stp_bank_fpx_affin_bank@3x.png */, + 1F6237EC7D3A866814A7DC88 /* stp_bank_fpx_alliance_bank@3x.png */, + 89CA73D6080D5B68F6465BC1 /* stp_bank_fpx_ambank@3x.png */, + 87DB9F06FDC13DD3BFEAA27F /* stp_bank_fpx_bank_islam@3x.png */, + A7B538B1223FBD7F409A4B29 /* stp_bank_fpx_bank_muamalat@3x.png */, + 34D8B80455B75885088E9DAC /* stp_bank_fpx_bank_rakyat@3x.png */, + BCFA6BB95613ED3F668A0DC9 /* stp_bank_fpx_bsn@3x.png */, + 1FD3467084C9B5F7B4C15364 /* stp_bank_fpx_cimb@3x.png */, + 7F90C21B77335533395C4321 /* stp_bank_fpx_hong_leong_bank@3x.png */, + 05A3E36498CBA9E5FB8323C9 /* stp_bank_fpx_hsbc@3x.png */, + 43398AFBACB8BA1DB29A437C /* stp_bank_fpx_kfh@3x.png */, + DA581592426609EDBE9B1CF4 /* stp_bank_fpx_maybank2e@3x.png */, + A982DCED06CCAEBE2C8835F3 /* stp_bank_fpx_maybank2u@3x.png */, + 7EB4A04665E4B0FBE6BA24F6 /* stp_bank_fpx_ocbc@3x.png */, + CAC5FD19B4843C454E6881DB /* stp_bank_fpx_public_bank@3x.png */, + 152E45642578F76AAEB1CF61 /* stp_bank_fpx_rhb@3x.png */, + 40E3C9AF9FA4576DAC027275 /* stp_bank_fpx_standard_chartered@3x.png */, + 8EC1D194936D061E3255BCF4 /* stp_bank_fpx_uob@3x.png */, + CEAF8A072B0BF86AD02655B0 /* stp_fpx_big_logo@3x.png */, + B2C7ED5A39AB3BB2248CEACA /* stp_fpx_logo@3x.png */, + ); + path = FPX; + sourceTree = ""; + }; + 2EC2475EABAA6AB3ADF1A1DC /* Images */ = { + isa = PBXGroup; + children = ( + 3CF7E190074CFBA173441126 /* Cards */, + 0360377163257AE558A2CDAA /* FPX */, + EC5115C4E72B99EE743139F1 /* stp_icon_add@3x.png */, + A1003B17EB955A29BC77BF01 /* stp_icon_bank@3x.png */, + A95FC67D6E0C11B6FE40DC92 /* stp_icon_checkmark@3x.png */, + 21BC7D5A0DDD653571B76168 /* stp_shipping_form@3x.png */, + ); + path = Images; + sourceTree = ""; + }; + 3CF7E190074CFBA173441126 /* Cards */ = { + isa = PBXGroup; + children = ( + 0227131370F51B0533585BEB /* stp_card_form_amex_cvc@3x.png */, + 2A0E991FB88CBE8C9135DFE0 /* stp_card_form_back@3x.png */, + 5C0462C8B2776183A2B03D51 /* stp_card_form_front@3x.png */, + ); + path = Cards; + sourceTree = ""; + }; + 437EB711F100257495D02BD7 /* Resources */ = { + isa = PBXGroup; + children = ( + 147D2DC1FFDFC99269039377 /* LaunchScreen.storyboard */, + 884C01B087B4D820395BD374 /* Main.storyboard */, + 6955B3A3353F8442E4FBBBF6 /* Assets.xcassets */, + ); + path = Resources; + sourceTree = ""; + }; + 7EA3633D452F7A9C017F4D52 /* Source */ = { + isa = PBXGroup; + children = ( + 8F7E3CE2105E4A39032CD919 /* Enums+CustomStringConvertible.swift */, + 273EE407039913F0B644172B /* PKAddPaymentPassRequest+Stripe_Error.swift */, + E5D9F97ABC88302478220267 /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift */, + FA793904C7B2D3AA0A4D5EFB /* STPAddCardViewController.swift */, + 498C3FB07CFD532779C755D3 /* STPAddress+BasicUI.swift */, + 72903593DC432D01720DC9D9 /* STPAddressFieldTableViewCell.swift */, + 28A50AFA4603E488FF3D82D0 /* STPAddressViewModel.swift */, + 12DBB3F72AEFB52DE27C27ED /* STPAnalyticsClient+BasicUI.swift */, + BD89580A3E41D7167C30B287 /* STPAnalyticsClient+Payments.swift */, + 1E8AFAE24610EC983727F860 /* STPAPIClient+BasicUI.swift */, + 807FF966F1DE05F3496B817B /* STPAPIClient+PushProvisioning.swift */, + E452877E5D11120B1E28A6E7 /* STPApplePayContextDelegate.swift */, + 03ACDC7EEC28D1FE50008F65 /* STPApplePayPaymentOption.swift */, + 9B782E1D974A4C131E60E2BD /* STPBackendAPIAdapter.swift */, + 458F8576215E0F8ECE1D74CE /* STPBankSelectionTableViewCell.swift */, + 81352A0CBE46A59E6B1A712E /* STPBankSelectionViewController.swift */, + E3B42EBAC0DC7ED0D9200DB7 /* STPBlocks.swift */, + 61F8308B7250B642D19827D8 /* STPCameraView.swift */, + 3E5FB20B2BEFC00D54FDD87D /* STPCard+BasicUI.swift */, + 69BD038947E8E2376A0D240B /* STPCardScanner.swift */, + 0B91C4D5B93FF71C61B140F1 /* STPCardScannerTableViewCell.swift */, + 6A9E7B637A8747431B38FD1D /* STPCardValidationState.swift */, + 2D1525AF65BDEF691F8BCBE8 /* STPCoreScrollViewController.swift */, + B78C72B0DB434EC7F700FDE0 /* STPCoreTableViewController.swift */, + 02FC9ED423D40C88D5A24441 /* STPCoreViewController.swift */, + BB8FCDBC63A79CD1571A2DFB /* STPCustomerContext.swift */, + 890660C21E3666CE7B82695B /* STPEphemeralKey.swift */, + 588C260880FFC584A00A89F5 /* STPEphemeralKeyManager.swift */, + 485E747DA1F72F091986787B /* STPEphemeralKeyProvider.swift */, + 0669B4CA326CE74D125C789C /* STPFakeAddPaymentPassViewController.swift */, + BA08DCDD421CE92ECB61EF5C /* STPFPXBankStatusResponse.swift */, + D93C23F55BEADF9BC74DFBDB /* STPImageLibrary.swift */, + 9EB24EC81CE2C8D1C863B044 /* STPIntentActionLinkAuthenticateAccount.swift */, + AFF957F38AABE5F748C38C0B /* STPLocalizedString.swift */, + 1FFBAA4B44967B157A4F4E91 /* STPPaymentActivityIndicatorView.swift */, + 01B42BE6FB5EC1F708875AB8 /* STPPaymentCardTextFieldCell.swift */, + 18FCB69CD3B8C3DAB216A5F0 /* STPPaymentConfiguration.swift */, + C750C2C4AB33BC232D1592BA /* STPPaymentContext.swift */, + A1C876DC7F3E31D7189506A8 /* STPPaymentContextAmountModel.swift */, + 6F01150CD0255164FE2CF3A4 /* STPPaymentIntentParams+BasicUI.swift */, + 35C1E9B0EE03825DABF6471A /* STPPaymentMethod+BasicUI.swift */, + 53C5AB22D6328E85A6DDF663 /* STPPaymentMethodParams+BasicUI.swift */, + 30B694A39D54886392AA5DE3 /* STPPaymentOption.swift */, + D5C4A4CC7D2E9B5AB3EC3B79 /* STPPaymentOptionsInternalViewController.swift */, + A9A1BB31C7B514984231125B /* STPPaymentOptionsViewController.swift */, + 6215A9BF343775B1BD0F62AF /* STPPaymentOptionTableViewCell.swift */, + EDD30E5DB8DB3AA3567F5C20 /* STPPaymentOptionTuple.swift */, + 7D964D9E01627B419B4BD23C /* STPPaymentResult.swift */, + 3EBB07171F6FDCE6E20C454A /* STPPinManagementService.swift */, + 6887F19BB9804BF45FD703FF /* STPPushProvisioningContext.swift */, + 89E5DA3029F141B5111A5B2C /* STPPushProvisioningDetails.swift */, + 2B28B8A547CD846277ECD578 /* STPPushProvisioningDetailsParams.swift */, + 96E1FED5CE5974C9C1162E93 /* STPSectionHeaderView.swift */, + 71711FC8E2FB66E52A5FDD9A /* STPShippingAddressViewController.swift */, + 21E4B84223DDA131544DBBA7 /* STPShippingMethodsViewController.swift */, + 98544B08552407D41D398C68 /* STPShippingMethodTableViewCell.swift */, + C3B75875C55D2C2723DC5090 /* STPSource+BasicUI.swift */, + 8E8CA4361964E1BA400EFC89 /* STPTheme.swift */, + B1217AD643A9E8F88B60F645 /* STPUserInformation.swift */, + A5398E1156E0BFEBBF56FD2F /* String+Localized.swift */, + B70DF0B659009041F485EE0F /* Stripe+Exports.swift */, + B4D31B0D7BD9F97AF3BB61E6 /* StripeBundleLocator.swift */, + 3B3000668A75E095B514241F /* UIBarButtonItem+Stripe.swift */, + A7E369CEC9F5B3758F78E88F /* UINavigationBar+Stripe_Theme.swift */, + 2C078573F46762353664AC92 /* UINavigationController+Stripe_Completion.swift */, + 5476BD87E0480A93958F0328 /* UITableViewCell+Stripe_Borders.swift */, + E2307F2C5E53540D4ACAA1F6 /* UIToolbar+Stripe_InputAccessory.swift */, + F0293B82D7078EE11F9B5639 /* UIView+Helpers.swift */, + 1B76DF0FE363F59BF0940A8B /* UIView+Stripe_FirstResponder.swift */, + 86798C95A778362EF815B4C6 /* UIView+Stripe_SafeAreaBounds.swift */, + 2F08757CA6F6B2DA65C14E0A /* UIViewController+Stripe_KeyboardAvoiding.swift */, + 6F017A08C7E633FB4297D274 /* UIViewController+Stripe_NavigationItemProxy.swift */, + 5804C2B9C0704E386B3D25A4 /* UIViewController+Stripe_ParentViewController.swift */, + ); + path = Source; + sourceTree = ""; + }; + 85B4F3432AB3A8B94C9D2591 /* Products */ = { + isa = PBXGroup; + children = ( + 4259421D2CD26E37B96F97B2 /* Stripe.framework */, + 33FDC634FD5D79E824240DDC /* Stripe3DS2.framework */, + 52F8AEC50D4623F80F04A533 /* StripeApplePay.framework */, + 43B4E4B85C598D7A9AFCB4D4 /* StripeCore.framework */, + 512A0E7C246D5F044245E069 /* StripeCoreTestUtils.framework */, + CD5AC2BFBC8141F98C00CF9F /* StripeiOS_Tests.xctest */, + 1DD6897858F46976A946394E /* StripeiOSAppHostedTests.xctest */, + 294CD46E24BB2743042872D7 /* StripeiOSTestHostApp.app */, + 22D1C6EB5826E2D7C80B6CF3 /* StripePayments.framework */, + DDC55CC034022DFAC9366E2E /* StripePaymentSheet.framework */, + E136A967522048B313E3C62F /* StripePaymentsUI.framework */, + D794C5E6396B4A19DC4F6921 /* StripeUICore.framework */, + ); + name = Products; + sourceTree = ""; + }; + 97BE66955C28DB78DA12F5F6 /* BuildConfigurations */ = { + isa = PBXGroup; + children = ( + EB71A4A2762CF864DB198BCF /* Project-Debug.xcconfig */, + 969E196AB597EEF68C38103E /* Project-Release.xcconfig */, + D609FFD051FC01BF566665A0 /* StripeiOS Tests-Debug.xcconfig */, + E3C8833E7EBA7A1EBF349D19 /* StripeiOS Tests-Release.xcconfig */, + ); + name = BuildConfigurations; + path = ../BuildConfigurations; + sourceTree = ""; + }; + 9C700DBA02FFEC23B92EF107 /* StripeiOSAppHostedTests */ = { + isa = PBXGroup; + children = ( + 65DCC9BDA647E58D3882C698 /* Info.plist */, + 10BAD33BEC6C2894F9266902 /* LinkSecureCookieStoreTests.swift */, + ); + path = StripeiOSAppHostedTests; + sourceTree = ""; + }; + AF41AE099441C2A09DECC1AC /* Localizations */ = { + isa = PBXGroup; + children = ( + C2427C1CDFA85BFC6570F1E9 /* Localizable.strings */, + ); + path = Localizations; + sourceTree = ""; + }; + B035E857851EAF160C88DC2B /* StripeiOS */ = { + isa = PBXGroup; + children = ( + FBC7A77342A98B1DE8E416B7 /* Resources */, + 7EA3633D452F7A9C017F4D52 /* Source */, + C5923A4DD3CD39CB64B8A8C9 /* Info.plist */, + A8598727045C6268B57A5FC7 /* Stripe-umbrella.h */, + ); + path = StripeiOS; + sourceTree = ""; + }; + B7A38600F42999DA8F83AD27 /* Frameworks */ = { + isa = PBXGroup; + children = ( + 005650A59D692F820EF20F5F /* XCTest.framework */, + ); + name = Frameworks; + sourceTree = ""; + }; + CFD249F84390F2310542962A /* StripeiOSTestHostApp */ = { + isa = PBXGroup; + children = ( + 437EB711F100257495D02BD7 /* Resources */, + 9BBCE3A905041A709E8F279A /* AppDelegate.swift */, + B3E0745D13EB19BAA24F3BA3 /* Info.plist */, + 1F29C15B47C7CB0941CD4C9E /* ViewController.swift */, + ); + path = StripeiOSTestHostApp; + sourceTree = ""; + }; + D0F67D3A4FA2B67D2AB0A49F /* BuildConfigurations */ = { + isa = PBXGroup; + children = ( + 1D23EB567F573612E0794B3A /* Stripe Tests-Debug.xcconfig */, + 5E4EA6394497D1BD57ED0032 /* Stripe Tests-Release.xcconfig */, + E1A2173E0891B7138687D544 /* Stripe-Debug.xcconfig */, + DA82BB67D434E76B9ABA4CEC /* Stripe-Release.xcconfig */, + ); + path = BuildConfigurations; + sourceTree = ""; + }; + E4802964A9471C082CE01BA9 = { + isa = PBXGroup; + children = ( + 00AA0CCB7109D2B82EF8EEA0 /* Project */, + B7A38600F42999DA8F83AD27 /* Frameworks */, + 85B4F3432AB3A8B94C9D2591 /* Products */, + ); + sourceTree = ""; + }; + EB228B1D404A48918221D9D6 /* StripeiOSTests */ = { + isa = PBXGroup; + children = ( + EB90CE5EA5682921B4C247A0 /* Resources */, + 5430F6704CFF793CBFAAF549 /* AddPaymentMethodViewControllerSnapshotTests.swift */, + A9F06DC32147A8ABCD709E79 /* AddressViewControllerSnapshotTests.swift */, + 3C742844915B96CFD25BFFF9 /* AfterpayPriceBreakdownViewSnapshotTests.swift */, + 61AF6E95FE0DD913204CAB32 /* AnalyticsHelperTests.swift */, + 85AAB72218409F85FE29E69E /* APIRequestTest.swift */, + EFD2F6A5A046A620BAB75B41 /* AutoCompleteViewControllerSnapshotTests.swift */, + 86673B13C7762666C774F124 /* ButtonLinkSnapshotTests.swift */, + 153071C69A0BEE033E035DCF /* CardExpiryDateTests.swift */, + 46AC0B5EC7433E081825D31B /* CircularButtonSnapshotTests.swift */, + 806124200E77795DCFC8418E /* ConfirmButtonSnapshotTests.swift */, + CFFE40AD9D875709F643D2E5 /* ConfirmButtonTests.swift */, + E6312182B5BCAB940D216650 /* ConsumerSessionTests.swift */, + 8CE85C770AEEDBE4AEC93EAA /* Error+PaymentSheetTests.swift */, + 180CF848E3ABF0236C494D8B /* FBSnapshotTestCase+STPViewControllerLoading.swift */, + 7C04AFC9CDE50D09D38A3232 /* FormSpecProviderTest.swift */, + 45FF7A07CFC3B9B7AD6B49EE /* FraudDetectionDataTest.swift */, + AAF368BCD5990EE5DC17D299 /* ImageTest.swift */, + 0E10C4D64477638398251FFB /* Info.plist */, + 8D866DCC403994A0D3CFB1D7 /* KlarnaHelperTest.swift */, + BC7B7ED388B6C6AE57D3D627 /* LinkAccountServiceTests.swift */, + 72B1E4D9DA48CED66CD96308 /* LinkBadgeViewSnapshotTest.swift */, + 7F9659E92E94316FE4955156 /* LinkCardEditElementSnapshotTests.swift */, + 83A5A0074F69DE467F94ABFB /* LinkInlineSignupElementSnapshotTests.swift */, + CEB30E3846E24FD57A44764A /* LinkInMemoryCookieStoreTests.swift */, + 6F29142B51DC99E2258B4F60 /* LinkInstantDebitMandateViewSnapshotTests.swift */, + 835CB781FBC19773ACC20676 /* LinkLegalTermsViewSnapshotTests.swift */, + 157826E456AC3ABFE8000B81 /* LinkNavigationBarSnapshotTests.swift */, + 382A01724CE8659A5210073F /* LinkNoticeViewSnapshotTests.swift */, + 3024CAAF54280B0357014D39 /* LinkPaymentMethodPickerSnapshotTests.swift */, + E7C7D85A7FAAFDF4F59BA85E /* LinkSignupViewModelTests.swift */, + 52FA1AFD632CCE18C4315FC6 /* LinkStubs.swift */, + 5BB7453E4E1AD36D604A00E8 /* LinkToastSnapshotTests.swift */, + 24AE1B8096363F6C51A5AF7E /* LinkVerificationViewSnapshotTests.swift */, + 9DCCA0E8A02B4F4B23837FB4 /* MKPlacemark+PaymentSheetTests.swift */, + C17799DC7FA54E758EED31A6 /* NSArray+StripeTest.swift */, + 20552E792B8E7BA15821AB5D /* NSDecimalNumber+StripeTest.swift */, + D07275F94914B7E7937D24FE /* NSDictionary+StripeTest.swift */, + 725845FAFFAE11297BCDFD29 /* NSLocale+STPSwizzling.h */, + 86D961D8952114BE479E1EB7 /* NSLocale+STPSwizzling.m */, + 63F5F35DB97D8A176FB6ED24 /* NSString+StripeTest.swift */, + E1CD20E00EAD41091B71ABD5 /* NSURLComponents_StripeTest.swift */, + 9D85FA7B714BDD8D1FD83B75 /* OneTimeCodeTextFieldSnapshotTests.swift */, + 98F9CB667BC68767DFB5FACD /* OneTimeCodeTextFieldTests.swift */, + A61763BA2CCA86F9B8FD4F1F /* OperationDebouncerTests.swift */, + C8F8FCC84601E4ADC6B7F3CE /* PaymentAnalyticTest.swift */, + D8099EC47BB1527E2B9A2446 /* PaymentMethodMessagingViewFunctionalTest.swift */, + F4F116895CA31148DB6CE0DF /* PaymentMethodMessagingViewSnapshotTests.swift */, + 9D73D989C9D488D820029FA1 /* PaymentSheet+APITest.swift */, + D61146D58FD219AE39E71D7D /* PaymentSheetAddressTests.swift */, + 83760BB83F6391E0CF66234D /* PaymentSheetFormFactorySnapshotTest.swift */, + EEEC1EA82490933B169915C4 /* PaymentSheetFormFactoryTest.swift */, + E9A962F163EC016697A82124 /* PaymentSheetLinkAccountTests.swift */, + A1CF1C43EEE33FC0AF31A2BB /* PaymentSheetPaymentMethodTypeTest.swift */, + CDDB4BC3C16AB7218E6FF1FA /* PaymentSheetTestUtils.swift */, + 7B9A4A2B0FB9F8C743BBED48 /* PaymentTypeCellSnapshotTests.swift */, + AF342CBC167F9CAB5B49CC32 /* PayWithLinkButtonSnapshotTests.swift */, + 83707086116ED364BE3C1F2E /* PayWithLinkViewController-WalletViewModelTests.swift */, + B4D12508C2F1056A7EAFEC86 /* PKPayment+StripeTest.swift */, + FDA32D0C9E8A7A69F4899EDC /* RotatingCardBrandsViewSnapshotTests.swift */, + 12632E6710DE8861CAF1BAA4 /* RotatingCardBrandsViewTests.swift */, + 20879436DCFB1F03BE1608B3 /* ServerErrorMapperTest.swift */, + 78BA15D1F0844B1DA71A5348 /* STPAddCardViewControllerLocalizationTests.swift */, + 8704ABFA91A5226847F4A69A /* STPAddCardViewControllerTest.swift */, + 81B23BF8256AA75F9903AC0C /* STPAddressTests.m */, + D3D2DB08C335695B705F544C /* STPAddressViewModelTest.swift */, + 0241DB84973B21393BEC703E /* STPAnalyticsClientPaymentSheetTest.swift */, + E7ACB4FAFAD33296DE34D036 /* STPAnalyticsClientPaymentsTest.swift */, + 51F49F2466BBBE64799209B5 /* STPAPIClientNetworkBridgeTest.m */, + E4194E605BB5F31E9CBB8F96 /* STPAPIClientStubbedTest.swift */, + B8DD70E5ED8E9DE8E9752C9E /* STPAPIClientTest.swift */, + 4A57C346FA98A057AF31C8B1 /* STPAPISettingsBridgeTest.m */, + E3D73CE47141F7D1E44283D2 /* STPApplePayContextFunctionalTest.m */, + A1272F2E05A0E294DD9ECA26 /* STPApplePayContextFunctionalTestExtras.swift */, + 5F7AB40A5A10C2D267323ABE /* STPApplePayContextTest.swift */, + 3C77C7BC4BA57EC296CF2F1C /* STPApplePayFunctionalTest.swift */, + 0320B47C1DD836F7A7144B58 /* STPApplePayPaymentOptionTest.m */, + 47A1AA1C80AE910FCFF58996 /* STPApplePayTest.m */, + CFC4BC1AB047ED88C4D13C89 /* STPApplePayTest.swift */, + BCBEA9E4823F08C1F5057B5A /* STPAUBECSDebitFormViewSnapshotTests.swift */, + 4AAE5EE11611F9F7762B64C6 /* STPAUBECSFormViewModelTests.swift */, + 6152335A2DCCBB5490B39BB0 /* STPBankAccountFunctionalTest.m */, + 9F304A4028D563EBD2A523C8 /* STPBankAccountParamsTest.m */, + 6FA025CE30EB3DA1C4321505 /* STPBankAccountTest.m */, + 23997D61DF41CA84BFC33080 /* STPBECSDebitAccountNumberValidatorTests.swift */, + 3FC0560A312147C37CFE6CF9 /* STPBinRangeTest.swift */, + 4FFA8B446217CDE678D7287F /* STPBlocks.h */, + F3C732C25FD961631BD44FDD /* STPBSBNumberValidatorTests.swift */, + 3E1C5E08678292561255B1C5 /* STPCardBINMetadataTests.swift */, + 4866C83CD6596D781A053636 /* STPCardBrandTest.m */, + 6E1F6514E7530C2A3478B2F5 /* STPCardCVCInputTextFieldFormatterTests.swift */, + D1859673CAD068B345F5DD7D /* STPCardCVCInputTextFieldSnapshotTests.swift */, + 94A7104C1C470515616E4D2B /* STPCardCVCInputTextFieldTests.swift */, + D42F83F785EAF24F5DC7ED1A /* STPCardCVCInputTextFieldValidatorTests.swift */, + 40BB87E28719FE0C6B946BB5 /* STPCardExpiryInputTextFieldFormatterTests.swift */, + 0C75157665428685C7A4FD20 /* STPCardExpiryInputTextFieldSnapshotTests.swift */, + 4418164D75002AE6A0273176 /* STPCardExpiryInputTextFieldValidatorTests.swift */, + 1D983E089196152DA1C69469 /* STPCardFormViewSnapshotTests.swift */, + FF5E08A1651D9DFE502DA021 /* STPCardFormViewTests.swift */, + 94042E0B406F6AA45593D154 /* STPCardFunctionalTest.m */, + D3803D0DED98501AA26B2EAC /* STPCardNumberInputTextFieldFormatterTests.swift */, + FD289E1EA9F0CE1C848AC0BB /* STPCardNumberInputTextFieldSnapshotTests.swift */, + D7C4773C2D193BEDF1CBB530 /* STPCardNumberInputTextFieldValidatorTests.swift */, + 20A74D4FA51D15AF12B58767 /* STPCardParamsTest.m */, + 1D575C31524E596E9C1A8E9B /* STPCardTest.swift */, + E1644AA33E81233EF33022BA /* STPCardValidatorTest.swift */, + 1D51B04D83D4FEF7F90DF16A /* STPCertTest.swift */, + 444A6B234A4BD3CB2C1C977D /* STPConfirmCardOptionsTest.m */, + 0B989830717894A60B74E830 /* STPConfirmPaymentMethodOptionsTest.m */, + D82418EB84598A2CEE2DCB28 /* STPConnectAccountAddressTest.m */, + D45CA902199FABE1254E8D1F /* STPConnectAccountFunctionalTest.m */, + DF470B57E61F36BD1C0A1224 /* STPConnectAccountParamsTest.m */, + 8D49257A97E71A475A9F6E08 /* STPCountryPickerInputFieldSnapshotTests.swift */, + 2CC4B06AB5C02FF54091E5A8 /* STPCustomerContextTest.swift */, + FABC6FF8D581998A773A8656 /* STPCustomerTest.m */, + 1C1548BA518F7AC2A9ECF9D5 /* STPE2ETest.swift */, + 63BDA70DB74745C5F457CF88 /* STPElementsSessionTest.swift */, + F0D010B03BB8B290806267A0 /* STPEphemeralKeyManagerTest.m */, + C5BEAA15B53AC5662A33D0E1 /* STPEphemeralKeyTest.swift */, + 9EAE9A2AE65771403CE57C11 /* STPErrorBridgeTest.m */, + 547F890FCD6D6E8B0774E919 /* STPFileFunctionalTest.m */, + 0841697DF701B4A4637E67D9 /* STPFileTest.m */, + D3B00A0DC03407F2E67C3B4E /* STPFixtures.h */, + 17DDEC7D4C1D5FF6488D71D5 /* STPFixtures.m */, + CFDBAD933B23E26CBA816018 /* STPFixtures+Swift.swift */, + F7385193226663A5B79E69ED /* STPFloatingPlaceholderTextFieldSnapshotTests.swift */, + 30964128998473CAA9F2DD7E /* STPFormEncoderTest.swift */, + 2E1862744F23286D1FB9D4AE /* STPFormTextFieldTest.swift */, + F44327A2B2C9483F52EE343B /* STPFormViewSnapshotTests.swift */, + 966BA151AA0E428D1DD8A97C /* STPFPXBankBrandTest.m */, + 284C67269D2606DA147AE01D /* STPGenericInputPickerFieldSnapshotTests.swift */, + 2D63B73C5773432CA134D1FC /* STPGenericInputPickerFieldValidatorTest.swift */, + 58A53F005EA8FDDAA66126BA /* STPGenericInputTextFieldSnapshotTests.swift */, + F546088BA4F763334CFD3D34 /* STPImageLibraryTest.swift */, + A22E5B87755C1F05C3DB438C /* STPInputTextFieldFormatterTests.swift */, + 9FEE395C4DD0E0112AF3720C /* STPInputTextFieldValidatorTests.swift */, + AD06AED0AF8A9A7FB4A2E66F /* STPIntentActionAlipayHandleRedirectTest.swift */, + 9CE65B72986B6B404B036183 /* STPIntentActionTest.m */, + D2F205F920E971DEA59E3C31 /* STPIntentActionTypeTest.swift */, + 4CA5D11C977A95B8E936E907 /* STPIntentActionWeChatPayRedirectToAppTest.swift */, + 6C8FE751333F580004BD72BA /* STPIntentWithPreferencesTest.swift */, + CF5E437205850C2039BBEBCB /* STPLabeledFormTextFieldViewSnapshotTests.m */, + 7F68637E75142DCD46710796 /* STPLabeledMultiFormTextFieldViewSnapshotTests.swift */, + C713F58BC61A962C720AE0AE /* STPMandateCustomerAcceptanceParamsTest.swift */, + B7F7AA0B7B86BA5BB2FE92CE /* STPMandateDataParamsTest.swift */, + DB58AC5E2E0A68221260FD44 /* STPMandateOnlineParamsTest.swift */, + F153420EA142B5CA76E89A04 /* STPMocks.h */, + 88AEABC15CCBB9EA393C175F /* STPMocks.m */, + 131B856C6C1053E5FACD44D1 /* STPNetworkStubbingTestCase.h */, + 9475002E030E98735A0FD2EC /* STPNetworkStubbingTestCase.m */, + B0D081A26B59DE51FDDD1798 /* STPNetworkStubbingTestCase.swift */, + 6223E57D3A198F956A37ED89 /* STPNumericDigitInputTextFormatterTests.swift */, + 4AA36705DED9164663A98B6A /* STPNumericStringValidatorTests.swift */, + B7439CC6C0FB060EA312FF6D /* STPPaymentCardTextFieldTest.m */, + 1F16C36797D978E72E612100 /* STPPaymentCardTextFieldTestsSwift.swift */, + E6DEE912364C9F4B51B374D0 /* STPPaymentCardTextFieldViewModelTest.swift */, + 79ABE6A14AF9D14103050876 /* STPPaymentConfigurationTest.m */, + 3C995125252BED1EEC018B9D /* STPPaymentContextApplePayTest.swift */, + D106EFD1BF019522BE7BEBBE /* STPPaymentContextSnapshotTests.m */, + 683F7735569D22CBEC9CA2E6 /* STPPaymentHandlerFunctionalTest.m */, + EF48EC440E1ED5D6BAA567FF /* STPPaymentHandlerStubbedMockedFilesTests.swift */, + FC30E6129279F14506219E98 /* STPPaymentHandlerTests.swift */, + 46C31CAD0FA74B58BA2B8530 /* STPPaymentIntentEnumsTest.swift */, + A8CF4EBC85BDC771DF2213C0 /* STPPaymentIntentFunctionalTest.m */, + CDDEAB86BE4711841D426F3B /* STPPaymentIntentFunctionalTest.swift */, + 6C7B8DACB0A7294BC235E3BC /* STPPaymentIntentLastPaymentErrorTest.swift */, + CF902DC49DD90860BD0E5E80 /* STPPaymentIntentParamsTest.swift */, + ACE8998EAF997A78759E49B5 /* STPPaymentIntentTest.swift */, + 1DECFBCFDD8FB9362D2FB4E1 /* STPPaymentMethodAddressTest.m */, + D87817A1D3D213AA4ADF6A4C /* STPPaymentMethodAffirmParamsTest.swift */, + 1F0DF2ED9232A7CC51F5FCB1 /* STPPaymentMethodAffirmTests.swift */, + CF8E26B31C0F2B256763BDDB /* STPPaymentMethodAfterpayClearpayParamsTest.m */, + 26D0BC36C2610307A01C52CB /* STPPaymentMethodAfterpayClearpayTest.m */, + 8A7FBDBADC1EBC62FFC4A366 /* STPPaymentMethodAUBECSDebitParamsTests.m */, + 54426CBF6F77ABEFBDFDA8C4 /* STPPaymentMethodAUBECSDebitTests.swift */, + 71C175861E0F8882BCF85F78 /* STPPaymentMethodBacsDebitTest.m */, + 130483A5043ACC4D60E0B0D6 /* STPPaymentMethodBancontactParamsTests.m */, + F4E5416F6AE8BED88980D6F8 /* STPPaymentMethodBancontactTests.swift */, + A00F4D8DB80EBD753E9AC961 /* STPPaymentMethodBillingDetailsTest.m */, + 7AC0A18C441FCA394BEF6A3D /* STPPaymentMethodBillingDetailsTests+Link.swift */, + A1C67AC5D415615E9F27D3E3 /* STPPaymentMethodBoletoParamsTests.swift */, + 4E371E9B3B2E343FE954531C /* STPPaymentMethodBoletoTests.swift */, + E0A27E1D77216CE0267C981C /* STPPaymentMethodCardChecksTest.m */, + 1671EC46C713D51013AD7D8B /* STPPaymentMethodCardParamsTest.swift */, + 8BD02D8298877F10F2EF2A9D /* STPPaymentMethodCardTest.swift */, + 1443F134C35E15AB9A1EA560 /* STPPaymentMethodCardWalletMasterpassTest.m */, + 0DF76154CA9CACA9CDB0BA94 /* STPPaymentMethodCardWalletTest.m */, + B5BA1ABFF1739E4051FAE6CB /* STPPaymentMethodCardWalletVisaCheckoutTest.m */, + 0ABB2CA7E96BE249CE8C0566 /* STPPaymentMethodCashAppParamsTests.swift */, + 7DDE50CBC86AD77084C877B6 /* STPPaymentMethodCashAppTests.swift */, + 28B8ADA1C10FED0733B4FAD3 /* STPPaymentMethodEPSParamsTests.m */, + 0C44B4366D6C4FD4B11662C8 /* STPPaymentMethodEPSTests.swift */, + BF8A27B4CE855392E3E3F735 /* STPPaymentMethodFPXTest.m */, + 45BBD120FDBA434BC7E64435 /* STPPaymentMethodFunctionalTest.m */, + 2AE3A8CA366854F49C28BD93 /* STPPaymentMethodGiropayParamsTests.m */, + 583DB466066B47C0F716E474 /* STPPaymentMethodGiropayTests.swift */, + 03CEA8792C992D865ECC46AA /* STPPaymentMethodGrabPayParamsTest.m */, + 415554EBF13241D12094103B /* STPPaymentMethodiDEALTest.m */, + 47E12A0CBFA259A032F7AF0C /* STPPaymentMethodKlarnaParamsTests.swift */, + 4FD94FF270165D699DA89B24 /* STPPaymentMethodKlarnaTests.swift */, + E4442901582C822007C3BB67 /* STPPaymentMethodNetBankingParamsTest.m */, + 739934737B9A09775CD278C9 /* STPPaymentMethodNetBankingTests.swift */, + 1E2638F7AA0906914117C2D5 /* STPPaymentMethodOptionsTest.swift */, + 5F23996EC0E06D47B20316AC /* STPPaymentMethodOXXOParamsTests.m */, + EAD0042713B4C2FEB8916EDC /* STPPaymentMethodOXXOTests.m */, + 7B4DEF058786BD9D3D5EE2FB /* STPPaymentMethodParamsTest.m */, + FE1929CD1B3D3DCC2401E2F8 /* STPPaymentMethodPayPalParamsTests.m */, + 02DEC25F0E12F9F4396E87B2 /* STPPaymentMethodPayPalTests.m */, + CFB4BBE0C4760DB457DE9A57 /* STPPaymentMethodPrzelewy24ParamsTests.m */, + F6558C62376C2397030BD4A6 /* STPPaymentMethodPrzelewy24Tests.swift */, + 8F4D19337AD2936DA69EAC4A /* STPPaymentMethodSEPADebitTest.m */, + 3871630C99B378C67B6E9F7C /* STPPaymentMethodSofortParamsTests.m */, + E8063E8073A32E0B081A1DFA /* STPPaymentMethodSofortTests.swift */, + 148C1D7D1BBBC6B74894A869 /* STPPaymentMethodTest.swift */, + 8B5AC148E7D8B2EBAEDCB364 /* STPPaymentMethodThreeDSecureUsageTest.m */, + EAB51503D64F3464BFBC18CB /* STPPaymentMethodUPIParamsTest.m */, + 940209E5D30E86E856016906 /* STPPaymentMethodUPITests.swift */, + CF13BAEF86594C9CABD4F42A /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift */, + 74FDEF9F687C63BADFB96480 /* STPPaymentMethodUSBankAccountParamsTest.swift */, + 3B112FFF3FCA82094281493F /* STPPaymentMethodUSBankAccountTest.swift */, + 6C7094644EE056260D6F3B67 /* STPPaymentOptionsViewControllerLocalizationTests.swift */, + A2922A32A754CFC9AB8B48AE /* STPPaymentOptionsViewControllerTest.swift */, + 924E878428D15506711CA628 /* STPPhoneNumberValidatorTest.swift */, + AFB9F0B171497353AADD6544 /* STPPIIFunctionalTest.m */, + 336CC555B845DED30208D39D /* STPPinManagementServiceFunctionalTest.swift */, + 917154477796779ECFA1334A /* STPPostalCodeInputTextFieldFormatterTests.swift */, + 090EF7D598B8DE779C275395 /* STPPostalCodeInputTextFieldSnapshotTests.swift */, + 6618739767139C25C05B3631 /* STPPostalCodeInputTextFieldTests.swift */, + EA9975553E669AF69F3CE437 /* STPPostalCodeInputTextFieldValidatorTests.swift */, + FDF7394DDD552EDE996EAD8E /* STPPostalCodeValidatorTest.swift */, + 1A8A6B88797870BC71CCB3AF /* STPPushProvisioningDetailsFunctionalTest.swift */, + 2D878F923A1F69B58D6B2812 /* STPRadarSessionFunctionalTest.swift */, + E3810757AF83E98307634814 /* STPRedirectContextTest.m */, + 6CC0B1FC92A573AAEA4F4E94 /* STPSetupIntentConfirmParamsTest.swift */, + 60E21B0DB0D3B307040EA88E /* STPSetupIntentFunctionalTest.m */, + 49AA313E068FB99CEAA5F7D3 /* STPSetupIntentFunctionalTest.swift */, + 6C0F5E7908FE3C7606DEAAD5 /* STPSetupIntentLastSetupErrorTest.m */, + 3BCC4BB345F320EEECE0437C /* STPSetupIntentTest.m */, + DE3D0F21FB3B3E8BCA15FF7C /* STPShippingAddressViewControllerLocalizationTests.swift */, + 789D0B49B0788794739E3DD4 /* STPShippingAddressViewControllerTest.swift */, + C7F54E1E8FFA1505D24538A6 /* STPShippingMethodsViewControllerLocalizationTests.swift */, + 63114D0EAAE2606732DF5AA0 /* STPSourceCardDetailsTest.swift */, + 20D90EE8CC6606CAE9B06C63 /* STPSourceFunctionalTest.m */, + D1681629DCEA0765515D3D49 /* STPSourceOwnerTest.m */, + 1F6CB4B8FAD14B4D70A63595 /* STPSourceParamsTest.swift */, + 620CEE3CCA05E49698C4DE35 /* STPSourceReceiverTest.m */, + B6799CC0D9C5FED26EE3494F /* STPSourceRedirectTest.m */, + 342398F7CB61693AAADE3F26 /* STPSourceSEPADebitDetailsTest.m */, + B7B5B626BCED2938C256BF49 /* STPSourceTest.m */, + 5377FE4A168F45625923AF37 /* STPSourceVerificationTest.m */, + 51E62BB62EA9B782778CA880 /* STPStackViewWithSeparatorSnapshotTests.swift */, + 9290B75648F2D23764FA71E9 /* STPSTPViewWithSeparatorSnapshotTests.m */, + 6B527265D186B66150D2787E /* STPStringUtilsTest.m */, + EB6AE83989B0596F0C111E13 /* STPStringUtilsTest.swift */, + E315168EF07F52B733EA77F8 /* STPSwiftFixtures.swift */, + E3AC82FE96A69220B300FBE6 /* STPTestAPIClient+Swift.swift */, + AA0FCB6E05CDF74CC1C42042 /* STPTestingAPIClient.h */, + C1AAF76C3F4850217C28E362 /* STPTestingAPIClient.m */, + 72AC063833D29BC984B008A8 /* STPTestUtils.h */, + 9F8F79CBFDC930A18998D7B1 /* STPTestUtils.m */, + 2A8A2CD759D465290066EF65 /* STPTextFieldDelegateProxyTests.swift */, + B6F3B966470A530E0DC53F8C /* STPThreeDSButtonCustomizationTest.swift */, + 483B243268646AE65B06E98C /* STPThreeDSFooterCustomizationTest.swift */, + 5FDD4956E0A04D33F0856F31 /* STPThreeDSLabelCustomizationTest.swift */, + 51408DE266D0345784ADD4FA /* STPThreeDSNavigationBarCustomizationTest.swift */, + C23D612FD5AD7772E1B30DCC /* STPThreeDSSelectionCustomizationTest.swift */, + A583966A33DCDCF04322A592 /* STPThreeDSTextFieldCustomizationTest.swift */, + 3ED44491EB0AC72B1B1A773C /* STPThreeDSUICustomizationTest.swift */, + A14673FA54CB8B85CF8D71C3 /* STPTokenTest.m */, + F22EE02CE12F3EBFACDAE9A4 /* STPUIVCStripeParentViewControllerTests.m */, + DC3AD586DDED620B9E68F461 /* StripeErrorTest.swift */, + B407FE2D39775902A95B1118 /* StripeiOS Tests-Bridging-Header.h */, + 57852C506CC1A2B0F978D98B /* SWHttpTrafficRecorder.h */, + 40DC59F285B3AB0F60B33443 /* SWHttpTrafficRecorder.m */, + 5CE0ABE46E1C27A6AD8D8BD9 /* TextFieldElement+CardTest.swift */, + 51BD2CE41E4F0CF648F44E4A /* TextFieldElement+IBANTest.swift */, + E68F6B90F3BC61A49570FAF4 /* UINavigationBar+StripeTest.m */, + 3B0E131538728BC4802627B1 /* UserDefaults+StripeTest.swift */, + 0DB03E83746FE78361831546 /* WalletHeaderViewSnapshotTests.swift */, + ); + path = StripeiOSTests; + sourceTree = ""; + }; + EB90CE5EA5682921B4C247A0 /* Resources */ = { + isa = PBXGroup; + children = ( + A8F5797C145751AD55858B80 /* 3DSSource.json */, + A32EE0B1FC16687797E7C9DA /* AlipaySource.json */, + 848AAE69CBF96A057F734365 /* ApplePayPaymentMethod.json */, + 2EADE1EE40283DEF52B0B5D4 /* BacsDebitPaymentMethod.json */, + 31F32B2852B1683819BD02F3 /* BancontactSource.json */, + 62169E37BD48681A914FA4CB /* BankAccount.json */, + C861CC96A54A24BDC1303C1B /* Card.json */, + 344C5B4D8789FEFA4CFC0E43 /* CardPaymentMethod.json */, + 582C9C6F076A0FF1780C9769 /* CardSource.json */, + 4928F057B093B6DB5D76D32D /* Customer.json */, + CFE6A87CDA08F216E1DFE4CD /* ElementsSession.json */, + F8CEB050CB2231F1459824EC /* EphemeralKey.json */, + 1020F76D0722D685A97EF202 /* EPSSource.json */, + 3F31C891BDDD871F48F6E60B /* FileUpload.json */, + 682296D57E9486BBCC6ED7F5 /* GiropaySource.json */, + F24B1E06C600CA30A97F5AEA /* iDEALSource.json */, + DFA7A75BA785EBBE4C05DAA3 /* Images.xcassets */, + FE2DED6ABA7407C17C1391B6 /* MockFiles */, + 63A16B33FCF0B351D4A60247 /* MultibancoSource.json */, + A9B50EE8ABC2506A6397C056 /* P24Source.json */, + 68778692A4A89E311FC9DEAE /* PaymentIntent.json */, + FCA64B2C21FCFAC433EA3781 /* recorded_network_traffic */, + 252FD94C68761E893E53D5F9 /* SEPADebitSource.json */, + 0364E74C0FB269E93D18966D /* SetupIntent.json */, + E412BB731BC34BB1A7EC3B8D /* SofortSource.json */, + FD3398E2352CEA0264F20AEA /* stp_test_upload_image.jpeg */, + 0A726D87E8C916E8FEA0C780 /* WeChatPaySource.json */, + ); + path = Resources; + sourceTree = ""; + }; + FBC7A77342A98B1DE8E416B7 /* Resources */ = { + isa = PBXGroup; + children = ( + 2EC2475EABAA6AB3ADF1A1DC /* Images */, + AF41AE099441C2A09DECC1AC /* Localizations */, + ); + path = Resources; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXHeadersBuildPhase section */ + 8D1F9C232F0DF9D129FE872E /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 30D48C62B2FA6B28EC23A5BB /* Stripe-umbrella.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXHeadersBuildPhase section */ + +/* Begin PBXNativeTarget section */ + 1628E8B14F1F7C63CF8C9962 /* StripeiOSTestHostApp */ = { + isa = PBXNativeTarget; + buildConfigurationList = B6D8846CE184BBD9139D6D15 /* Build configuration list for PBXNativeTarget "StripeiOSTestHostApp" */; + buildPhases = ( + C4ED1B5A417CECDB38B21698 /* Sources */, + 9BE857B21623552BA4F3FAA1 /* Resources */, + 9F30C6D40956861F588C2CD0 /* Embed Frameworks */, + C04854A8658964DDD0A15E1C /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = StripeiOSTestHostApp; + productName = StripeiOSTestHostApp; + productReference = 294CD46E24BB2743042872D7 /* StripeiOSTestHostApp.app */; + productType = "com.apple.product-type.application"; + }; + 49A14BBD10B1E97F2B43C448 /* StripeiOSAppHostedTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = FCBB942DEBE57423D79373B4 /* Build configuration list for PBXNativeTarget "StripeiOSAppHostedTests" */; + buildPhases = ( + E160E75F2870000E2FA9C5DC /* Sources */, + C501B175E87CBBB0075EB9C5 /* Resources */, + 16FB342935F756BD2EA7CE4C /* Embed Frameworks */, + 7129AA5AD64208D3B1A76AAE /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + 4E57D9EF1E91091C57E49111 /* PBXTargetDependency */, + 11BFD72CD1F291E0E86EC784 /* PBXTargetDependency */, + ); + name = StripeiOSAppHostedTests; + productName = StripeiOSAppHostedTests; + productReference = 1DD6897858F46976A946394E /* StripeiOSAppHostedTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; + 8BE23AD5D9A3D939AF46F31E /* StripeiOSTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 54F4DB1E9C991950FD9EB4B4 /* Build configuration list for PBXNativeTarget "StripeiOSTests" */; + buildPhases = ( + FB03810F22F4E0919BB2EF68 /* Sources */, + BADDFD2C3E190FF24FEB93C7 /* Resources */, + 5789BE2CD297329ED86678C0 /* Embed Frameworks */, + 6ACBDD66E08F7CEE34A8FFFA /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + C4F1FB6EFF8D25D54580374A /* PBXTargetDependency */, + ); + name = StripeiOSTests; + packageProductDependencies = ( + 62887B4538E4E41E735685E1 /* OHHTTPStubs */, + 911CA85A1610303FA0AF0643 /* OHHTTPStubsSwift */, + E804AA8C4156CC85FFD9595F /* OCMock */, + C55551F29B99CF6D6DD9EE2F /* iOSSnapshotTestCase */, + ); + productName = StripeiOS_Tests; + productReference = CD5AC2BFBC8141F98C00CF9F /* StripeiOS_Tests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; + ADF894AA8F6022D9BED17346 /* StripeiOS */ = { + isa = PBXNativeTarget; + buildConfigurationList = F0FCB32AE130ABA66178DD9B /* Build configuration list for PBXNativeTarget "StripeiOS" */; + buildPhases = ( + 8D1F9C232F0DF9D129FE872E /* Headers */, + 24CDC502E3D468F309116FE1 /* Sources */, + 0F32E5D5E633BAB349B6EB50 /* Resources */, + CAAA5D4C8E87896995960E8C /* Embed Frameworks */, + 1587AD7E52759FBC80315765 /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = StripeiOS; + productName = Stripe; + productReference = 4259421D2CD26E37B96F97B2 /* Stripe.framework */; + productType = "com.apple.product-type.framework"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + E63832AA5BB4225708B7C838 /* Project object */ = { + isa = PBXProject; + attributes = { + TargetAttributes = { + 49A14BBD10B1E97F2B43C448 = { + TestTargetID = 1628E8B14F1F7C63CF8C9962; + }; + }; + }; + buildConfigurationList = 2804CCD7A91C99DF32596147 /* Build configuration list for PBXProject "Stripe" */; + compatibilityVersion = "Xcode 13.0"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + Base, + "bg-BG", + "ca-ES", + "cs-CZ", + da, + de, + "el-GR", + en, + "en-GB", + es, + "es-419", + "et-EE", + fi, + fil, + fr, + "fr-CA", + hr, + hu, + id, + it, + ja, + ko, + "lt-LT", + "lv-LV", + "ms-MY", + mt, + nb, + nl, + "nn-NO", + "pl-PL", + "pt-BR", + "pt-PT", + "ro-RO", + ru, + "sk-SK", + "sl-SI", + sv, + tk, + tr, + vi, + "zh-HK", + "zh-Hans", + "zh-Hant", + ); + mainGroup = E4802964A9471C082CE01BA9; + packageReferences = ( + D244E8B5402CB2CF89CE7872 /* XCRemoteSwiftPackageReference "ocmock" */, + 44C6210EB47ACF468D255723 /* XCRemoteSwiftPackageReference "OHHTTPStubs" */, + 71086E1440B890A9DC01C9DF /* XCRemoteSwiftPackageReference "ios-snapshot-test-case" */, + ); + productRefGroup = 85B4F3432AB3A8B94C9D2591 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + ADF894AA8F6022D9BED17346 /* StripeiOS */, + 8BE23AD5D9A3D939AF46F31E /* StripeiOSTests */, + 1628E8B14F1F7C63CF8C9962 /* StripeiOSTestHostApp */, + 49A14BBD10B1E97F2B43C448 /* StripeiOSAppHostedTests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 0F32E5D5E633BAB349B6EB50 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 01F2C8715D740FDE683FAECF /* stp_card_form_amex_cvc@3x.png in Resources */, + 62D65741A6D49C7EBFB7CB61 /* stp_card_form_back@3x.png in Resources */, + 8CE31C2917AC0B9084C90650 /* stp_card_form_front@3x.png in Resources */, + 5A75EF4349F6B444B6D56552 /* stp_bank_fpx_affin_bank@3x.png in Resources */, + E7071A87146B640C2DEE6B80 /* stp_bank_fpx_alliance_bank@3x.png in Resources */, + F5C771A7C98E78F99E13DBA1 /* stp_bank_fpx_ambank@3x.png in Resources */, + 3F4356512B60B90BFCDA7E01 /* stp_bank_fpx_bank_islam@3x.png in Resources */, + C1E52E633CCF8D18AC804C97 /* stp_bank_fpx_bank_muamalat@3x.png in Resources */, + 83FE814F7C472EB31DD9D28F /* stp_bank_fpx_bank_rakyat@3x.png in Resources */, + 7797B149DF5D5CA201089BC2 /* stp_bank_fpx_bsn@3x.png in Resources */, + C5EAA26F7A4C71DCB08015A5 /* stp_bank_fpx_cimb@3x.png in Resources */, + 187966C4EDE4FEE1FC6B9F24 /* stp_bank_fpx_hong_leong_bank@3x.png in Resources */, + 369C7A7A0DA46C58B02DBA00 /* stp_bank_fpx_hsbc@3x.png in Resources */, + D8D9DF85FE4BF1AB14DBEB95 /* stp_bank_fpx_kfh@3x.png in Resources */, + C87E17137F8E2CBA52B43A98 /* stp_bank_fpx_maybank2e@3x.png in Resources */, + B1B3E2766CB2B4AA48D130F6 /* stp_bank_fpx_maybank2u@3x.png in Resources */, + E611EAD0DBDBAE1AD3442CDC /* stp_bank_fpx_ocbc@3x.png in Resources */, + 7BEC4847DDD51C9C36C758E9 /* stp_bank_fpx_public_bank@3x.png in Resources */, + 2F2923C176FFD66CC5F96A2D /* stp_bank_fpx_rhb@3x.png in Resources */, + F3F2B3317A63067ECA26F3E2 /* stp_bank_fpx_standard_chartered@3x.png in Resources */, + 42B74740CEF8F04A3F34971A /* stp_bank_fpx_uob@3x.png in Resources */, + 31B49323CC357B478431B716 /* stp_fpx_big_logo@3x.png in Resources */, + 2BA38A013A538479C3518424 /* stp_fpx_logo@3x.png in Resources */, + CDBF169718F4882A946BF22D /* stp_icon_add@3x.png in Resources */, + AF0FE00E669705885EAF6F20 /* stp_icon_bank@3x.png in Resources */, + F228C86E3F2172FE6A349243 /* stp_icon_checkmark@3x.png in Resources */, + D5BD5B1657C9D0FD7F6593F3 /* stp_shipping_form@3x.png in Resources */, + 22BE2ABB29F77362FF16D945 /* Localizable.strings in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 9BE857B21623552BA4F3FAA1 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + EEB5E5E9C4E06B148A91C7BD /* Assets.xcassets in Resources */, + 08111F4AD3CA0755420E05F7 /* LaunchScreen.storyboard in Resources */, + DE23FEF74E860620A334FDF5 /* Main.storyboard in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + BADDFD2C3E190FF24FEB93C7 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 0D4211C0767F66A35914FE07 /* 3DSSource.json in Resources */, + 03A2D8DD31042AC83AD2EFC5 /* AlipaySource.json in Resources */, + 75F81259F0DEE1ED09FDECBA /* ApplePayPaymentMethod.json in Resources */, + 5AC2B7D5912DBD0FF7DB2349 /* BacsDebitPaymentMethod.json in Resources */, + A9A77D698551322BF3A067BC /* BancontactSource.json in Resources */, + 77ED42569FEC0EC5AF83538A /* BankAccount.json in Resources */, + 659E5859BD4F4F5BACB6F3C1 /* Card.json in Resources */, + 89C927D276A90357F06ECE3C /* CardPaymentMethod.json in Resources */, + 2CE1FB3A85DE7A74D0A80901 /* CardSource.json in Resources */, + 395E5AE58C4B5B4EE511B182 /* Customer.json in Resources */, + 66ED28E18385B2644F4EF3CC /* EPSSource.json in Resources */, + 6509C8C171F9A76E14F59050 /* ElementsSession.json in Resources */, + E7DE88AF9BDC4EB9E6DE73F1 /* EphemeralKey.json in Resources */, + 45A2E2904C48F19F4D3AD1BF /* FileUpload.json in Resources */, + 5224B2FFB1786C8AF3095248 /* GiropaySource.json in Resources */, + 10342D659764A88A695EF38B /* Images.xcassets in Resources */, + F49D9C4030829D13A6EB45BE /* MockFiles in Resources */, + 7EB99B1286C38DD944D0D9DC /* MultibancoSource.json in Resources */, + 15772D8E06236054D11A1034 /* P24Source.json in Resources */, + 53D0248D7927B0E62B8C967B /* PaymentIntent.json in Resources */, + 87061C07E2B17DD8B7052B72 /* SEPADebitSource.json in Resources */, + 314B144D765DB6CD8ABE8B7D /* SetupIntent.json in Resources */, + EEC27B2210C2353DDDDEEAD1 /* SofortSource.json in Resources */, + 2197EAE8DE995628312490BE /* WeChatPaySource.json in Resources */, + 1AB740655742E5D00E905A4B /* iDEALSource.json in Resources */, + E4BA3AF73442897BCA3A6962 /* recorded_network_traffic in Resources */, + 2AE9ABA774B430E174279FEA /* stp_test_upload_image.jpeg in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + C501B175E87CBBB0075EB9C5 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 24CDC502E3D468F309116FE1 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 315713352C770DA3ED9CBDCD /* Enums+CustomStringConvertible.swift in Sources */, + B6784B7F4B9B04617C0EE510 /* PKAddPaymentPassRequest+Stripe_Error.swift in Sources */, + C0688E067AE4FFDFFDDC03BB /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift in Sources */, + 6BF6ECC4A4E61E2FFC3EA20B /* STPAPIClient+BasicUI.swift in Sources */, + 5212C7875C07F9BF16AFD98D /* STPAPIClient+PushProvisioning.swift in Sources */, + BB46077C256C26418420F240 /* STPAddCardViewController.swift in Sources */, + E63B5BAF6B5645C979BFBA71 /* STPAddress+BasicUI.swift in Sources */, + 58240AA66FAE55131268E4A0 /* STPAddressFieldTableViewCell.swift in Sources */, + 2BD45625F6F665B60C6CAD30 /* STPAddressViewModel.swift in Sources */, + 605EFBDD21426FD30581563F /* STPAnalyticsClient+BasicUI.swift in Sources */, + 7589E37795D21AB818B0C333 /* STPAnalyticsClient+Payments.swift in Sources */, + F86F2DF6E46EFABE23AD5D27 /* STPApplePayContextDelegate.swift in Sources */, + FC166455478EAF51F7C34E68 /* STPApplePayPaymentOption.swift in Sources */, + C314A5C55064C51C2B999E6B /* STPBackendAPIAdapter.swift in Sources */, + CEF318C74D2E44C78EF85306 /* STPBankSelectionTableViewCell.swift in Sources */, + 172D96526023A80534D54CC0 /* STPBankSelectionViewController.swift in Sources */, + A0AA0B8AEF5B429858D71F6B /* STPBlocks.swift in Sources */, + DDBF5AAE607C698618DDE865 /* STPCameraView.swift in Sources */, + 66065B1D65D7D5502D4E2F2B /* STPCard+BasicUI.swift in Sources */, + BAFD06E994739E1C38DFFBBC /* STPCardScanner.swift in Sources */, + 2FFA7C2D1C7337FDB4C608A5 /* STPCardScannerTableViewCell.swift in Sources */, + 2F18A1903244E144C7802E09 /* STPCardValidationState.swift in Sources */, + DF73457BF349BC962A6AC502 /* STPCoreScrollViewController.swift in Sources */, + EEFFE199D9769FF449BFD7FF /* STPCoreTableViewController.swift in Sources */, + B1BF689B91D538BDCA4C8578 /* STPCoreViewController.swift in Sources */, + 429DBA641E926EBC2D049FE7 /* STPCustomerContext.swift in Sources */, + 62B91808A088C4F9FDB62C53 /* STPEphemeralKey.swift in Sources */, + 4FB67F10A0B7106A8142B842 /* STPEphemeralKeyManager.swift in Sources */, + B8385576DC25BDEEB92D812F /* STPEphemeralKeyProvider.swift in Sources */, + AF23CB4EF17E87007CFC3E96 /* STPFPXBankStatusResponse.swift in Sources */, + 0B9C0E9A7A750607413C9E53 /* STPFakeAddPaymentPassViewController.swift in Sources */, + 5D6B52EB4D7258129F134D07 /* STPImageLibrary.swift in Sources */, + D2869246B446B8B31F1CD368 /* STPIntentActionLinkAuthenticateAccount.swift in Sources */, + 98EE8326C1D133E1C998114F /* STPLocalizedString.swift in Sources */, + 609E4D384B75F6A111DC0E27 /* STPPaymentActivityIndicatorView.swift in Sources */, + 4EFF8B46B12DA4D9AAB22523 /* STPPaymentCardTextFieldCell.swift in Sources */, + 446A108C8EB6C338A1D774F8 /* STPPaymentConfiguration.swift in Sources */, + B00F7FC372E376C6B2170D37 /* STPPaymentContext.swift in Sources */, + 447C19BDB2CF5445045F81F7 /* STPPaymentContextAmountModel.swift in Sources */, + EA7FEC518AA07BA59405A5E3 /* STPPaymentIntentParams+BasicUI.swift in Sources */, + 7BC98BE168781C5B3EC8A8DB /* STPPaymentMethod+BasicUI.swift in Sources */, + 812682EA323986B8F698FF3C /* STPPaymentMethodParams+BasicUI.swift in Sources */, + F10FC337254A34ED8F13E341 /* STPPaymentOption.swift in Sources */, + 4ED44ACF24949F516867235C /* STPPaymentOptionTableViewCell.swift in Sources */, + 5ECED204FD22CFEA3A806767 /* STPPaymentOptionTuple.swift in Sources */, + 1F432D0B37949217E4299A20 /* STPPaymentOptionsInternalViewController.swift in Sources */, + 69AC1EDE2A3C03B1D980CA54 /* STPPaymentOptionsViewController.swift in Sources */, + 307FD6A103EF7AF3CE451598 /* STPPaymentResult.swift in Sources */, + 7EAA7334372DBC38DF8FA0AA /* STPPinManagementService.swift in Sources */, + 2E35B0FB60FCBE7608080642 /* STPPushProvisioningContext.swift in Sources */, + FEE74744B657F86873EA2F3D /* STPPushProvisioningDetails.swift in Sources */, + 4E09E54E7FEC35C49C59A379 /* STPPushProvisioningDetailsParams.swift in Sources */, + 23D1246A5DAB5333650F104F /* STPSectionHeaderView.swift in Sources */, + 124D43C1A633922B1DA3E1E7 /* STPShippingAddressViewController.swift in Sources */, + 7B9C0D039EA9EF593AEC682D /* STPShippingMethodTableViewCell.swift in Sources */, + 7F9D08AC5A448C7693162D7D /* STPShippingMethodsViewController.swift in Sources */, + 3CE88568CB9648D6F1503B88 /* STPSource+BasicUI.swift in Sources */, + F835CEC935464FF32726A0A0 /* STPTheme.swift in Sources */, + 279D2BA91198E18730626CE6 /* STPUserInformation.swift in Sources */, + 542610492B38FEB652C6823E /* String+Localized.swift in Sources */, + DCF615643A22D0A7B739547C /* Stripe+Exports.swift in Sources */, + BF4ED4828114E2E89A3D4AB7 /* StripeBundleLocator.swift in Sources */, + 98E2332DE7F54E970BE5EEF7 /* UIBarButtonItem+Stripe.swift in Sources */, + 9D9692DFC4F06F8C70145000 /* UINavigationBar+Stripe_Theme.swift in Sources */, + 9C13E8A017A4E23BCCDE618B /* UINavigationController+Stripe_Completion.swift in Sources */, + 54331380F5AC68846DBE94D5 /* UITableViewCell+Stripe_Borders.swift in Sources */, + 093FE3D65978E3DB6B79AE05 /* UIToolbar+Stripe_InputAccessory.swift in Sources */, + C8A6CA6352B7C8FEE3D91476 /* UIView+Helpers.swift in Sources */, + 7F235CD649F6E97E4E7DD180 /* UIView+Stripe_FirstResponder.swift in Sources */, + 44672917D3AC4B83F9EC3BC3 /* UIView+Stripe_SafeAreaBounds.swift in Sources */, + CC072EBAD035AA54A2AD3ABC /* UIViewController+Stripe_KeyboardAvoiding.swift in Sources */, + 3FA556CF8B11E2486F505161 /* UIViewController+Stripe_NavigationItemProxy.swift in Sources */, + 3930ECBEE003772C1245D25B /* UIViewController+Stripe_ParentViewController.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + C4ED1B5A417CECDB38B21698 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 234C71F480318E9062075924 /* AppDelegate.swift in Sources */, + 3C1A7B9810B038177FF1CF52 /* ViewController.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + E160E75F2870000E2FA9C5DC /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 64801CF2D2CAC9008C17D154 /* LinkSecureCookieStoreTests.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + FB03810F22F4E0919BB2EF68 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 3EB3745F556EA12AB27A8545 /* APIRequestTest.swift in Sources */, + 26D73AA93A48A1112FECD86D /* AddPaymentMethodViewControllerSnapshotTests.swift in Sources */, + 34F3ABB5BF9D3645D3B6DA80 /* AddressViewControllerSnapshotTests.swift in Sources */, + 6EF3F611E6EA3CB479D62450 /* AfterpayPriceBreakdownViewSnapshotTests.swift in Sources */, + DF85F5EC6E16CAD21491891A /* AnalyticsHelperTests.swift in Sources */, + 5910FCB9822259D5EC7E4051 /* AutoCompleteViewControllerSnapshotTests.swift in Sources */, + 3EFC86756F23D8AE6708ECE2 /* ButtonLinkSnapshotTests.swift in Sources */, + C1E70FD29BBE36D76A7E6929 /* CardExpiryDateTests.swift in Sources */, + D375ADBD1F4B48380D5347D1 /* CircularButtonSnapshotTests.swift in Sources */, + 2F9FA9CBCA3C0CE52FAC9B6B /* ConfirmButtonSnapshotTests.swift in Sources */, + 3C1E9069CD03ED9981D7F3E2 /* ConfirmButtonTests.swift in Sources */, + 795F3783D62AB8E2A00DCD05 /* ConsumerSessionTests.swift in Sources */, + 0DFA17378D894C70D72C9F62 /* Error+PaymentSheetTests.swift in Sources */, + C9E66A22494C02050AE34A9B /* FBSnapshotTestCase+STPViewControllerLoading.swift in Sources */, + 35C1CF73701EECC7DB6AB722 /* FormSpecProviderTest.swift in Sources */, + ACF6CFE0F8B88FDBBB16968C /* FraudDetectionDataTest.swift in Sources */, + AF18D569B296BFC1EB5A7338 /* ImageTest.swift in Sources */, + 87AACDD643A998FFDD505D22 /* KlarnaHelperTest.swift in Sources */, + 5D1A9F97F79DAA3F46C82A28 /* LinkAccountServiceTests.swift in Sources */, + A0C3E888938414D185B71CF7 /* LinkBadgeViewSnapshotTest.swift in Sources */, + FBBDAEA5DF5CB3A138D82E8A /* LinkCardEditElementSnapshotTests.swift in Sources */, + 229E25F8DFCC55CA9EDD15AB /* LinkInMemoryCookieStoreTests.swift in Sources */, + 67E18B3DA020900C83AE29B9 /* LinkInlineSignupElementSnapshotTests.swift in Sources */, + 4F315E738D87C06682C4C504 /* LinkInstantDebitMandateViewSnapshotTests.swift in Sources */, + D0C81317E0AA8EB0370B1BA1 /* LinkLegalTermsViewSnapshotTests.swift in Sources */, + 6326FD82A0E6AFB10C773B03 /* LinkNavigationBarSnapshotTests.swift in Sources */, + AB3BD6A5E8660EC6D2C299BD /* LinkNoticeViewSnapshotTests.swift in Sources */, + 2F866A52BA39ED5D32BCA9DD /* LinkPaymentMethodPickerSnapshotTests.swift in Sources */, + 450FAE41FB4538462D05F2E4 /* LinkSignupViewModelTests.swift in Sources */, + 6964CE3F9E07A53AA2954E8E /* LinkStubs.swift in Sources */, + 91F90088C3F1102E3C041014 /* LinkToastSnapshotTests.swift in Sources */, + BAEE96DF60B42EE2A559DC9F /* LinkVerificationViewSnapshotTests.swift in Sources */, + FDD1858CAEFCEBB22BEC9BBC /* MKPlacemark+PaymentSheetTests.swift in Sources */, + 609C2C8F10AFAA2711639CD0 /* NSArray+StripeTest.swift in Sources */, + B98D71ED9ACC2E1B47372F53 /* NSDecimalNumber+StripeTest.swift in Sources */, + 6E7AD3CCC966A7F34922B172 /* NSDictionary+StripeTest.swift in Sources */, + 317275D3FB6DC708BD905040 /* NSLocale+STPSwizzling.m in Sources */, + C4C1295E7DA618DFB944A534 /* NSString+StripeTest.swift in Sources */, + 68318DB86DFCD19505FC47BA /* NSURLComponents_StripeTest.swift in Sources */, + D83F76F584BC345CFBA71CF8 /* OneTimeCodeTextFieldSnapshotTests.swift in Sources */, + 951344464ACF84F0F6D43D10 /* OneTimeCodeTextFieldTests.swift in Sources */, + 8E423294AB602BF25DB11D8E /* OperationDebouncerTests.swift in Sources */, + C4DC3F4FA93A3BAF6EE782A0 /* PKPayment+StripeTest.swift in Sources */, + DC57D2DC40C6BA0C9CF7EC92 /* PayWithLinkButtonSnapshotTests.swift in Sources */, + 44A68EC2EE889D75474422F9 /* PayWithLinkViewController-WalletViewModelTests.swift in Sources */, + AD9B9F3FF697D4A3892E86F2 /* PaymentAnalyticTest.swift in Sources */, + C71B55CAA87D9B20B68E0608 /* PaymentMethodMessagingViewFunctionalTest.swift in Sources */, + B92C22D02FDA5BC7FCDF8EEF /* PaymentMethodMessagingViewSnapshotTests.swift in Sources */, + 9708289AA2B48C0828F21FA1 /* PaymentSheet+APITest.swift in Sources */, + 5DD402F5E453D4A2194A346B /* PaymentSheetAddressTests.swift in Sources */, + 8DCE81502850557222978D6A /* PaymentSheetFormFactorySnapshotTest.swift in Sources */, + 181B8C34A3FFCF2DBEF0086E /* PaymentSheetFormFactoryTest.swift in Sources */, + 5A6CF4BEE7E5B3587217C848 /* PaymentSheetLinkAccountTests.swift in Sources */, + 9252E3E10BD2FEC4CDD05959 /* PaymentSheetPaymentMethodTypeTest.swift in Sources */, + 9668F3E0DC7FAC4725B8C446 /* PaymentSheetTestUtils.swift in Sources */, + C861BB9EAAD04949E338D7FF /* PaymentTypeCellSnapshotTests.swift in Sources */, + BC694A1642DC30D530B60635 /* RotatingCardBrandsViewSnapshotTests.swift in Sources */, + C5D295FE9988CA80ABA57801 /* RotatingCardBrandsViewTests.swift in Sources */, + 49A77DA7B76E498C9F23D428 /* STPAPIClientNetworkBridgeTest.m in Sources */, + 9A24970C5FB6D3F7314AE550 /* STPAPIClientStubbedTest.swift in Sources */, + 5E5EE69D140F6FEDA5F0A346 /* STPAPIClientTest.swift in Sources */, + 825CBE20F190E96EFA95B35A /* STPAPISettingsBridgeTest.m in Sources */, + 17BD7C0391F3182E32A63D6B /* STPAUBECSDebitFormViewSnapshotTests.swift in Sources */, + 583DE9869C885BA02E0A071E /* STPAUBECSFormViewModelTests.swift in Sources */, + B6B89E3F7DE0811BD5CB9D31 /* STPAddCardViewControllerLocalizationTests.swift in Sources */, + FB34906C9215D0E03850064B /* STPAddCardViewControllerTest.swift in Sources */, + 4ABF7BEF11E1F4715DCFF446 /* STPAddressTests.m in Sources */, + F53E04785DB804EA5C2AAC18 /* STPAddressViewModelTest.swift in Sources */, + 0CBBE909CA773D7D45B9AD4C /* STPAnalyticsClientPaymentSheetTest.swift in Sources */, + E6F428CFAD64979A8874B00B /* STPAnalyticsClientPaymentsTest.swift in Sources */, + F30F6DA35482988D8EC9FCEB /* STPApplePayContextFunctionalTest.m in Sources */, + 6F9525063D76A9F86A10CCBF /* STPApplePayContextFunctionalTestExtras.swift in Sources */, + F2655328479314A9C8718DE4 /* STPApplePayContextTest.swift in Sources */, + 9B149DA42FB38C3542E0CB4B /* STPApplePayFunctionalTest.swift in Sources */, + 2990B7B1513ACA58065A1D0B /* STPApplePayPaymentOptionTest.m in Sources */, + AE9AB2E4E82CC2AFF8B2DC96 /* STPApplePayTest.m in Sources */, + FBBA3B39598BBECB664C5E7F /* STPApplePayTest.swift in Sources */, + D0342D50F9AC319919D93D59 /* STPBECSDebitAccountNumberValidatorTests.swift in Sources */, + 66B7EF2DC1CBF813707C767C /* STPBSBNumberValidatorTests.swift in Sources */, + B7CE5D774F8BA17274995BA2 /* STPBankAccountFunctionalTest.m in Sources */, + A7244AE4E4D7E5BC66303F62 /* STPBankAccountParamsTest.m in Sources */, + E1EA6387717F8ACD3637AA78 /* STPBankAccountTest.m in Sources */, + 6F4FBB4F10B5DB2CF8BB3460 /* STPBinRangeTest.swift in Sources */, + B917BF282C84507292112B9D /* STPCardBINMetadataTests.swift in Sources */, + C3C01719939575E1519EEC2C /* STPCardBrandTest.m in Sources */, + FE6647242714D9BEA1EBC055 /* STPCardCVCInputTextFieldFormatterTests.swift in Sources */, + 8EC1820299152F8565D30A40 /* STPCardCVCInputTextFieldSnapshotTests.swift in Sources */, + F35E090A607EB5F86FFC3D31 /* STPCardCVCInputTextFieldTests.swift in Sources */, + D2C062CE4E54094B1AC33E78 /* STPCardCVCInputTextFieldValidatorTests.swift in Sources */, + AC35943F1EAD50E9D5D509B3 /* STPCardExpiryInputTextFieldFormatterTests.swift in Sources */, + A66C279957B6AC8F72DE05C7 /* STPCardExpiryInputTextFieldSnapshotTests.swift in Sources */, + C57BDA835AED735321906977 /* STPCardExpiryInputTextFieldValidatorTests.swift in Sources */, + 07A5CDBFDF2340BAD99D6EB3 /* STPCardFormViewSnapshotTests.swift in Sources */, + 246920234EE8382FB4E56516 /* STPCardFormViewTests.swift in Sources */, + 50D084E50661D174297FC30B /* STPCardFunctionalTest.m in Sources */, + CF2E17AC77EB08393B8A3F98 /* STPCardNumberInputTextFieldFormatterTests.swift in Sources */, + EEBA9A95E8057A06E5E7C103 /* STPCardNumberInputTextFieldSnapshotTests.swift in Sources */, + 1A058C42C4703458CA1CA522 /* STPCardNumberInputTextFieldValidatorTests.swift in Sources */, + 222F49985A1817D8D41D8B56 /* STPCardParamsTest.m in Sources */, + D15160C0F0763078DBB434E4 /* STPCardTest.swift in Sources */, + 1E8D8E2494062262A332879C /* STPCardValidatorTest.swift in Sources */, + A781FB0F586B26655FAEC3C0 /* STPCertTest.swift in Sources */, + 9FFC5B74959F202EDF277DF8 /* STPConfirmCardOptionsTest.m in Sources */, + 9482B2A9A13CA7F5F8C79780 /* STPConfirmPaymentMethodOptionsTest.m in Sources */, + FF00A8AACAA255F89D9034A0 /* STPConnectAccountAddressTest.m in Sources */, + 9BD0FF43A32AAAA30A5973A5 /* STPConnectAccountFunctionalTest.m in Sources */, + DF7C5EA613A5B8F5D1431622 /* STPConnectAccountParamsTest.m in Sources */, + 9D464A252FBD0D4E2A0A7398 /* STPCountryPickerInputFieldSnapshotTests.swift in Sources */, + 4C3B161481D11385352B06D4 /* STPCustomerContextTest.swift in Sources */, + 89E246E74FACFAB5EFBA6980 /* STPCustomerTest.m in Sources */, + CBCA59D39B30D869B4FDC04B /* STPE2ETest.swift in Sources */, + 0F0F35439565AA0D284A6A70 /* STPElementsSessionTest.swift in Sources */, + 53853E75323918758A1A3B35 /* STPEphemeralKeyManagerTest.m in Sources */, + B6656829DEC006DBEED2AA0E /* STPEphemeralKeyTest.swift in Sources */, + 7435E6BB6971012A9B0DB52E /* STPErrorBridgeTest.m in Sources */, + B9EE4AC9206FD77A2FB8C702 /* STPFPXBankBrandTest.m in Sources */, + 97F2F817247B1CB07F1E6600 /* STPFileFunctionalTest.m in Sources */, + B3C3DCC5BD2BE09DEC0906F0 /* STPFileTest.m in Sources */, + FDDA6674A050209FF8CB7E30 /* STPFixtures+Swift.swift in Sources */, + 8A3B74851E988FC00BE172E1 /* STPFixtures.m in Sources */, + BC6912C0DE15008C8D8C303C /* STPFloatingPlaceholderTextFieldSnapshotTests.swift in Sources */, + 91A839DEDA7D1EAF6FC66BE0 /* STPFormEncoderTest.swift in Sources */, + 96098727EFA6A72087A35A52 /* STPFormTextFieldTest.swift in Sources */, + 013F991AB34E38BDBA6E4521 /* STPFormViewSnapshotTests.swift in Sources */, + 51044B947A7FDB99451466D8 /* STPGenericInputPickerFieldSnapshotTests.swift in Sources */, + 8532FEBF4F2E0EB282D466CE /* STPGenericInputPickerFieldValidatorTest.swift in Sources */, + 4B0917FC15BF56D0100E0ED1 /* STPGenericInputTextFieldSnapshotTests.swift in Sources */, + 3AD22E0BD44B02D968C6569A /* STPImageLibraryTest.swift in Sources */, + D7C555B36C282B99E22B8D45 /* STPInputTextFieldFormatterTests.swift in Sources */, + 903FFB756C6ED520BE38EF6F /* STPInputTextFieldValidatorTests.swift in Sources */, + 45FA9B8CC2D18E29BE81CF8F /* STPIntentActionAlipayHandleRedirectTest.swift in Sources */, + AE90123C3DCF6C56329ABD72 /* STPIntentActionTest.m in Sources */, + 4935C8B3ECFBAD947E694934 /* STPIntentActionTypeTest.swift in Sources */, + 8F0326E98C74EB62E34B9FEA /* STPIntentActionWeChatPayRedirectToAppTest.swift in Sources */, + 1F417D0874CC86F4C9AB2790 /* STPIntentWithPreferencesTest.swift in Sources */, + 5660A88BD792A7A844084F16 /* STPLabeledFormTextFieldViewSnapshotTests.m in Sources */, + A22D548084E7DE1FE5ABE8E7 /* STPLabeledMultiFormTextFieldViewSnapshotTests.swift in Sources */, + 331924F0801287BAD413FDCB /* STPMandateCustomerAcceptanceParamsTest.swift in Sources */, + 781EC0163AC001C6A66045B6 /* STPMandateDataParamsTest.swift in Sources */, + B82859A4444B9F735720F232 /* STPMandateOnlineParamsTest.swift in Sources */, + 9FD92B3ADEBEC96660B70409 /* STPMocks.m in Sources */, + 542B23E3AC288AF2F4E5D6C8 /* STPNetworkStubbingTestCase.m in Sources */, + F481DAE25F9957D23F529CF7 /* STPNetworkStubbingTestCase.swift in Sources */, + A77C5769B20D7884FC8FC4FB /* STPNumericDigitInputTextFormatterTests.swift in Sources */, + F975CE029DF30419B8DB0D8F /* STPNumericStringValidatorTests.swift in Sources */, + 41A4FD6DA3754A61880111D4 /* STPPIIFunctionalTest.m in Sources */, + 82B4D3FC63C069728459BABD /* STPPaymentCardTextFieldTest.m in Sources */, + 86BAF121184D71F5F4FFAD7B /* STPPaymentCardTextFieldTestsSwift.swift in Sources */, + B4719234E4BBDAD260E31373 /* STPPaymentCardTextFieldViewModelTest.swift in Sources */, + 3AAE488F2461A46143B3A687 /* STPPaymentConfigurationTest.m in Sources */, + 829D43B6705D125FEC9926DA /* STPPaymentContextApplePayTest.swift in Sources */, + 258E3F6C9DBB2B510CCEC525 /* STPPaymentContextSnapshotTests.m in Sources */, + FEF2E0DAC862FF42B814AFCA /* STPPaymentHandlerFunctionalTest.m in Sources */, + 194154708E1A9E013DCE2C72 /* STPPaymentHandlerStubbedMockedFilesTests.swift in Sources */, + 2A528B7B2579E5F977797822 /* STPPaymentHandlerTests.swift in Sources */, + 1BC4044802EE7D3E2643DC84 /* STPPaymentIntentEnumsTest.swift in Sources */, + 672B820564FFBAFFAD93B27E /* STPPaymentIntentFunctionalTest.m in Sources */, + 37E9160706C9EEEFEF133617 /* STPPaymentIntentFunctionalTest.swift in Sources */, + F729E784CFFC1F79EF5F2ABE /* STPPaymentIntentLastPaymentErrorTest.swift in Sources */, + CA189278AD606BEAC62D545F /* STPPaymentIntentParamsTest.swift in Sources */, + 73AFE2A8839EFAB8330F6CF0 /* STPPaymentIntentTest.swift in Sources */, + 6D578C4501AD509C88010ABB /* STPPaymentMethodAUBECSDebitParamsTests.m in Sources */, + D567569568C0D8F2D7B179B3 /* STPPaymentMethodAUBECSDebitTests.swift in Sources */, + D8C7D8B5749708833F89024B /* STPPaymentMethodAddressTest.m in Sources */, + 4993037E5386D0AF87B24871 /* STPPaymentMethodAffirmParamsTest.swift in Sources */, + F5CC4F320D09A06F0B21ABE6 /* STPPaymentMethodAffirmTests.swift in Sources */, + 0A181790DAF17BD039F01B15 /* STPPaymentMethodAfterpayClearpayParamsTest.m in Sources */, + 621CBBB2E116055B85E860B8 /* STPPaymentMethodAfterpayClearpayTest.m in Sources */, + 64A685F033DC45A99FB3E300 /* STPPaymentMethodBacsDebitTest.m in Sources */, + F530ECEDE5FFDD9B4321CCD4 /* STPPaymentMethodBancontactParamsTests.m in Sources */, + 1CCFC43F7FCD273E2100D321 /* STPPaymentMethodBancontactTests.swift in Sources */, + DEC21E8B6DD667AD876ABA0E /* STPPaymentMethodBillingDetailsTest.m in Sources */, + 29428CDB658E6F504402D844 /* STPPaymentMethodBillingDetailsTests+Link.swift in Sources */, + 8C977F8D224A7360AE8E15A7 /* STPPaymentMethodBoletoParamsTests.swift in Sources */, + 5170651536332C4842E9D009 /* STPPaymentMethodBoletoTests.swift in Sources */, + EE4E023ABD6C611BA3EDDD17 /* STPPaymentMethodCardChecksTest.m in Sources */, + 8378F2A4B0796819BB1C6C54 /* STPPaymentMethodCardParamsTest.swift in Sources */, + 43FFF2881D4EFA7B57A60E09 /* STPPaymentMethodCardTest.swift in Sources */, + 35048D07C8323A96161F63C5 /* STPPaymentMethodCardWalletMasterpassTest.m in Sources */, + D5EECD0F9CD1DBC927E3E4E2 /* STPPaymentMethodCardWalletTest.m in Sources */, + 088B79E49A52A57B74B23F6E /* STPPaymentMethodCardWalletVisaCheckoutTest.m in Sources */, + 0185AC6B123CD73E877D4FCE /* STPPaymentMethodCashAppParamsTests.swift in Sources */, + CBAF9C6F87F746F17495ADC2 /* STPPaymentMethodCashAppTests.swift in Sources */, + D597AF40C2DFEE3321566DD0 /* STPPaymentMethodEPSParamsTests.m in Sources */, + C7EB8FB325BF491FDE25FE66 /* STPPaymentMethodEPSTests.swift in Sources */, + 3B6EF1B8C8C6A37D220AF282 /* STPPaymentMethodFPXTest.m in Sources */, + 06FB8C17B3AA4957FD0F3CD2 /* STPPaymentMethodFunctionalTest.m in Sources */, + CF6AB51AA76503F2EDD42BED /* STPPaymentMethodGiropayParamsTests.m in Sources */, + B8ED1F697519A6FCD3D79431 /* STPPaymentMethodGiropayTests.swift in Sources */, + ACE783C5E133EB9962908BA8 /* STPPaymentMethodGrabPayParamsTest.m in Sources */, + 7844BB705AEB002965EF82B0 /* STPPaymentMethodKlarnaParamsTests.swift in Sources */, + 27F1783CBFEC06BFD6C114F6 /* STPPaymentMethodKlarnaTests.swift in Sources */, + 2199D89054CACD1658CDC2F1 /* STPPaymentMethodNetBankingParamsTest.m in Sources */, + FF0F9BA6FE4B88297A434EA7 /* STPPaymentMethodNetBankingTests.swift in Sources */, + 7A1F658DC9D1494153AF215E /* STPPaymentMethodOXXOParamsTests.m in Sources */, + 3573889F65E85478DE770A49 /* STPPaymentMethodOXXOTests.m in Sources */, + DD8E2B99BAE917F83258DC35 /* STPPaymentMethodOptionsTest.swift in Sources */, + AE94F473534AE66F984D0254 /* STPPaymentMethodParamsTest.m in Sources */, + 25E98D9074E582E91A10F5FF /* STPPaymentMethodPayPalParamsTests.m in Sources */, + B7519705686EAA1BC4F0BC5A /* STPPaymentMethodPayPalTests.m in Sources */, + 8DB0749467DC61349532FB7B /* STPPaymentMethodPrzelewy24ParamsTests.m in Sources */, + D7D24DCC9402153965AF7F1B /* STPPaymentMethodPrzelewy24Tests.swift in Sources */, + D4362E8B9ACB60F1DCDEDBCF /* STPPaymentMethodSEPADebitTest.m in Sources */, + 284BCD4B0744CDD91F7D8B15 /* STPPaymentMethodSofortParamsTests.m in Sources */, + 5C5E1CE53D89DE8F0B867115 /* STPPaymentMethodSofortTests.swift in Sources */, + A08C2F0E7F642515B1D263ED /* STPPaymentMethodTest.swift in Sources */, + 4048B897F3AD627A0BF36D62 /* STPPaymentMethodThreeDSecureUsageTest.m in Sources */, + 78F39965489F75A82584F7E9 /* STPPaymentMethodUPIParamsTest.m in Sources */, + AF44725558E654548FED2A2B /* STPPaymentMethodUPITests.swift in Sources */, + 4AAA2CD5AEF1F913395B3B95 /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift in Sources */, + 7D251ABF1EBF65ACA8A4BDD4 /* STPPaymentMethodUSBankAccountParamsTest.swift in Sources */, + 225140E0BD9C0630116DDE4A /* STPPaymentMethodUSBankAccountTest.swift in Sources */, + 50FB2476C611B94E5D283836 /* STPPaymentMethodiDEALTest.m in Sources */, + 66C38EA9CFB1ED4DF2F974BF /* STPPaymentOptionsViewControllerLocalizationTests.swift in Sources */, + 2C7991FDF7B374E0E65E253F /* STPPaymentOptionsViewControllerTest.swift in Sources */, + 07BF3CF1656AF5F5A0678873 /* STPPhoneNumberValidatorTest.swift in Sources */, + EBD436689635CC28A24DECD4 /* STPPinManagementServiceFunctionalTest.swift in Sources */, + 385CAC4D2FF119D2E925916B /* STPPostalCodeInputTextFieldFormatterTests.swift in Sources */, + 9B1AC278FDCDABF26C5E468C /* STPPostalCodeInputTextFieldSnapshotTests.swift in Sources */, + 4E31B1864DA407598FB1BBC6 /* STPPostalCodeInputTextFieldTests.swift in Sources */, + 91558F51B87C72E745244958 /* STPPostalCodeInputTextFieldValidatorTests.swift in Sources */, + 6FCA954C32AB351F902BA876 /* STPPostalCodeValidatorTest.swift in Sources */, + 3172C789DF2CE133ECA359D7 /* STPPushProvisioningDetailsFunctionalTest.swift in Sources */, + 4A61DC36F10B9C9C24345613 /* STPRadarSessionFunctionalTest.swift in Sources */, + 3D0C7EA60D59DFC66F0E02FE /* STPRedirectContextTest.m in Sources */, + 47E93085491222CB4C7FA9D4 /* STPSTPViewWithSeparatorSnapshotTests.m in Sources */, + 044B7BECFBDB1F6C8CA08514 /* STPSetupIntentConfirmParamsTest.swift in Sources */, + 72CB8A91146F3EC17CEFA235 /* STPSetupIntentFunctionalTest.m in Sources */, + C32D7ACEBC852CBC295BBEF2 /* STPSetupIntentFunctionalTest.swift in Sources */, + C83D461C72BE7E4A309C8452 /* STPSetupIntentLastSetupErrorTest.m in Sources */, + 3DE8D85607C0F78C67E35DF4 /* STPSetupIntentTest.m in Sources */, + A77EC5CE65161573062E9F98 /* STPShippingAddressViewControllerLocalizationTests.swift in Sources */, + 08ED7A4EB7E64FDAED2C2D39 /* STPShippingAddressViewControllerTest.swift in Sources */, + A7A1D3C0D75DCD7217D297FF /* STPShippingMethodsViewControllerLocalizationTests.swift in Sources */, + AE747ADA2841AA06F32558D8 /* STPSourceCardDetailsTest.swift in Sources */, + 33742D8595DCB200A2507B0A /* STPSourceFunctionalTest.m in Sources */, + E2FECEC25C1C69E4969B0E2A /* STPSourceOwnerTest.m in Sources */, + D8BECFB70834CC42BA6706D8 /* STPSourceParamsTest.swift in Sources */, + E32AB1FF0BFE8979650FDF15 /* STPSourceReceiverTest.m in Sources */, + AE5C68FE305F63791B59CDD1 /* STPSourceRedirectTest.m in Sources */, + 65E6B1133364DA6854F570A2 /* STPSourceSEPADebitDetailsTest.m in Sources */, + 64D168991F7FF6CB9A223F54 /* STPSourceTest.m in Sources */, + FAD79E127796E97B3C1693FD /* STPSourceVerificationTest.m in Sources */, + D7956073A8FD3785193E0577 /* STPStackViewWithSeparatorSnapshotTests.swift in Sources */, + 3DC523AEEA01D68424C8B41B /* STPStringUtilsTest.m in Sources */, + CA4F392070740C56FE2BB461 /* STPStringUtilsTest.swift in Sources */, + 9D8354BDB04CEC5D1EFCF54F /* STPSwiftFixtures.swift in Sources */, + B5A0B3CF303CD41FF73D4310 /* STPTestAPIClient+Swift.swift in Sources */, + 59D4B49B6C7EEF121EE266E4 /* STPTestUtils.m in Sources */, + AD731C41CE95233A733E0289 /* STPTestingAPIClient.m in Sources */, + E3F1BAD22CC6E90B761B0502 /* STPTextFieldDelegateProxyTests.swift in Sources */, + 58A8B3F57FE98C22D8F90C77 /* STPThreeDSButtonCustomizationTest.swift in Sources */, + EA571ECEFDDF10AF87CE2B74 /* STPThreeDSFooterCustomizationTest.swift in Sources */, + C35CF837D67AE8DB7CBDAD98 /* STPThreeDSLabelCustomizationTest.swift in Sources */, + 590DB84AC15709E3C6F1FC3B /* STPThreeDSNavigationBarCustomizationTest.swift in Sources */, + DD16FC7ABCA7817794ECC407 /* STPThreeDSSelectionCustomizationTest.swift in Sources */, + 2AC91F23CF3949ADC60D27F7 /* STPThreeDSTextFieldCustomizationTest.swift in Sources */, + 701C464523173C6809544935 /* STPThreeDSUICustomizationTest.swift in Sources */, + D2F19E21D5EC275A3CF23F7C /* STPTokenTest.m in Sources */, + 2F6814A67EA18BFD306A9E5D /* STPUIVCStripeParentViewControllerTests.m in Sources */, + 4DA70CAD042B5968D833B5BF /* SWHttpTrafficRecorder.m in Sources */, + 71116C2D5831E271E12DB059 /* ServerErrorMapperTest.swift in Sources */, + A8B0DB753CAA2223C8BED099 /* StripeErrorTest.swift in Sources */, + 9299ECF19D79770F54AC4732 /* TextFieldElement+CardTest.swift in Sources */, + 3B237145902E3DB07E747E32 /* TextFieldElement+IBANTest.swift in Sources */, + CEE483EB7B06B3C607BC755C /* UINavigationBar+StripeTest.m in Sources */, + 51D515315F02D4C03BA12366 /* UserDefaults+StripeTest.swift in Sources */, + 8B80FB6FC88D411A90E9D487 /* WalletHeaderViewSnapshotTests.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 11BFD72CD1F291E0E86EC784 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeiOSTestHostApp; + target = 1628E8B14F1F7C63CF8C9962 /* StripeiOSTestHostApp */; + targetProxy = 32221E5BA07FB5AA4EBFE81C /* PBXContainerItemProxy */; + }; + 4E57D9EF1E91091C57E49111 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeiOS; + target = ADF894AA8F6022D9BED17346 /* StripeiOS */; + targetProxy = D90CE98566BBDA0E7340E1D7 /* PBXContainerItemProxy */; + }; + C4F1FB6EFF8D25D54580374A /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeiOS; + target = ADF894AA8F6022D9BED17346 /* StripeiOS */; + targetProxy = 8F25D3DFD6D65DAE4B581911 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin PBXVariantGroup section */ + 147D2DC1FFDFC99269039377 /* LaunchScreen.storyboard */ = { + isa = PBXVariantGroup; + children = ( + AD8BED2A6066514B51693172 /* Base */, + ); + name = LaunchScreen.storyboard; + sourceTree = ""; + }; + 884C01B087B4D820395BD374 /* Main.storyboard */ = { + isa = PBXVariantGroup; + children = ( + C980D24DDC884FECCE39139F /* Base */, + ); + name = Main.storyboard; + sourceTree = ""; + }; + C2427C1CDFA85BFC6570F1E9 /* Localizable.strings */ = { + isa = PBXVariantGroup; + children = ( + 7B8ADF2EA2D5C4C90BCDDDD5 /* bg-BG */, + 26E70469F4032553B4BB62DA /* ca-ES */, + 8ED737CB1253C3C5704B6C05 /* cs-CZ */, + 27A4D83C0E7D4AE0CCD38B89 /* da */, + 2560B4EEE60D40FB31B8552F /* de */, + DE24B3BAD7E890661CCA817D /* el-GR */, + 7B10B1A15063FEDBA4A59953 /* en */, + 90D0900C5FDBB7952BCF2C3A /* en-GB */, + A6F6634AD12771A9BB100DD3 /* es */, + AB1A92C77AC61335184BBDBC /* es-419 */, + 12D757B36030685C401A6990 /* et-EE */, + BAFD938F3A149A56CC99FE96 /* fi */, + 0A16326394D71637A2CF68C3 /* fil */, + 04838ACE779F5CC949C276CB /* fr */, + DFA4E34E459EA612E065DE64 /* fr-CA */, + 7CAE7444CEFE1D1EFB888996 /* hr */, + BFB4A210A30D1D4F3D3100E5 /* hu */, + 084B1B9FCCF4AB727B4ECFB2 /* id */, + ABAABB969BFB7102D5AA5598 /* it */, + 064CFCA8FCEA9E4BAB3547D0 /* ja */, + 1007571188950D7FBF745A4E /* ko */, + 43ADFC4EF612D7C4A46E81B9 /* lt-LT */, + 13DDBEA7D444A8AC14E0F1C8 /* lv-LV */, + 9D3BBCE8C46A38D0E20DBF4E /* ms-MY */, + 04FE0C74090AE8A871CCE5EC /* mt */, + 482693A3D9A527B0E671F757 /* nb */, + D3667F97B7665F699D6704BE /* nl */, + B334078D1C3E629BEB498BAB /* nn-NO */, + 0C9D6F99E303A17A91101723 /* pl-PL */, + 002634603200AABECC9686B1 /* pt-BR */, + C45852D37E323E65C47348B0 /* pt-PT */, + AA3CA5B4B91838CC7ED4D5EB /* ro-RO */, + 121B7EDCCD0957C9A444A8E3 /* ru */, + 0F735744F27D46F005BB5D67 /* sk-SK */, + 618DE183886175AF23C4E668 /* sl-SI */, + 58158E8A117299834246100F /* sv */, + 35ABE696542469B79A9D52E6 /* tk */, + 313263D9DC0629C5EE279FEB /* tr */, + 4077600B9B3C1ABFF38383BE /* vi */, + F13E3DE09A463B4501733B87 /* zh-Hans */, + 4DFDFC019C67AFF7C0F7630B /* zh-Hant */, + EA20E0E29EDC1F61ADA226A2 /* zh-HK */, + ); + name = Localizable.strings; + sourceTree = ""; + }; +/* End PBXVariantGroup section */ + +/* Begin XCBuildConfiguration section */ + 006D8C78BFD35DF18042E040 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 1D23EB567F573612E0794B3A /* Stripe Tests-Debug.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeiOSTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeiOSTests; + PRODUCT_NAME = StripeiOS_Tests; + SDKROOT = iphoneos; + }; + name = Debug; + }; + 008139FD6C1D3C3903E1FBC4 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 969E196AB597EEF68C38103E /* Project-Release.xcconfig */; + buildSettings = { + }; + name = Release; + }; + 160D95A170FDDD08C46E4C35 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 5E4EA6394497D1BD57ED0032 /* Stripe Tests-Release.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeiOSTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeiOSTests; + PRODUCT_NAME = StripeiOS_Tests; + SDKROOT = iphoneos; + }; + name = Release; + }; + 2473A3BC44750A2ADE1CE6A4 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = E1A2173E0891B7138687D544 /* Stripe-Debug.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeiOS/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-ios"; + PRODUCT_NAME = Stripe; + SDKROOT = iphoneos; + }; + name = Debug; + }; + 4CFD88FE3BC957D5059302C0 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = D609FFD051FC01BF566665A0 /* StripeiOS Tests-Debug.xcconfig */; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeiOSAppHostedTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeiOSAppHostedTests; + PRODUCT_NAME = StripeiOSAppHostedTests; + SDKROOT = iphoneos; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/StripeiOSTestHostApp.app/StripeiOSTestHostApp"; + TEST_TARGET_NAME = StripeiOSTestHostApp; + }; + name = Debug; + }; + 50C10A6D37E11392F6B0E7F2 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = E3C8833E7EBA7A1EBF349D19 /* StripeiOS Tests-Release.xcconfig */; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeiOSAppHostedTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeiOSAppHostedTests; + PRODUCT_NAME = StripeiOSAppHostedTests; + SDKROOT = iphoneos; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/StripeiOSTestHostApp.app/StripeiOSTestHostApp"; + TEST_TARGET_NAME = StripeiOSTestHostApp; + }; + name = Release; + }; + 610AD56E6CBB9E6781424682 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = EB71A4A2762CF864DB198BCF /* Project-Debug.xcconfig */; + buildSettings = { + }; + name = Debug; + }; + A71DB2DB45060267821F91EF /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = E3C8833E7EBA7A1EBF349D19 /* StripeiOS Tests-Release.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeiOSTestHostApp/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeiOSTestHostApp; + PRODUCT_NAME = StripeiOSTestHostApp; + SDKROOT = iphoneos; + }; + name = Release; + }; + D3B5E5117D5D1C186CC75505 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = D609FFD051FC01BF566665A0 /* StripeiOS Tests-Debug.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeiOSTestHostApp/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeiOSTestHostApp; + PRODUCT_NAME = StripeiOSTestHostApp; + SDKROOT = iphoneos; + }; + name = Debug; + }; + D88B8DCAA56931FBE4963518 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = DA82BB67D434E76B9ABA4CEC /* Stripe-Release.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeiOS/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-ios"; + PRODUCT_NAME = Stripe; + SDKROOT = iphoneos; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 2804CCD7A91C99DF32596147 /* Build configuration list for PBXProject "Stripe" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 610AD56E6CBB9E6781424682 /* Debug */, + 008139FD6C1D3C3903E1FBC4 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 54F4DB1E9C991950FD9EB4B4 /* Build configuration list for PBXNativeTarget "StripeiOSTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 006D8C78BFD35DF18042E040 /* Debug */, + 160D95A170FDDD08C46E4C35 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + B6D8846CE184BBD9139D6D15 /* Build configuration list for PBXNativeTarget "StripeiOSTestHostApp" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + D3B5E5117D5D1C186CC75505 /* Debug */, + A71DB2DB45060267821F91EF /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + F0FCB32AE130ABA66178DD9B /* Build configuration list for PBXNativeTarget "StripeiOS" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 2473A3BC44750A2ADE1CE6A4 /* Debug */, + D88B8DCAA56931FBE4963518 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + FCBB942DEBE57423D79373B4 /* Build configuration list for PBXNativeTarget "StripeiOSAppHostedTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 4CFD88FE3BC957D5059302C0 /* Debug */, + 50C10A6D37E11392F6B0E7F2 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + +/* Begin XCRemoteSwiftPackageReference section */ + 44C6210EB47ACF468D255723 /* XCRemoteSwiftPackageReference "OHHTTPStubs" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/eurias-stripe/OHHTTPStubs"; + requirement = { + branch = master; + kind = branch; + }; + }; + 71086E1440B890A9DC01C9DF /* XCRemoteSwiftPackageReference "ios-snapshot-test-case" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/uber/ios-snapshot-test-case"; + requirement = { + kind = upToNextMajorVersion; + minimumVersion = 8.0.0; + }; + }; + D244E8B5402CB2CF89CE7872 /* XCRemoteSwiftPackageReference "ocmock" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/erikdoe/ocmock"; + requirement = { + branch = master; + kind = branch; + }; + }; +/* End XCRemoteSwiftPackageReference section */ + +/* Begin XCSwiftPackageProductDependency section */ + 62887B4538E4E41E735685E1 /* OHHTTPStubs */ = { + isa = XCSwiftPackageProductDependency; + productName = OHHTTPStubs; + }; + 911CA85A1610303FA0AF0643 /* OHHTTPStubsSwift */ = { + isa = XCSwiftPackageProductDependency; + productName = OHHTTPStubsSwift; + }; + C55551F29B99CF6D6DD9EE2F /* iOSSnapshotTestCase */ = { + isa = XCSwiftPackageProductDependency; + productName = iOSSnapshotTestCase; + }; + E804AA8C4156CC85FFD9595F /* OCMock */ = { + isa = XCSwiftPackageProductDependency; + productName = OCMock; + }; +/* End XCSwiftPackageProductDependency section */ + }; + rootObject = E63832AA5BB4225708B7C838 /* Project object */; +} diff --git a/Stripe/Stripe.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/Stripe/Stripe.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/Stripe/Stripe.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOS.xcscheme b/Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOS.xcscheme new file mode 100644 index 00000000..e730b6df --- /dev/null +++ b/Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOS.xcscheme @@ -0,0 +1,83 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOSTestHostApp.xcscheme b/Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOSTestHostApp.xcscheme new file mode 100644 index 00000000..eb6c42eb --- /dev/null +++ b/Stripe/Stripe.xcodeproj/xcshareddata/xcschemes/StripeiOSTestHostApp.xcscheme @@ -0,0 +1,89 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Stripe/StripeiOS/Info.plist b/Stripe/StripeiOS/Info.plist new file mode 100644 index 00000000..abe21473 --- /dev/null +++ b/Stripe/StripeiOS/Info.plist @@ -0,0 +1,24 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + Stripe + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundlePackageType + FMWK + CFBundleShortVersionString + $(CURRENT_PROJECT_VERSION) + CFBundleSignature + ???? + CFBundleVersion + $(CURRENT_PROJECT_VERSION) + NSPrincipalClass + + + diff --git a/Stripe/StripeiOS/Resources/Images/Cards/stp_card_form_amex_cvc@3x.png b/Stripe/StripeiOS/Resources/Images/Cards/stp_card_form_amex_cvc@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..0ef2077b73b55e96b8319c746495cd05a0f61c76 GIT binary patch literal 4550 zcmZuz2{_aLA788)W?Kj`Y$GPQCO2W`zOPE@a6}_#?h>=PGq;>+t|X~9s zxsv;ca#iyGJ}sW#|M~yF&*%9(ulMWv?6YUjo^QOBg&`M*AO{Ep;xaZO+JHbTNDyd` zIg}M(+$72n03mB-W~UE~pex&;ux-%ge=T5SGXLHIL;Ut%Gr&jw@$WhTl}Q47#lLL- zt_32%irL9BBY+FSOXP@{MHnT5==05cQplFX{z5m1>1UwjfOGyF0hpN$w4FF$2;ITV)0xe? zg}?|zfFR)9jRWd#ipc{uW}F!WNM;WZ0T^ZjfSDcuFc&Z~{Y*1cWoG})?3w|vdszVQ zdYAyH`U48sh30@C%+mlH3tJGz)IX=OcVvEZTj&5yk-xB!Kp!a`X<3_$X+}yKq(eup^zN#L6+lH}cpQN(h zaR(xwiYmXTd5Ph%qk>#)axfh`6nL*}lK2Vs8%L~oRuc^k&Z(=N<`@&{G82~2R>Ikz zifW^|nw;!&8LV-W;VoKNAtd+DO*^Ll7{2R6e>dy<(x@)?`y*Yg#}5OaAIN?toia*^ ztcU`E;7P_rUAvGy^Ud}}<2p!C(@T2c=d`v@pFVB~}1+s^}BEYvI#$ zT^|-!aIE;4)c&vewd={26VT0HArTa|@bGPcRE^g1&EHW!bGrB@cpqe&`-3c#$#b+Xs)yISWL+dlV=EsE|7KEoGJ)oYF%nOYsCT`-Fj!i_)4G%D+D;!3kP zsQ6@m&kK7s6_b@{XLeORlQx3L!9|p#gt)cMJv*}R?UPc)B@#`0S0M?xca@Rek4~BM zD$V*HyOmhvjevd#3+D~g@HKFc?D7RuFMr11)8YAcOIXDMGAw0)H^*Dp0y)`nI1A0` zTBlQnX?h@zzXR<6$5*BpzerWW&vJJ_q;97&n)kGc2F-YA`HLkKeeM35pSP!4rZ&gA zo8N7cXg=80)pf0>UMdIaXq~iaOlS<1w>O>sS(qv(An<*LS2N;*s!l^;<8><~*pehMe2aAj zELI$FSV5f|-GexrXkv%OOVE8%+(Ys`obgIZTA!PxbcVaW+j;m$d0Y_)ZAg71@LO4$ z-jnL-eM6i-hVS|)?~~FtOriMHRLa~X_Sb~lcx%xJGK%#mBmU)i)ng~lUA}#hk~Wm+9!U1oQL^VIY4D+64fI~-ID8CCm73X#u4x`R_m%;p!~Sm5Vo>V|DU{B z{>qOTq1nST$Uv5xC+sB6S|)zLE5FQFrnkt4qbmfOa>`}1*BD0*PnIy2K{2w{?mFq^ zD?R5_B_%DK6x>~+P~@>{e{1s)^ejuvXj*>sKlu*+MaclAA*L5M%YGxlgD{a>PhQOzS`kc@u(n(_vs1*@P6$^{4W*KZx%yfym#k&tTsx#x z-{)Z!<7(o<;OxzB^cx~wp7Xo%!?>fU=e{xy03v2ZO#x{w{dl2$z;`J2*Jk96*;zev0373cFMaE>!K zMnIaga^H8DLUk+bWK77z4xRwmvRkUkw1BgFs&D|z?_ozgjoN{?hGLK`e^!wQ0S(?+ zL@lZg^zSN}DM|dJKIvmv^sfuw# zxel)DIxL)=QHyw93c16VkIyQXjhJ9l^lTggT{}bjRQ}9;aNq~O?2?&TSAt}YD#K}b zGkh8$+>l~+xzO^Ea^vG0pQaOt&z=>h-b-W8`nlZS$oqaii}3n&USdZSx5_u(_14@aR1OAE!MpO^ifQ*IvQ48*csid8g> zZ>H0mTi~aB>SQSPx<9{F zo&cAJ%Dlc&JX_@MaMYu2ip74a8b;8LXV<)6*4t|p_UZ*C9!z;-KEu8L_=n#7q1F9| zWKCfNxoDW zE%n#=hVpzfIAH`tDGYUcFNs0mpqa(%PNV(U+viB6x8FTgWqDxO4L8X{TJ>pJ&ntKp z6gtr?ODI}jO?^`U2kjDMUjVCbHs=bKQv$^b!}ilgn*HSD0uYAZvvt8IwNb$<&cVLl z730vS@HdBO9f?Up1cfYh;ns>z5_h=O{WueH_(dnoamVUgba@w)rA(&hoBhNZo{(Lt zT68BClo_V|t?P8hgKhi_Y244{Zpd5rTH`|MYdGq|c)F)*kh&1n?4y^0lvDa(v~#@A zVcfI2ZpgmNp0&+`+0@s{DBJNoDb>dsLe%=u!5EQYqiI*VV@aFJ3@2^?&W3qlJm{XT zz+T|WVUoO2&Sr9G;4M0%;fmisYYY#C5lKwV-Y)?TS8A?soWJufT#hGgtt z9^l{=dx>Uo6EPuP;V0vJ0Lb-HyGjt++T0N?_ z4b39KauAce6_tDTIpQ=sZXaZyyfu!p($Cfx5Gwiqg1PRZzPE~$vb2##cr-}T%K+v-{R{o`kUi`CSr4_BUiDS#kqEEJW zZluX9EB%6hj#XE?TF!(1m65#hKyFzTG)~5}ibco8(yU!Hvr@e|Es!$v{Ug18D-+(Z$FQ^0u!xCR6=4Fv3VV>2hvK$m>VgIj>VL`>x+zX2|gQv<5=RChDEb~xOBKn`P zUi(Oqzt+;!)>O)e)Ishc>Xa%GFNj7*f^|}L%F}1<{^78l{iQdl)Fj!%y74tiR9*l~ zMkB#dI!GvsE*kn@!z|DLwh^k!|B_i6M;ZVw`=&-$>o zAMt6_K{o5>UHq#AAd|iP2kN*`E%jTyx|%YNe7>!AdyNWAE*ytB``2uw{i+N8bE*Nd znPLC)ewcvwSoYMyadvFKcDUNra1BJQFDkos=@HN0YZ4Xy_r|-|AMYXjpD$!bUYhT3 z3llQs1&=;;Tc5-y=V0XFii%0)z?)px24R_Xa*k*AE$+Vkt!;CA(w#BhndUH>VOhgm zzE_llB9=A=e0P*YW}9J<6SYl!Mn%H{u<3!`OnSvvf5SZ4$@2U9l#y#Ced z`a>|=k8ob6U-oq3k0BrH_RD%c3vplRCBax;md;5yD*OGm@2Ue(nUaLBnMQK*xNbM{|Oi0 zQZA_rwk1`4eS4ofIMv%VLwt+2%2XDrxaC>wq3QXr zGp<^V&t-J_I&Ya|G)*WSS}ua;fFFbutzIiEN{_bro%5ck$%jsV%5PG?j9jt3&0Qti zyh*P2M96aY#YKT4-*1>D2`vH-ToV0Z98!6rE&ZMTWbo(igpt)VHHo4DlJb&qUF$d_ zNc;`4ca*&%u1|x|+)hi_&t(1FBY16fG<(TAgR)ZcL7TmGdf*4~AEI}Z8 z7zjjX3T6NlH}PU9AXuB9vpNH`ps)i_=mF^VzZTHalz;C45PR?s4Cr@%^M@obrHQ~^ z{x9s`v490wgg;)I1>k`KxI@hEf)+~~Mf@RwY1E&5S|IH#0Hzg(A}#8$1*S9r*ntp0 zKR5tI|EYm|8VQVmsPNzIVPMGbmgWL#00=w)k^W#bBOuZ+njJ`J`F$d7O54-_+<_KF zyAm)0V8Hd;SUmi@1xANPT20#%e(N-zW(T5ZBS54Then!01AraK2LRg900XqNqiOu1 z_mBs48hkjS@jy#cXvKp+4+nUL0DpLm-;8s=^)m3g>h!lXy8uF%+)8hG`)O(SK=5+j z48-}>FUj03#q#|p{ErztSN3|d1JBFDk1(%#^~@)kLFZ0f*e*OnzkZ*O1g`oS#|l)U3$dLQ??3qM&lzno%-d(ME~UCTH`( zX(p_Qp`$*vPnAbp^FZ6B{!K>L@}qaGbvhgYU-3p7a(DiATgM)k<(-1R^Vt%z3J#job9>!ARsN`H8;YDZEB@uh>RDDvXhBC)IrPN>mn7;kbg4)X&4iW1`j zwf1(ZxPu+iKX$9B_1-TBCk5+4CZW|cdLL0L$oy?dRdcxU9aPk_FGYlK(#=<~mdj!) zcn0X1{hxGVhK9yl)54KQv#M7`NQI)3t5hcoXtlyNW|Z;lIZ{?e&OOzv^ln?y!(;8H zF`9tfF{U zp1F#2M3z*@p!oFu%MmEnbhMK0+@i}DZrOR+01{Ig&gslkx_XI)Q)#5RQG1)-5Y02< z&Ob~>wLyXfIc=0p;z(7RvguZ^q848*9rBEJD2qPJ`$onG>L$%DEF8-sv1exrN=^As zX3sgIE;gD3x?v}H0=QBQ!ddj$zYV^w{t$P*?)6G`sY$);!fk%LZaGwnpt!yB&gD^8 z2d+(q6X}Mx1QX`#x;r>d=)}ClJt>GVnENxRdU@%nE?#xc&d^yGk)%hZi}eU;VIBMJw(83|wz^2U1!g2;o8Owa(BemAZW|nd z0W;?c;jcEM-|Ae_vfTxCzSXjdKDv+q{E=f$#mZ*3J9-<%M}bV0t>(9-Tp6I@+D`aem92!L^ty z&k`J^hDUJRLnty5Uc>n5^|9c8HDUb!A^stDWDZH(VnXvXyW@Td)6(M!(apDgugjRu zqJF{oM~-8uvZ)4aAL-S3oelk2Qw!Ya*t;H!+*40svc>Y_m4~7}+NNssDvBv*i05QX ze2w*O&U}#l@%w(n{z%HUwo)ik^5Suv>Nk^<0Vk(BP1dN0b8iVl_3RTwwI5VE~(aBX=F##uZYrDiC1mDO6E= zwjPH~fn2IEbF><4b+vb9!yPN}$zB|_{-N$rVi<$Nj`Z{0{*)j-LS@CxQ$t-X#%(Jp zJ<(!dwLJ7QIh{m_54tEE*2n_$<>=a(c_GUUNe;B)BEO}XMDAC0Oy@-~4jWDESe(oJ zfhyA9jKs1C;L8MdOVyarn&#Tz6lvXZQe`=jP3lb~))sS(dAA1@`N&!mti~6q_}9b1 z4QFt*dU7PTI|OW(5tjN6-Y0RC6Zf~i^aW54@$3R~E3XjPBAY%>6PCr~8i~~;nn}VW zh=EoRoTmotBr_AaOV>y3C6d0~JqK=>-uFD2%!Yfa-?jC9<5`6jOqRhJ7mB6V zGc1Eo_<>m$?lI!ONHevH=%<~tgd!ynS{Eh3DFj%4PK+)_hKKPdvtCx{2~MhD6b|d8 zJ5i7`fm?~vLsy~j)(f%LU|&G~gbqHtyz$Vf@_=$m=TuYTWH1|56VNLWqhE#HA~?Q9 zw`^*86*5z60UsFe`SS8iko(U2<>rLR2?ESQ60A1beegI=C+}P83+;qSj4r>WB$ziU zZ=^Qj9`BtEl$U-q%vus0$WE4YTs-f-dM$BQ%@8p)11IygI}M$v`{q1#9=&2h@MVAc zYr%QylESgD(mg@Dz3|bt`by5^y}I=h_f=@(ta;PlE;q6r7+1LEetq+uGUOs>02lH` zG!-Qr7=#@pl&q@ehxelK-`k!hIu%l#KCLT$Z-lJ;&6Ff~`^!MLd6|~-7sp0|p|d;{e%UY2>O?P&QsCtFXVjjtvIh(5Dv1(82BlW-C<2uhU?248JT+)*QC>>J7|S$6GcJW(m0Kvijev7snX3<17t zsmj*J$pN7Xp$)r*YX(9FHV}hhN*4cIy6;RLLPJ-`AZ9Gl7xDa@h}<P`$Q#{4;_ zszcR-IrMM`)89h{70j!#*QXJ@e?`(~F!OyQ@UjeBq4EZ%l={zWuw5am=!o)G{eH1p z-q%Qxj?b4$F|~+3t=u1*fZOR(@jk*nHKirr-;sK;ltIR1)z%>+w=m(OT7Tj+XF-49 zu!8COg)6Nl-!=bvg~u-?2Mv>@#LE0~Gd`z!dxv^HcifN}*~<#OC%1$O)|b~z!qmk} zC?)k%Ap-;SAKSka?~YAoUY@+kxBo#U*TTE1KaB3uIl~$IC+BZWvWELxoxeGrQ{nYS z^kIDg@l#P|h+Br!i}xD>_f`%9m#$PM$+9u*M7LcJD)I>;`w!<7K~k|SK|w)+LAUQt zZ+cB#)BNSoW=b(){m!%9@~ZK+7N?+4-(&eK*f(k~=)?27snBYp3QIQ@we}G;A?-#!mnja~WNd>YJc-?9Hp}7ZIUcsd z<+)G@)uEU#eah`Km3Nm-qQnPt zad~tGTRfR2m`??ZBSdVF2w{PCuD-N``J=k-|A4P zF4+{cjw~u4D!<~DL2$7cG3{d?vbKBsE>e+1OG{c~Gw;3ZDz_BNBz#vJq|-R?g}qyY9Xh`A0HL$sBl@Bgk`ll{x7p z#m-7^H+YoC^}R{>SQ#Hd2}CESaBWU{W{a&$QNF-Z_x_QGh;n^_P5_W8IWhXMlA9VcvbH1eKlWIM4alP zp@lajt`va8&t6rw$B&MdzAmCLv2tynVkPZeNox>T@C^HGT=&wBukZePx15mOS+<6w zVpo(a*6#OY_}i}LJ3CeY)<8J*{sh_b!E=9Hd>Y``YXMTTBx z-QGsnzTLmAUL$YXx^%fsc~)1W(@a>bdK3?AW*KWi<#f z_QpJz+GYdC4( mue{d>zczoq?wn-T+&yl4Rm;Vjfngm4e2mYS8C2+DasLNwsh+d| literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/Cards/stp_card_form_front@3x.png b/Stripe/StripeiOS/Resources/Images/Cards/stp_card_form_front@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..6512912e9f5a1847667085431c008e0cb40dbfde GIT binary patch literal 3414 zcmZ{mXH-+^7KW365PCwdLkmq22-0hSfI(_#0;3cuD$>LOX+nS?p-3H26p*4^I$n`B zR2xJoK|}%JN@&tM5m149h6pli-Fwzv-`elq&wjss*7=<_378n8I_Temb1mJ-L*nx16P;IHLh(i+S#vb}p4XJklFf}ukOPK1MQ z&G+qenmOB)nDthXA!H+gCms6adlEbBh3s?Ew)eHZQN`+J)>5lCV7Qcu!FdNXx2hVC zu$l0qvvxZCEcEisC>udqxf6DG5cc*Ac}kWXl*F?}E*k0=8OML?HWiZ8RhBle<%8kI(s(ciVS+nUAB5*NBQI7Vle zb|*O)Zt4WC`1!X~uLvV@zU}NqFWftAhY+m!)a|d{M8mKp7}SuoYBCybes^S&vO}X6 zY>(hw{oedreOJ7IzAIQRKfycaifkUpHx;dw^{N)b*n2Kfy374Wh)B+xnWubz#$Q!4 zBNS9v%7X7=O6^e&4mggJwto)Hvxtm(4P28hwIi2!GD%dbz^ZCBrLQWF6FhmcorIxV zblUE>&D2Wz^*xp7PKZpx6)PD9MNCt(dr@$rcY>)c?iSMhi=HKvJx81)Mw`j~S%@Wb z`k+|GZK0F!{RcC8dYFjh{rHnZVzO&&O`G`$WA0e+cFI~#d-pRR%`l(e9lnj!`3P9F zTn|W7Ef0k!|8>3X@-{?Qc|tToHV;!5CgUk8HC?-fm_Q1%UHxGUY0G?WBNw$2X(ZWJ zPjdTsqj?W3oE@>}1esQ7b{3sq5KetAELbR!Q{XDh7qRVVW5}Q{&c02@_gVNczd(wG zsd=U-le&j>2FKbOvh3Yf+PDBi64~!P`?lbmr>Aj1Rb&L%q1WNw$`&Rdh;B~jZoAT+ zmGu0id}{HaxO!a!?I|^r#4bb&_|eXU8#Z>^x*0RT*k4hVwh+%^|8f{pk#VgYQzKR; zoviDf5V~cJZmuy5Xad_^52`cvZHXaRjB(Y*x0heUh7mrAHMW7f)CA}e^{`qguPv^D zf|6x+bVa(>$(X>Lr*7%x1nD{MKq$p%04f$bNEEk!qb@lZ3SY}AQ3WZaiO++I`BFyZ zV={A|-s*o4HHWh4rNwbzD_7c;One`{u0Krw)eT3yck>HIN+uQx1})<9%kzd*mW0XRxZpVmzFr< ze>lH6rIyts7D<7ZaZxx)=ntZseA$jTx5ilH`WgH@TKnYm;SPWljqXBpLG`xUNX))-n5Ft1hs zV=|=}vl1DLN%QcpIB!(EcOII)tt!!0V?wXY9jgEWE#NfkM0)CRY`*uLII4%0vfn8z zwEiGAUF+BvIim|j21R(MTrnYxy&>f~D@nXkb08p>e=M`MSRhkJbi~S&LuG{#pbD{dWBIjt2f8t?@r;STx;n zOKQ%4WcDkWNO5>N@Kym7ePt|~6EO+99!D^F9RgY(#2`kOE?yt18=JnXuDO_6w&(pj>l$d@xMoDnhH>eBuC}H$ z7zA>G4Yg4&F;B_G=q~xuMv1^^qfsy@L{9|zw?K0qR|l(5+0+fYkJ|ik06^7<=Eam1YBv_x}SNRu;Lr%IOI2_4|$Ch<5vFt3aUeD8ouOfF8Wm#Hs3nZUd zW?(7RXbJAJ=VEfdtk1mG1qw3!&>EAE4Df(w6@0ZLB57N!PkHgQlM@S7;7ar90<1~y z>H?266Y|Zl@~jj6OofiHeie)5cFac=^|9NjPS{9&pLnkb4$&Kkn}WQZ0d>Z`qEkJd z*xxNll@gMD@)@HS@U9iFS<<`WWu~u@#^0uw+|)HBYIY&c+qrU{W4VkvBo_WAjPT&6`Y z2hUb-qI7M?z7s+u&pVRCYET1%x*{D5bH$<+%W`Z;-SE?8Gzr$Q*AO+H-LIk$yg-`2G~q@RuVDMs);OfRXHR?aerPjFE^F{iU~Jon zkKF|qg z{!>eoX};bYfwwj$muIRwr4T#|b8Ca7CSldV^G`^^XIxn$%xO#dK)_>cXpXMZcZvQl DPdo00 literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_affin_bank@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_affin_bank@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..44e7685aab26e59d860856a112b5281b76f90a20 GIT binary patch literal 2603 zcmV+`3e@$9P)XkZqGB^({#hjqYe$&6u9YpLTFM>c7E|=tvCB3K_5;!E_5g^Dcv2A z{N1cm54PBJi>UcWzdh6UN2FvI3L{<|dB_(D>bm-}Fhf#SLHiXKA84`Bi}$KF?7+55 ziKV5&6pa_0^F@Nj?+z2zh)m6WXZ+EdzsnPCw@ls}UII7P7_>Ra7YSN;G+Nk#hnluO z;%0W-8)J{erss>RVcf1TUnFSB@fdLpI2xArqmJ8dVE(qtE@Ib&HR3b!*PMxCA!7}F zk)ZiU3@a!hGe0e-P?!N<^UJO~u50ps8v=y|wqHuBH}}jRd+qXdH;;E0I-HD1f3)@H zZfh>%cZC81g6BGHyGirUV)QtyC+jb*@a?2W8?5kMg4!*=D0E28Db$YtqRWmw0m(wa zkxyFt*0xDIztI;}NB#12|z3OI* z&g(DeyCFa*czw(fs|lf%RVIuGPsx?+*WYVShZp^Lw%!sS)naF(GlUL1uOvwqOxhDJ zjJO<^Wwp`ZSB*J;-$ORW4>VjISh1Rio#Rzs^>x+j|p^nclbhH{;J%c@cIFS9|RrHK+M5L4WPDM;I~dV5D@-zX$A^p|glvI&yT8?( zUiF*t>@vT}2faV>m@wjaxa`iKX}>k6zz`g*KNoMg=1ndct_@$wIcrmF7TUby8Kl{}%#A5Ado-ooNiRh_*5`4GM5n*Lf%zsV?88FD~X z+k}z&Yf$#Whz}-u46wYv)}D=5@6zUjj(UQ2)p*CBQm!Mmg{V?q&k{|9L3Y^ zwhAK}%s;E%qT;ucQvUOA4?+FjY3@%1-QRp8T1+UIb12G_Im7M<9kyId)Q%H63L}2o zeW&SN>V4cJxMG#~a^?_Rh>ux(6&Od7^Zu_h^D3TKq z>DuvmuYB#C`VY0j+di-?C384d#ksL1YzgLnR-B48jRr{-NFl5#LfFN~n(rJZ`b z&1M!C)lt*ooc{I59_9Ant!fv6f5D6JhxrL2L>vh$c{P3_P3Z%q+v@Cu56j z>xHW5+Z&kj&CF9Qb0AL zEIOOXS&m&+U%nj#eKz%ksmbKv>_d?_V4I&EOi}2Du7t1JY2n9%dvWqzKsMI{uO+l)k3kahZ9F3NE#Hf)Bu>Xbe zjOy#e_H#$?69fgc;RqwRy?s|5W{92`5NmdE?1no+mzyHQSo7jj#J`1Yzs!OzJ6H;L=NgGf~qH%A3P>Q;LG3Q<* z2nQ*=0e&qUnLc;XYh8e_1#3;eGYjT2B}EYneYY(cfCUpst-${x!4;3ku?4Gs*T{FFoFn^Jc$&=Y&d7~s$Uv;hz3P* zqc{|jN_NG|Zk{K6Yupj`x#T;oz;c*-IO^5Whpo)#y0h_;Ai&tE-Lgm!B10=DMNf?6 z3T(jk#5`lsZ>`jzRRa;&sH8H*E}#Q<+4WcOTmMaruPBPf|Tiy*0;0lhBey@U~nTW1l3 z&=oq6|JR@WX!3DM#Np$dMG(Ds(l=kavvQLelFeyeOm3t{1Q6g^H3{zvN^u-PZ!2xU zeB`OAs{~wQ8oO%A8A( z+S4k9KLkfkCy3lDxmuwBQ%yHM7dV|D23?r=a?4MkT!I+XW>KWFc6ABT530r*JJi~# z#JL2S49ffze`I!Cf*khO1iA#d1i8Z{$R)@n$R)@n$R)@n$R)@f{{`jiyTRXoMw|cu N002ovPDHLkV1hB4>_h+n literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_alliance_bank@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_alliance_bank@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..2a20f56ecc06ab5749619b39ecdc1d7a606777e0 GIT binary patch literal 858 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 JPDHLkV1n7j0Qdj^ literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bank_islam@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bank_islam@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..bb5281cce5ad0fbc94e6d2fc36e344933aae9eae GIT binary patch literal 1214 zcmV;v1VQ_WP)ZQ!lH;&Uhk@L*r z?zYwJuF=?1oyQ1t$P0D&;O+0Z*xFsA&MJoNuhHIbt=wj)@V?sSlEKFbbnm;__0;C# zf4A$Z&(JZ8-D#@caIXCI`O!3t&M1Y}OPT4P$>M&t`|b7COqt0Lc7B^hU!>xI zx8#ex(Ke0!`1|?h@&5e%?5)t(O`G}U@adq*h{^`&nk!5RiEE;u*eQ~ z$`pFfEQrk`g8cIL%Nu^>kH7H1+tfjm=9I$HI*{LVvCuGz;CHg~%i-5fo7PR5)=HS? zl*7|NlhZws{Pg(cjK2Eo^!@kx)k&D?p~~4-pwvW_=99wVeYN%0=ihO!?zPp>FN)7C zh~ay*^UdScNSE!i)b!8fhlWT($7hvt#M`RDQ6XsO8(dCeq)<&ePW zoX6Q$py-*!_uK33vD5wf{Px@G`swoCZmro_q2!9a>Z8l*q|5s1^WAH!+FPRc-0SSD z(DB0D_~P&J!Q9h6lFS}}#|m}I7kuj1#`ORI0#`{yK~#8N?bbt+WNQF~;e3lm+qP|E zY}>YtYumQ%|99g=^lWOnrn{=ox0s0YEOV6+xk)|Xa5x+ehr{8W8d?$UY>pm$t1q0l z?qoqx;SSlQp}K5;z~#Q0HPR^i<(5Xzv!^a=uS^J*yIrh)W-K8i3ZB0J?v_TGQB@1P zd_acmt^=;GlOf%Qv%z1MFjn6UyfYwEP6U9q3Yij)0V`xo4!}kkQx0%W#?-rj+6I}j zD;sFqDPuMP9G5Ze0Ch4Z^ew=MjQI|rN9Jq>Xpu3u0bIE<<|x21nX?w4Qsz7X@TAOn z3Shg;ISY^@b36cLGUqwdBliIG%A9=w@trc}GL|eE6Uqi?mN_#3t|PKf(kDNJWv%S< zT2fF85IZBgwBAYzR;~fq5|T{@mCT3bl+LgRYa)=7Z)McLrei<-w3 zKfv&qmaOlZ=9L!*@Nd%+-MFa~>Q@1{CbT4X?iMPQdmb$JBg$CuhNpu{ZS+|z>+6-F zT{W9t(4$O`W8o#QGBEA)R~BmlQ~6k4@f9jvCHcdzYGD;!ek_64-%vWL-^9IDi!$0D z!{U0oq+5#<>Ufvc?`hFi&cv~>eq)gq_WlPRK3x8>((%cs0W7fjvr?~83x58^m*6gK zRJsz|zWSOu-^?x>*r8;Rs_%cOalxt|3zV5i-cLXCKrFwrvTNd(j=}KI_VOHGf6XM8 zrrBSET7rhE_ULb<;AsoD$_|n0EZ^^cZ1FH>&7Xg5SibD-v5gAm~l89 c4u`|>KOI|D#(?(6QUCw|07*qoM6N<$f@Uwl?EnA( literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bank_muamalat@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_bank_muamalat@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..f1627fd02b94981e8b62e00a97eb4df85d1917c3 GIT binary patch literal 2207 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 FV1iEx{Br;R literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_hsbc@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_hsbc@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..3db758fc0fdb26defad933e8cf86425cbe456df2 GIT binary patch literal 922 zcmV;L17-Y)P)U(*FJZ{`mOeDL4N6`}MiL{`~ylAus1tU+sa2>vMbHA~4`1G2tvZ_Qc5e)7IfC zH~sSS?0SOZIYs0_P4ld={O|Dm?e6V?iRMgK<2y*|X>#p?isCLi;xs_;m7DB*gz08* z=TlzfLQdl~LiWMM@tvgLB{KEB!St}T{`mRmUup1}pWq}i`Ptj#JW1+mbMmCB=T>0o zS!DkC`ugJI0xW`hK=Z8YVxP8>TP!Moul-%y8P-pB^#1z#<2pv{ zdV~4Y*6)#+?ud`;bbarQlvRI7DXAnYI7`0sl!vK~#8N?bqXyBtaO3;d83m zHpaGXu5H`K{cEGQw|5#1TwY5|VVLtmMH0|-8*uQ#X% zScbMi{tIB>8GV$|2yhUZhWI~#dlh!#lufNX^kFfS$Xo3tWa@fOC+D(>H+b zE3El=4gLYfA++QQ0=N+(>E41U;4VR6Wrm^vE|Dxi7~sqb{C%M~fQS4BNq|}S+sI1; z?`47f1_Wj!(oghR9Ei+9X!Eo`c@q-L(q$SFEz@Plr1nm;(e_f1>fezb6(3&v1a|tN zptNmz*lm}fWHn@VT!&Vns9<*C6rgOPJdSA($KP4 z)f>F3i_jEX(-|GtV$haQ7=C{>0tpz{RT$sD)&sjR5e=(8K*CWN2_883L9wHsa1Men wR17OT0(-+n7%QqgV}Ii;uq?~6EX%U2KLxi&_ZIgcWdHyG07*qoM6N<$f{a)1tN;K2 literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_kfh@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_kfh@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..3531c3d83bc1c37371528f5b27fc6d190eff4c66 GIT binary patch literal 2634 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 Ef?KJaJpcdz literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_maybank2u@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_maybank2u@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..5edd26bdfc0a192eca1f46a435d82c7f8c0704b5 GIT binary patch literal 4570 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 Ef?KJaJpcdz literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_ocbc@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_ocbc@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..9bff169f086a1d6f60bdbba239e276460ad7147f GIT binary patch literal 2054 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 Q=l}o!07*qoM6N<$g2wJ;K>z>% literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_standard_chartered@3x.png b/Stripe/StripeiOS/Resources/Images/FPX/stp_bank_fpx_standard_chartered@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..c8d75605ad116a298f03adb01ff1f432a82ecaf4 GIT binary patch literal 2596 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 literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/stp_icon_add@3x.png b/Stripe/StripeiOS/Resources/Images/stp_icon_add@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..fd7daea53728e5604f097a179f7dbfe169bbd9e4 GIT binary patch literal 114 zcmeAS@N?(olHy`uVBq!ia0vp^#vshd3?v!y7i9uTwg8_HR}i=Af9%%dc0dMWNswPK zgTu2MX+VyYr;B4q1!HnTLc)J$6*k+y@;q_@Ui_l}nHabZWN4}#xGV?K?CI*~vd$@? F2>>q8DFE69Q8sZR$k>S!X98s>PX>_?1T+n(1&Ah21ZtcLg68a1DfPG1jUoZoE@g2s<`~MwIYi*X8`DYe)!xtXrwC6t8d`N?T7?B5338;`9ycDSTKiwGSH~Mo-U3d6^w7L zIyX5xinKlyo_AwgZuS4uulhblm~!t6eE9D%cdpp#vssN6yAOL_dbz=Q!s)Zd+qso& z;_g-28k~O4{~=J}MYH6>f|m744>fv@JmOsCc_jJ5RG&a^E#BOjI$xg{1x(i$S-C#q zg!9Ufbr&L*#vIz5wbrlr)mAMzlUXL~uO0~9?QA&LtZ$cX*Xz>8e+@=#wH2DjyA_}E z7%rM4cx;NXTd9-eWUYpmIeJ-UN3TcftPiuuy8kX$W^LYg)_IndTaR6A3zwEzTlw_Q z)cv2r|DE#J*#5I#Id17EceA*gpWHR$di$>HHr(CkQF!z(ANwiG6ZzEv*Y3AX^JCZ? Xm3+JN{hsr{fMD=+^>bP0l+XkKCh#9S literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Images/stp_icon_checkmark@3x.png b/Stripe/StripeiOS/Resources/Images/stp_icon_checkmark@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..88ee3f86e75e9948a9379a9f3b0bbb91ff16100b GIT binary patch literal 653 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 JjjTn;{{a@R+ur~H literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOS/Resources/Localizations/bg-BG.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/bg-BG.lproj/Localizable.strings new file mode 100644 index 00000000..9b4e5b19 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/bg-BG.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Офлайн"; + +"3D Secure" = "3D Secure удостоверяване"; + +"Add New Card…" = "Добавяне на нова карта..."; + +"Add a Card" = "Добавяне на карта"; + +"Apt." = "Aп."; + +"Contact" = "Контакт"; + +"Delivery" = "Доставка"; + +"Delivery Address" = "Адрес за доставка"; + +"Free" = "Безплатно"; + +"Invalid Shipping Address" = "Невалиден адрес за доставка"; + +"Loading…" = "Зареждане..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Следващо"; + +"Online Banking (FPX)" = "Онлайн банкиране (FPX)"; + +"Payment Method" = "Метод на плащане"; + +"Shipping" = "Доставка"; + +"Shipping Method" = "Метод на доставка"; + +"Use Billing" = "Да се използва адреса за фактуриране"; + +"Use Delivery" = "Да се използва адреса за доставка"; + +"Use Shipping" = "Да се използва адреса за изпращане"; diff --git a/Stripe/StripeiOS/Resources/Localizations/ca-ES.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/ca-ES.lproj/Localizable.strings new file mode 100644 index 00000000..b7979c0a --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/ca-ES.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Fora de línia"; + +"3D Secure" = "3D Segur"; + +"Add New Card…" = "Afegir nova targeta..."; + +"Add a Card" = "Afegir una targeta"; + +"Apt." = "Apt."; + +"Contact" = "Contacte"; + +"Delivery" = "Lliurament"; + +"Delivery Address" = "Adreça de lliurament"; + +"Free" = "Sense cost"; + +"Invalid Shipping Address" = "Adreça d'enviament invàlida"; + +"Loading…" = "Carregant..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Següent"; + +"Online Banking (FPX)" = "Banca Online (FPX)"; + +"Payment Method" = "Mètode de pagament"; + +"Shipping" = "Enviament"; + +"Shipping Method" = "Mètode d'enviament"; + +"Use Billing" = "Fer servir Facturació"; + +"Use Delivery" = "Fer servir Lliurament"; + +"Use Shipping" = "Fes servir Enviament"; diff --git a/Stripe/StripeiOS/Resources/Localizations/cs-CZ.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/cs-CZ.lproj/Localizable.strings new file mode 100644 index 00000000..a0a89390 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/cs-CZ.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Přidat novou kartu"; + +"Add a Card" = "Přidat kartu"; + +"Apt." = "Byt"; + +"Contact" = "Kontakt"; + +"Delivery" = "Dodávka"; + +"Delivery Address" = "Dodací adresa"; + +"Free" = "Zdarma"; + +"Invalid Shipping Address" = "Neplatná dodací adresa"; + +"Loading…" = "Načítání..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Další"; + +"Online Banking (FPX)" = "Online bankovnictví (FPX)"; + +"Payment Method" = "Způsob platby"; + +"Shipping" = "Doprava"; + +"Shipping Method" = "Způsob dopravy"; + +"Use Billing" = "Použít fakturaci"; + +"Use Delivery" = "Použít dodání"; + +"Use Shipping" = "Použít dopravu"; diff --git a/Stripe/StripeiOS/Resources/Localizations/da.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/da.lproj/Localizable.strings new file mode 100644 index 00000000..8415e65b --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/da.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Tilføj nyt kort …"; + +"Add a Card" = "Tilføj et kort"; + +"Apt." = "Lejlighed"; + +"Contact" = "Kontaktperson"; + +"Delivery" = "Levering"; + +"Delivery Address" = "Leveringsadresse"; + +"Free" = "Gratis"; + +"Invalid Shipping Address" = "Ugyldig forsendelsesadresse"; + +"Loading…" = "Indlæser ..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Næste"; + +"Online Banking (FPX)" = "Onlinebanking (FPX)"; + +"Payment Method" = "Betalingsmetode"; + +"Shipping" = "Forsendelse"; + +"Shipping Method" = "Forsendelsesmetode"; + +"Use Billing" = "Brug faktureringsadresse"; + +"Use Delivery" = "Brug leveringsadresse"; + +"Use Shipping" = "Brug forsendelsesadresse"; diff --git a/Stripe/StripeiOS/Resources/Localizations/de.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/de.lproj/Localizable.strings new file mode 100644 index 00000000..1818eb85 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/de.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – offline"; + +"3D Secure" = "Secure"; + +"Add New Card…" = "Neue Karte hinzufügen…"; + +"Add a Card" = "Karte hinzufügen"; + +"Apt." = "Adresszeile 2"; + +"Contact" = "Kontakt"; + +"Delivery" = "Lieferinformationen"; + +"Delivery Address" = "Lieferadresse"; + +"Free" = "Kostenlos"; + +"Invalid Shipping Address" = "Ungültige Versandadresse"; + +"Loading…" = "Wird geladen..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Weiter"; + +"Online Banking (FPX)" = "Online-Banking (FPX)"; + +"Payment Method" = "Zahlungsmethode"; + +"Shipping" = "Versand"; + +"Shipping Method" = "Versandmethode"; + +"Use Billing" = "Rechnungsadresse verwenden"; + +"Use Delivery" = "Lieferadresse verwenden"; + +"Use Shipping" = "Versandadresse verwenden"; diff --git a/Stripe/StripeiOS/Resources/Localizations/el-GR.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/el-GR.lproj/Localizable.strings new file mode 100644 index 00000000..a95851a2 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/el-GR.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Εκτός σύνδεσης"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Προσθήκη νέας κάρτας…"; + +"Add a Card" = "Προσθήκη κάρτας"; + +"Apt." = "Διαμ."; + +"Contact" = "Επικοινωνία"; + +"Delivery" = "Παράδοση"; + +"Delivery Address" = "Διεύθυνση παράδοσης"; + +"Free" = "Δωρεάν"; + +"Invalid Shipping Address" = "Μη έγκυρη διεύθυνση αποστολής"; + +"Loading…" = "Φόρτωση…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Επόμενο"; + +"Online Banking (FPX)" = "Τραπεζικές υπηρεσίες μέσω διαδικτύου (FPX)"; + +"Payment Method" = "Μέθοδος πληρωμής"; + +"Shipping" = "Αποστολή"; + +"Shipping Method" = "Μέθοδος αποστολής"; + +"Use Billing" = "Χρήση χρέωσης"; + +"Use Delivery" = "Χρήση παράδοσης"; + +"Use Shipping" = "Χρήση αποστολής"; diff --git a/Stripe/StripeiOS/Resources/Localizations/en-GB.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/en-GB.lproj/Localizable.strings new file mode 100644 index 00000000..6d515362 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/en-GB.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ — Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Add New Card…"; + +"Add a Card" = "Add a Card"; + +"Apt." = "Apt."; + +"Contact" = "Contact"; + +"Delivery" = "Delivery"; + +"Delivery Address" = "Delivery Address"; + +"Free" = "Free"; + +"Invalid Shipping Address" = "Invalid Shipping Address"; + +"Loading…" = "Loading…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Next"; + +"Online Banking (FPX)" = "Online Banking (FPX)"; + +"Payment Method" = "Payment Method"; + +"Shipping" = "Shipping"; + +"Shipping Method" = "Shipping Method"; + +"Use Billing" = "Use Billing"; + +"Use Delivery" = "Use Delivery"; + +"Use Shipping" = "Use Shipping"; diff --git a/Stripe/StripeiOS/Resources/Localizations/en.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/en.lproj/Localizable.strings new file mode 100644 index 00000000..40b3a294 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/en.lproj/Localizable.strings @@ -0,0 +1,60 @@ +/* Bank name when bank is offline for maintenance. */ +"%@ - Offline" = "%@ - Offline"; + +/* Source type brand name */ +"3D Secure" = "3D Secure"; + +/* Title for Add a Card view */ +"Add a Card" = "Add a Card"; + +/* Button to add a new credit card. */ +"Add New Card…" = "Add New Card…"; + +/* Caption for Apartment/Address line 2 field on address form */ +"Apt." = "Apt."; + +/* Title for contact info form */ +"Contact" = "Contact"; + +/* Title for delivery info form */ +"Delivery" = "Delivery"; + +/* Title for delivery address entry section */ +"Delivery Address" = "Delivery Address"; + +/* Label for free shipping method */ +"Free" = "Free"; + +/* Shipping form error message */ +"Invalid Shipping Address" = "Invalid Shipping Address"; + +/* Title for screen when data is still loading from the network. */ +"Loading…" = "Loading…"; + +/* Source type brand name */ +"Multibanco" = "Multibanco"; + +/* Button to move to the next text entry field */ +"Next" = "Next"; + +/* Button to pay with a Bank Account (using FPX). */ +"Online Banking (FPX)" = "Online Banking (FPX)"; + +/* Title for Payment Method screen */ +"Payment Method" = "Payment Method"; + +/* Title for shipping info form */ +"Shipping" = "Shipping"; + +/* Label for shipping method form */ +"Shipping Method" = "Shipping Method"; + +/* Button to fill shipping address from billing address. */ +"Use Billing" = "Use Billing"; + +/* Button to fill billing address from delivery address. */ +"Use Delivery" = "Use Delivery"; + +/* Button to fill billing address from shipping address. */ +"Use Shipping" = "Use Shipping"; + diff --git a/Stripe/StripeiOS/Resources/Localizations/es-419.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/es-419.lproj/Localizable.strings new file mode 100644 index 00000000..0ff4ff68 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/es-419.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Fuera de línea"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Agregar nueva tarjeta..."; + +"Add a Card" = "Agrega una tarjeta"; + +"Apt." = "Apto."; + +"Contact" = "Contacto"; + +"Delivery" = "Entrega"; + +"Delivery Address" = "Dirección de entrega"; + +"Free" = "Gratis"; + +"Invalid Shipping Address" = "Dirección de envío inválida"; + +"Loading…" = "Cargando..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Siguiente"; + +"Online Banking (FPX)" = "Banca electrónica (FPX)"; + +"Payment Method" = "Método de pago"; + +"Shipping" = "Envío"; + +"Shipping Method" = "Método de envío"; + +"Use Billing" = "Usar la de facturación"; + +"Use Delivery" = "Usar la de entrega"; + +"Use Shipping" = "Usar la de envío"; diff --git a/Stripe/StripeiOS/Resources/Localizations/es.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/es.lproj/Localizable.strings new file mode 100644 index 00000000..13311015 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/es.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Fuera de línea"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Añadir tarjeta nueva…"; + +"Add a Card" = "Añade una tarjeta"; + +"Apt." = "Piso"; + +"Contact" = "Contacto"; + +"Delivery" = "Entrega"; + +"Delivery Address" = "Dirección de entrega"; + +"Free" = "Gratuito"; + +"Invalid Shipping Address" = "Dirección de envío no válida"; + +"Loading…" = "Cargando…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Siguiente"; + +"Online Banking (FPX)" = "Banca electrónica (FPX)"; + +"Payment Method" = "Método de pago"; + +"Shipping" = "Envío"; + +"Shipping Method" = "Método de envío"; + +"Use Billing" = "Usar la de facturación"; + +"Use Delivery" = "Usar la de entrega"; + +"Use Shipping" = "Usar la de envío"; diff --git a/Stripe/StripeiOS/Resources/Localizations/et-EE.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/et-EE.lproj/Localizable.strings new file mode 100644 index 00000000..afe2d85c --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/et-EE.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Võrguühenduseta"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Lisa uus kaart ..."; + +"Add a Card" = "Kaardi lisamine"; + +"Apt." = "Korter"; + +"Contact" = "Kontakt"; + +"Delivery" = "Tarnimine"; + +"Delivery Address" = "Tarneaadress"; + +"Free" = "Tasuta"; + +"Invalid Shipping Address" = "Kehtetu tarneaadress"; + +"Loading…" = "Laadimine ..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Järgmine"; + +"Online Banking (FPX)" = "Internetipank (FPX)"; + +"Payment Method" = "Makseviis"; + +"Shipping" = "Tarnimine"; + +"Shipping Method" = "Tarneviis"; + +"Use Billing" = "Kasuta arveldusaadressi"; + +"Use Delivery" = "Kasuta saaja aadressi"; + +"Use Shipping" = "Kasuta tarneaadressi"; diff --git a/Stripe/StripeiOS/Resources/Localizations/fi.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/fi.lproj/Localizable.strings new file mode 100644 index 00000000..71096979 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/fi.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ Offline-tila"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Lisää uusi kortti..."; + +"Add a Card" = "Lisää kortti"; + +"Apt." = "Asunto"; + +"Contact" = "Yhteystiedot"; + +"Delivery" = "Toimitus"; + +"Delivery Address" = "Toimitusosoite"; + +"Free" = "Maksuton"; + +"Invalid Shipping Address" = "Toimitusosoite ei kelpaa"; + +"Loading…" = "Lataa..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Seuraava"; + +"Online Banking (FPX)" = "Verkkopankki (FPX)"; + +"Payment Method" = "Maksutapa"; + +"Shipping" = "Toimitus"; + +"Shipping Method" = "Toimitustapa"; + +"Use Billing" = "Käytä laskutusosoitetta"; + +"Use Delivery" = "Käytä toimitusosoitetta"; + +"Use Shipping" = "Käytä lähetysosoitetta"; diff --git a/Stripe/StripeiOS/Resources/Localizations/fil.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/fil.lproj/Localizable.strings new file mode 100644 index 00000000..a4252268 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/fil.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Magdagdag ng Bagong Kard..."; + +"Add a Card" = "Magdagdag ng isang Kard"; + +"Apt." = "Apt."; + +"Contact" = "Kontak"; + +"Delivery" = "Pagpapadala"; + +"Delivery Address" = "Adres ng Paghahatarin"; + +"Free" = "Libre"; + +"Invalid Shipping Address" = "Adres ng Nagpadala ay di balido"; + +"Loading…" = "Naglo-load..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Susunod"; + +"Online Banking (FPX)" = "Online Banking (FPX)"; + +"Payment Method" = "Paraan ng Pagbabayad"; + +"Shipping" = "Pagpapadala"; + +"Shipping Method" = "Paraan ng Pagpapadala"; + +"Use Billing" = "Gamitin ang Billing"; + +"Use Delivery" = "Gamitin ang Padadalhan"; + +"Use Shipping" = "Gamitin ang Nagpadala"; diff --git a/Stripe/StripeiOS/Resources/Localizations/fr-CA.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/fr-CA.lproj/Localizable.strings new file mode 100644 index 00000000..8ba185a4 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/fr-CA.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Hors ligne"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Ajouter une carte…"; + +"Add a Card" = "Ajouter une carte"; + +"Apt." = "Nº d'app."; + +"Contact" = "Contact"; + +"Delivery" = "Livraison"; + +"Delivery Address" = "Adresse de livraison"; + +"Free" = "Gratuit"; + +"Invalid Shipping Address" = "Adresse de livraison non valide"; + +"Loading…" = "Chargement…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Suivant"; + +"Online Banking (FPX)" = "Services bancaires en ligne (FPX)"; + +"Payment Method" = "Moyen de paiement"; + +"Shipping" = "Livraison"; + +"Shipping Method" = "Mode de livraison"; + +"Use Billing" = "Utiliser l'adresse de facturation"; + +"Use Delivery" = "Utiliser l'adresse de livraison"; + +"Use Shipping" = "Utiliser l'adresse de livraison"; diff --git a/Stripe/StripeiOS/Resources/Localizations/fr.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/fr.lproj/Localizable.strings new file mode 100644 index 00000000..191697b3 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/fr.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Hors ligne"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Ajouter une nouvelle carte..."; + +"Add a Card" = "Ajouter une carte"; + +"Apt." = "Numéro d'appartement"; + +"Contact" = "Contact"; + +"Delivery" = "Livraison"; + +"Delivery Address" = "Adresse de livraison"; + +"Free" = "Gratuit"; + +"Invalid Shipping Address" = "Adresse de livraison incorrecte"; + +"Loading…" = "Chargement en cours..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Suivant"; + +"Online Banking (FPX)" = "Compte bancaire (FPX)"; + +"Payment Method" = "Moyen de paiement"; + +"Shipping" = "Expédition"; + +"Shipping Method" = "Mode de livraison"; + +"Use Billing" = "Utiliser l'adresse de facturation"; + +"Use Delivery" = "Utiliser l'adresse de livraison"; + +"Use Shipping" = "Utiliser l'adresse de livraison"; diff --git a/Stripe/StripeiOS/Resources/Localizations/hr.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/hr.lproj/Localizable.strings new file mode 100644 index 00000000..207042cc --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/hr.lproj/Localizable.strings @@ -0,0 +1,37 @@ +"%@ - Offline" = "%@ - Izvan mreže"; + +"Add New Card…" = "Dodaj novu karticu..."; + +"Add a Card" = "Dodaj karticu"; + +"Apt." = "Stan br."; + +"Contact" = "Kontakt"; + +"Delivery" = "Dostava"; + +"Delivery Address" = "Adresa za dostavu"; + +"Free" = "Besplatno"; + +"Invalid Shipping Address" = "Adresa za otpremu nije valjana"; + +"Loading…" = "Učitava se..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Sljedeće"; + +"Online Banking (FPX)" = "Online Banking (FPX)"; + +"Payment Method" = "Način plaćanja"; + +"Shipping" = "Otprema"; + +"Shipping Method" = "Način otpreme"; + +"Use Billing" = "Koristi istu adresu kao i za naplatu"; + +"Use Delivery" = "Koristi istu adresu kao i za dostavu"; + +"Use Shipping" = "Koristi istu adresu kao i za otpremu"; diff --git a/Stripe/StripeiOS/Resources/Localizations/hu.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/hu.lproj/Localizable.strings new file mode 100644 index 00000000..355f47cd --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/hu.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Új kártya hozzáadása..."; + +"Add a Card" = "Kártya hozzáadása"; + +"Apt." = "Házszám"; + +"Contact" = "Kapcsolat"; + +"Delivery" = "Szállítás"; + +"Delivery Address" = "Szállítási cím"; + +"Free" = "Ingyenes"; + +"Invalid Shipping Address" = "Érvénytelen szállítási cím"; + +"Loading…" = "Betöltés..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Tovább"; + +"Online Banking (FPX)" = "Online bankolás (FPX)"; + +"Payment Method" = "Fizetési mód"; + +"Shipping" = "Szállítás"; + +"Shipping Method" = "Szállítási mód"; + +"Use Billing" = "Számlázási használata"; + +"Use Delivery" = "Szállítási használata"; + +"Use Shipping" = "Szállítási használata"; diff --git a/Stripe/StripeiOS/Resources/Localizations/id.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/id.lproj/Localizable.strings new file mode 100644 index 00000000..c53fed57 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/id.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Tambahkan Kartu Baru..."; + +"Add a Card" = "Tambahkan sebuah Kartu"; + +"Apt." = "Apt."; + +"Contact" = "Kontak"; + +"Delivery" = "Pengiriman"; + +"Delivery Address" = "Alamat Pengiriman"; + +"Free" = "Gratis"; + +"Invalid Shipping Address" = "Alamat Pengiriman Tidak Valid"; + +"Loading…" = "Memuat…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Berikutnya"; + +"Online Banking (FPX)" = "Perbankan Online (FPX)"; + +"Payment Method" = "Metode Pembayaran"; + +"Shipping" = "Pengiriman"; + +"Shipping Method" = "Metode Pengiriman"; + +"Use Billing" = "Gunakan Tagihan"; + +"Use Delivery" = "Gunakan Pengiriman"; + +"Use Shipping" = "Gunakan Pengiriman"; diff --git a/Stripe/StripeiOS/Resources/Localizations/it.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/it.lproj/Localizable.strings new file mode 100644 index 00000000..e59f1ff5 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/it.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Aggiungi nuova carta…"; + +"Add a Card" = "Aggiungi una carta"; + +"Apt." = "Int."; + +"Contact" = "Contatto"; + +"Delivery" = "Consegna"; + +"Delivery Address" = "Indirizzo di consegna"; + +"Free" = "Gratuita"; + +"Invalid Shipping Address" = "Indirizzo di spedizione non valido"; + +"Loading…" = "Caricamento..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Avanti"; + +"Online Banking (FPX)" = "Online Banking (FPX)"; + +"Payment Method" = "Modalità di pagamento"; + +"Shipping" = "Spedizione"; + +"Shipping Method" = "Modalità di spedizione"; + +"Use Billing" = "Usa indirizzo di fatturazione"; + +"Use Delivery" = "Usa indirizzo di consegna"; + +"Use Shipping" = "Usa indirizzo di spedizione"; diff --git a/Stripe/StripeiOS/Resources/Localizations/ja.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/ja.lproj/Localizable.strings new file mode 100644 index 00000000..4b98d6cc --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/ja.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - オフライン"; + +"3D Secure" = "3D セキュア"; + +"Add New Card…" = "新しいクレジットカードを追加..."; + +"Add a Card" = "カードを追加"; + +"Apt." = "建物名"; + +"Contact" = "連絡先"; + +"Delivery" = "配達先情報"; + +"Delivery Address" = "配達先住所"; + +"Free" = "無料"; + +"Invalid Shipping Address" = "配送先住所が無効です"; + +"Loading…" = "ロード中..."; + +"Multibanco" = "Multibanco"; + +"Next" = "次へ"; + +"Online Banking (FPX)" = "オンラインバンキング (FPX)"; + +"Payment Method" = "お支払い方法"; + +"Shipping" = "配送先情報"; + +"Shipping Method" = "配送方法"; + +"Use Billing" = "請求先を使用"; + +"Use Delivery" = "配達先を使用"; + +"Use Shipping" = "配送先を使用"; diff --git a/Stripe/StripeiOS/Resources/Localizations/ko.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/ko.lproj/Localizable.strings new file mode 100644 index 00000000..2e2e2d79 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/ko.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - 오프라인"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "새 카드 추가..."; + +"Add a Card" = "카드 추가"; + +"Apt." = "아파트"; + +"Contact" = "연락처"; + +"Delivery" = "배달"; + +"Delivery Address" = "배달 주소"; + +"Free" = "무료"; + +"Invalid Shipping Address" = "잘못된 배송 주소"; + +"Loading…" = "로드 중..."; + +"Multibanco" = "Multibanco"; + +"Next" = "다음"; + +"Online Banking (FPX)" = "온라인 뱅킹(FPX)"; + +"Payment Method" = "결제 수단"; + +"Shipping" = "배송"; + +"Shipping Method" = "배송 방법"; + +"Use Billing" = "청구 사용"; + +"Use Delivery" = "배달 사용"; + +"Use Shipping" = "배송 사용"; diff --git a/Stripe/StripeiOS/Resources/Localizations/lt-LT.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/lt-LT.lproj/Localizable.strings new file mode 100644 index 00000000..83a835e9 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/lt-LT.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – neprisijungęs"; + +"3D Secure" = "3D saugi"; + +"Add New Card…" = "Pridėti naują kortelę..."; + +"Add a Card" = "Pridėti kortelę"; + +"Apt." = "But."; + +"Contact" = "Susisiekti"; + +"Delivery" = "Pristatymas"; + +"Delivery Address" = "Pristatymo adresas"; + +"Free" = "Nemokamai"; + +"Invalid Shipping Address" = "Netinkamas siuntimo adresas"; + +"Loading…" = "Įkeliama..."; + +"Multibanco" = "„Multibanco“"; + +"Next" = "Toliau"; + +"Online Banking (FPX)" = "Interneto banko paslaugos (FPX)"; + +"Payment Method" = "Mokėjimo būdas"; + +"Shipping" = "Siuntimas"; + +"Shipping Method" = "Siuntimo būdas"; + +"Use Billing" = "Naudoti sąskaitų siuntimo adresą"; + +"Use Delivery" = "Naudoti pristatymo adresą"; + +"Use Shipping" = "Naudoti siuntimo adresą"; diff --git a/Stripe/StripeiOS/Resources/Localizations/lv-LV.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/lv-LV.lproj/Localizable.strings new file mode 100644 index 00000000..c06a3481 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/lv-LV.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ — bezsaistē"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Pievienot jaunu karti…"; + +"Add a Card" = "Kartes pievienošana"; + +"Apt." = "Dzīv."; + +"Contact" = "Kontaktinformācija"; + +"Delivery" = "Piegāde"; + +"Delivery Address" = "Piegādes adrese"; + +"Free" = "Bezmaksas"; + +"Invalid Shipping Address" = "Nederīga piegādes adrese"; + +"Loading…" = "Ielādē…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Tālāk"; + +"Online Banking (FPX)" = "Internetbanka (FPX)"; + +"Payment Method" = "Maksājuma metode"; + +"Shipping" = "Piegāde"; + +"Shipping Method" = "Piegādes metode"; + +"Use Billing" = "Izmantot norēķinu adresi"; + +"Use Delivery" = "Izmantot piegādes adresi"; + +"Use Shipping" = "Izmantot piegādes adresi"; diff --git a/Stripe/StripeiOS/Resources/Localizations/ms-MY.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/ms-MY.lproj/Localizable.strings new file mode 100644 index 00000000..a4c59d97 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/ms-MY.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – Luar Talian"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Tambah Kad Baharu..."; + +"Add a Card" = "Tambah Kad"; + +"Apt." = "Apt."; + +"Contact" = "Maklumat Hubungan"; + +"Delivery" = "Penghantaran"; + +"Delivery Address" = "Alamat Penghantaran"; + +"Free" = "Percuma"; + +"Invalid Shipping Address" = "Alamat Pengiriman Tidak Sah"; + +"Loading…" = "Sedang dimuatkan..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Seterusnya"; + +"Online Banking (FPX)" = "Perbankan Dalam Talian (FPX)"; + +"Payment Method" = "Kaedah Pembayaran"; + +"Shipping" = "Pengiriman"; + +"Shipping Method" = "Kaedah Pengiriman"; + +"Use Billing" = "Gunakan Alamat Pengebilan"; + +"Use Delivery" = "Gunakan Alamat Penghantaran"; + +"Use Shipping" = "Gunakan Alamat Pengiriman"; diff --git a/Stripe/StripeiOS/Resources/Localizations/mt.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/mt.lproj/Localizable.strings new file mode 100644 index 00000000..d7513a3c --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/mt.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@- Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Żid Kard Ġdida…"; + +"Add a Card" = "Żid Kard"; + +"Apt." = "Appartament"; + +"Contact" = "Kuntatt"; + +"Delivery" = "Kunsinna"; + +"Delivery Address" = "Indirizz tal-Kunsinna"; + +"Free" = "Bla ħlas"; + +"Invalid Shipping Address" = "L-Indirizz għat-Trasport tal-Merkanzija Mhux Validu"; + +"Loading…" = "Qed jillowdja…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Li jmiss"; + +"Online Banking (FPX)" = "Bankar online"; + +"Payment Method" = "Metodu tal-Ħlas"; + +"Shipping" = "Trasport tal-Merkanzija"; + +"Shipping Method" = "Il-Mod tat-Trasport tal-Merkanzija"; + +"Use Billing" = "Uża tal-Kont"; + +"Use Delivery" = "Uża tal-Kunsinna"; + +"Use Shipping" = "Uża tat-Trasport tal-Merkanzija"; diff --git a/Stripe/StripeiOS/Resources/Localizations/nb.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/nb.lproj/Localizable.strings new file mode 100644 index 00000000..336da04f --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/nb.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – frakoblet"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Legg til nytt kort…"; + +"Add a Card" = "Legg til kort"; + +"Apt." = "Leil."; + +"Contact" = "Kontakt"; + +"Delivery" = "Leveranse"; + +"Delivery Address" = "Leveringsadresse"; + +"Free" = "Gratis"; + +"Invalid Shipping Address" = "Ugyldig leveringsadresse"; + +"Loading…" = "Laster ..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Neste"; + +"Online Banking (FPX)" = "Banktjenester på nett (FPX)"; + +"Payment Method" = "Betalingsmåte"; + +"Shipping" = "Frakt"; + +"Shipping Method" = "Forsendelsesmetode"; + +"Use Billing" = "Bruk fakturaadresse"; + +"Use Delivery" = "Bruk leveringsadresse"; + +"Use Shipping" = "Bruk forsendelsesadresse"; diff --git a/Stripe/StripeiOS/Resources/Localizations/nl.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/nl.lproj/Localizable.strings new file mode 100644 index 00000000..d31b0f2f --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/nl.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Nieuwe kaart toevoegen…"; + +"Add a Card" = "Een kaart toevoegen"; + +"Apt." = "Adresregel 2"; + +"Contact" = "Contact"; + +"Delivery" = "Bezorging"; + +"Delivery Address" = "Bezorgadres"; + +"Free" = "Gratis"; + +"Invalid Shipping Address" = "Ongeldig verzendadres"; + +"Loading…" = "Laden…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Volgende"; + +"Online Banking (FPX)" = "Online bankieren (FPX)"; + +"Payment Method" = "Betaalmethode"; + +"Shipping" = "Verzending"; + +"Shipping Method" = "Verzendmethode"; + +"Use Billing" = "Factuuradres gebruiken"; + +"Use Delivery" = "Bezorgadres gebruiken"; + +"Use Shipping" = "Verzendadres gebruiken"; diff --git a/Stripe/StripeiOS/Resources/Localizations/nn-NO.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/nn-NO.lproj/Localizable.strings new file mode 100644 index 00000000..559c25dd --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/nn-NO.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Fråkopla"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Legg til nytt kort …"; + +"Add a Card" = "Legg til kort"; + +"Apt." = "Leilegheit"; + +"Contact" = "Kontakt"; + +"Delivery" = "Leveranse"; + +"Delivery Address" = "Leveringsadresse"; + +"Free" = "Gratis"; + +"Invalid Shipping Address" = "Ugyldig sendingsadresse"; + +"Loading…" = "Lastar …"; + +"Multibanco" = "Multibanco"; + +"Next" = "Neste"; + +"Online Banking (FPX)" = "Nettbank (FPX)"; + +"Payment Method" = "Betalingsmetode"; + +"Shipping" = "Sending"; + +"Shipping Method" = "Sendingsmåte"; + +"Use Billing" = "Bruk fakturering"; + +"Use Delivery" = "Bruk leveringsadressa"; + +"Use Shipping" = "Bruk sendingsadressa"; diff --git a/Stripe/StripeiOS/Resources/Localizations/pl-PL.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/pl-PL.lproj/Localizable.strings new file mode 100644 index 00000000..a5946070 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/pl-PL.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ — offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Dodaj nową kartę…"; + +"Add a Card" = "Dodaj kartę"; + +"Apt." = "Nr lokalu"; + +"Contact" = "Kontakt"; + +"Delivery" = "Dostawa"; + +"Delivery Address" = "Adres dostawy"; + +"Free" = "Bezpłatnie"; + +"Invalid Shipping Address" = "Nieprawidłowy adres dostawy"; + +"Loading…" = "Ładowanie…"; + +"Multibanco" = "Multibanco"; + +"Next" = "Dalej"; + +"Online Banking (FPX)" = "Bankowość elektroniczna (FPX)"; + +"Payment Method" = "Metoda płatności"; + +"Shipping" = "Dostawa"; + +"Shipping Method" = "Metoda dostawy"; + +"Use Billing" = "Użyj adresu płatności"; + +"Use Delivery" = "Użyj adresu dostawy"; + +"Use Shipping" = "Użyj adresu dostawy"; diff --git a/Stripe/StripeiOS/Resources/Localizations/pt-BR.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/pt-BR.lproj/Localizable.strings new file mode 100644 index 00000000..7cdcc17e --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/pt-BR.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Adicionar novo cartão..."; + +"Add a Card" = "Adicionar um cartão"; + +"Apt." = "Apt."; + +"Contact" = "Contato"; + +"Delivery" = "Entrega"; + +"Delivery Address" = "Endereço de entrega"; + +"Free" = "Grátis"; + +"Invalid Shipping Address" = "Endereço de envio inválido"; + +"Loading…" = "Carregando..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Próximo"; + +"Online Banking (FPX)" = "Online Banking (FPX)"; + +"Payment Method" = "Forma de pagamento"; + +"Shipping" = "Dados de entrega"; + +"Shipping Method" = "Método de entrega"; + +"Use Billing" = "Usar cobrança"; + +"Use Delivery" = "Usar entrega"; + +"Use Shipping" = "Usar envio"; diff --git a/Stripe/StripeiOS/Resources/Localizations/pt-PT.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/pt-PT.lproj/Localizable.strings new file mode 100644 index 00000000..0b428ecc --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/pt-PT.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Adicionar Novo Cartão..."; + +"Add a Card" = "Adicionar um cartão"; + +"Apt." = "Apt."; + +"Contact" = "Contacto"; + +"Delivery" = "Entrega"; + +"Delivery Address" = "Endereço de entrega"; + +"Free" = "Gratuito"; + +"Invalid Shipping Address" = "Endereço de envio inválido"; + +"Loading…" = "A carregar..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Seguinte"; + +"Online Banking (FPX)" = "Serviços bancários online (FPX)"; + +"Payment Method" = "Método de Pagamento"; + +"Shipping" = "Envio"; + +"Shipping Method" = "Método de envio"; + +"Use Billing" = "Utilizar endereço de faturação"; + +"Use Delivery" = "Utilizar endereço de entrega"; + +"Use Shipping" = "Utilizar endereço de envio"; diff --git a/Stripe/StripeiOS/Resources/Localizations/ro-RO.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/ro-RO.lproj/Localizable.strings new file mode 100644 index 00000000..666d8c8a --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/ro-RO.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Adăugare card nou..."; + +"Add a Card" = "Adăugare card"; + +"Apt." = "Apartament"; + +"Contact" = "Detalii de contact"; + +"Delivery" = "Informații privind livrarea"; + +"Delivery Address" = "Adresa de livrare"; + +"Free" = "Expediere gratuită"; + +"Invalid Shipping Address" = "Adresa de expediere nu este validă"; + +"Loading…" = "Se încarcă..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Următorul"; + +"Online Banking (FPX)" = "Servicii bancare electronice (FPX)"; + +"Payment Method" = "Metoda de plată"; + +"Shipping" = "Expediere"; + +"Shipping Method" = "Metoda de expediere"; + +"Use Billing" = "Utilizare adresă de facturare"; + +"Use Delivery" = "Utilizare adresă de livrare"; + +"Use Shipping" = "Utilizare adresă de expediere"; diff --git a/Stripe/StripeiOS/Resources/Localizations/ru.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/ru.lproj/Localizable.strings new file mode 100644 index 00000000..9a9d68d5 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/ru.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - не в сети"; + +"3D Secure" = "Метод 3D Secure"; + +"Add New Card…" = "Добавить новую карту..."; + +"Add a Card" = "Добавить карту"; + +"Apt." = "Кв."; + +"Contact" = "Контактные данные"; + +"Delivery" = "Доставка"; + +"Delivery Address" = "Адрес доставки"; + +"Free" = "Бесплатно"; + +"Invalid Shipping Address" = "Ошибочный адрес доставки"; + +"Loading…" = "Идет загрузка..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Далее"; + +"Online Banking (FPX)" = "Онлайн-банкинг (FPX)"; + +"Payment Method" = "Метод оплаты"; + +"Shipping" = "Доставка"; + +"Shipping Method" = "Метод доставки"; + +"Use Billing" = "Использовать адрес выставления счетов"; + +"Use Delivery" = "Использовать адрес доставки"; + +"Use Shipping" = "Использовать адрес доставки"; diff --git a/Stripe/StripeiOS/Resources/Localizations/sk-SK.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/sk-SK.lproj/Localizable.strings new file mode 100644 index 00000000..fb8fa880 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/sk-SK.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Pridať novú kartu..."; + +"Add a Card" = "Pridať kartu"; + +"Apt." = "Byt"; + +"Contact" = "Kontakt"; + +"Delivery" = "Dodanie"; + +"Delivery Address" = "Dodacia adresa"; + +"Free" = "Zdarma"; + +"Invalid Shipping Address" = "Neplatná dodacia adresa"; + +"Loading…" = "Nahrávanie..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Ďalej"; + +"Online Banking (FPX)" = "Online bankovníctvo (FPX)"; + +"Payment Method" = "Spôsob platby"; + +"Shipping" = "Dodanie"; + +"Shipping Method" = "Spôsob dodania"; + +"Use Billing" = "Použiť fakturačnú adresu"; + +"Use Delivery" = "Použiť adresu doručenia"; + +"Use Shipping" = "Použiť dodaciu adresu"; diff --git a/Stripe/StripeiOS/Resources/Localizations/sl-SI.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/sl-SI.lproj/Localizable.strings new file mode 100644 index 00000000..77eccc7d --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/sl-SI.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – ni dosegljiva"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Dodaj novo kartico ..."; + +"Add a Card" = "Dodajte kartico"; + +"Apt." = "Stanovanje"; + +"Contact" = "Stik"; + +"Delivery" = "Dostava"; + +"Delivery Address" = "Dostavni naslov"; + +"Free" = "Brezplačno"; + +"Invalid Shipping Address" = "Neveljaven dostavni naslov"; + +"Loading…" = "Nalaganje ..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Naprej"; + +"Online Banking (FPX)" = "Spletno bančništvo (FPX)"; + +"Payment Method" = "Načina plačila"; + +"Shipping" = "Dostava"; + +"Shipping Method" = "Način dostave"; + +"Use Billing" = "Uporabi naslov plačnika računa"; + +"Use Delivery" = "Uporabi dostavo"; + +"Use Shipping" = "Uporabi dostavo"; diff --git a/Stripe/StripeiOS/Resources/Localizations/sv.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/sv.lproj/Localizable.strings new file mode 100644 index 00000000..77af2122 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/sv.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ – Offline"; + +"3D Secure" = "3D Secure"; + +"Add New Card…" = "Lägg till nytt kort …"; + +"Add a Card" = "Lägg till kort"; + +"Apt." = "Lägenhetsnr."; + +"Contact" = "Kontaktinformation"; + +"Delivery" = "Leverans"; + +"Delivery Address" = "Leveransadress"; + +"Free" = "Gratis"; + +"Invalid Shipping Address" = "Ogiltig leveransadress"; + +"Loading…" = "Laddar ..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Nästa"; + +"Online Banking (FPX)" = "Internetbank (FPX)"; + +"Payment Method" = "Betalningsmetod"; + +"Shipping" = "Frakt"; + +"Shipping Method" = "Leveransmetod"; + +"Use Billing" = "Använd faktureringsadress"; + +"Use Delivery" = "Använd leveransadress"; + +"Use Shipping" = "Använd fraktadress"; diff --git a/Stripe/StripeiOS/Resources/Localizations/tk.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/tk.lproj/Localizable.strings new file mode 100644 index 00000000..e69de29b diff --git a/Stripe/StripeiOS/Resources/Localizations/tr.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/tr.lproj/Localizable.strings new file mode 100644 index 00000000..f660bf8d --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/tr.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Çevrimdışı"; + +"3D Secure" = "3D Güvenli"; + +"Add New Card…" = "Yeni Kart Ekle..."; + +"Add a Card" = "Kart Ekle"; + +"Apt." = "Apt."; + +"Contact" = "İletişim"; + +"Delivery" = "Teslimat"; + +"Delivery Address" = "Teslimat Adresi"; + +"Free" = "Ücretsiz"; + +"Invalid Shipping Address" = "Geçersiz Gönderi Adresi"; + +"Loading…" = "Yükleniyor..."; + +"Multibanco" = "Multibanco"; + +"Next" = "İleri"; + +"Online Banking (FPX)" = "Çevrimiçi Bankacılık (FPX)"; + +"Payment Method" = "Ödeme Metodu"; + +"Shipping" = "Gönderi"; + +"Shipping Method" = "Gönderi Metodu"; + +"Use Billing" = "Fatura Adresini Kullan"; + +"Use Delivery" = "Teslimat Adresini Kullan"; + +"Use Shipping" = "Gönderi Adresini Kullan"; diff --git a/Stripe/StripeiOS/Resources/Localizations/vi.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/vi.lproj/Localizable.strings new file mode 100644 index 00000000..3234969e --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/vi.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - Ngoại tuyến"; + +"3D Secure" = "Bảo mật 3D"; + +"Add New Card…" = "Thêm thẻ mới..."; + +"Add a Card" = "Thêm thẻ"; + +"Apt." = "Căn hộ"; + +"Contact" = "Liên hệ"; + +"Delivery" = "Giao hàng"; + +"Delivery Address" = "Địa chỉ giao hàng"; + +"Free" = "Miễn phí"; + +"Invalid Shipping Address" = "Địa chỉ vận chuyển không hợp lệ"; + +"Loading…" = "Đang tải..."; + +"Multibanco" = "Multibanco"; + +"Next" = "Tiếp"; + +"Online Banking (FPX)" = "Ngân hàng Trực tuyến (FPX)"; + +"Payment Method" = "Phương thức thanh toán"; + +"Shipping" = "Vận chuyển"; + +"Shipping Method" = "Phương thức vận chuyển"; + +"Use Billing" = "Sử dụng Hóa đơn"; + +"Use Delivery" = "Sử dụng Giao hàng"; + +"Use Shipping" = "Chọn Vận chuyển"; diff --git a/Stripe/StripeiOS/Resources/Localizations/zh-HK.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/zh-HK.lproj/Localizable.strings new file mode 100644 index 00000000..58d0da4c --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/zh-HK.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - 離線"; + +"3D Secure" = "3DS 驗證"; + +"Add New Card…" = "添加新卡…"; + +"Add a Card" = "添加卡"; + +"Apt." = "公寓"; + +"Contact" = "聯絡資訊"; + +"Delivery" = "收貨"; + +"Delivery Address" = "收貨地址"; + +"Free" = "免費"; + +"Invalid Shipping Address" = "收货地址無效"; + +"Loading…" = "正在載入..."; + +"Multibanco" = "Multibanco"; + +"Next" = "下一步"; + +"Online Banking (FPX)" = "網上銀行 (FPX)"; + +"Payment Method" = "支付方式"; + +"Shipping" = "配送"; + +"Shipping Method" = "配送方式"; + +"Use Billing" = "使用帳單地址"; + +"Use Delivery" = "使用收貨地址"; + +"Use Shipping" = "使用收货地址"; diff --git a/Stripe/StripeiOS/Resources/Localizations/zh-Hans.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/zh-Hans.lproj/Localizable.strings new file mode 100644 index 00000000..e774f908 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/zh-Hans.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - 离线"; + +"3D Secure" = "3DS 验证"; + +"Add New Card…" = "添加新卡……"; + +"Add a Card" = "添加卡"; + +"Apt." = "公寓"; + +"Contact" = "联系信息"; + +"Delivery" = "收货信息"; + +"Delivery Address" = "收货地址"; + +"Free" = "免费"; + +"Invalid Shipping Address" = "无效的送货地址"; + +"Loading…" = "正在加载……"; + +"Multibanco" = "Multibanco"; + +"Next" = "下一步"; + +"Online Banking (FPX)" = "网上银行 (FPX)"; + +"Payment Method" = "支付方式"; + +"Shipping" = "配送"; + +"Shipping Method" = "配送方式"; + +"Use Billing" = "使用账单地址"; + +"Use Delivery" = "使用收货地址"; + +"Use Shipping" = "使用送货地址"; diff --git a/Stripe/StripeiOS/Resources/Localizations/zh-Hant.lproj/Localizable.strings b/Stripe/StripeiOS/Resources/Localizations/zh-Hant.lproj/Localizable.strings new file mode 100644 index 00000000..dbcc27f7 --- /dev/null +++ b/Stripe/StripeiOS/Resources/Localizations/zh-Hant.lproj/Localizable.strings @@ -0,0 +1,39 @@ +"%@ - Offline" = "%@ - 離線"; + +"3D Secure" = "3DS 驗證"; + +"Add New Card…" = "添加新卡..."; + +"Add a Card" = "新增卡"; + +"Apt." = "公寓"; + +"Contact" = "聯絡資訊"; + +"Delivery" = "寄送"; + +"Delivery Address" = "寄送地址"; + +"Free" = "免費"; + +"Invalid Shipping Address" = "送貨地址無效"; + +"Loading…" = "正在載入..."; + +"Multibanco" = "Multibanco"; + +"Next" = "下一步"; + +"Online Banking (FPX)" = "網上銀行 (FPX)"; + +"Payment Method" = "付款方式"; + +"Shipping" = "送貨"; + +"Shipping Method" = "送貨方式"; + +"Use Billing" = "使用帳單地址"; + +"Use Delivery" = "使用寄送地址"; + +"Use Shipping" = "使用送貨地址"; diff --git a/Stripe/StripeiOS/Source/Enums+CustomStringConvertible.swift b/Stripe/StripeiOS/Source/Enums+CustomStringConvertible.swift new file mode 100644 index 00000000..a7f24c21 --- /dev/null +++ b/Stripe/StripeiOS/Source/Enums+CustomStringConvertible.swift @@ -0,0 +1,63 @@ +// +// Enums+CustomStringConvertible.swift +// Stripe +// +// Autogenerated by generate_objc_enum_string_values.rb +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import SwiftUI + +/// :nodoc: +extension STPBankSelectionMethod: CustomStringConvertible { + public var description: String { + switch self { + case .FPX: + return "FPX" + case .unknown: + return "unknown" + } + } +} + +/// :nodoc: +extension STPBillingAddressFields: CustomStringConvertible { + public var description: String { + switch self { + case .full: + return "full" + case .name: + return "name" + case .none: + return "none" + case .postalCode: + return "postalCode" + case .zip: + return "zip" + } + } +} + +/// :nodoc: +extension STPShippingStatus: CustomStringConvertible { + public var description: String { + switch self { + case .invalid: + return "invalid" + case .valid: + return "valid" + } + } +} + +/// :nodoc: +extension STPShippingType: CustomStringConvertible { + public var description: String { + switch self { + case .delivery: + return "delivery" + case .shipping: + return "shipping" + } + } +} diff --git a/Stripe/StripeiOS/Source/PKAddPaymentPassRequest+Stripe_Error.swift b/Stripe/StripeiOS/Source/PKAddPaymentPassRequest+Stripe_Error.swift new file mode 100644 index 00000000..ab0f3bbe --- /dev/null +++ b/Stripe/StripeiOS/Source/PKAddPaymentPassRequest+Stripe_Error.swift @@ -0,0 +1,30 @@ +// +// PKAddPaymentPassRequest+Stripe_Error.swift +// StripeiOS +// +// Created by Jack Flintermann on 9/29/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +import ObjectiveC +import PassKit + +var stpAddPaymentPassRequest: UInt8 = 0 + +// This is used to store an error on a PKAddPaymentPassRequest +// so that STPFakeAddPaymentPassViewController can inspect it for debugging. +extension PKAddPaymentPassRequest { + @objc var stp_error: NSError? { + get { + return objc_getAssociatedObject(self, &stpAddPaymentPassRequest) as? NSError + } + set(stp_error) { + objc_setAssociatedObject( + self, + &stpAddPaymentPassRequest, + stp_error, + .OBJC_ASSOCIATION_RETAIN_NONATOMIC + ) + } + } +} diff --git a/Stripe/StripeiOS/Source/PKPaymentAuthorizationViewController+Stripe_Blocks.swift b/Stripe/StripeiOS/Source/PKPaymentAuthorizationViewController+Stripe_Blocks.swift new file mode 100644 index 00000000..590d3342 --- /dev/null +++ b/Stripe/StripeiOS/Source/PKPaymentAuthorizationViewController+Stripe_Blocks.swift @@ -0,0 +1,187 @@ +// +// PKPaymentAuthorizationViewController+Stripe_Blocks.swift +// StripeiOS +// +// Created by Ben Guo on 4/19/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import ObjectiveC +import PassKit +@_spi(STP) import StripeCore + +typealias STPApplePayPaymentMethodHandlerBlock = (STPPaymentMethod, @escaping STPPaymentStatusBlock) + -> Void +typealias STPPaymentCompletionBlock = (STPPaymentStatus, Error?) -> Void +typealias STPPaymentSummaryItemCompletionBlock = ([PKPaymentSummaryItem]) -> Void +typealias STPShippingMethodSelectionBlock = ( + PKShippingMethod, @escaping STPPaymentSummaryItemCompletionBlock +) -> Void +typealias STPShippingAddressValidationBlock = ( + STPShippingStatus, [PKShippingMethod], [PKPaymentSummaryItem] +) -> Void +typealias STPShippingAddressSelectionBlock = ( + STPAddress, @escaping STPShippingAddressValidationBlock +) -> Void +typealias STPPaymentAuthorizationBlock = (PKPayment) -> Void +extension PKPaymentAuthorizationViewController { + class func stp_controller( + with paymentRequest: PKPaymentRequest, + apiClient: STPAPIClient, + onShippingAddressSelection: @escaping STPShippingAddressSelectionBlock, + onShippingMethodSelection: @escaping STPShippingMethodSelectionBlock, + onPaymentAuthorization: @escaping STPPaymentAuthorizationBlock, + onPaymentMethodCreation: @escaping STPApplePayPaymentMethodHandlerBlock, + onFinish: @escaping STPPaymentCompletionBlock + ) -> Self? { + let delegate = STPBlockBasedApplePayDelegate() + delegate.apiClient = apiClient + delegate.onShippingAddressSelection = onShippingAddressSelection + delegate.onShippingMethodSelection = onShippingMethodSelection + delegate.onPaymentAuthorization = onPaymentAuthorization + delegate.onPaymentMethodCreation = onPaymentMethodCreation + delegate.onFinish = onFinish + let viewController = self.init(paymentRequest: paymentRequest) + viewController?.delegate = delegate + if let viewController = viewController { + objc_setAssociatedObject( + viewController, + UnsafeRawPointer(&kSTPBlockBasedApplePayDelegateAssociatedObjectKey), + delegate, + .OBJC_ASSOCIATION_RETAIN_NONATOMIC + ) + } + return viewController + } +} + +private var kSTPBlockBasedApplePayDelegateAssociatedObjectKey = 0 +typealias STPApplePayShippingMethodCompletionBlock = ( + PKPaymentAuthorizationStatus, [PKPaymentSummaryItem]? +) -> Void +typealias STPApplePayShippingAddressCompletionBlock = ( + PKPaymentAuthorizationStatus, [PKShippingMethod]?, [PKPaymentSummaryItem]? +) -> Void +class STPBlockBasedApplePayDelegate: NSObject, PKPaymentAuthorizationViewControllerDelegate { + var apiClient: STPAPIClient? + var onShippingAddressSelection: STPShippingAddressSelectionBlock? + var onShippingMethodSelection: STPShippingMethodSelectionBlock? + var onPaymentAuthorization: STPPaymentAuthorizationBlock? + var onPaymentMethodCreation: STPApplePayPaymentMethodHandlerBlock? + var onFinish: STPPaymentCompletionBlock? + var lastError: Error? + var didSucceed = false + + // Remove all this once we drop iOS 11 support + func paymentAuthorizationViewController( + _ controller: PKPaymentAuthorizationViewController, + didAuthorizePayment payment: PKPayment, + completion: @escaping (PKPaymentAuthorizationStatus) -> Void + ) { + onPaymentAuthorization?(payment) + + let paymentMethodCreateCompletion: ((STPPaymentMethod?, Error?) -> Void)? = { + result, + paymentMethodCreateError in + if let paymentMethodCreateError = paymentMethodCreateError { + self.lastError = paymentMethodCreateError + completion(.failure) + return + } + guard let result = result else { + self.lastError = NSError.stp_genericFailedToParseResponseError() + completion(.failure) + return + } + self.onPaymentMethodCreation?( + result, + { status, error in + if status != .success || error != nil { + self.lastError = error + completion(.failure) + if controller.presentingViewController == nil { + // If we call completion() after dismissing, didFinishWithStatus is NOT called. + self._finish() + } + return + } + self.didSucceed = true + completion(.success) + if controller.presentingViewController == nil { + // If we call completion() after dismissing, didFinishWithStatus is NOT called. + self._finish() + } + } + ) + } + if let paymentMethodCreateCompletion = paymentMethodCreateCompletion { + apiClient?.createPaymentMethod(with: payment, completion: paymentMethodCreateCompletion) + } + } + + func paymentAuthorizationViewController( + _ controller: PKPaymentAuthorizationViewController, + didSelect shippingMethod: PKShippingMethod, + completion: @escaping (PKPaymentAuthorizationStatus, [PKPaymentSummaryItem]) -> Void + ) { + onShippingMethodSelection?( + shippingMethod, + { summaryItems in + completion(PKPaymentAuthorizationStatus.success, summaryItems) + } + ) + } + + func paymentAuthorizationViewController( + _ controller: PKPaymentAuthorizationViewController, + didSelectShippingContact contact: PKContact, + handler completion: @escaping (PKPaymentRequestShippingContactUpdate) -> Void + ) { + let stpAddress = STPAddress(pkContact: contact) + onShippingAddressSelection?( + stpAddress, + { status, shippingMethods, summaryItems in + if status == .invalid { + let genericShippingError = NSError( + domain: PKPaymentErrorDomain, + code: PKPaymentError.shippingContactInvalidError.rawValue, + userInfo: nil + ) + completion( + PKPaymentRequestShippingContactUpdate( + errors: [genericShippingError], + paymentSummaryItems: summaryItems, + shippingMethods: shippingMethods + ) + ) + } else { + completion( + PKPaymentRequestShippingContactUpdate( + errors: nil, + paymentSummaryItems: summaryItems, + shippingMethods: shippingMethods + ) + ) + } + } + ) + } + + func paymentAuthorizationViewControllerDidFinish( + _ controller: PKPaymentAuthorizationViewController + ) { + _finish() + } + + func _finish() { + if didSucceed { + onFinish?(.success, nil) + } else if let lastError = lastError { + onFinish?(.error, lastError) + } else { + onFinish?(.userCancellation, nil) + } + } +} + +typealias STPPaymentAuthorizationStatusCallback = (PKPaymentAuthorizationStatus) -> Void diff --git a/Stripe/StripeiOS/Source/STPAPIClient+BasicUI.swift b/Stripe/StripeiOS/Source/STPAPIClient+BasicUI.swift new file mode 100644 index 00000000..00e78896 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAPIClient+BasicUI.swift @@ -0,0 +1,178 @@ +// +// STPAPIClient+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments + +/// A client for making connections to the Stripe API. +extension STPAPIClient { + /// Initializes an API client with the given configuration. + /// - Parameter configuration: The configuration to use. + /// - Returns: An instance of STPAPIClient. + @available( + *, + deprecated, + message: + "This initializer previously configured publishableKey and stripeAccount via the STPPaymentConfiguration instance. This behavior is deprecated; set the STPAPIClient configuration, publishableKey, and stripeAccount properties directly on the STPAPIClient instead." + ) + public convenience init( + configuration: STPPaymentConfiguration + ) { + // For legacy reasons, we'll support this initializer and use the deprecated configuration.{publishableKey, stripeAccount} properties + self.init() + publishableKey = configuration.publishableKey + stripeAccount = configuration.stripeAccount + } +} + +extension STPAPIClient { + /// The client's configuration. + /// Defaults to `STPPaymentConfiguration.shared`. + @objc public var configuration: STPPaymentConfiguration { + get { + if let config = _stored_configuration as? STPPaymentConfiguration { + return config + } else { + return .shared + } + } + set { + _stored_configuration = newValue + } + } + + /// Update a customer with parameters + /// - seealso: https://stripe.com/docs/api#update_customer + func updateCustomer( + withParameters parameters: [String: Any], + using ephemeralKey: STPEphemeralKey, + completion: @escaping STPCustomerCompletionBlock + ) { + let endpoint = "\(APIEndpointCustomers)/\(ephemeralKey.customerID ?? "")" + APIRequest.post( + with: self, + endpoint: endpoint, + additionalHeaders: authorizationHeader(using: ephemeralKey), + parameters: parameters + ) { object, _, error in + completion(object, error) + } + } + + /// Attach a Payment Method to a customer + /// - seealso: https://stripe.com/docs/api/payment_methods/attach + func attachPaymentMethod( + _ paymentMethodID: String, + toCustomerUsing ephemeralKey: STPEphemeralKey, + completion: @escaping STPErrorBlock + ) { + guard let customerID = ephemeralKey.customerID else { + assertionFailure() + completion(nil) + return + } + let endpoint = "\(APIEndpointPaymentMethods)/\(paymentMethodID)/attach" + APIRequest.post( + with: self, + endpoint: endpoint, + additionalHeaders: authorizationHeader(using: ephemeralKey), + parameters: [ + "customer": customerID + ] + ) { _, _, error in + completion(error) + } + } + + /// Detach a Payment Method from a customer + /// - seealso: https://stripe.com/docs/api/payment_methods/detach + func detachPaymentMethod( + _ paymentMethodID: String, + fromCustomerUsing ephemeralKey: STPEphemeralKey, + completion: @escaping STPErrorBlock + ) { + let endpoint = "\(APIEndpointPaymentMethods)/\(paymentMethodID)/detach" + APIRequest.post( + with: self, + endpoint: endpoint, + additionalHeaders: authorizationHeader(using: ephemeralKey), + parameters: [:] + ) { _, _, error in + completion(error) + } + } + + /// Retrieves a list of Payment Methods attached to a customer. + /// @note This only fetches card type Payment Methods + func listPaymentMethodsForCustomer( + using ephemeralKey: STPEphemeralKey, + completion: @escaping STPPaymentMethodsCompletionBlock + ) { + let header = authorizationHeader(using: ephemeralKey) + let params: [String: Any] = [ + "customer": ephemeralKey.customerID ?? "", + "type": "card", + ] + APIRequest.getWith( + self, + endpoint: APIEndpointPaymentMethods, + additionalHeaders: header, + parameters: params as [String: Any] + ) { deserializer, _, error in + if let error = error { + completion(nil, error) + } else if let paymentMethods = deserializer?.paymentMethods { + completion(paymentMethods, nil) + } + } + } + + /// Retrieve a customer + /// - seealso: https://stripe.com/docs/api#retrieve_customer + func retrieveCustomer( + using ephemeralKey: STPEphemeralKey, + completion: @escaping STPCustomerCompletionBlock + ) { + let endpoint = "\(APIEndpointCustomers)/\(ephemeralKey.customerID ?? "")" + APIRequest.getWith( + self, + endpoint: endpoint, + additionalHeaders: authorizationHeader(using: ephemeralKey), + parameters: [:] + ) { object, _, error in + completion(object, error) + } + } + + // MARK: FPX + /// Retrieves the online status of the FPX banks from the Stripe API. + /// - Parameter completion: The callback to run with the returned FPX bank list, or an error. + func retrieveFPXBankStatus( + withCompletion completion: @escaping STPFPXBankStatusCompletionBlock + ) { + APIRequest.getWith( + self, + endpoint: APIEndpointFPXStatus, + parameters: [ + "account_holder_type": "individual" + ] + ) { statusResponse, _, error in + completion(statusResponse, error) + } + } + + // MARK: Helpers + + /// A helper method that returns the Authorization header to use for API requests. If ephemeralKey is nil, uses self.publishableKey instead. + func authorizationHeader(using ephemeralKey: STPEphemeralKey? = nil) -> [String: String] { + return authorizationHeader(using: ephemeralKey?.secret) + } +} + +private let APIEndpointFPXStatus = "fpx/bank_statuses" diff --git a/Stripe/StripeiOS/Source/STPAPIClient+PushProvisioning.swift b/Stripe/StripeiOS/Source/STPAPIClient+PushProvisioning.swift new file mode 100644 index 00000000..1acda148 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAPIClient+PushProvisioning.swift @@ -0,0 +1,40 @@ +// +// STPAPIClient+PushProvisioning.swift +// StripeiOS +// +// Created by Jack Flintermann on 9/27/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments +@_spi(STP) import StripePaymentsUI + +typealias STPPushProvisioningDetailsCompletionBlock = (STPPushProvisioningDetails?, Error?) -> Void +extension STPAPIClient { + func retrievePushProvisioningDetails( + with params: STPPushProvisioningDetailsParams, + ephemeralKey: STPEphemeralKey, + completion: @escaping STPPushProvisioningDetailsCompletionBlock + ) { + + let endpoint = "issuing/cards/\(params.cardId)/push_provisioning_details" + let parameters = [ + "ios": [ + "certificates": params.certificatesBase64, + "nonce": params.nonceHex, + "nonce_signature": params.nonceSignatureHex, + ] as [String: Any], + ] + + APIRequest.getWith( + self, + endpoint: endpoint, + additionalHeaders: authorizationHeader(using: ephemeralKey), + parameters: parameters + ) { details, _, error in + completion(details, error) + } + } +} diff --git a/Stripe/StripeiOS/Source/STPAddCardViewController.swift b/Stripe/StripeiOS/Source/STPAddCardViewController.swift new file mode 100644 index 00000000..91391796 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAddCardViewController.swift @@ -0,0 +1,972 @@ +// +// STPAddCardViewController.swift +// StripeiOS +// +// Created by Jack Flintermann on 3/23/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments +@_spi(STP) import StripePaymentsUI +@_spi(STP) import StripeUICore +import UIKit + +/// This view controller contains a credit card entry form that the user can fill out. On submission, it will use the Stripe API to convert the user's card details to a Stripe token. It renders a right bar button item that submits the form, so it must be shown inside a `UINavigationController`. +public class STPAddCardViewController: STPCoreTableViewController, STPAddressViewModelDelegate, + STPCardScannerDelegate, STPPaymentCardTextFieldDelegate, UITableViewDelegate, + UITableViewDataSource +{ + + /// A convenience initializer; equivalent to calling `init(configuration: STPPaymentConfiguration.shared, theme: STPTheme.defaultTheme)`. + @objc + public convenience init() { + self.init(configuration: STPPaymentConfiguration.shared, theme: STPTheme.defaultTheme) + } + + /// Initializes a new `STPAddCardViewController` with the provided configuration and theme. Don't forget to set the `delegate` property after initialization. + /// - Parameters: + /// - configuration: The configuration to use (this determines the Stripe publishable key to use, the required billing address fields, whether or not to use SMS autofill, etc). - seealso: STPPaymentConfiguration + /// - theme: The theme to use to inform the view controller's visual appearance. - seealso: STPTheme + @objc(initWithConfiguration:theme:) + public init( + configuration: STPPaymentConfiguration, + theme: STPTheme + ) { + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + super.init(theme: theme) + commonInit(with: configuration) + } + + /// The view controller's delegate. This must be set before showing the view controller in order for it to work properly. - seealso: STPAddCardViewControllerDelegate + @objc public weak var delegate: STPAddCardViewControllerDelegate? + /// You can set this property to pre-fill any information you've already collected from your user. - seealso: STPUserInformation.h + @objc public var prefilledInformation: STPUserInformation? { + didSet { + if let address = prefilledInformation?.billingAddress { + addressViewModel.address = address + } + } + } + + private var _customFooterView: UIView? + /// Provide this view controller with a footer view. + /// When the footer view needs to be resized, it will be sent a + /// `sizeThatFits:` call. The view should respond correctly to this method in order + /// to be sized and positioned properly. + @objc public var customFooterView: UIView? { + get { + _customFooterView + } + set(footerView) { + _customFooterView = footerView + _configureFooterView() + } + } + + func _configureFooterView() { + if isViewLoaded, let footerView = _customFooterView { + let size = footerView.sizeThatFits( + CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) + ) + footerView.frame = CGRect(x: 0, y: 0, width: size.width, height: size.height) + + tableView?.tableFooterView = footerView + } + } + + /// The API Client to use to make requests. + /// Defaults to `STPAPIClient.shared` + public var apiClient: STPAPIClient = STPAPIClient.shared + + /// Use init: or initWithConfiguration:theme: + required init( + theme: STPTheme? + ) { + let configuration = STPPaymentConfiguration.shared + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + super.init(theme: theme) + } + + /// Use init: or initWithConfiguration:theme: + required init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + let configuration = STPPaymentConfiguration.shared + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil) + } + + /// Use init: or initWithConfiguration:theme: + required init?( + coder aDecoder: NSCoder + ) { + let configuration = STPPaymentConfiguration.shared + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + super.init(coder: aDecoder) + } + + private var _alwaysEnableDoneButton = false + @objc var alwaysEnableDoneButton: Bool { + get { + _alwaysEnableDoneButton + } + set(alwaysEnableDoneButton) { + if alwaysEnableDoneButton != _alwaysEnableDoneButton { + _alwaysEnableDoneButton = alwaysEnableDoneButton + updateDoneButton() + } + } + } + private var configuration: STPPaymentConfiguration? + @objc var shippingAddress: STPAddress? + private var hasUsedShippingAddress = false + private weak var cardImageView: UIImageView? + private var doneItem: UIBarButtonItem? + private var cardHeaderView: STPSectionHeaderView? + + @available(iOS 13, macCatalyst 14, *) + private var cardScanner: STPCardScanner? { + get { + _cardScanner as? STPCardScanner + } + set { + _cardScanner = newValue + } + } + + /// Storage for `cardScanner`. + private var _cardScanner: NSObject? + + @available(macCatalyst 14, *) + private var scannerCell: STPCardScannerTableViewCell? { + get { + _scannerCell as? STPCardScannerTableViewCell + } + set { + _scannerCell = newValue + } + } + + /// Storage for `scannerCell`. + private var _scannerCell: NSObject? + + private var _isScanning = false + private var isScanning: Bool { + get { + _isScanning + } + set(isScanning) { + if _isScanning == isScanning { + return + } + _isScanning = isScanning + + cardHeaderView?.button?.isEnabled = !isScanning + let indexPath = IndexPath( + row: 0, + section: STPPaymentCardSection.stpPaymentCardScannerSection.rawValue + ) + tableView?.beginUpdates() + if isScanning { + tableView?.insertRows(at: [indexPath], with: .automatic) + } else { + tableView?.deleteRows(at: [indexPath], with: .automatic) + } + tableView?.endUpdates() + if isScanning { + tableView?.scrollToRow(at: indexPath, at: .middle, animated: true) + } + updateInputAccessoryVisiblity() + } + } + private var addressHeaderView: STPSectionHeaderView? + var paymentCell: STPPaymentCardTextFieldCell? + + private var _loading = false + @objc var loading: Bool { + get { + _loading + } + set(loading) { + if loading == _loading { + return + } + _loading = loading + stp_navigationItemProxy?.setHidesBackButton(loading, animated: true) + stp_navigationItemProxy?.leftBarButtonItem?.isEnabled = !loading + activityIndicator?.animating = loading + if loading { + tableView?.endEditing(true) + var loadingItem: UIBarButtonItem? + if let activityIndicator = activityIndicator { + loadingItem = UIBarButtonItem(customView: activityIndicator) + } + stp_navigationItemProxy?.setRightBarButton(loadingItem, animated: true) + cardHeaderView?.buttonHidden = true + } else { + stp_navigationItemProxy?.setRightBarButton(doneItem, animated: true) + cardHeaderView?.buttonHidden = false + } + var cells = addressViewModel.addressCells as [UITableViewCell] + + if let paymentCell = paymentCell { + cells.append(paymentCell) + } + for cell in cells { + cell.isUserInteractionEnabled = !loading + UIView.animate( + withDuration: 0.1, + animations: { + cell.alpha = loading ? 0.7 : 1.0 + } + ) + } + } + } + private var activityIndicator: STPPaymentActivityIndicatorView? + private weak var lookupActivityIndicator: STPPaymentActivityIndicatorView? + var addressViewModel: STPAddressViewModel + private var inputAccessoryToolbar: UIToolbar? + private var lookupSucceeded = false + private var scannerCompleteAnimationTimer: Timer? + + @objc(commonInitWithConfiguration:) func commonInit(with configuration: STPPaymentConfiguration) + { + STPAnalyticsClient.sharedClient.addClass( + toProductUsageIfNecessary: STPAddCardViewController.self + ) + + self.configuration = configuration + shippingAddress = nil + hasUsedShippingAddress = false + addressViewModel.delegate = self + title = STPLocalizedString("Add a Card", "Title for Add a Card view") + + if #available(iOS 13.0, macCatalyst 14, *) { + cardScanner = STPCardScanner() + } + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + estimatedHeightForRowAt indexPath: IndexPath + ) -> CGFloat { + return 44.0 + } + + @objc override func createAndSetupViews() { + super.createAndSetupViews() + + let doneItem = UIBarButtonItem( + barButtonSystemItem: .done, + target: self, + action: #selector(nextPressed(_:)) + ) + self.doneItem = doneItem + stp_navigationItemProxy?.rightBarButtonItem = doneItem + updateDoneButton() + + stp_navigationItemProxy?.leftBarButtonItem?.accessibilityIdentifier = + "AddCardViewControllerNavBarCancelButtonIdentifier" + stp_navigationItemProxy?.rightBarButtonItem?.accessibilityIdentifier = + "AddCardViewControllerNavBarDoneButtonIdentifier" + + let cardImageView = UIImageView(image: STPLegacyImageLibrary.largeCardFrontImage()) + cardImageView.contentMode = .center + cardImageView.frame = CGRect( + x: 0, + y: 0, + width: view.bounds.size.width, + height: cardImageView.bounds.size.height + (57 * 2) + ) + self.cardImageView = cardImageView + tableView?.tableHeaderView = cardImageView + + let paymentCell = STPPaymentCardTextFieldCell( + style: .default, + reuseIdentifier: "STPAddCardViewControllerPaymentCardTextFieldCell" + ) + paymentCell.paymentField?.delegate = self + if configuration?.requiredBillingAddressFields == .postalCode { + // If postal code collection is enabled, move the postal code field into the card entry field. + // Otherwise, this will be picked up by the billing address fields below. + paymentCell.paymentField?.postalCodeEntryEnabled = true + } + self.paymentCell = paymentCell + + activityIndicator = STPPaymentActivityIndicatorView( + frame: CGRect(x: 0, y: 0, width: 20.0, height: 20.0) + ) + + inputAccessoryToolbar = UIToolbar.stp_inputAccessoryToolbar( + withTarget: self, + action: #selector(paymentFieldNextTapped) + ) + inputAccessoryToolbar?.stp_setEnabled(false) + updateInputAccessoryVisiblity() + tableView?.dataSource = self + tableView?.delegate = self + tableView?.reloadData() + if let address = prefilledInformation?.billingAddress { + addressViewModel.address = address + } + + let addressHeaderView = STPSectionHeaderView() + addressHeaderView.theme = theme + addressHeaderView.title = String.Localized.billing_address + switch configuration?.shippingType { + case .shipping: + addressHeaderView.button?.setTitle( + STPLocalizedString( + "Use Shipping", + "Button to fill billing address from shipping address." + ), + for: .normal + ) + case .delivery: + addressHeaderView.button?.setTitle( + STPLocalizedString( + "Use Delivery", + "Button to fill billing address from delivery address." + ), + for: .normal + ) + default: + break + } + addressHeaderView.button?.addTarget( + self, + action: #selector(useShippingAddress(_:)), + for: .touchUpInside + ) + let requiredFields = configuration?.requiredBillingAddressFields ?? .none + let needsAddress = requiredFields != .none && !addressViewModel.isValid + let buttonVisible = + needsAddress && shippingAddress?.containsContent(for: requiredFields) != nil + && !hasUsedShippingAddress + addressHeaderView.buttonHidden = !buttonVisible + addressHeaderView.setNeedsLayout() + self.addressHeaderView = addressHeaderView + let cardHeaderView = STPSectionHeaderView() + cardHeaderView.theme = theme + cardHeaderView.title = STPPaymentMethodType.card.displayName + cardHeaderView.buttonHidden = true + self.cardHeaderView = cardHeaderView + + // re-set the custom footer view if it was added before we loaded + _configureFooterView() + + view.addGestureRecognizer( + UITapGestureRecognizer(target: self, action: #selector(endEditing)) + ) + + setUpCardScanningIfAvailable() + + STPAnalyticsClient.sharedClient.clearAdditionalInfo() + } + + /// :nodoc: + @objc + public override func viewDidLayoutSubviews() { + super.viewDidLayoutSubviews() + + // Resetting it re-calculates the size based on new view width + // UITableView requires us to call setter again to actually pick up frame + // change on footers + if tableView?.tableFooterView != nil { + customFooterView = tableView?.tableFooterView + } + } + + func setUpCardScanningIfAvailable() { + if #available(iOS 13.0, macCatalyst 14, *) { + if !STPCardScanner.cardScanningAvailable || configuration?.cardScanningEnabled != true { + return + } + let scannerCell = STPCardScannerTableViewCell() + self.scannerCell = scannerCell + + let cardScanner = STPCardScanner(delegate: self) + cardScanner.cameraView = scannerCell.cameraView + self.cardScanner = cardScanner + + cardHeaderView?.buttonHidden = false + cardHeaderView?.button?.setTitle( + String.Localized.scan_card_title_capitalization, + for: .normal + ) + cardHeaderView?.button?.addTarget( + self, + action: #selector(scanCard), + for: .touchUpInside + ) + cardHeaderView?.setNeedsLayout() + } + } + + @objc func scanCard() { + if #available(iOS 13.0, macCatalyst 14.0, *) { + view.endEditing(true) + isScanning = true + cardScanner?.start() + } + } + + @objc func endEditing() { + view.endEditing(false) + } + + /// :nodoc: + @objc + public override func updateAppearance() { + super.updateAppearance() + + view.backgroundColor = theme.primaryBackgroundColor + + let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme + doneItem?.stp_setTheme(navBarTheme) + tableView?.allowsSelection = false + + cardImageView?.tintColor = theme.accentColor + activityIndicator?.tintColor = theme.accentColor + + paymentCell?.theme = theme + cardHeaderView?.theme = theme + addressHeaderView?.theme = theme + for cell in addressViewModel.addressCells { + cell.theme = theme + } + setNeedsStatusBarAppearanceUpdate() + } + + /// :nodoc: + @objc + public override func viewDidAppear(_ animated: Bool) { + super.viewDidAppear(animated) + stp_beginObservingKeyboardAndInsettingScrollView( + tableView, + onChange: nil + ) + firstEmptyField()?.becomeFirstResponder() + } + + func firstEmptyField() -> UIResponder? { + + if paymentCell?.isEmpty != nil { + return paymentCell! + } + + for cell in addressViewModel.addressCells { + if cell.contents?.count ?? 0 == 0 { + return cell + } + } + return nil + } + + /// :nodoc: + @objc + public override func handleCancelTapped(_ sender: Any?) { + delegate?.addCardViewControllerDidCancel(self) + } + + @objc func nextPressed(_ sender: Any?) { + loading = true + guard let cardParams = paymentCell?.paymentField?.paymentMethodParams.card else { + return + } + // Create and return a Payment Method + let billingDetails = STPPaymentMethodBillingDetails() + if configuration?.requiredBillingAddressFields == .postalCode { + let address = STPAddress() + address.postalCode = paymentCell?.paymentField?.postalCode + billingDetails.address = STPPaymentMethodAddress(address: address) + } else { + billingDetails.address = STPPaymentMethodAddress(address: addressViewModel.address) + billingDetails.email = addressViewModel.address.email + billingDetails.name = addressViewModel.address.name + billingDetails.phone = addressViewModel.address.phone + } + let paymentMethodParams = STPPaymentMethodParams( + card: cardParams, + billingDetails: billingDetails, + metadata: nil + ) + apiClient.createPaymentMethod(with: paymentMethodParams) { + paymentMethod, + createPaymentMethodError in + if let createPaymentMethodError = createPaymentMethodError { + self.handleError(createPaymentMethodError) + } else { + if let paymentMethod = paymentMethod { + self.delegate?.addCardViewController( + self, + didCreatePaymentMethod: paymentMethod + ) { + attachToCustomerError in + stpDispatchToMainThreadIfNecessary({ + if let attachToCustomerError = attachToCustomerError { + self.handleError(attachToCustomerError) + } else { + self.loading = false + } + }) + } + } + } + } + } + + func handleError(_ error: Error) { + loading = false + firstEmptyField()?.becomeFirstResponder() + + let alertController = UIAlertController( + title: error.localizedDescription, + message: (error as NSError).localizedFailureReason, + preferredStyle: .alert + ) + + alertController.addAction( + UIAlertAction( + title: String.Localized.ok, + style: .cancel, + handler: nil + ) + ) + + present(alertController, animated: true) + } + + func updateDoneButton() { + stp_navigationItemProxy?.rightBarButtonItem?.isEnabled = + (paymentCell?.paymentField?.isValid ?? false && addressViewModel.isValid) + || alwaysEnableDoneButton + } + + func updateInputAccessoryVisiblity() { + // The inputAccessoryToolbar switches from the paymentCell to the first address field. + // It should only be shown when there *is* an address field. This compensates for the lack + // of a 'Return' key on the number pad used for paymentCell entry + let hasAddressCells = (addressViewModel.addressCells.count) > 0 + paymentCell?.inputAccessoryView = hasAddressCells ? inputAccessoryToolbar : nil + } + + // MARK: - STPPaymentCardTextField + @objc + public func paymentCardTextFieldDidChange(_ textField: STPPaymentCardTextField) { + inputAccessoryToolbar?.stp_setEnabled(textField.isValid) + updateDoneButton() + } + + @objc func paymentFieldNextTapped() { + _ = addressViewModel.addressCells.stp_boundSafeObject(at: 0)?.becomeFirstResponder() + } + + @objc + public func paymentCardTextFieldWillEndEditing(forReturn textField: STPPaymentCardTextField) { + paymentFieldNextTapped() + } + + @objc + public func paymentCardTextFieldDidBeginEditingCVC(_ textField: STPPaymentCardTextField) { + let isAmex = STPCardValidator.brand(forNumber: textField.cardNumber ?? "") == .amex + var newImage: UIImage? + var animationTransition: UIView.AnimationOptions + + if isAmex { + newImage = STPLegacyImageLibrary.largeCardAmexCVCImage() + animationTransition = .transitionCrossDissolve + } else { + newImage = STPLegacyImageLibrary.largeCardBackImage() + animationTransition = .transitionFlipFromRight + } + + if let cardImageView = cardImageView { + UIView.transition( + with: cardImageView, + duration: 0.2, + options: animationTransition, + animations: { + self.cardImageView?.image = newImage + } + ) + } + } + + @objc + public func paymentCardTextFieldDidEndEditingCVC(_ textField: STPPaymentCardTextField) { + let isAmex = STPCardValidator.brand(forNumber: textField.cardNumber ?? "") == .amex + let animationTransition: UIView.AnimationOptions = + isAmex ? .transitionCrossDissolve : .transitionFlipFromLeft + + if let cardImageView = cardImageView { + UIView.transition( + with: cardImageView, + duration: 0.2, + options: animationTransition, + animations: { + self.cardImageView?.image = STPLegacyImageLibrary.largeCardFrontImage() + } + ) + } + } + + @objc + public func paymentCardTextFieldDidBeginEditing(_ textField: STPPaymentCardTextField) { + if #available(iOS 13.0, macCatalyst 14.0, *) { + cardScanner?.stop() + } + } + + // MARK: - STPAddressViewModelDelegate + func addressViewModel(_ addressViewModel: STPAddressViewModel, addedCellAt index: Int) { + let indexPath = IndexPath( + row: index, + section: STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue + ) + tableView?.insertRows(at: [indexPath], with: .automatic) + updateInputAccessoryVisiblity() + } + + func addressViewModel(_ addressViewModel: STPAddressViewModel, removedCellAt index: Int) { + let indexPath = IndexPath( + row: Int(index), + section: STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue + ) + tableView?.deleteRows(at: [indexPath], with: .automatic) + updateInputAccessoryVisiblity() + } + + func addressViewModelDidChange(_ addressViewModel: STPAddressViewModel) { + updateDoneButton() + } + + func addressViewModelWillUpdate(_ addressViewModel: STPAddressViewModel) { + tableView?.beginUpdates() + } + + func addressViewModelDidUpdate(_ addressViewModel: STPAddressViewModel) { + tableView?.endUpdates() + } + + // MARK: - UITableView + /// :nodoc: + @objc + public func numberOfSections(in tableView: UITableView) -> Int { + return 3 + } + + /// :nodoc: + @objc + public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { + if section == STPPaymentCardSection.stpPaymentCardNumberSection.rawValue { + return 1 + } else if section == STPPaymentCardSection.stpPaymentCardScannerSection.rawValue { + return isScanning ? 1 : 0 + } else if section == STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue { + return addressViewModel.addressCells.count + } + return 0 + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + cellForRowAt indexPath: IndexPath + ) -> UITableViewCell { + var cell: UITableViewCell? + switch indexPath.section { + case STPPaymentCardSection.stpPaymentCardNumberSection.rawValue: + cell = paymentCell + case STPPaymentCardSection.stpPaymentCardScannerSection.rawValue: + if #available(macCatalyst 14, *) { + cell = scannerCell + } else { + assertionFailure() + cell = UITableViewCell() + } + case STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue: + cell = addressViewModel.addressCells.stp_boundSafeObject(at: indexPath.row) + default: + return UITableViewCell() // won't be called; exists to make the static analyzer happy + } + cell?.backgroundColor = theme.secondaryBackgroundColor + cell?.contentView.backgroundColor = UIColor.clear + return cell! + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + willDisplay cell: UITableViewCell, + forRowAt indexPath: IndexPath + ) { + let topRow = indexPath.row == 0 + let bottomRow = + self.tableView(tableView, numberOfRowsInSection: indexPath.section) - 1 == indexPath.row + cell.stp_setBorderColor(theme.tertiaryBackgroundColor) + cell.stp_setTopBorderHidden(!topRow) + cell.stp_setBottomBorderHidden(!bottomRow) + cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) + cell.stp_setFakeSeparatorLeftInset(15.0) + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + heightForFooterInSection section: Int + ) + -> CGFloat + { + if self.tableView(tableView, numberOfRowsInSection: section) == 0 { + return 0.01 + } + return 27.0 + } + + /// :nodoc: + @objc + public override func tableView( + _ tableView: UITableView, + heightForHeaderInSection section: Int + ) -> CGFloat { + let fittingSize = CGSize( + width: view.bounds.size.width, + height: CGFloat.greatestFiniteMagnitude + ) + let numberOfRows = self.tableView(tableView, numberOfRowsInSection: section) + if section == STPPaymentCardSection.stpPaymentCardNumberSection.rawValue { + return cardHeaderView?.sizeThatFits(fittingSize).height ?? 0.0 + } else if section == STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue + && numberOfRows != 0 + { + return addressHeaderView?.sizeThatFits(fittingSize).height ?? 0.0 + } else if section == STPPaymentCardSection.stpPaymentCardScannerSection.rawValue { + return 0.01 + } else if numberOfRows != 0 { + return tableView.sectionHeaderHeight + } + return 0.01 + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + viewForHeaderInSection section: Int + ) + -> UIView? + { + if self.tableView(tableView, numberOfRowsInSection: section) == 0 { + return UIView() + } else { + if section == STPPaymentCardSection.stpPaymentCardNumberSection.rawValue { + return cardHeaderView + } else if section == STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue + { + return addressHeaderView + } + } + return nil + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + viewForFooterInSection section: Int + ) + -> UIView? + { + return UIView() + } + + @objc func useShippingAddress(_ sender: UIButton) { + tableView?.beginUpdates() + addressViewModel.address = shippingAddress ?? STPAddress() + hasUsedShippingAddress = true + firstEmptyField()?.becomeFirstResponder() + UIView.animate( + withDuration: 0.2, + animations: { + self.addressHeaderView?.buttonHidden = true + } + ) + tableView?.endUpdates() + } + + // MARK: - STPCardScanner + /// :nodoc: + @objc + public override func viewWillTransition( + to size: CGSize, + with coordinator: UIViewControllerTransitionCoordinator + ) { + super.viewWillTransition(to: size, with: coordinator) + if #available(iOS 13.0, macCatalyst 14.0, *) { + let orientation = UIDevice.current.orientation + if orientation.isPortrait || orientation.isLandscape { + cardScanner?.deviceOrientation = orientation + } + if isScanning { + let indexPath = IndexPath( + row: 0, + section: STPPaymentCardSection.stpPaymentCardScannerSection.rawValue + ) + DispatchQueue.main.async(execute: { + self.tableView?.scrollToRow(at: indexPath, at: .middle, animated: true) + }) + } + } + } + + static let cardScannerKSTPCardScanAnimationTime: TimeInterval = 0.04 + + @available(iOS 13, macCatalyst 14, *) + func cardScanner( + _ scanner: STPCardScanner, + didFinishWith cardParams: STPPaymentMethodCardParams?, + error: Error? + ) { + if let error = error { + handleError(error) + } + if let cardParams = cardParams { + view.isUserInteractionEnabled = false + paymentCell?.paymentField?.inputView = UIView() as? UIInputView + var i = 0 + scannerCompleteAnimationTimer = Timer.scheduledTimer( + withTimeInterval: STPAddCardViewController.cardScannerKSTPCardScanAnimationTime, + repeats: true, + block: { timer in + i += 1 + let newParams = STPPaymentMethodCardParams() + guard let number = cardParams.number else { + timer.invalidate() + self.view.isUserInteractionEnabled = false + return + } + if i < number.count { + newParams.number = String( + number[...number.index(number.startIndex, offsetBy: i)] + ) + } else { + newParams.number = number + } + self.paymentCell?.paymentField?.paymentMethodParams = STPPaymentMethodParams( + card: newParams, + billingDetails: nil, + metadata: nil + ) + if i > number.count { + self.paymentCell?.paymentField?.paymentMethodParams = + STPPaymentMethodParams( + card: cardParams, + billingDetails: nil, + metadata: nil + ) + self.isScanning = false + self.paymentCell?.paymentField?.inputView = nil + // Force the inputView to reload by asking the text field to resign/become first responder: + _ = self.paymentCell?.paymentField?.resignFirstResponder() + _ = self.paymentCell?.paymentField?.becomeFirstResponder() + timer.invalidate() + self.view.isUserInteractionEnabled = true + } + } + ) + } else { + isScanning = false + } + } + +} + +/// An `STPAddCardViewControllerDelegate` is notified when an `STPAddCardViewController` +/// successfully creates a card token or is cancelled. It has internal error-handling +/// logic, so there's no error case to deal with. +@objc public protocol STPAddCardViewControllerDelegate: NSObjectProtocol { + /// Called when the user cancels adding a card. You should dismiss (or pop) the + /// view controller at this point. + /// - Parameter addCardViewController: the view controller that has been cancelled + func addCardViewControllerDidCancel(_ addCardViewController: STPAddCardViewController) + + /// This is called when the user successfully adds a card and Stripe returns a + /// Payment Method. + /// You should send the PaymentMethod to your backend to store it on a customer, and then + /// call the provided `completion` block when that call is finished. If an error + /// occurs while talking to your backend, call `completion(error)`, otherwise, + /// dismiss (or pop) the view controller. + /// - Parameters: + /// - addCardViewController: the view controller that successfully created a token + /// - paymentMethod: the Payment Method that was created. - seealso: STPPaymentMethod + /// - completion: call this callback when you're done sending the token to your backend + @objc func addCardViewController( + _ addCardViewController: STPAddCardViewController, + didCreatePaymentMethod paymentMethod: STPPaymentMethod, + completion: @escaping STPErrorBlock + ) + + // MARK: - Deprecated + + /// This method is deprecated as of v16.0.0 (https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md#migrating-from-versions--1600). + /// To use this class, migrate your integration from Charges to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read + @available( + *, + deprecated, + message: + "Use addCardViewController(_:didCreatePaymentMethod:completion:) instead and migrate your integration to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read", + renamed: "addCardViewController(_:didCreatePaymentMethod:completion:)" + ) + @objc optional func addCardViewController( + _ addCardViewController: STPAddCardViewController, + didCreateToken token: STPToken, + completion: STPErrorBlock + ) + /// This method is deprecated as of v16.0.0 (https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md#migrating-from-versions--1600). + /// To use this class, migrate your integration from Charges to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read + @available( + *, + deprecated, + message: + "Use addCardViewController(_:didCreatePaymentMethod:completion:) instead and migrate your integration to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read", + renamed: "addCardViewController(_:didCreatePaymentMethod:completion:)" + ) + @objc optional func addCardViewController( + _ addCardViewController: STPAddCardViewController, + didCreateSource source: STPSource, + completion: STPErrorBlock + ) +} + +private let STPPaymentCardCellReuseIdentifier = "STPPaymentCardCellReuseIdentifier" +enum STPPaymentCardSection: Int { + case stpPaymentCardNumberSection = 0 + case stpPaymentCardScannerSection = 1 + case stpPaymentCardBillingAddressSection = 2 +} + +/// :nodoc: +@_spi(STP) extension STPAddCardViewController: STPAnalyticsProtocol { + @_spi(STP) public static let stp_analyticsIdentifier = "STPAddCardViewController" +} diff --git a/Stripe/StripeiOS/Source/STPAddress+BasicUI.swift b/Stripe/StripeiOS/Source/STPAddress+BasicUI.swift new file mode 100644 index 00000000..d99df351 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAddress+BasicUI.swift @@ -0,0 +1,212 @@ +// +// STPAddress+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripePayments +@_spi(STP) import StripePaymentsUI +@_spi(STP) import StripeUICore + +/// What set of billing address information you need to collect from your user. +/// +/// @note If the user is from a country that does not use zip/postal codes, +/// the user may not be asked for one regardless of this setting. +@objc +public enum STPBillingAddressFields: UInt { + /// No billing address information + case none + /// Just request the user's billing postal code + case postalCode + /// Request the user's full billing address + case full + /// Just request the user's billing name + case name + /// Just request the user's billing ZIP (synonym for STPBillingAddressFieldsZip) + @available(*, deprecated, message: "Use STPBillingAddressFields.postalCode instead") + case zip +} + +extension STPAddress { + + /// Checks if this STPAddress has the level of valid address information + /// required by the passed in setting. + /// - Parameter requiredFields: The required level of billing address information to + /// check against. + /// - Returns: YES if this address contains at least the necessary information, + /// NO otherwise. + @objc + public func containsRequiredFields(_ requiredFields: STPBillingAddressFields) -> Bool { + switch requiredFields { + case .none: + return true + case .postalCode: + return STPPostalCodeValidator.validationState( + forPostalCode: postalCode, + countryCode: country + ) == .valid + case .full: + return hasValidPostalAddress() + case .name: + return (name?.count ?? 0) > 0 + default: + fatalError() + } + } + + /// Checks if this STPAddress has any content (possibly invalid) in any of the + /// desired billing address fields. + /// Where `containsRequiredFields:` validates that this STPAddress contains valid data in + /// all of the required fields, this method checks for the existence of *any* data. + /// For example, if `desiredFields` is `STPBillingAddressFieldsZip`, this will check + /// if the postalCode is empty. + /// Note: When `desiredFields == STPBillingAddressFieldsNone`, this method always returns + /// NO. + /// @parameter desiredFields The billing address information the caller is interested in. + /// - Returns: YES if there is any data in this STPAddress that's relevant for those fields. + @objc(containsContentForBillingAddressFields:) + public func containsContent(for desiredFields: STPBillingAddressFields) -> Bool { + switch desiredFields { + case .none: + return false + case .postalCode: + return (postalCode?.count ?? 0) > 0 + case .full: + return hasPartialPostalAddress() + case .name: + return (name?.count ?? 0) > 0 + default: + fatalError() + } + } + + /// Checks if this STPAddress has the level of valid address information + /// required by the passed in setting. + /// Note: When `requiredFields == nil`, this method always returns + /// YES. + /// - Parameter requiredFields: The required shipping address information to check against. + /// - Returns: YES if this address contains at least the necessary information, + /// NO otherwise. + @objc + public func containsRequiredShippingAddressFields( + _ requiredFields: Set? + ) + -> Bool + { + guard let requiredFields = requiredFields else { + return true + } + var containsFields = true + + if requiredFields.contains(.name) { + containsFields = containsFields && (name?.count ?? 0) > 0 + } + if requiredFields.contains(.emailAddress) { + containsFields = + containsFields && STPEmailAddressValidator.stringIsValidEmailAddress(email) + } + if requiredFields.contains(.phoneNumber) { + containsFields = + containsFields + && STPPhoneNumberValidator.stringIsValidPhoneNumber( + phone ?? "", + forCountryCode: country + ) + } + if requiredFields.contains(.postalAddress) { + containsFields = containsFields && hasValidPostalAddress() + } + return containsFields + } + + /// Checks if this STPAddress has any content (possibly invalid) in any of the + /// desired shipping address fields. + /// Where `containsRequiredShippingAddressFields:` validates that this STPAddress + /// contains valid data in all of the required fields, this method checks for the + /// existence of *any* data. + /// Note: When `desiredFields == nil`, this method always returns + /// NO. + /// @parameter desiredFields The shipping address information the caller is interested in. + /// - Returns: YES if there is any data in this STPAddress that's relevant for those fields. + @objc + public func containsContent( + forShippingAddressFields desiredFields: Set? + ) + -> Bool + { + guard let desiredFields = desiredFields else { + return false + } + return (desiredFields.contains(.name) && (name?.count ?? 0) > 0) + || (desiredFields.contains(.emailAddress) && (email?.count ?? 0) > 0) + || (desiredFields.contains(.phoneNumber) && (phone?.count ?? 0) > 0) + || (desiredFields.contains(.postalAddress) && hasPartialPostalAddress()) + } + + /// Converts an STPBillingAddressFields enum value into the closest equivalent + /// representation of PKContactField options + /// - Parameter billingAddressFields: Stripe billing address fields enum value to convert. + /// - Returns: The closest representation of the billing address requirement as + /// a PKContactField value. + @objc(applePayContactFieldsFromBillingAddressFields:) + public class func applePayContactFields( + from billingAddressFields: STPBillingAddressFields + ) + -> Set + { + switch billingAddressFields { + case .none: + return Set([]) + case .postalCode, .full: + return Set([.name, .postalAddress]) + case .name: + return Set([.name]) + case .zip: + return Set() + @unknown default: + fatalError() + } + } + + /// Converts a set of STPContactField values into the closest equivalent + /// representation of PKContactField options + /// - Parameter contactFields: Stripe contact fields values to convert. + /// - Returns: The closest representation of the contact fields as + /// a PKContactField value. + @objc + public class func pkContactFields( + fromStripeContactFields contactFields: Set? + ) -> Set? { + guard let contactFields = contactFields else { + return nil + } + + var pkFields: Set = Set() + let stripeToPayKitContactMap: [STPContactField: PKContactField] = [ + STPContactField.postalAddress: PKContactField.postalAddress, + STPContactField.emailAddress: PKContactField.emailAddress, + STPContactField.phoneNumber: PKContactField.phoneNumber, + STPContactField.name: PKContactField.name, + ] + + for contactField in contactFields { + if let convertedField = stripeToPayKitContactMap[contactField] { + pkFields.insert(convertedField) + } + } + return pkFields + } + + private func hasValidPostalAddress() -> Bool { + return (line1?.count ?? 0) > 0 && (city?.count ?? 0) > 0 && (country?.count ?? 0) > 0 + && ((state?.count ?? 0) > 0 || !(country == "US")) + && (STPPostalCodeValidator.validationState( + forPostalCode: postalCode, + countryCode: country + ) == .valid) + } +} diff --git a/Stripe/StripeiOS/Source/STPAddressFieldTableViewCell.swift b/Stripe/StripeiOS/Source/STPAddressFieldTableViewCell.swift new file mode 100644 index 00000000..7ae4b4e7 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAddressFieldTableViewCell.swift @@ -0,0 +1,509 @@ +// +// STPAddressFieldTableViewCell.swift +// StripeiOS +// +// Created by Ben Guo on 4/13/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +@_spi(STP) import StripeUICore +import UIKit + +enum STPAddressFieldType: Int { + case name + case line1 + case line2 + case city + case state + case zip + case country + case email + case phone +} + +protocol STPAddressFieldTableViewCellDelegate: AnyObject { + func addressFieldTableViewCellDidUpdateText(_ cell: STPAddressFieldTableViewCell) + + func addressFieldTableViewCellDidReturn(_ cell: STPAddressFieldTableViewCell) + func addressFieldTableViewCellDidEndEditing(_ cell: STPAddressFieldTableViewCell) + var addressFieldTableViewCountryCode: String? { get set } + var availableCountries: Set? { get set } +} + +class STPAddressFieldTableViewCell: UITableViewCell, UITextFieldDelegate, UIPickerViewDelegate, + UIPickerViewDataSource +{ + + init( + type: STPAddressFieldType, + contents: String?, + lastInList: Bool, + delegate: STPAddressFieldTableViewCellDelegate? + ) { + textField = { + if type == .phone { + // We have very specific US-based phone formatting that's built into STPFormTextField + let formTextField = STPFormTextField() + formTextField.preservesContentsOnPaste = false + formTextField.selectionEnabled = false + return formTextField + } else { + return STPValidatedTextField() + } + }() + + super.init(style: .default, reuseIdentifier: nil) + self.delegate = delegate + theme = STPTheme() + _contents = contents + + textField.delegate = self + textField.addTarget( + self, + action: #selector(STPAddressFieldTableViewCell.textFieldTextDidChange(textField:)), + for: .editingChanged + ) + contentView.addSubview(textField) + + let toolbar = UIToolbar() + let flexibleItem = UIBarButtonItem( + barButtonSystemItem: .flexibleSpace, + target: nil, + action: nil + ) + let nextItem = UIBarButtonItem( + title: STPLocalizedString("Next", nil), + style: .done, + target: self, + action: #selector(nextTapped(sender:)) + ) + toolbar.items = [flexibleItem, nextItem] + inputAccessoryToolbar = toolbar + + var countryCode = NSLocale.autoupdatingCurrent.regionCode + var otherCountryCodes = Array( + self.delegate?.availableCountries ?? Set(NSLocale.isoCountryCodes) + ) + if otherCountryCodes.contains(countryCode ?? "") { + // Remove the current country code to re-add it once we sort the list. + otherCountryCodes.removeAll { $0 == countryCode } + } else { + // If it isn't in the list (if we've been configured to not show that country), don't re-add it. + countryCode = nil + } + let locale = NSLocale.current as NSLocale + otherCountryCodes = + (otherCountryCodes as NSArray).sortedArray(comparator: { code1, code2 in + guard let code1 = code1 as? String, let code2 = code2 as? String else { + return .orderedDescending + } + let localeID1 = NSLocale.localeIdentifier(fromComponents: [ + NSLocale.Key.countryCode.rawValue: code1 + ]) + let localeID2 = NSLocale.localeIdentifier(fromComponents: [ + NSLocale.Key.countryCode.rawValue: code2 + ]) + if let name1 = locale.displayName(forKey: .identifier, value: localeID1), + let name2 = locale.displayName(forKey: .identifier, value: localeID2) + { + return name1.compare(name2) + } else { + return .orderedDescending + } + }) as? [String] ?? [] + if let countryCode = countryCode { + countryCodes = ["", countryCode] + otherCountryCodes + } else { + countryCodes = [""] + otherCountryCodes + } + let pickerView = UIPickerView() + pickerView.dataSource = self + pickerView.delegate = self + countryPickerView = pickerView + + self.lastInList = lastInList + self.type = type + self.textField.text = contents + + var ourCountryCode = self.delegate?.addressFieldTableViewCountryCode + + if ourCountryCode == nil { + ourCountryCode = countryCode + } + delegateCountryCodeDidChange(countryCode: ourCountryCode ?? "") + updateAppearance() + + self.textField.translatesAutoresizingMaskIntoConstraints = false + + NSLayoutConstraint.activate( + [ + textField.leadingAnchor.constraint( + equalTo: contentView.safeAreaLayoutGuide.leadingAnchor, + constant: 15 + ), + textField.trailingAnchor.constraint( + equalTo: contentView.safeAreaLayoutGuide.trailingAnchor, + constant: -15 + ), + textField.topAnchor.constraint( + equalTo: contentView.safeAreaLayoutGuide.topAnchor, + constant: 1 + ), + contentView.safeAreaLayoutGuide.bottomAnchor.constraint( + greaterThanOrEqualTo: textField.bottomAnchor + ), + textField.heightAnchor.constraint(greaterThanOrEqualToConstant: 43), + inputAccessoryToolbar?.heightAnchor.constraint(equalToConstant: 44), + ].compactMap { $0 } + ) + } + + var type: STPAddressFieldType = .name + var caption: String? { + get { + self.textField.placeholder + } + set { + self.textField.placeholder = newValue + } + } + + private(set) var textField: STPValidatedTextField + private var _contents: String? + var contents: String? { + get { + // iOS 11 QuickType completions from textContentType have a space at the end. + // This *keeps* that space in the `textField`, but removes leading/trailing spaces from + // the logical contents of this field, so they're ignored for validation and persisting + return _contents?.trimmingCharacters(in: CharacterSet.whitespaces) + } + set { + _contents = newValue + if self.type == .country { + self.updateTextFieldsAndCaptions() + } else { + self.textField.text = contents + } + if self.textField.isFirstResponder { + self.textField.validText = self.potentiallyValidContents + } else { + self.textField.validText = self.validContents + } + self.delegate?.addressFieldTableViewCellDidUpdateText(self) + } + } + + var theme: STPTheme = .defaultTheme { + didSet { + updateAppearance() + } + } + + var lastInList: Bool = false { + didSet { + updateTextFieldsAndCaptions() + } + } + + private var inputAccessoryToolbar: UIToolbar? + private var countryPickerView: UIPickerView? + private var countryCodes: [AnyHashable]? + private weak var delegate: STPAddressFieldTableViewCellDelegate? + private var ourCountryCode: String? + + func updateTextFieldsAndCaptions() { + textField.placeholder = placeholder(for: type) + + if !lastInList { + textField.returnKeyType = .next + } else { + textField.returnKeyType = .default + } + switch type { + case .name: + textField.keyboardType = .default + textField.textContentType = .name + case .line1: + textField.keyboardType = .numbersAndPunctuation + textField.textContentType = .streetAddressLine1 + case .line2: + textField.keyboardType = .numbersAndPunctuation + textField.textContentType = .streetAddressLine2 + case .city: + textField.keyboardType = .default + textField.textContentType = .addressCity + case .state: + textField.keyboardType = .default + textField.textContentType = .addressState + case .zip: + textField.keyboardType = .numbersAndPunctuation + textField.textContentType = .postalCode + case .country: + textField.keyboardType = .default + // Don't set textContentType for Country, because we don't want iOS to skip the UIPickerView for input + textField.inputView = countryPickerView + // If we're being set directly to a country we don't allow, add it to the allowed list + let countryCodes = self.countryCodes ?? [] + if let contents = contents, + !countryCodes.contains(contents) && NSLocale.isoCountryCodes.contains(contents) + { + self.countryCodes = countryCodes + [contents] + } + let index = countryCodes.firstIndex(of: contents ?? "") ?? NSNotFound + if index == NSNotFound { + textField.text = "" + } else { + countryPickerView?.selectRow(index, inComponent: 0, animated: false) + if let countryPickerView = countryPickerView { + textField.text = pickerView( + countryPickerView, + titleForRow: index, + forComponent: 0 + ) + } + } + textField.validText = validContents + case .phone: + self.textField.keyboardType = .numbersAndPunctuation + self.textField.textContentType = .telephoneNumber + let behavior: STPFormTextFieldAutoFormattingBehavior = + (self.countryCodeIsUnitedStates ? .phoneNumbers : .none) + (self.textField as? STPFormTextField)?.autoFormattingBehavior = behavior + case .email: + self.textField.keyboardType = .emailAddress + self.textField.textContentType = .emailAddress + } + + if !self.lastInList { + self.textField.inputAccessoryView = self.inputAccessoryToolbar + } else { + self.textField.inputAccessoryView = nil + } + self.textField.accessibilityLabel = self.textField.placeholder + self.textField.accessibilityIdentifier = self.accessibilityIdentifierForAddressField( + type: self.type + ) + } + + func accessibilityIdentifierForAddressField(type: STPAddressFieldType) -> String { + switch type { + case .name: + return "ShippingAddressFieldTypeNameIdentifier" + case .line1: + return "ShippingAddressFieldTypeLine1Identifier" + case .line2: + return "ShippingAddressFieldTypeLine2Identifier" + case .city: + return "ShippingAddressFieldTypeCityIdentifier" + case .state: + return "ShippingAddressFieldTypeStateIdentifier" + case .zip: + return "ShippingAddressFieldTypeZipIdentifier" + case .country: + return "ShippingAddressFieldTypeCountryIdentifier" + case .email: + return "ShippingAddressFieldTypeEmailIdentifier" + case .phone: + return "ShippingAddressFieldTypePhoneIdentifier" + } + } + + func stateFieldCaption(forCountryCode countryCode: String?) -> String { + return StripeSharedStrings.localizedStateString(for: countryCode) + } + + func placeholder(for addressFieldType: STPAddressFieldType) -> String { + switch addressFieldType { + case .name: + return String.Localized.name + case .line1: + return String.Localized.address + case .line2: + return STPLocalizedString( + "Apt.", + "Caption for Apartment/Address line 2 field on address form" + ) + case .city: + return String.Localized.city + case .state: + return stateFieldCaption(forCountryCode: self.ourCountryCode) + case .zip: + return StripeSharedStrings.localizedPostalCodeString(for: self.ourCountryCode) + case .country: + return String.Localized.country + case .email: + return String.Localized.email + case .phone: + return String.Localized.phone + } + } + + func delegateCountryCodeDidChange(countryCode: String) { + if self.type == .country { + self.contents = countryCode + } + + self.ourCountryCode = countryCode + self.updateTextFieldsAndCaptions() + self.setNeedsLayout() + } + + func updateAppearance() { + self.backgroundColor = self.theme.secondaryBackgroundColor + self.contentView.backgroundColor = .clear + self.textField.placeholderColor = theme.tertiaryForegroundColor + self.textField.defaultColor = theme.primaryForegroundColor + self.textField.errorColor = self.theme.errorColor + self.textField.font = self.theme.font + self.setNeedsLayout() + } + + var countryCodeIsUnitedStates: Bool { + self.ourCountryCode == "US" + } + + public override func becomeFirstResponder() -> Bool { + return self.textField.becomeFirstResponder() + } + + @objc func nextTapped(sender: NSObject) { + delegate?.addressFieldTableViewCellDidReturn(self) + } + + @objc + public func textFieldShouldReturn(_ textField: UITextField) -> Bool { + self.delegate?.addressFieldTableViewCellDidReturn(self) + return false + } + + @objc + public func textFieldDidEndEditing(_ textField: UITextField) { + (textField as? STPFormTextField)?.validText = validContents + self.delegate?.addressFieldTableViewCellDidEndEditing(self) + } + + public override func accessibilityElementCount() -> Int { + return 1 + } + + public override func accessibilityElement(at index: Int) -> Any? { + return textField + } + + public override func index(ofAccessibilityElement element: Any) -> Int { + return 0 + } + + @objc func textFieldTextDidChange(textField: STPValidatedTextField) { + if self.type != .country { + _contents = textField.text + if textField.isFirstResponder { + textField.validText = self.potentiallyValidContents + } else { + textField.validText = self.validContents + } + } + self.delegate?.addressFieldTableViewCellDidUpdateText(self) + } + + // pragma mark - UITextFieldDelegate + + var validContents: Bool { + switch self.type { + case .name, .line1, .city, .state, .country: + return self.contents?.count ?? 0 > 0 + case .line2: + return true + case .zip: + return STPPostalCodeValidator.validationState( + forPostalCode: self.contents, + countryCode: self.ourCountryCode + ) == .valid + case .email: + return STPEmailAddressValidator.stringIsValidEmailAddress(self.contents) + case .phone: + return STPPhoneNumberValidator.stringIsValidPhoneNumber( + self.contents ?? "", + forCountryCode: self.ourCountryCode + ) + } + } + + var potentiallyValidContents: Bool { + switch self.type { + case .name, .line1, .city, .state, .country, .line2, .phone: + return true + case .zip: + let validationState = STPPostalCodeValidator.validationState( + forPostalCode: self.contents, + countryCode: self.ourCountryCode + ) + return validationState == .valid || validationState == .incomplete + case .email: + return STPEmailAddressValidator.stringIsValidPartialEmailAddress(self.contents) + } + } + + public func pickerView( + _ pickerView: UIPickerView, + didSelectRow row: Int, + inComponent component: Int + ) { + guard let countryCode = self.countryCodes?[row] as? String + else { + return + } + self.ourCountryCode = countryCode + self.contents = self.ourCountryCode + textField.text = self.pickerView(pickerView, titleForRow: row, forComponent: component) + // UIControlEvent not fired for programmatic changes + self.textFieldTextDidChange(textField: textField) + self.delegate?.addressFieldTableViewCountryCode = self.ourCountryCode ?? "" + + } + + public func pickerView( + _ pickerView: UIPickerView, + titleForRow row: Int, + forComponent component: Int + ) -> String? { + guard let countryCode = self.countryCodes?[row] as? String else { + return nil + } + let identifier = Locale.identifier(fromComponents: [ + NSLocale.Key.countryCode.rawValue: countryCode + ]) + return (NSLocale.autoupdatingCurrent as NSLocale).displayName( + forKey: NSLocale.Key(rawValue: NSLocale.Key.identifier.rawValue), + value: identifier + ) + } + + public func numberOfComponents(in pickerView: UIPickerView) -> Int { + return 1 + } + + public func pickerView( + _ pickerView: UIPickerView, + numberOfRowsInComponent component: Int + ) + -> Int + { + self.countryCodes?.count ?? 0 + } + + required convenience init?( + coder aDecoder: NSCoder + ) { + assertionFailure("Use initWithType: instead.") + self.init( + type: .name, + contents: nil, + lastInList: false, + delegate: nil + ) + } + +} diff --git a/Stripe/StripeiOS/Source/STPAddressViewModel.swift b/Stripe/StripeiOS/Source/STPAddressViewModel.swift new file mode 100644 index 00000000..aa3fa68b --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAddressViewModel.swift @@ -0,0 +1,434 @@ +// +// STPAddressViewModel.swift +// StripeiOS +// +// Created by Jack Flintermann on 4/21/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Contacts +import CoreLocation +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +import UIKit + +protocol STPAddressViewModelDelegate: AnyObject { + func addressViewModelDidChange(_ addressViewModel: STPAddressViewModel) + func addressViewModel(_ addressViewModel: STPAddressViewModel, addedCellAt index: Int) + func addressViewModel(_ addressViewModel: STPAddressViewModel, removedCellAt index: Int) + func addressViewModelWillUpdate(_ addressViewModel: STPAddressViewModel) + func addressViewModelDidUpdate(_ addressViewModel: STPAddressViewModel) +} + +class STPAddressViewModel: STPAddressFieldTableViewCellDelegate { + private(set) var addressCells: [STPAddressFieldTableViewCell] = [] + weak var delegate: STPAddressViewModelDelegate? + + var addressFieldTableViewCountryCode: String? = Locale.autoupdatingCurrent.regionCode { + didSet { + updatePostalCodeCellIfNecessary() + if let addressFieldTableViewCountryCode = addressFieldTableViewCountryCode { + for cell in addressCells { + cell.delegateCountryCodeDidChange(countryCode: addressFieldTableViewCountryCode) + } + } + } + } + + var address: STPAddress { + get { + let address = STPAddress() + for cell in addressCells { + + switch cell.type { + case .name: + address.name = cell.contents + case .line1: + address.line1 = cell.contents + case .line2: + address.line2 = cell.contents + case .city: + address.city = cell.contents + case .state: + address.state = cell.contents + case .zip: + address.postalCode = cell.contents + case .country: + address.country = cell.contents + case .email: + address.email = cell.contents + case .phone: + address.phone = cell.contents + } + } + // Prefer to use the contents of STPAddressFieldTypeCountry, but fallback to + // `addressFieldTableViewCountryCode` if nil (important for STPBillingAddressFieldsPostalCode) + address.country = address.country ?? addressFieldTableViewCountryCode + return address + } + set(address) { + if let country = address.country { + addressFieldTableViewCountryCode = country + } + + for cell in addressCells { + switch cell.type { + case .name: + cell.contents = address.name + case .line1: + cell.contents = address.line1 + case .line2: + cell.contents = address.line2 + case .city: + cell.contents = address.city + case .state: + cell.contents = address.state + case .zip: + cell.contents = address.postalCode + case .country: + cell.contents = address.country + case .email: + cell.contents = address.email + case .phone: + cell.contents = address.phone + } + } + } + } + + // The default value of availableCountries is nil, which will allow all known countries. + var availableCountries: Set? + + var isValid: Bool { + if isBillingAddress { + // The AddressViewModel is only for address fields. + // Determining whether the postal code is present is up to the + // STPCardTextFieldViewModel. + if requiredBillingAddressFields == .postalCode { + return true + } else { + return address.containsRequiredFields(requiredBillingAddressFields) + } + + } else { + if let requiredShippingAddressFields = requiredShippingAddressFields { + return address.containsRequiredShippingAddressFields(requiredShippingAddressFields) + } + return false + } + } + + // The default value of availableCountries is nil, which will allow all known countries. + init( + requiredBillingFields requiredBillingAddressFields: STPBillingAddressFields, + availableCountries: Set? = nil + ) { + isBillingAddress = true + self.availableCountries = availableCountries + self.requiredBillingAddressFields = requiredBillingAddressFields + switch requiredBillingAddressFields { + case .none: + addressCells = [] + case .zip, .postalCode: + addressCells = [] // Postal code cell will be added later if necessary + case .full: + addressCells = [ + STPAddressFieldTableViewCell( + type: .name, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .line1, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .line2, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .country, + contents: addressFieldTableViewCountryCode, + lastInList: false, + delegate: self + ), + // Postal code cell will be added here later if necessary + STPAddressFieldTableViewCell( + type: .city, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .state, + contents: "", + lastInList: true, + delegate: self + ), + ] + case .name: + addressCells = [ + STPAddressFieldTableViewCell( + type: .name, + contents: "", + lastInList: true, + delegate: self + ), + ] + default: + fatalError() + } + commonInit() + } + + init( + requiredShippingFields requiredShippingAddressFields: Set, + availableCountries: Set? = nil + ) { + isBillingAddress = false + self.availableCountries = availableCountries + self.requiredShippingAddressFields = requiredShippingAddressFields + var cells: [STPAddressFieldTableViewCell] = [] + + if requiredShippingAddressFields.contains(STPContactField.name) { + cells.append( + STPAddressFieldTableViewCell( + type: .name, + contents: "", + lastInList: false, + delegate: self + ) + ) + } + if requiredShippingAddressFields.contains(.emailAddress) { + cells.append( + STPAddressFieldTableViewCell( + type: .email, + contents: "", + lastInList: false, + delegate: self + ) + ) + } + if requiredShippingAddressFields.contains(STPContactField.postalAddress) { + var postalCells = [ + STPAddressFieldTableViewCell( + type: .name, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .line1, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .line2, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .country, + contents: addressFieldTableViewCountryCode, + lastInList: false, + delegate: self + ), + // Postal code cell will be added here later if necessary + STPAddressFieldTableViewCell( + type: .city, + contents: "", + lastInList: false, + delegate: self + ), + STPAddressFieldTableViewCell( + type: .state, + contents: "", + lastInList: false, + delegate: self + ), + ] + if requiredShippingAddressFields.contains(.name) { + postalCells.remove(at: 0) + } + cells.append(contentsOf: postalCells.compactMap { $0 }) + } + if requiredShippingAddressFields.contains(.phoneNumber) { + cells.append( + STPAddressFieldTableViewCell( + type: .phone, + contents: "", + lastInList: false, + delegate: self + ) + ) + } + if let lastCell = cells.last { + lastCell.lastInList = true + } + addressCells = cells + commonInit() + } + + private func cell(at index: Int) -> STPAddressFieldTableViewCell? { + guard index > 0, + index < addressCells.count + else { + return nil + } + return addressCells[index] + } + + private var isBillingAddress = false + private var requiredBillingAddressFields: STPBillingAddressFields = .none + private var requiredShippingAddressFields: Set? + private var showingPostalCodeCell = false + private var geocodeInProgress = false + + private func commonInit() { + if let countryCode = Locale.autoupdatingCurrent.regionCode { + addressFieldTableViewCountryCode = countryCode + } + updatePostalCodeCellIfNecessary() + } + + private func updatePostalCodeCellIfNecessary() { + delegate?.addressViewModelWillUpdate(self) + let shouldBeShowingPostalCode = STPPostalCodeValidator.postalCodeIsRequired( + forCountryCode: addressFieldTableViewCountryCode + ) + + if shouldBeShowingPostalCode && !showingPostalCodeCell { + if containsStateAndPostalFields() { + // Add before city + let zipFieldIndex = addressCells.firstIndex(where: { $0.type == .city }) ?? 0 + + var mutableAddressCells = addressCells + mutableAddressCells.insert( + STPAddressFieldTableViewCell( + type: .zip, + contents: "", + lastInList: false, + delegate: self + ), + at: zipFieldIndex + ) + addressCells = mutableAddressCells + delegate?.addressViewModel(self, addedCellAt: zipFieldIndex) + delegate?.addressViewModelDidChange(self) + } + } else if !shouldBeShowingPostalCode && showingPostalCodeCell { + if containsStateAndPostalFields() { + if let zipFieldIndex = addressCells.firstIndex(where: { $0.type == .zip }) { + + var mutableAddressCells = addressCells + mutableAddressCells.remove(at: zipFieldIndex) + addressCells = mutableAddressCells + delegate?.addressViewModel(self, removedCellAt: zipFieldIndex) + delegate?.addressViewModelDidChange(self) + } + } + } + showingPostalCodeCell = shouldBeShowingPostalCode + delegate?.addressViewModelDidUpdate(self) + } + + private func containsStateAndPostalFields() -> Bool { + if isBillingAddress { + return requiredBillingAddressFields == .full + } else { + return requiredShippingAddressFields?.contains(.postalAddress) ?? false + } + } + + func updateCityAndState(fromZipCodeCell zipCell: STPAddressFieldTableViewCell?) { + + let zipCode = zipCell?.contents + + if geocodeInProgress || zipCode == nil || !(zipCell?.textField.validText ?? false) + || !(addressFieldTableViewCountryCode == "US") + { + return + } + + var cityCell: STPAddressFieldTableViewCell? + var stateCell: STPAddressFieldTableViewCell? + for cell in addressCells { + if cell.type == .city { + cityCell = cell + } else if cell.type == .state { + stateCell = cell + } + } + + if (cityCell == nil && stateCell == nil) + || ((cityCell?.contents?.count ?? 0) > 0 || (stateCell?.contents?.count ?? 0) > 0) + { + // Don't auto fill if either have text already + // Or if neither are non-nil + return + } else { + geocodeInProgress = true + let geocoder = CLGeocoder() + + let onCompletion: CLGeocodeCompletionHandler = { placemarks, error in + stpDispatchToMainThreadIfNecessary({ + if (placemarks?.count ?? 0) > 0 && error == nil { + let placemark = placemarks?.first + if (cityCell?.contents?.count ?? 0) == 0 + && (stateCell?.contents?.count ?? 0) == 0 + && (zipCell?.contents == zipCode) + { + // Check contents again to make sure they're still empty + // And that zipcode hasn't changed to something else + cityCell?.contents = placemark?.locality + stateCell?.contents = placemark?.administrativeArea + } + } + self.geocodeInProgress = false + }) + } + + let address = CNMutablePostalAddress() + address.postalCode = zipCode ?? "" + address.isoCountryCode = addressFieldTableViewCountryCode ?? "" + + geocoder.geocodePostalAddress( + address, + completionHandler: onCompletion + ) + } + } + + private func cell(after cell: STPAddressFieldTableViewCell?) -> STPAddressFieldTableViewCell? { + guard let cell = cell, + let cellIndex = addressCells.firstIndex(of: cell), + cellIndex + 1 < addressCells.count + else { + return nil + } + return addressCells[cellIndex + 1] + } + + func addressFieldTableViewCellDidUpdateText(_ cell: STPAddressFieldTableViewCell) { + delegate?.addressViewModelDidChange(self) + } + + func addressFieldTableViewCellDidReturn(_ cell: STPAddressFieldTableViewCell) { + _ = self.cell(after: cell)?.becomeFirstResponder() + } + + func addressFieldTableViewCellDidEndEditing(_ cell: STPAddressFieldTableViewCell) { + if cell.type == .zip { + updateCityAndState(fromZipCodeCell: cell) + } + } + +} diff --git a/Stripe/StripeiOS/Source/STPAnalyticsClient+BasicUI.swift b/Stripe/StripeiOS/Source/STPAnalyticsClient+BasicUI.swift new file mode 100644 index 00000000..a0f9aaa1 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAnalyticsClient+BasicUI.swift @@ -0,0 +1,84 @@ +// +// STPAnalyticsClient+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments + +@objc(STPBasicUIAnalyticsSerializer) +class STPBasicUIAnalyticsSerializer: NSObject, STPAnalyticsSerializer { + static func serializeConfiguration( + _ configuration: NSObject + ) -> [String: + String] + { + var dictionary: [String: String] = [:] + dictionary["publishable_key"] = STPAPIClient.shared.publishableKey ?? "unknown" + + guard let configuration = configuration as? STPPaymentConfiguration else { + return dictionary + } + + if configuration.applePayEnabled && !configuration.fpxEnabled { + dictionary["additional_payment_methods"] = "default" + } else if !configuration.applePayEnabled && !configuration.fpxEnabled { + dictionary["additional_payment_methods"] = "none" + } else if !configuration.applePayEnabled && configuration.fpxEnabled { + dictionary["additional_payment_methods"] = "fpx" + } else if configuration.applePayEnabled && configuration.fpxEnabled { + dictionary["additional_payment_methods"] = "applepay,fpx" + } + + switch configuration.requiredBillingAddressFields { + case .none: + dictionary["required_billing_address_fields"] = "none" + case .postalCode: + dictionary["required_billing_address_fields"] = "zip" + case .full: + dictionary["required_billing_address_fields"] = "full" + case .name: + dictionary["required_billing_address_fields"] = "name" + default: + fatalError() + } + + var shippingFields: [String] = [] + if let shippingAddressFields = configuration.requiredShippingAddressFields { + if shippingAddressFields.contains(.name) { + shippingFields.append("name") + } + if shippingAddressFields.contains(.emailAddress) { + shippingFields.append("email") + } + if shippingAddressFields.contains(.postalAddress) { + shippingFields.append("address") + } + if shippingAddressFields.contains(.phoneNumber) { + shippingFields.append("phone") + } + } + + if shippingFields.isEmpty { + shippingFields.append("none") + } + dictionary["required_shipping_address_fields"] = shippingFields.joined(separator: "_") + + switch configuration.shippingType { + case .shipping: + dictionary["shipping_type"] = "shipping" + case .delivery: + dictionary["shipping_type"] = "delivery" + @unknown default: + break + } + + dictionary["company_name"] = configuration.companyName + dictionary["apple_merchant_identifier"] = configuration.appleMerchantIdentifier ?? "unknown" + return dictionary + } +} diff --git a/Stripe/StripeiOS/Source/STPAnalyticsClient+Payments.swift b/Stripe/StripeiOS/Source/STPAnalyticsClient+Payments.swift new file mode 100644 index 00000000..ce381e66 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPAnalyticsClient+Payments.swift @@ -0,0 +1,28 @@ +// +// STPAnalyticsClient+Payments.swift +// StripeiOS +// +// Created by David Estes on 1/24/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +/// An analytic specific to payments that serializes payment-specific +/// information into its params. +@_spi(STP) public protocol PaymentAnalytic: Analytic { + var productUsage: Set { get } + var additionalParams: [String: Any] { get } +} + +@_spi(STP) extension PaymentAnalytic { + public var params: [String: Any] { + var params = additionalParams + + params["apple_pay_enabled"] = NSNumber(value: StripeAPI.deviceSupportsApplePay()) + params["ocr_type"] = PaymentsSDKVariant.ocrTypeString + params["pay_var"] = PaymentsSDKVariant.variant + return params + } +} diff --git a/Stripe/StripeiOS/Source/STPApplePayContextDelegate.swift b/Stripe/StripeiOS/Source/STPApplePayContextDelegate.swift new file mode 100644 index 00000000..0814a5de --- /dev/null +++ b/Stripe/StripeiOS/Source/STPApplePayContextDelegate.swift @@ -0,0 +1,98 @@ +// +// STPApplePayContextDelegate.swift +// StripeiOS +// +// Created by David Estes on 9/15/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripeApplePay +@_spi(STP) import StripeCore + +/// Implement the required methods of this delegate to supply a PaymentIntent to STPApplePayContext and be notified of the completion of the Apple Pay payment. +/// You may also implement the optional delegate methods to handle shipping methods and shipping address changes e.g. to verify you can ship to the address, or update the payment amount. +@objc public protocol STPApplePayContextDelegate: _stpinternal_STPApplePayContextDelegateBase { + /// Called after the customer has authorized Apple Pay. Implement this method to call the completion block with the client secret of a PaymentIntent or SetupIntent. + /// - Parameters: + /// - paymentMethod: The PaymentMethod that represents the customer's Apple Pay payment method. + /// If you create the PaymentIntent with confirmation_method=manual, pass `paymentMethod.stripeId` as the payment_method and confirm=true. Otherwise, you can ignore this parameter. + /// - paymentInformation: The underlying PKPayment created by Apple Pay. + /// If you create the PaymentIntent with confirmation_method=manual, you can collect shipping information using its `shippingContact` and `shippingMethod` properties. + /// - completion: Call this with the PaymentIntent or SetupIntent client secret, or the error that occurred creating the PaymentIntent or SetupIntent. + @objc(applePayContext:didCreatePaymentMethod:paymentInformation:completion:) + func applePayContext( + _ context: STPApplePayContext, + didCreatePaymentMethod paymentMethod: STPPaymentMethod, + paymentInformation: PKPayment, + completion: @escaping STPIntentClientSecretCompletionBlock + ) + + /// Called after the Apple Pay sheet is dismissed with the result of the payment. + /// Your implementation could stop a spinner and display a receipt view or error to the customer, for example. + /// - Parameters: + /// - status: The status of the payment + /// - error: The error that occurred, if any. + @objc(applePayContext:didCompleteWithStatus:error:) + func applePayContext( + _ context: STPApplePayContext, + didCompleteWith status: STPPaymentStatus, + error: Error? + ) +} + +/// A helper class used to bridge StripeApplePay.framework with the legacy Stripe.framework objects. +@objc(STPApplePayContextLegacyHelper) +class STPApplePayContextLegacyHelper: NSObject { + @objc class func performDidCreatePaymentMethod( + _ storage: _stpinternal_ApplePayContextDidCreatePaymentMethodStorage + ) { + let delegate = storage.delegate as! STPApplePayContextDelegate + // Convert the PaymentMethod to an STPPaymentMethod: + guard + let stpPaymentMethod = STPPaymentMethod.decodedObject( + fromAPIResponse: storage.paymentMethod.allResponseFields + ) + else { + assertionFailure("Failed to convert PaymentMethod to STPPaymentMethod") + return + } + delegate.applePayContext( + storage.context, + didCreatePaymentMethod: stpPaymentMethod, + paymentInformation: storage.paymentInformation, + completion: storage.completion + ) + } + + @objc class func performDidComplete(_ storage: _stpinternal_ApplePayContextDidCompleteStorage) { + let delegate = storage.delegate as! STPApplePayContextDelegate + let stpStatus = STPPaymentStatus(applePayStatus: storage.status) + + // If this is a modern API error, convert it down to a legacy STPError. + // This is to avoid changing the API experience for users. + // We can re-evaluate this as we release more of the modern API. + if let modernError = storage.error as? StripeError { + storage.error = NSError.stp_error(from: modernError) + } + + delegate.applePayContext(storage.context, didCompleteWith: stpStatus, error: storage.error) + } + +} + +extension STPPaymentStatus { + init( + applePayStatus: STPApplePayContext.PaymentStatus + ) { + switch applePayStatus { + case .success: + self = .success + case .error: + self = .error + case .userCancellation: + self = .userCancellation + } + } +} diff --git a/Stripe/StripeiOS/Source/STPApplePayPaymentOption.swift b/Stripe/StripeiOS/Source/STPApplePayPaymentOption.swift new file mode 100644 index 00000000..14edba1f --- /dev/null +++ b/Stripe/StripeiOS/Source/STPApplePayPaymentOption.swift @@ -0,0 +1,51 @@ +// +// STPApplePayPaymentOption.swift +// StripeiOS +// +// Created by Ben Guo on 4/19/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +import UIKit + +/// An empty class representing that the user wishes to pay via Apple Pay. This can +/// be checked on an `STPPaymentContext`, e.g: +/// ``` +/// if paymentContext.selectedPaymentOption is STPApplePayPaymentOption { +/// // Don't ask the user for their card number; they want to pay with apple pay. +/// } +/// ``` +@objc public class STPApplePayPaymentOption: NSObject, STPPaymentOption { + // MARK: - STPPaymentOption + @objc public var image: UIImage { + return STPImageLibrary.applePayCardImage() + } + + @objc public var templateImage: UIImage { + // No template for Apple Pay + return STPImageLibrary.applePayCardImage() + } + + @objc public var label: String { + return String.Localized.apple_pay + } + + @objc public var isReusable: Bool { + return true + } + + // MARK: - Equality + /// :nodoc: + @objc + public override func isEqual(_ object: Any?) -> Bool { + return object is STPApplePayPaymentOption + } + + /// :nodoc: + @objc public override var hash: Int { + return NSStringFromClass(STPApplePayPaymentOption.self).hash + } +} diff --git a/Stripe/StripeiOS/Source/STPBackendAPIAdapter.swift b/Stripe/StripeiOS/Source/STPBackendAPIAdapter.swift new file mode 100644 index 00000000..2cd6ed58 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPBackendAPIAdapter.swift @@ -0,0 +1,86 @@ +// +// STPBackendAPIAdapter.swift +// StripeiOS +// +// Created by Jack Flintermann on 1/12/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation +import UIKit + +/// A "bridge" from our pre-built UI (`STPPaymentContext`, `STPPaymentOptionsViewController`) +/// to your backend to fetch Customer-related information needed to power those views. +/// Typically, you will not need to implement this protocol yourself. You +/// should instead use `STPCustomerContext`, which implements +/// and manages retrieving and updating a Stripe customer for you. +/// - seealso: STPCustomerContext.h +/// If you would prefer retrieving and updating your Stripe customer object via +/// your own backend instead of using `STPCustomerContext`, you should make your +/// application's API client conform to this interface. +@objc public protocol STPBackendAPIAdapter: NSObjectProtocol { + /// Retrieve the customer to be displayed inside a payment context. + /// If you are not using STPCustomerContext: + /// On your backend, retrieve the Stripe customer associated with your currently + /// logged-in user ( https://stripe.com/docs/api#retrieve_customer ), and return + /// the raw JSON response from the Stripe API. Back in your iOS app, after you've + /// called this API, deserialize your API response into an `STPCustomer` object + /// (you can use the `STPCustomerDeserializer` class to do this). + /// - seealso: STPCard + /// - Parameter completion: call this callback when you're done fetching and parsing the above information from your backend. For example, `completion(customer, nil)` (if your call succeeds) or `completion(nil, error)` if an error is returned. + func retrieveCustomer(_ completion: STPCustomerCompletionBlock?) + /// Retrieves a list of Payment Methods attached to a customer. + /// If you are implementing your own : + /// Call the list method ( https://stripe.com/docs/api/payment_methods/list ) + /// with the Stripe customer. If this API call succeeds, call `completion(paymentMethods)` + /// with the list of PaymentMethods. Otherwise, call `completion(error)` with the error + /// that occurred. + /// - Parameter completion: Call this callback with the list of Payment Methods attached to the + /// customer. For example, `completion(paymentMethods)` (if your call succeeds) or + /// `completion(error)` if an error is returned. + func listPaymentMethodsForCustomer(completion: STPPaymentMethodsCompletionBlock?) + /// Adds a Payment Method to a customer. + /// If you are implementing your own : + /// On your backend, retrieve the Stripe customer associated with your logged-in user. + /// Then, call the Attach method on the Payment Method with that customer's ID + /// ( https://stripe.com/docs/api/payment_methods/attach ). If this API call succeeds, + /// call `completion(nil)`. Otherwise, call `completion(error)` with the error that + /// occurred. + /// - Parameters: + /// - paymentMethod: A valid Payment Method + /// - completion: Call this callback when you're done adding the payment method + /// to the customer on your backend. For example, `completion(nil)` (if your call succeeds) + /// or `completion(error)` if an error is returned. + func attachPaymentMethod(toCustomer paymentMethod: STPPaymentMethod, completion: STPErrorBlock?) + + /// Deletes the given Payment Method from the customer. + /// If you are implementing your own : + /// Call the Detach method ( https://stripe.com/docs/api/payment_methods/detach ) + /// on the Payment Method. If this API call succeeds, call `completion(nil)`. + /// Otherwise, call `completion(error)` with the error that occurred. + /// - Parameters: + /// - paymentMethod: The Payment Method to delete from the customer + /// - completion: Call this callback when you're done deleting the Payment Method + /// from the customer on your backend. For example, `completion(nil)` (if your call + /// succeeds) or `completion(error)` if an error is returned. + @objc optional func detachPaymentMethod( + fromCustomer paymentMethod: STPPaymentMethod, + completion: STPErrorBlock? + ) + /// Sets the given shipping address on the customer. + /// If you are implementing your own : + /// On your backend, retrieve the Stripe customer associated with your logged-in user. + /// Then, call the Customer Update method ( https://stripe.com/docs/api#update_customer ) + /// specifying shipping to be the given shipping address. If this API call succeeds, + /// call `completion(nil)`. Otherwise, call `completion(error)` with the error that occurred. + /// - Parameters: + /// - shipping: The shipping address to set on the customer + /// - completion: call this callback when you're done updating the customer on + /// your backend. For example, `completion(nil)` (if your call succeeds) or + /// `completion(error)` if an error is returned. + /// - seealso: https://stripe.com/docs/api#update_customer + @objc optional func updateCustomer( + withShippingAddress shipping: STPAddress, + completion: STPErrorBlock? + ) +} diff --git a/Stripe/StripeiOS/Source/STPBankSelectionTableViewCell.swift b/Stripe/StripeiOS/Source/STPBankSelectionTableViewCell.swift new file mode 100644 index 00000000..052f0976 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPBankSelectionTableViewCell.swift @@ -0,0 +1,131 @@ +// +// STPBankSelectionTableViewCell.swift +// StripeiOS +// +// Created by David Estes on 8/9/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +import UIKit + +class STPBankSelectionTableViewCell: UITableViewCell { + func configure( + withBank bankBrand: STPFPXBankBrand, + theme: STPTheme, + selected: Bool, + offline: Bool, + enabled: Bool + ) { + bank = bankBrand + self.theme = theme + + backgroundColor = theme.secondaryBackgroundColor + + // Left icon + leftIcon?.image = STPLegacyImageLibrary.fpxBrandImage(for: bank) + leftIcon?.tintColor = primaryColorForPaymentOption(withSelected: selected, enabled: enabled) + + // Title label + titleLabel?.font = theme.font + titleLabel?.text = STPFPXBank.stringFrom(bank) + if offline { + let format = STPLocalizedString( + "%@ - Offline", + "Bank name when bank is offline for maintenance." + ) + titleLabel?.text = String(format: format, STPFPXBank.stringFrom(bank) ?? "") + } + titleLabel?.textColor = primaryColorForPaymentOption( + withSelected: isSelected, + enabled: enabled + ) + + // Loading indicator + activityIndicator?.tintColor = theme.accentColor + if selected { + activityIndicator?.startAnimating() + } else { + activityIndicator?.stopAnimating() + } + + setNeedsLayout() + } + + private var bank: STPFPXBankBrand! + private var theme: STPTheme = .defaultTheme + private var leftIcon: UIImageView? + private var titleLabel: UILabel? + private var activityIndicator: UIActivityIndicatorView? + + override init( + style: UITableViewCell.CellStyle, + reuseIdentifier: String? + ) { + super.init(style: style, reuseIdentifier: reuseIdentifier) + // Left icon + let leftIcon = UIImageView() + self.leftIcon = leftIcon + contentView.addSubview(leftIcon) + + // Title label + let titleLabel = UILabel() + self.titleLabel = titleLabel + contentView.addSubview(titleLabel) + + // Loading indicator + var activityIndicator: UIActivityIndicatorView? + if #available(iOS 13.0, *) { + activityIndicator = UIActivityIndicatorView(style: .medium) + } else { + activityIndicator = UIActivityIndicatorView(style: .gray) + } + self.activityIndicator = activityIndicator + self.activityIndicator?.hidesWhenStopped = true + if let activityIndicator = activityIndicator { + contentView.addSubview(activityIndicator) + } + } + + override func layoutSubviews() { + super.layoutSubviews() + + let midY = bounds.midY + let padding: CGFloat = 15.0 + let iconWidth: CGFloat = 26.0 + + // Left icon + leftIcon?.sizeToFit() + leftIcon?.center = CGPoint(x: padding + (iconWidth / 2.0), y: midY) + + // Activity indicator + activityIndicator?.center = CGPoint( + x: bounds.width - padding - (activityIndicator?.bounds.midX ?? 0.0), + y: midY + ) + + // Title label + var labelFrame = bounds + // not every icon is `iconWidth` wide, but give them all the same amount of space: + labelFrame.origin.x = padding + iconWidth + padding + labelFrame.size.width = + (activityIndicator?.frame.minX ?? 0.0) - padding - labelFrame.origin.x + titleLabel?.frame = labelFrame + } + + func primaryColorForPaymentOption(withSelected selected: Bool, enabled: Bool) -> UIColor { + if selected { + return theme.accentColor + } else { + return + (enabled + ? theme.primaryForegroundColor + : theme.primaryForegroundColor.withAlphaComponent(0.6)) + } + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/STPBankSelectionViewController.swift b/Stripe/StripeiOS/Source/STPBankSelectionViewController.swift new file mode 100644 index 00000000..b6fb8cdd --- /dev/null +++ b/Stripe/StripeiOS/Source/STPBankSelectionViewController.swift @@ -0,0 +1,300 @@ +// +// STPBankSelectionViewController.swift +// StripeiOS +// +// Created by David Estes on 8/9/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +import PassKit +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments +@_spi(STP) import StripePaymentsUI +import UIKit + +/// The payment methods supported by STPBankSelectionViewController. +@objc public enum STPBankSelectionMethod: Int { + /// FPX (Malaysia) + case FPX + /// An unknown payment method + case unknown +} + +/// This view controller displays a list of banks of the specified type, allowing the user to select one to pay from. +/// Once a bank is selected, it will return a PaymentMethodParams object, which you can use to confirm a PaymentIntent +/// or inspect to obtain details about the selected bank. +public class STPBankSelectionViewController: STPCoreTableViewController, UITableViewDataSource, + UITableViewDelegate +{ + /// A convenience initializer; equivalent to calling `init( bankMethod:bankMethod configuration:STPPaymentConfiguration.shared theme:STPTheme.defaultTheme`. + @objc + public convenience init( + bankMethod: STPBankSelectionMethod + ) { + self.init( + bankMethod: bankMethod, + configuration: STPPaymentConfiguration.shared, + theme: STPTheme.defaultTheme + ) + } + + @objc public convenience required init( + theme: STPTheme? + ) { + self.init( + bankMethod: .FPX, + configuration: STPPaymentConfiguration.shared, + theme: theme ?? .defaultTheme + ) + } + + /// Initializes a new `STPBankSelectionViewController` with the provided configuration and theme. Don't forget to set the `delegate` property after initialization. + /// - Parameters: + /// - bankMethod: The user will be presented with a list of banks for this payment method. STPBankSelectionMethodFPX is currently the only supported payment method. + /// - configuration: The configuration to use. This determines the Stripe publishable key to use when querying metadata about the banks. - seealso: STPPaymentConfiguration + /// - theme: The theme to use to inform the view controller's visual appearance. - seealso: STPTheme + @objc + public init( + bankMethod: STPBankSelectionMethod, + configuration: STPPaymentConfiguration, + theme: STPTheme + ) { + super.init(theme: theme) + STPAnalyticsClient.sharedClient.addClass( + toProductUsageIfNecessary: STPBankSelectionViewController.self + ) + assert(bankMethod == .FPX, "STPBankSelectionViewController currently only supports FPX.") + self.bankMethod = bankMethod + self.configuration = configuration + selectedBank = .unknown + apiClient = STPAPIClient.shared + if bankMethod == .FPX { + _refreshFPXStatus() + NotificationCenter.default.addObserver( + self, + selector: #selector(_refreshFPXStatus), + name: UIApplication.didBecomeActiveNotification, + object: nil + ) + } + title = String.Localized.bank_account + } + + /// The view controller's delegate. This must be set before showing the view controller in order for it to work properly. - seealso: STPBankSelectionViewControllerDelegate + @objc public weak var delegate: STPBankSelectionViewControllerDelegate? + + /// The API Client to use to make requests. + /// Defaults to `STPAPIClient.shared` + public var apiClient: STPAPIClient = .shared + + private var bankMethod: STPBankSelectionMethod = .unknown + private var selectedBank: STPFPXBankBrand = .unknown + private var configuration: STPPaymentConfiguration? + private weak var imageView: UIImageView? + private var headerView: STPSectionHeaderView? + private var loading = false + private var bankStatus: STPFPXBankStatusResponse? + + deinit { + NotificationCenter.default.removeObserver(self) + } + + @objc func _refreshFPXStatus() { + apiClient.retrieveFPXBankStatus(withCompletion: { bankStatusResponse, error in + if error == nil && bankStatusResponse != nil { + if let bankStatusResponse = bankStatusResponse { + self._update(withBankStatus: bankStatusResponse) + } + } + }) + } + + @objc override func createAndSetupViews() { + super.createAndSetupViews() + + tableView?.register( + STPBankSelectionTableViewCell.self, + forCellReuseIdentifier: STPBankSelectionCellReuseIdentifier + ) + + tableView?.dataSource = self + tableView?.delegate = self + tableView?.reloadData() + } + + @objc override func updateAppearance() { + super.updateAppearance() + + tableView?.reloadData() + } + + @objc override func useSystemBackButton() -> Bool { + return true + } + + func _update(withBankStatus bankStatusResponse: STPFPXBankStatusResponse) { + bankStatus = bankStatusResponse + + tableView?.beginUpdates() + if let indexPathsForVisibleRows = tableView?.indexPathsForVisibleRows { + tableView?.reloadRows(at: indexPathsForVisibleRows, with: .none) + } + tableView?.endUpdates() + } + + // MARK: - UITableView + + /// :nodoc: + @objc + public func numberOfSections(in tableView: UITableView) -> Int { + return 1 + } + + /// :nodoc: + @objc + public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { + return STPFPXBankBrand.unknown.rawValue + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + cellForRowAt indexPath: IndexPath + ) -> UITableViewCell { + let cell = + tableView.dequeueReusableCell( + withIdentifier: STPBankSelectionCellReuseIdentifier, + for: indexPath + ) + as? STPBankSelectionTableViewCell + let bankBrand = STPFPXBankBrand(rawValue: indexPath.row) + let selected = selectedBank == bankBrand + var offline: Bool? + if let bankBrand = bankBrand { + offline = bankStatus != nil && !(bankStatus?.bankBrandIsOnline(bankBrand) ?? false) + } + if let bankBrand = bankBrand { + cell?.configure( + withBank: bankBrand, + theme: theme, + selected: selected, + offline: offline ?? false, + enabled: !loading + ) + } + return cell! + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + willDisplay cell: UITableViewCell, + forRowAt indexPath: IndexPath + ) { + let topRow = indexPath.row == 0 + let bottomRow = + self.tableView(tableView, numberOfRowsInSection: indexPath.section) - 1 == indexPath.row + cell.stp_setBorderColor(theme.tertiaryBackgroundColor) + cell.stp_setTopBorderHidden(!topRow) + cell.stp_setBottomBorderHidden(!bottomRow) + cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) + cell.stp_setFakeSeparatorLeftInset(15.0) + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + heightForFooterInSection section: Int + ) + -> CGFloat + { + return 27.0 + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + shouldHighlightRowAt indexPath: IndexPath + ) + -> Bool + { + return !loading + } + + /// :nodoc: + @objc + public func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { + if loading { + return // Don't allow user interaction if we're currently setting up a payment method + } + loading = true + tableView.deselectRow(at: indexPath, animated: true) + let bankIndex = indexPath.row + selectedBank = STPFPXBankBrand(rawValue: bankIndex) ?? .unknown + tableView.reloadSections( + NSIndexSet(index: indexPath.section) as IndexSet, + with: .none + ) + + let fpx = STPPaymentMethodFPXParams() + fpx.bank = STPFPXBankBrand(rawValue: bankIndex) ?? .unknown + // Create and return a Payment Method Params object + let paymentMethodParams = STPPaymentMethodParams( + fpx: fpx, + billingDetails: nil, + metadata: nil + ) + if delegate?.responds( + to: #selector( + STPBankSelectionViewControllerDelegate.bankSelectionViewController( + _: + didCreatePaymentMethodParams: + )) + ) ?? false { + delegate?.bankSelectionViewController( + self, + didCreatePaymentMethodParams: paymentMethodParams + ) + } + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } + + required init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + fatalError("init(nibName:bundle:) has not been implemented") + } +} + +/// An `STPBankSelectionViewControllerDelegate` is notified when a user selects a bank. +@objc public protocol STPBankSelectionViewControllerDelegate: NSObjectProtocol { + /// This is called when the user selects a bank. + /// You can use the returned PaymentMethodParams to confirm a PaymentIntent, or inspect + /// it to obtain details about the selected bank. + /// Once you're done, you'll want to dismiss (or pop) the view controller. + /// - Parameters: + /// - bankViewController: the view controller that created the PaymentMethodParams + /// - paymentMethodParams: the PaymentMethodParams that was created. - seealso: STPPaymentMethodParams + @objc(bankSelectionViewController:didCreatePaymentMethodParams:) + func bankSelectionViewController( + _ bankViewController: STPBankSelectionViewController, + didCreatePaymentMethodParams paymentMethodParams: STPPaymentMethodParams + ) +} + +private let STPBankSelectionCellReuseIdentifier = "STPBankSelectionCellReuseIdentifier" + +/// :nodoc: +@_spi(STP) extension STPBankSelectionViewController: STPAnalyticsProtocol { + @_spi(STP) public static var stp_analyticsIdentifier = "STPBankSelectionViewController" +} diff --git a/Stripe/StripeiOS/Source/STPBlocks.swift b/Stripe/StripeiOS/Source/STPBlocks.swift new file mode 100644 index 00000000..e750cc71 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPBlocks.swift @@ -0,0 +1,44 @@ +// +// STPBlocks.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit + +/// A callback to be run with a response from the Stripe API containing information about the online status of FPX banks. +/// - Parameters: +/// - bankStatusResponse: The response from Stripe containing the status of the various banks. Will be nil if an error occurs. - seealso: STPFPXBankStatusResponse +/// - error: The error returned from the response, or nil if none occurs. +typealias STPFPXBankStatusCompletionBlock = (STPFPXBankStatusResponse?, Error?) -> Void + +/// These values control the labels used in the shipping info collection form. +@objc public enum STPShippingType: Int { + /// Shipping the purchase to the provided address using a third-party + /// shipping company. + case shipping + /// Delivering the purchase by the seller. + case delivery +} + +/// An enum representing the status of a shipping address validation. +@objc public enum STPShippingStatus: Int { + /// The shipping address is valid. + case valid + /// The shipping address is invalid. + case invalid +} + +/// A callback to be run with a validation result and shipping methods for a +/// shipping address. +/// - Parameters: +/// - status: An enum representing whether the shipping address is valid. +/// - shippingValidationError: If the shipping address is invalid, an error describing the issue with the address. If no error is given and the address is invalid, the default error message will be used. +/// - shippingMethods: The shipping methods available for the address. +/// - selectedShippingMethod: The default selected shipping method for the address. +public typealias STPShippingMethodsCompletionBlock = ( + STPShippingStatus, Error?, [PKShippingMethod]?, PKShippingMethod? +) -> Void diff --git a/Stripe/StripeiOS/Source/STPCameraView.swift b/Stripe/StripeiOS/Source/STPCameraView.swift new file mode 100644 index 00000000..17fa4247 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCameraView.swift @@ -0,0 +1,77 @@ +// +// STPCameraView.swift +// StripeiOS +// +// Created by David Estes on 8/17/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import AVFoundation +import UIKit + +@available(macCatalyst 14.0, *) +class STPCameraView: UIView { + private var flashLayer: CALayer? + + var captureSession: AVCaptureSession? { + get { + return (videoPreviewLayer.session)! + } + set(captureSession) { + videoPreviewLayer.session = captureSession + } + } + + var videoPreviewLayer: AVCaptureVideoPreviewLayer { + return layer as! AVCaptureVideoPreviewLayer + } + + func playSnapshotAnimation() { + CATransaction.begin() + CATransaction.setValue( + kCFBooleanTrue, + forKey: kCATransactionDisableActions + ) + flashLayer?.frame = CGRect( + x: 0, + y: 0, + width: layer.bounds.size.width, + height: layer.bounds.size.height + ) + flashLayer?.opacity = 1.0 + CATransaction.commit() + DispatchQueue.main.async(execute: { + let fadeAnim = CABasicAnimation(keyPath: "opacity") + fadeAnim.fromValue = NSNumber(value: 1.0) + fadeAnim.toValue = NSNumber(value: 0.0) + fadeAnim.duration = 1.0 + self.flashLayer?.add(fadeAnim, forKey: "opacity") + self.flashLayer?.opacity = 0.0 + }) + } + + override init( + frame: CGRect + ) { + super.init(frame: frame) + flashLayer = CALayer() + if let flashLayer = flashLayer { + layer.addSublayer(flashLayer) + } + flashLayer?.masksToBounds = true + flashLayer?.backgroundColor = UIColor.black.cgColor + flashLayer?.opacity = 0.0 + layer.masksToBounds = true + videoPreviewLayer.videoGravity = .resizeAspectFill + } + + override class var layerClass: AnyClass { + return AVCaptureVideoPreviewLayer.self + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/STPCard+BasicUI.swift b/Stripe/StripeiOS/Source/STPCard+BasicUI.swift new file mode 100644 index 00000000..2b90f892 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCard+BasicUI.swift @@ -0,0 +1,30 @@ +// +// STPCard+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import UIKit + +extension STPCard: STPPaymentOption { + // MARK: - STPPaymentOption + @objc public var image: UIImage { + return STPImageLibrary.cardBrandImage(for: brand) + } + + @objc public var templateImage: UIImage { + return STPImageLibrary.templatedBrandImage(for: brand) + } + + @objc public var label: String { + let brand = STPCard.string(from: self.brand) + return "\(brand) \(last4 )" + } + + @objc public var isReusable: Bool { + return true + } +} diff --git a/Stripe/StripeiOS/Source/STPCardScanner.swift b/Stripe/StripeiOS/Source/STPCardScanner.swift new file mode 100644 index 00000000..d4985c5f --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCardScanner.swift @@ -0,0 +1,518 @@ +// +// STPCardScanner.swift +// StripeiOS +// +// Created by David Estes on 8/17/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import AVFoundation +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments +@_spi(STP) import StripePaymentsUI +import UIKit +import Vision + +enum STPCardScannerError: Int { + /// Camera not available. + case cameraNotAvailable +} + +@available(iOS 13, macCatalyst 14, *) +@objc protocol STPCardScannerDelegate: NSObjectProtocol { + @objc(cardScanner:didFinishWithCardParams:error:) func cardScanner( + _ scanner: STPCardScanner, + didFinishWith cardParams: STPPaymentMethodCardParams?, + error: Error? + ) +} + +@available(iOS 13, macCatalyst 14, *) +@objc(STPCardScanner_legacy) +class STPCardScanner: NSObject, AVCaptureVideoDataOutputSampleBufferDelegate, + STPCardScanningProtocol +{ + // iOS will kill the app if it tries to request the camera without an NSCameraUsageDescription + static let cardScanningAvailableCameraHasUsageDescription = { + return + (Bundle.main.infoDictionary?["NSCameraUsageDescription"] != nil + || Bundle.main.localizedInfoDictionary?["NSCameraUsageDescription"] != nil) + }() + + static var cardScanningAvailable: Bool { + // Always allow in tests: + if NSClassFromString("XCTest") != nil { + return true + } + return cardScanningAvailableCameraHasUsageDescription + } + + weak var cameraView: STPCameraView? + + var feedbackGenerator: UINotificationFeedbackGenerator? + + @objc public var deviceOrientation: UIDeviceOrientation { + get { + return stp_deviceOrientation + } + set(newDeviceOrientation) { + stp_deviceOrientation = newDeviceOrientation + + // This is an optimization for portrait mode: The card will be centered in the screen, + // so we can ignore the top and bottom. We'll use the whole frame in landscape. + let kSTPCardScanningScreenCenter = CGRect( + x: 0, + y: CGFloat(0.3), + width: 1, + height: CGFloat(0.4) + ) + + // iOS camera image data is returned in LandcapeLeft orientation by default. We'll flip it as needed: + switch newDeviceOrientation { + case .portraitUpsideDown: + videoOrientation = .portraitUpsideDown + textOrientation = .left + regionOfInterest = kSTPCardScanningScreenCenter + case .landscapeLeft: + videoOrientation = .landscapeRight + textOrientation = .up + regionOfInterest = CGRect(x: 0, y: 0, width: 1, height: 1) + case .landscapeRight: + videoOrientation = .landscapeLeft + textOrientation = .down + regionOfInterest = CGRect(x: 0, y: 0, width: 1, height: 1) + case .portrait, .faceUp, .faceDown, .unknown: + // swift-format-ignore: NoCasesWithOnlyFallthrough + fallthrough + default: + videoOrientation = .portrait + textOrientation = .right + regionOfInterest = kSTPCardScanningScreenCenter + } + cameraView?.videoPreviewLayer.connection?.videoOrientation = videoOrientation + } + } + + override init() { + } + + init( + delegate: STPCardScannerDelegate? + ) { + super.init() + self.delegate = delegate + captureSessionQueue = DispatchQueue(label: "com.stripe.CardScanning.CaptureSessionQueue") + deviceOrientation = UIDevice.current.orientation + } + + func start() { + if isScanning { + return + } + STPAnalyticsClient.sharedClient.addClass(toProductUsageIfNecessary: STPCardScanner.self) + startTime = Date() + + isScanning = true + didTimeout = false + timeoutStarted = false + feedbackGenerator = UINotificationFeedbackGenerator() + feedbackGenerator?.prepare() + + captureSessionQueue?.async(execute: { + #if targetEnvironment(simulator) + // Camera not supported on Simulator + self.stopWithError(STPCardScanner.stp_cardScanningError()) + return + #else + self.detectedNumbers = NSCountedSet() // capacity: 5 + self.detectedExpirations = NSCountedSet() // capacity: 5 + self.setupCamera() + DispatchQueue.main.async(execute: { + self.cameraView?.captureSession = self.captureSession + self.cameraView?.videoPreviewLayer.connection?.videoOrientation = + self.videoOrientation + }) + #endif + }) + } + + func stop() { + stopWithError(nil) + } + + private weak var delegate: STPCardScannerDelegate? + private var captureDevice: AVCaptureDevice? + private var captureSession: AVCaptureSession? + private var captureSessionQueue: DispatchQueue? + private var videoDataOutput: AVCaptureVideoDataOutput? + private var videoDataOutputQueue: DispatchQueue? + private var textRequest: VNRecognizeTextRequest? + private var isScanning = false + private var didTimeout = false + private var timeoutStarted = false + private var stp_deviceOrientation: UIDeviceOrientation! + private var videoOrientation: AVCaptureVideoOrientation! + private var textOrientation: CGImagePropertyOrientation! + private var regionOfInterest = CGRect.zero + private var detectedNumbers: NSCountedSet? + private var detectedExpirations: NSCountedSet? + private var startTime: Date? + + // MARK: Public + + class func stp_cardScanningError() -> Error { + let userInfo = [ + NSLocalizedDescriptionKey: String.Localized.allow_camera_access, + STPError.errorMessageKey: "The camera couldn't be used.", + ] + return NSError( + domain: STPCardScannerErrorDomain, + code: STPCardScannerError.cameraNotAvailable.rawValue, + userInfo: userInfo + ) + } + + deinit { + if isScanning { + captureDevice?.unlockForConfiguration() + captureSession?.stopRunning() + } + } + + func stopWithError(_ error: Error?) { + if isScanning { + finish(with: nil, error: error) + } + } + + // MARK: Setup + func setupCamera() { + weak var weakSelf = self + textRequest = VNRecognizeTextRequest(completionHandler: { request, error in + let strongSelf = weakSelf + if !(strongSelf?.isScanning ?? false) { + return + } + if error != nil { + strongSelf?.stopWithError(STPCardScanner.stp_cardScanningError()) + return + } + strongSelf?.processVNRequest(request) + }) + + let captureDevice = AVCaptureDevice.default( + .builtInWideAngleCamera, + for: .video, + position: .back + ) + self.captureDevice = captureDevice + + captureSession = AVCaptureSession() + captureSession?.sessionPreset = .hd1920x1080 + + var deviceInput: AVCaptureDeviceInput? + do { + if let captureDevice = captureDevice { + deviceInput = try AVCaptureDeviceInput(device: captureDevice) + } + } catch { + stopWithError(STPCardScanner.stp_cardScanningError()) + return + } + + if let deviceInput = deviceInput { + if captureSession?.canAddInput(deviceInput) ?? false { + captureSession?.addInput(deviceInput) + } else { + stopWithError(STPCardScanner.stp_cardScanningError()) + return + } + } + + videoDataOutputQueue = DispatchQueue(label: "com.stripe.CardScanning.VideoDataOutputQueue") + videoDataOutput = AVCaptureVideoDataOutput() + videoDataOutput?.alwaysDiscardsLateVideoFrames = true + videoDataOutput?.setSampleBufferDelegate(self, queue: videoDataOutputQueue) + + // This is the recommended pixel buffer format for Vision: + videoDataOutput?.videoSettings = [ + kCVPixelBufferPixelFormatTypeKey as String: + kCVPixelFormatType_420YpCbCr8BiPlanarFullRange, + ] + + if let videoDataOutput = videoDataOutput { + if captureSession?.canAddOutput(videoDataOutput) ?? false { + captureSession?.addOutput(videoDataOutput) + } else { + stopWithError(STPCardScanner.stp_cardScanningError()) + return + } + } + + // This improves recognition quality, but means the VideoDataOutput buffers won't match what we're seeing on screen. + videoDataOutput?.connection(with: .video)?.preferredVideoStabilizationMode = .auto + + captureSession?.startRunning() + + do { + try self.captureDevice?.lockForConfiguration() + self.captureDevice?.autoFocusRangeRestriction = .near + } catch { + } + } + + // MARK: Processing + func captureOutput( + _ output: AVCaptureOutput, + didOutput sampleBuffer: CMSampleBuffer, + from connection: AVCaptureConnection + ) { + if !isScanning { + return + } + let pixelBuffer = CMSampleBufferGetImageBuffer(sampleBuffer) + if pixelBuffer == nil { + return + } + textRequest?.recognitionLevel = .accurate + textRequest?.usesLanguageCorrection = false + textRequest?.regionOfInterest = regionOfInterest + var handler: VNImageRequestHandler? + if let pixelBuffer = pixelBuffer { + handler = VNImageRequestHandler( + cvPixelBuffer: pixelBuffer, + orientation: textOrientation, + options: [:] + ) + } + do { + try handler?.perform([textRequest].compactMap { $0 }) + } catch { + } + } + + func processVNRequest(_ request: VNRequest) { + var allNumbers: [String] = [] + for observation in request.results ?? [] { + guard let observation = observation as? VNRecognizedTextObservation else { + continue + } + let candidates = observation.topCandidates(5) + let topCandidate = candidates.first?.string + if STPCardValidator.sanitizedNumericString(for: topCandidate ?? "").count >= 4 { + allNumbers.append(topCandidate ?? "") + } + for recognizedText in candidates { + let possibleNumber = STPCardValidator.sanitizedNumericString( + for: recognizedText.string + ) + // This probably isn't something we're interested in, so don't bother processing it. + if possibleNumber.count < 4 { + continue + } + + // First strategy: We check if Vision sent us a number in a group on its own. If that fails, we'll try + // to catch it later when we iterate over all the numbers. + if STPCardValidator.validationState( + forNumber: possibleNumber, + validatingCardBrand: true + ) + == .valid + { + addDetectedNumber(possibleNumber) + } else if possibleNumber.count >= 4 && possibleNumber.count <= 6 + && STPStringUtils.stringMayContainExpirationDate(recognizedText.string) + { + // Try to parse anything that looks like an expiration date. + let expirationString = STPStringUtils.expirationDateString( + from: recognizedText.string + ) + let sanitizedExpiration = STPCardValidator.sanitizedNumericString( + for: expirationString ?? "" + ) + let month = (sanitizedExpiration as NSString).substring(to: 2) + let year = (sanitizedExpiration as NSString).substring(from: 2) + + // Ignore expiration dates 10+ years in the future, as they're likely to be incorrect recognitions + let calendar = Calendar(identifier: .gregorian) + let presentYear = calendar.component(.year, from: Date()) + let maxYear = (presentYear % 100) + 10 + + if STPCardValidator.validationState(forExpirationYear: year, inMonth: month) + == .valid + && Int(year) ?? 0 < maxYear + { + addDetectedExpiration(sanitizedExpiration) + } + } + } + } + // Second strategy: We look for consecutive groups of 4/4/4/4 or 4/6/5 + // Vision is sending us groups like ["1234 565", "1234 1"], so we'll normalize these into groups with spaces: + let allGroups = allNumbers.joined(separator: " ").components(separatedBy: " ") + if allGroups.count < 3 { + return + } + for i in 0..<(allGroups.count - 3) { + let string1 = allGroups[i] + let string2 = allGroups[i + 1] + let string3 = allGroups[i + 2] + var string4 = "" + if i + 3 < allGroups.count { + string4 = allGroups[i + 3] + } + // Then we'll go through each group and build a potential match: + let potentialCardString = "\(string1)\(string2)\(string3)\(string4)" + let potentialAmexString = "\(string1)\(string2)\(string3)" + + // Then we'll add valid matches. It's okay if we add a number a second time after doing so above, as the success of that first pass means it's more likely to be a good match. + if STPCardValidator.validationState( + forNumber: potentialCardString, + validatingCardBrand: true + ) + == .valid + { + addDetectedNumber(potentialCardString) + } else if STPCardValidator.validationState( + forNumber: potentialAmexString, + validatingCardBrand: true + ) == .valid { + addDetectedNumber(potentialAmexString) + } + } + } + + func addDetectedNumber(_ number: String) { + detectedNumbers?.add(number) + + // Set a timeout: If we don't get enough scans in the next 1 second, we'll use the best option we have. + if !timeoutStarted { + timeoutStarted = true + weak var weakSelf = self + DispatchQueue.main.async(execute: { + let strongSelf = weakSelf + strongSelf?.cameraView?.playSnapshotAnimation() + strongSelf?.feedbackGenerator?.notificationOccurred(.success) + }) + videoDataOutputQueue?.asyncAfter( + deadline: DispatchTime.now() + Double( + Int64(kSTPCardScanningTimeout * Double(NSEC_PER_SEC)) + ) + / Double(NSEC_PER_SEC), + execute: { + let strongSelf = weakSelf + if strongSelf?.isScanning ?? false { + strongSelf?.didTimeout = true + strongSelf?.finishIfReady() + } + } + ) + } + + if (detectedNumbers?.count(for: number) ?? 0) >= kSTPCardScanningMinimumValidScans { + finishIfReady() + } + } + + func addDetectedExpiration(_ expiration: String) { + detectedExpirations?.add(expiration) + if (detectedExpirations?.count(for: expiration) ?? 0) >= kSTPCardScanningMinimumValidScans { + finishIfReady() + } + } + + // MARK: Completion + func finishIfReady() { + if !isScanning { + return + } + let detectedNumbers = self.detectedNumbers + let detectedExpirations = self.detectedExpirations + + let topNumber = (detectedNumbers?.allObjects as NSArray?)?.sortedArray(comparator: { + obj1, + obj2 in + let c1 = detectedNumbers?.count(for: obj1) ?? 0 + let c2 = detectedNumbers?.count(for: obj2) ?? 0 + if c1 < c2 { + return .orderedAscending + } else if c1 > c2 { + return .orderedDescending + } else { + return .orderedSame + } + }).last + let topExpiration = (detectedExpirations?.allObjects as NSArray?)?.sortedArray(comparator: { + obj1, + obj2 in + let c1 = detectedExpirations?.count(for: obj1) ?? 0 + let c2 = detectedExpirations?.count(for: obj2) ?? 0 + if c1 < c2 { + return .orderedAscending + } else if c1 > c2 { + return .orderedDescending + } else { + return .orderedSame + } + }).last + + if didTimeout + || (((detectedNumbers?.count(for: topNumber ?? 0) ?? 0) + >= kSTPCardScanningMinimumValidScans) + && ((detectedExpirations?.count(for: topExpiration ?? 0) ?? 0) + >= kSTPCardScanningMinimumValidScans)) + || ((detectedNumbers?.count(for: topNumber ?? 0) ?? 0) >= kSTPCardScanningMaxValidScans) + { + let params = STPPaymentMethodCardParams() + params.number = topNumber as? String + if let topExpiration = topExpiration { + params.expMonth = NSNumber( + value: Int((topExpiration as! NSString).substring(to: 2)) ?? 0 + ) + params.expYear = NSNumber( + value: Int((topExpiration as! NSString).substring(from: 2)) ?? 0 + ) + } + finish(with: params, error: nil) + } + } + + func finish(with params: STPPaymentMethodCardParams?, error: Error?) { + var duration: TimeInterval? + if let startTime = startTime { + duration = Date().timeIntervalSince(startTime) + } + isScanning = false + captureDevice?.unlockForConfiguration() + captureSession?.stopRunning() + + DispatchQueue.main.async(execute: { + if params == nil { + STPAnalyticsClient.sharedClient.logCardScanCancelled(withDuration: duration ?? 0.0) + } else { + STPAnalyticsClient.sharedClient.logCardScanSucceeded(withDuration: duration ?? 0.0) + } + self.feedbackGenerator = nil + + self.cameraView?.captureSession = nil + self.delegate?.cardScanner(self, didFinishWith: params, error: error) + }) + } + + // MARK: Orientation +} + +// The number of successful scans required for both card number and expiration date before returning a result. +private let kSTPCardScanningMinimumValidScans = 2 +// If no expiration date is found, we'll return a result after this many successful scans. +private let kSTPCardScanningMaxValidScans = 3 +// Once one successful scan is found, we'll stop scanning after this many seconds. +private let kSTPCardScanningTimeout: TimeInterval = 1.0 +let STPCardScannerErrorDomain = "STPCardScannerErrorDomain" + +@available(iOS 13, macCatalyst 14, *) +/// :nodoc: +extension STPCardScanner: STPAnalyticsProtocol { + static var stp_analyticsIdentifier = "STPCardScanner" +} diff --git a/Stripe/StripeiOS/Source/STPCardScannerTableViewCell.swift b/Stripe/StripeiOS/Source/STPCardScannerTableViewCell.swift new file mode 100644 index 00000000..7692e148 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCardScannerTableViewCell.swift @@ -0,0 +1,67 @@ +// +// STPCardScannerTableViewCell.swift +// StripeiOS +// +// Created by David Estes on 8/17/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import UIKit + +@available(macCatalyst 14.0, *) +class STPCardScannerTableViewCell: UITableViewCell { + private(set) weak var cameraView: STPCameraView? + + private var _theme: STPTheme? + var theme: STPTheme? { + get { + _theme + } + set(theme) { + _theme = theme + updateAppearance() + } + } + + let cardSizeRatio: CGFloat = 2.125 / 3.370 // ID-1 card size (in inches) + + override init( + style: UITableViewCell.CellStyle, + reuseIdentifier: String? + ) { + super.init(style: style, reuseIdentifier: reuseIdentifier) + let cameraView = STPCameraView(frame: bounds) + contentView.addSubview(cameraView) + self.cameraView = cameraView + theme = STPTheme.defaultTheme + self.cameraView?.translatesAutoresizingMaskIntoConstraints = false + contentView.addConstraints( + [ + cameraView.heightAnchor.constraint( + equalTo: cameraView.widthAnchor, + multiplier: cardSizeRatio + ), + cameraView.bottomAnchor.constraint(equalTo: contentView.bottomAnchor, constant: 0), + cameraView.leftAnchor.constraint(equalTo: contentView.leftAnchor, constant: 0), + cameraView.rightAnchor.constraint(equalTo: contentView.rightAnchor, constant: 0), + cameraView.topAnchor.constraint(equalTo: contentView.topAnchor, constant: 0), + ]) + updateAppearance() + } + + override func layoutSubviews() { + + super.layoutSubviews() + } + + @objc func updateAppearance() { + // The first few frames of the camera view will be black, so our background should be black too. + cameraView?.backgroundColor = UIColor.black + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/STPCardValidationState.swift b/Stripe/StripeiOS/Source/STPCardValidationState.swift new file mode 100644 index 00000000..48e1f46b --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCardValidationState.swift @@ -0,0 +1,9 @@ +// +// STPCardValidationState.swift +// StripeiOS +// +// Created by Jack Flintermann on 8/7/15. +// Copyright (c) 2015 Stripe, Inc. All rights reserved. +// + +import Foundation diff --git a/Stripe/StripeiOS/Source/STPCoreScrollViewController.swift b/Stripe/StripeiOS/Source/STPCoreScrollViewController.swift new file mode 100644 index 00000000..de3b7bbf --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCoreScrollViewController.swift @@ -0,0 +1,62 @@ +// +// STPCoreScrollViewController.swift +// StripeiOS +// +// Created by Brian Dorfman on 1/6/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeUICore +import UIKit + +/// This is the base class for all Stripe scroll view controllers. It is intended +/// for use only by Stripe classes, you should not subclass it yourself in your app. +public class STPCoreScrollViewController: STPCoreViewController { + /// This returns the scroll view being managed by the view controller + @objc public lazy var scrollView: UIScrollView = { + createScrollView() + }() + + /// This method is used by the base implementation to create the object + /// backing the `scrollView` property. Subclasses can override to change the + /// type of the scroll view (eg UITableView or UICollectionView instead of + /// UIScrollView). + + func createScrollView() -> UIScrollView { + return UIScrollView() + } + + override func createAndSetupViews() { + super.createAndSetupViews() + view.addSubview(scrollView) + } + + /// :nodoc: + @objc + public override func viewDidLoad() { + super.viewDidLoad() + + scrollView.contentInsetAdjustmentBehavior = .automatic + } + + /// :nodoc: + @objc + public override func viewDidLayoutSubviews() { + super.viewDidLayoutSubviews() + scrollView.frame = view.bounds + } + + @objc override func updateAppearance() { + super.updateAppearance() + + scrollView.backgroundColor = theme.primaryBackgroundColor + scrollView.tintColor = theme.accentColor + + if theme.primaryBackgroundColor.isBright { + scrollView.indicatorStyle = .black + } else { + scrollView.indicatorStyle = .white + } + } +} diff --git a/Stripe/StripeiOS/Source/STPCoreTableViewController.swift b/Stripe/StripeiOS/Source/STPCoreTableViewController.swift new file mode 100644 index 00000000..5d7105e3 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCoreTableViewController.swift @@ -0,0 +1,54 @@ +// +// STPCoreTableViewController.swift +// StripeiOS +// +// Created by Brian Dorfman on 1/6/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import Foundation +import UIKit + +/// This is the base class for all Stripe scroll view controllers. It is intended +/// for use only by Stripe classes, you should not subclass it yourself in your app. +/// It inherits from STPCoreScrollViewController and changes the type of the +/// created scroll view to UITableView, as well as other shared table view logic. +public class STPCoreTableViewController: STPCoreScrollViewController { + /// This points to the same object as `STPCoreScrollViewController`'s `scrollView` + /// property but with the type cast to `UITableView` + + @objc public var tableView: UITableView? { + return (scrollView as? UITableView) + } + + override func createScrollView() -> UIScrollView { + let tableView = UITableView(frame: CGRect.zero, style: .grouped) + tableView.sectionHeaderHeight = 30 + + return tableView + } + + /// :nodoc: + @objc + public override func viewWillAppear(_ animated: Bool) { + super.viewWillAppear(animated) + + tableView?.reloadData() + } + + @objc override func updateAppearance() { + super.updateAppearance() + tableView?.separatorStyle = .none // handle this with fake separator views for flexibility + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + heightForHeaderInSection section: Int + ) + -> CGFloat + { + return 0.01 + } +} diff --git a/Stripe/StripeiOS/Source/STPCoreViewController.swift b/Stripe/StripeiOS/Source/STPCoreViewController.swift new file mode 100644 index 00000000..105b544d --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCoreViewController.swift @@ -0,0 +1,162 @@ +// +// STPCoreViewController.swift +// StripeiOS +// +// Created by Brian Dorfman on 1/6/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeUICore +import UIKit + +/// This is the base class for all Stripe view controllers. It is intended for use +/// only by Stripe classes, you should not subclass it yourself in your app. +/// It theming, back/cancel button management, and other shared logic for +/// Stripe view controllers. +public class STPCoreViewController: UIViewController { + /// A convenience initializer; equivalent to calling `init(theme: STPTheme.defaultTheme)`. + @objc + public convenience init() { + self.init(theme: STPTheme.defaultTheme) + } + + /// Initializes a new view controller with the specified theme + /// - Parameter theme: The theme to use to inform the view controller's visual appearance. - seealso: STPTheme + @objc public required init( + theme: STPTheme? + ) { + super.init(nibName: nil, bundle: nil) + commonInit(with: theme) + } + + /// Passes through to the default UIViewController behavior for this initializer, + /// and then also sets the default theme as in `init` + @objc public required override init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil) + commonInit(with: STPTheme.defaultTheme) + } + + /// Passes through to the default UIViewController behavior for this initializer, + /// and then also sets the default theme as in `init` + @objc public required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + commonInit(with: STPTheme.defaultTheme) + } + + private var _theme: STPTheme = STPTheme.defaultTheme + @objc var theme: STPTheme { + get { + _theme + } + set(theme) { + _theme = theme + updateAppearance() + } + } + @objc var cancelItem: UIBarButtonItem? + + /// All designated initializers funnel through this method to do their setup + /// - Parameter theme: Initial theme for this view controller + func commonInit(with theme: STPTheme?) { + if let theme = theme { + _theme = theme + } else { + _theme = .defaultTheme + } + + if !useSystemBackButton() { + cancelItem = UIBarButtonItem( + barButtonSystemItem: .cancel, + target: self, + action: #selector(STPAddCardViewController.handleCancelTapped(_:)) + ) + cancelItem?.accessibilityIdentifier = "CoreViewControllerCancelIdentifier" + + stp_navigationItemProxy?.leftBarButtonItem = cancelItem + } + + NotificationCenter.default.addObserver( + self, + selector: #selector(STPAddCardViewController.updateAppearance), + name: UIContentSizeCategory.didChangeNotification, + object: nil + ) + } + + /// Called in viewDidLoad after doing base implementation, before + /// calling updateAppearance + func createAndSetupViews() { + // do nothing + } + + // These viewDidX() methods have significant code done + // in the base class and super must be called if they are overidden + /// :nodoc: + @objc + public override func viewDidLoad() { + super.viewDidLoad() + + createAndSetupViews() + updateAppearance() + } + /// :nodoc: + @objc + public override func viewWillAppear(_ animated: Bool) { + super.viewWillAppear(animated) + updateAppearance() + } + /// :nodoc: + @objc + public override func viewWillDisappear(_ animated: Bool) { + super.viewWillDisappear(animated) + view.endEditing(true) + } + + /// Update views based on current STPTheme + @objc func updateAppearance() { + let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme + navigationItem.leftBarButtonItem?.stp_setTheme(navBarTheme) + navigationItem.rightBarButtonItem?.stp_setTheme(navBarTheme) + cancelItem?.stp_setTheme(navBarTheme) + + view.backgroundColor = theme.primaryBackgroundColor + + setNeedsStatusBarAppearanceUpdate() + } + + /// :nodoc: + @objc public override var preferredStatusBarStyle: UIStatusBarStyle { + let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme + return navBarTheme.secondaryBackgroundColor.isBright + ? .default + : .lightContent + } + + /// Called by the automatically-managed back/cancel button + /// By default pops the top item off the navigation stack, or if we are the + /// root of the navigation controller, dimisses presentation + /// - Parameter sender: Sender of the target action, if applicable. + @objc func handleCancelTapped(_ sender: Any?) { + if stp_isAtRootOfNavigationController() { + // if we're the root of the navigation controller, we've been presented modally. + presentingViewController?.dismiss(animated: true) + } else { + // otherwise, we've been pushed onto the stack. + navigationController?.popViewController(animated: true) + } + } + + /// If you override this and return YES, then your CoreVC implementation will not + /// create and set up a cancel and instead just use the default + /// UIViewController back button behavior. + /// You won't receive calls to `handleCancelTapped` if this is YES. + /// Defaults to NO. + func useSystemBackButton() -> Bool { + return false + } +} diff --git a/Stripe/StripeiOS/Source/STPCustomerContext.swift b/Stripe/StripeiOS/Source/STPCustomerContext.swift new file mode 100644 index 00000000..90c97d26 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPCustomerContext.swift @@ -0,0 +1,419 @@ +// +// STPCustomerContext.swift +// StripeiOS +// +// Created by Ben Guo on 5/2/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI + +/// An `STPCustomerContext` retrieves and updates a Stripe customer and their attached +/// payment methods using an ephemeral key, a short-lived API key scoped to a specific +/// customer object. If your current user logs out of your app and a new user logs in, +/// be sure to either create a new instance of `STPCustomerContext` or clear the current +/// instance's cache. On your backend, be sure to create and return a +/// new ephemeral key for the Customer object associated with the new user. +open class STPCustomerContext: NSObject, STPBackendAPIAdapter { + /// Initializes a new `STPCustomerContext` with the specified key provider. + /// Upon initialization, a CustomerContext will fetch a new ephemeral key from + /// your backend and use it to prefetch the customer object specified in the key. + /// Subsequent customer and payment method retrievals (e.g. by `STPPaymentContext`) + /// will return the prefetched customer / attached payment methods immediately if + /// its age does not exceed 60 seconds. + /// - Parameter keyProvider: The key provider the customer context will use. + /// - Returns: the newly-instantiated customer context. + @objc(initWithKeyProvider:) + public convenience init( + keyProvider: STPCustomerEphemeralKeyProvider + ) { + self.init(keyProvider: keyProvider, apiClient: STPAPIClient.shared) + } + + /// Initializes a new `STPCustomerContext` with the specified key provider. + /// Upon initialization, a CustomerContext will fetch a new ephemeral key from + /// your backend and use it to prefetch the customer object specified in the key. + /// Subsequent customer and payment method retrievals (e.g. by `STPPaymentContext`) + /// will return the prefetched customer / attached payment methods immediately if + /// its age does not exceed 60 seconds. + /// - Parameters: + /// - keyProvider: The key provider the customer context will use. + /// - apiClient: The API Client to use to make requests. + /// - Returns: the newly-instantiated customer context. + @objc(initWithKeyProvider:apiClient:) + public convenience init( + keyProvider: STPCustomerEphemeralKeyProvider?, + apiClient: STPAPIClient + ) { + let keyManager = STPEphemeralKeyManager( + keyProvider: keyProvider, + apiVersion: STPAPIClient.apiVersion, + performsEagerFetching: true + ) + self.init(keyManager: keyManager, apiClient: apiClient) + } + + /// `STPCustomerContext` will cache its customer object and associated payment methods + /// for up to 60 seconds. If your current user logs out of your app and a new user logs + /// in, be sure to either call this method or create a new instance of `STPCustomerContext`. + /// On your backend, be sure to create and return a new ephemeral key for the + /// customer object associated with the new user. + @objc + public func clearCache() { + clearCachedCustomer() + clearCachedPaymentMethods() + } + + private var _includeApplePayPaymentMethods = false + /// By default, `STPCustomerContext` will filter Apple Pay when it retrieves + /// Payment Methods. Apple Pay payment methods should generally not be re-used and + /// shouldn't be offered to customers as a new payment method (Apple Pay payment + /// methods may only be re-used for subscriptions). + /// If you are using `STPCustomerContext` to back your own UI and would like to + /// disable Apple Pay filtering, set this property to YES. + /// Note: If you are using `STPPaymentContext`, you should not change this property. + @objc public var includeApplePayPaymentMethods: Bool { + get { + _includeApplePayPaymentMethods + } + set(includeApplePayMethods) { + _includeApplePayPaymentMethods = includeApplePayMethods + customer?.updateSources(filteringApplePay: !includeApplePayMethods) + } + } + + private var _customer: STPCustomer? + private var customer: STPCustomer? { + get { + _customer + } + set(customer) { + _customer = customer + customerRetrievedDate = (customer) != nil ? Date() : nil + } + } + @objc internal var customerRetrievedDate: Date? + + private var _paymentMethods: [STPPaymentMethod]? + private var paymentMethods: [STPPaymentMethod]? { + get { + if !includeApplePayPaymentMethods { + var paymentMethodsExcludingApplePay: [STPPaymentMethod]? = [] + for paymentMethod in _paymentMethods ?? [] { + let isApplePay = + paymentMethod.type == .card && paymentMethod.card?.wallet?.type == .applePay + if !isApplePay { + paymentMethodsExcludingApplePay?.append(paymentMethod) + } + } + return paymentMethodsExcludingApplePay ?? [] + } else { + return _paymentMethods ?? [] + } + } + set(paymentMethods) { + _paymentMethods = paymentMethods + paymentMethodsRetrievedDate = paymentMethods != nil ? Date() : nil + } + } + @objc internal var paymentMethodsRetrievedDate: Date? + private var keyManager: STPEphemeralKeyManagerProtocol + private var apiClient: STPAPIClient + + init( + keyManager: STPEphemeralKeyManagerProtocol, + apiClient: STPAPIClient + ) { + STPAnalyticsClient.sharedClient.addClass(toProductUsageIfNecessary: STPCustomerContext.self) + self.keyManager = keyManager + self.apiClient = apiClient + _includeApplePayPaymentMethods = false + super.init() + retrieveCustomer(nil) + listPaymentMethodsForCustomer(completion: nil) + } + + func clearCachedCustomer() { + customer = nil + } + + func clearCachedPaymentMethods() { + paymentMethods = nil + } + + func shouldUseCachedCustomer() -> Bool { + if customer == nil || customerRetrievedDate == nil { + return false + } + let now = Date() + if let customerRetrievedDate = customerRetrievedDate { + return now.timeIntervalSince(customerRetrievedDate) < CachedCustomerMaxAge + } + return false + } + + func shouldUseCachedPaymentMethods() -> Bool { + if paymentMethods == nil || paymentMethodsRetrievedDate == nil { + return false + } + let now = Date() + if let paymentMethodsRetrievedDate = paymentMethodsRetrievedDate { + return now.timeIntervalSince(paymentMethodsRetrievedDate) < CachedCustomerMaxAge + } + return false + } + + // MARK: - STPBackendAPIAdapter + @objc + public func retrieveCustomer(_ completion: STPCustomerCompletionBlock? = nil) { + if shouldUseCachedCustomer() { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(self.customer, nil) + }) + } + return + } + keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in + guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(nil, retrieveKeyError) + }) + } + return + } + self.apiClient.retrieveCustomer(using: ephemeralKey) { customer, error in + if let customer = customer { + customer.updateSources(filteringApplePay: !self.includeApplePayPaymentMethods) + self.customer = customer + } + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(self.customer, error) + }) + } + } + }) + } + + @objc + public func updateCustomer( + withShippingAddress shipping: STPAddress, + completion: STPErrorBlock? + ) { + keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in + guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(retrieveKeyError) + }) + } + return + } + var params: [String: Any] = [:] + params["shipping"] = STPAddress.shippingInfoForCharge( + with: shipping, + shippingMethod: nil + ) + self.apiClient.updateCustomer( + withParameters: params, + using: ephemeralKey + ) { customer, error in + if let customer = customer { + customer.updateSources(filteringApplePay: !self.includeApplePayPaymentMethods) + self.customer = customer + } + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(error) + }) + } + } + }) + } + + /// A convenience method for attaching the PaymentMethod to the current Customer + @objc + public func attachPaymentMethodToCustomer( + paymentMethodId: String, + completion: STPErrorBlock? + ) { + keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in + guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(retrieveKeyError) + }) + } + return + } + + self.apiClient.attachPaymentMethod( + paymentMethodId, + toCustomerUsing: ephemeralKey + ) { error in + self.clearCachedPaymentMethods() + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(error) + }) + } + } + }) + } + + @objc + public func attachPaymentMethod( + toCustomer paymentMethod: STPPaymentMethod, + completion: STPErrorBlock? + ) { + attachPaymentMethodToCustomer( + paymentMethodId: paymentMethod.stripeId, + completion: completion + ) + } + + /// A convenience method for detaching the PaymentMethod to the current Customer + @objc + public func detachPaymentMethodFromCustomer( + paymentMethodId: String, + completion: STPErrorBlock? + ) { + keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in + guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(retrieveKeyError) + }) + } + return + } + + self.apiClient.detachPaymentMethod( + paymentMethodId, + fromCustomerUsing: ephemeralKey + ) { error in + self.clearCachedPaymentMethods() + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(error) + }) + } + } + }) + + } + + @objc + public func detachPaymentMethod( + fromCustomer paymentMethod: STPPaymentMethod, + completion: STPErrorBlock? + ) { + detachPaymentMethodFromCustomer( + paymentMethodId: paymentMethod.stripeId, + completion: completion + ) + } + + @objc + public func listPaymentMethodsForCustomer(completion: STPPaymentMethodsCompletionBlock? = nil) { + if shouldUseCachedPaymentMethods() { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(self.paymentMethods, nil) + }) + } + return + } + + keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in + guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(nil, retrieveKeyError) + }) + } + return + } + + self.apiClient.listPaymentMethodsForCustomer(using: ephemeralKey) { + paymentMethods, + error in + if paymentMethods != nil { + self.paymentMethods = paymentMethods + } + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(self.paymentMethods, error) + }) + } + } + }) + } + + func saveLastSelectedPaymentMethodID( + forCustomer paymentMethodID: String?, + completion: STPErrorBlock? + ) { + keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in + guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(retrieveKeyError) + }) + } + return + } + + var customerToDefaultPaymentMethodID = + (UserDefaults.standard.dictionary(forKey: kLastSelectedPaymentMethodDefaultsKey)) + as? [String: String] ?? [:] + if let customerID = ephemeralKey.customerID { + customerToDefaultPaymentMethodID[customerID] = paymentMethodID + UserDefaults.standard.set( + customerToDefaultPaymentMethodID, + forKey: kLastSelectedPaymentMethodDefaultsKey + ) + } + + if let completion = completion { + stpDispatchToMainThreadIfNecessary({ + completion(nil) + }) + } + }) + } + + func retrieveLastSelectedPaymentMethodIDForCustomer( + completion: @escaping (String?, Error?) -> Void + ) { + keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in + guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { + stpDispatchToMainThreadIfNecessary({ + completion(nil, retrieveKeyError) + }) + return + } + + let customerToDefaultPaymentMethodID = + (UserDefaults.standard.dictionary(forKey: kLastSelectedPaymentMethodDefaultsKey)) + as? [String: String] ?? [:] + stpDispatchToMainThreadIfNecessary({ + completion(customerToDefaultPaymentMethodID[ephemeralKey.customerID ?? ""], nil) + }) + }) + } +} + +/// Stores the key we use in NSUserDefaults to save a dictionary of Customer id to their last selected payment method ID +private let kLastSelectedPaymentMethodDefaultsKey = + UserDefaults.StripePaymentsUIKeys.customerToLastSelectedPaymentMethod.rawValue +private let CachedCustomerMaxAge: TimeInterval = 60 + +/// :nodoc: +@_spi(STP) extension STPCustomerContext: STPAnalyticsProtocol { + @_spi(STP) public static var stp_analyticsIdentifier = "STPCustomerContext" +} diff --git a/Stripe/StripeiOS/Source/STPEphemeralKey.swift b/Stripe/StripeiOS/Source/STPEphemeralKey.swift new file mode 100644 index 00000000..6ddf10b6 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPEphemeralKey.swift @@ -0,0 +1,104 @@ +// +// STPEphemeralKey.swift +// StripeiOS +// +// Created by Ben Guo on 5/4/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripePayments + +class STPEphemeralKey: NSObject, STPAPIResponseDecodable { + private(set) var stripeID: String + private(set) var created: Date + private(set) var livemode = false + private(set) var secret: String + private(set) var expires: Date + private(set) var customerID: String? + private(set) var issuingCardID: String? + + /// You cannot directly instantiate an `STPEphemeralKey`. You should instead use + /// `decodedObjectFromAPIResponse:` to create a key from a JSON response. + required init( + stripeID: String, + created: Date, + secret: String, + expires: Date + ) { + self.stripeID = stripeID + self.created = created + self.secret = secret + self.expires = expires + super.init() + } + + private(set) var allResponseFields: [AnyHashable: Any] = [:] + + class func decodedObject(fromAPIResponse response: [AnyHashable: Any]?) -> Self? { + guard let response = response else { + return nil + } + let dict = response.stp_dictionaryByRemovingNulls() + + // required fields + guard + let stripeId = dict.stp_string(forKey: "id"), + let created = dict.stp_date(forKey: "created"), + let secret = dict.stp_string(forKey: "secret"), + let expires = dict.stp_date(forKey: "expires"), + let associatedObjects = dict.stp_array(forKey: "associated_objects"), + dict["livemode"] != nil + else { + return nil + } + + var customerID: String? + var issuingCardID: String? + for obj in associatedObjects { + if let obj = obj as? [AnyHashable: Any] { + let type = obj.stp_string(forKey: "type") + if type == "customer" { + customerID = obj.stp_string(forKey: "id") + } + if type == "issuing.card" { + issuingCardID = obj.stp_string(forKey: "id") + } + } + } + if customerID == nil && issuingCardID == nil { + return nil + } + let key = self.init(stripeID: stripeId, created: created, secret: secret, expires: expires) + key.customerID = customerID + key.issuingCardID = issuingCardID + key.stripeID = stripeId + key.livemode = dict.stp_bool(forKey: "livemode", or: true) + key.created = created + key.secret = secret + key.expires = expires + key.allResponseFields = response + return key + } + + override var hash: Int { + return stripeID.hash + } + + override func isEqual(_ object: Any?) -> Bool { + if self === (object as? STPEphemeralKey) { + return true + } + if object == nil || !(object is STPEphemeralKey) { + return false + } + if let object = object as? STPEphemeralKey { + return isEqual(to: object) + } + return false + } + + func isEqual(to other: STPEphemeralKey) -> Bool { + return stripeID == other.stripeID + } +} diff --git a/Stripe/StripeiOS/Source/STPEphemeralKeyManager.swift b/Stripe/StripeiOS/Source/STPEphemeralKeyManager.swift new file mode 100644 index 00000000..b9955c98 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPEphemeralKeyManager.swift @@ -0,0 +1,179 @@ +// +// STPEphemeralKeyManager.swift +// StripeiOS +// +// Created by Ben Guo on 5/9/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +import UIKit + +protocol STPEphemeralKeyManagerProtocol { + /// If the retriever's stored ephemeral key has not expired, it will be + /// returned immediately to the given callback. If the stored key is expiring, a + /// new key will be requested from the key provider, and returned to the callback. + /// If the retriever is unable to provide an unexpired key, an error will be returned. + /// - Parameter completion: The callback to be run with the returned key, or an error. + func getOrCreateKey(_ completion: @escaping STPEphemeralKeyCompletionBlock) +} + +typealias STPEphemeralKeyCompletionBlock = (STPEphemeralKey?, Error?) -> Void +class STPEphemeralKeyManager: NSObject, STPEphemeralKeyManagerProtocol { + private var _expirationInterval: TimeInterval = 0.0 + /// If the current ephemeral key expires in less than this time interval, a call + /// to `getOrCreateKey` will request a new key from the manager's key provider. + /// The maximum allowed value is one hour – higher values will be clamped. + var expirationInterval: TimeInterval { + get { + _expirationInterval + } + set(expirationInterval) { + _expirationInterval = TimeInterval(min(expirationInterval, 60 * 60)) + } + } + /// If this value is YES, the manager will eagerly refresh its key on app foregrounding. + private(set) var performsEagerFetching = false + + /// Initializes a new `STPEphemeralKeyManager` with the specified key provider. + /// - Parameters: + /// - keyProvider: The key provider the manager will use. + /// - apiVersion: The Stripe API version the manager will use. + /// - performsEagerFetching: If the manager should eagerly refresh its key on app foregrounding. + /// - Returns: the newly-initiated `STPEphemeralKeyManager`. + @objc init( + keyProvider: Any?, + apiVersion: String, + performsEagerFetching: Bool + ) { + super.init() + assert( + keyProvider is STPCustomerEphemeralKeyProvider + || keyProvider is STPIssuingCardEphemeralKeyProvider, + "Your STPEphemeralKeyProvider must either implement `STPCustomerEphemeralKeyProvider` or `STPIssuingCardEphemeralKeyProvider`." + ) + expirationInterval = DefaultExpirationInterval + self.keyProvider = keyProvider + self.apiVersion = apiVersion + self.performsEagerFetching = performsEagerFetching + NotificationCenter.default.addObserver( + self, + selector: #selector(handleWillForegroundNotification), + name: UIApplication.willEnterForegroundNotification, + object: nil + ) + } + + @objc dynamic func getOrCreateKey(_ completion: @escaping STPEphemeralKeyCompletionBlock) { + if currentKeyIsUnexpired() { + completion(ephemeralKey, nil) + } else { + if let createKeyPromise = createKeyPromise { + // coalesce repeated calls into one request + createKeyPromise.onSuccess({ key in + completion(key, nil) + }).onFailure({ error in + completion(nil, error) + }) + } else { + createKeyPromise = STPPromise.init().onSuccess({ key in + self.ephemeralKey = key + completion(key, nil) + }).onFailure({ error in + completion(nil, error) + }) + _createKey() + } + } + } + + @objc internal var ephemeralKey: STPEphemeralKey? + private var apiVersion: String? + private var keyProvider: Any? + @objc internal var lastEagerKeyRefresh: Date? + private var createKeyPromise: STPPromise? + + deinit { + NotificationCenter.default.removeObserver( + self, + name: UIApplication.willEnterForegroundNotification, + object: nil + ) + } + + func currentKeyIsUnexpired() -> Bool { + return ephemeralKey != nil + && (ephemeralKey?.expires.timeIntervalSinceNow ?? 0.0) > expirationInterval + } + + func shouldPerformEagerRefresh() -> Bool { + return performsEagerFetching + && (lastEagerKeyRefresh == nil + || (lastEagerKeyRefresh?.timeIntervalSinceNow ?? 0.0) > MinEagerRefreshInterval) + } + + @objc func handleWillForegroundNotification() { + // To make sure we don't end up hitting the ephemeral keys endpoint on every + // foreground (e.g. if there's an issue decoding the ephemeral key), throttle + // eager refreshes to once per hour. + if !currentKeyIsUnexpired() && shouldPerformEagerRefresh() { + lastEagerKeyRefresh = Date() + getOrCreateKey({ _, _ in + // getOrCreateKey sets the self.ephemeralKey. Nothing left to do for us here + }) + } + } + + func _createKey() { + let jsonCompletion = + { (jsonResponse: [AnyHashable: Any]?, error: Error?) in + let key = STPEphemeralKey.decodedObject(fromAPIResponse: jsonResponse) + if let key = key { + self.createKeyPromise?.succeed(key) + } else { + // the API request failed + if let error = error { + self.createKeyPromise?.fail(error) + } else { + // the ephemeral key could not be decoded + self.createKeyPromise?.fail(NSError.stp_ephemeralKeyDecodingError()) + if self.keyProvider is STPCustomerEphemeralKeyProvider { + assert( + false, + "Could not parse the ephemeral key response following protocol STPCustomerEphemeralKeyProvider. Make sure your backend is sending the unmodified JSON of the ephemeral key to your app. For more info, see https://stripe.com/docs/mobile/ios/basic#prepare-your-api" + ) + } else if self.keyProvider is STPIssuingCardEphemeralKeyProvider { + assert( + false, + "Could not parse the ephemeral key response following protocol STPIssuingCardEphemeralKeyProvider. Make sure your backend is sending the unmodified JSON of the ephemeral key to your app. For more info, see https://stripe.com/docs/mobile/ios/basic#prepare-your-api" + ) + } + assert( + false, + "Could not parse the ephemeral key response. Make sure your backend is sending the unmodified JSON of the ephemeral key to your app. For more info, see https://stripe.com/docs/mobile/ios/basic#prepare-your-api" + ) + } + } + self.createKeyPromise = nil + } as STPJSONResponseCompletionBlock + + if keyProvider is STPCustomerEphemeralKeyProvider { + weak var provider = keyProvider as? STPCustomerEphemeralKeyProvider + provider?.createCustomerKey( + withAPIVersion: apiVersion ?? "", + completion: jsonCompletion + ) + } else if keyProvider is STPIssuingCardEphemeralKeyProvider { + weak var provider = keyProvider as? STPIssuingCardEphemeralKeyProvider + provider?.createIssuingCardKey( + withAPIVersion: apiVersion ?? "", + completion: jsonCompletion + ) + } + } +} + +private let DefaultExpirationInterval: TimeInterval = 60 +private let MinEagerRefreshInterval: TimeInterval = 60 * 60 diff --git a/Stripe/StripeiOS/Source/STPEphemeralKeyProvider.swift b/Stripe/StripeiOS/Source/STPEphemeralKeyProvider.swift new file mode 100644 index 00000000..9bac4bb2 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPEphemeralKeyProvider.swift @@ -0,0 +1,74 @@ +// +// STPEphemeralKeyProvider.swift +// StripeiOS +// +// Created by Ben Guo on 5/9/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import Foundation + +/// You should make your application's API client conform to this interface. +/// It provides a way for Stripe utility classes to request a new ephemeral key from +/// your backend, which it will use to retrieve and update Stripe API objects. +@objc public protocol STPCustomerEphemeralKeyProvider: NSObjectProtocol { + /// Creates a new ephemeral key for retrieving and updating a Stripe customer. + /// On your backend, you should create a new ephemeral key for the Stripe customer + /// associated with your user, and return the raw JSON response from the Stripe API. + /// For an example Ruby implementation of this API, refer to our example backend: + /// https://github.com/stripe/example-mobile-backend/blob/v18.1.0/web.rb + /// Back in your iOS app, once you have a response from this API, call the provided + /// completion block with the JSON response, or an error if one occurred. + /// - Parameters: + /// - apiVersion: The Stripe API version to use when creating a key. + /// You should pass this parameter to your backend, and use it to set the API version + /// in your key creation request. Passing this version parameter ensures that the + /// Stripe SDK can always parse the ephemeral key response from your server. + /// - completion: Call this callback when you're done fetching a new ephemeral + /// key from your backend. For example, `completion(json, nil)` (if your call succeeds) + /// or `completion(nil, error)` if an error is returned. + @objc(createCustomerKeyWithAPIVersion:completion:) func createCustomerKey( + withAPIVersion apiVersion: String, + completion: @escaping STPJSONResponseCompletionBlock + ) +} + +/// You should make your application's API client conform to this interface. +/// It provides a way for Stripe utility classes to request a new ephemeral key from +/// your backend, which it will use to retrieve and update Stripe API objects. +@objc public protocol STPIssuingCardEphemeralKeyProvider: NSObjectProtocol { + /// Creates a new ephemeral key for retrieving and updating a Stripe Issuing Card. + /// On your backend, you should create a new ephemeral key for your logged-in user's + /// primary Issuing Card, and return the raw JSON response from the Stripe API. + /// For an example Ruby implementation of this API, refer to our example backend: + /// https://github.com/stripe/example-mobile-backend/blob/v18.1.0/web.rb + /// Back in your iOS app, once you have a response from this API, call the provided + /// completion block with the JSON response, or an error if one occurred. + /// - Parameters: + /// - apiVersion: The Stripe API version to use when creating a key. + /// You should pass this parameter to your backend, and use it to set the API version + /// in your key creation request. Passing this version parameter ensures that the + /// Stripe SDK can always parse the ephemeral key response from your server. + /// - completion: Call this callback when you're done fetching a new ephemeral + /// key from your backend. For example, `completion(json, nil)` (if your call succeeds) + /// or `completion(nil, error)` if an error is returned. + @objc(createIssuingCardKeyWithAPIVersion:completion:) func createIssuingCardKey( + withAPIVersion apiVersion: String, + completion: @escaping STPJSONResponseCompletionBlock + ) +} + +/// You should make your application's API client conform to this interface. +/// It provides a way for Stripe utility classes to request a new ephemeral key from +/// your backend, which it will use to retrieve and update Stripe API objects. +/// @deprecated use `STPCustomerEphemeralKeyProvider` or `STPIssuingCardEphemeralKeyProvider` +/// depending on the type of key that will@objc be fetched. + +@available( + *, + deprecated, + message: + "use `STPCustomerEphemeralKeyProvider` or `STPIssuingCardEphemeralKeyProvider` depending on the type of key that will be fetched." +) +@objc public protocol STPEphemeralKeyProvider: STPCustomerEphemeralKeyProvider { +} diff --git a/Stripe/StripeiOS/Source/STPFPXBankStatusResponse.swift b/Stripe/StripeiOS/Source/STPFPXBankStatusResponse.swift new file mode 100644 index 00000000..73a2ff66 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPFPXBankStatusResponse.swift @@ -0,0 +1,44 @@ +// +// STPFPXBankStatusResponse.swift +// StripeiOS +// +// Created by David Estes on 10/21/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripePayments + +class STPFPXBankStatusResponse: NSObject, STPAPIResponseDecodable { + func bankBrandIsOnline(_ bankBrand: STPFPXBankBrand) -> Bool { + let bankCode = STPFPXBank.bankCodeFrom(bankBrand, false) + let bankStatus = bankList?[bankCode ?? ""] + if bankCode != nil && bankStatus != nil { + return bankStatus?.boolValue ?? false + } + // This status endpoint isn't reliable. If we don't know this bank's status, default to online. + // The worst that will happen here is that the user ends up at their bank's "Down For Maintenance" page when checking out. + return true + } + + private var bankList: [String: NSNumber]? + private(set) var allResponseFields: [AnyHashable: Any] = [:] + + required internal override init() { + super.init() + } + + class func decodedObject(fromAPIResponse response: [AnyHashable: Any]?) -> Self? { + guard let response = response else { + return nil + } + let dict = response.stp_dictionaryByRemovingNulls() + + let statusResponse = self.init() + statusResponse.bankList = + dict.stp_dictionary(forKey: "parsed_bank_status") as? [String: NSNumber] + statusResponse.allResponseFields = dict + + return statusResponse + } +} diff --git a/Stripe/StripeiOS/Source/STPFakeAddPaymentPassViewController.swift b/Stripe/StripeiOS/Source/STPFakeAddPaymentPassViewController.swift new file mode 100644 index 00000000..a26561de --- /dev/null +++ b/Stripe/StripeiOS/Source/STPFakeAddPaymentPassViewController.swift @@ -0,0 +1,287 @@ +// +// STPFakeAddPaymentPassViewController.swift +// StripeiOS +// +// Created by Jack Flintermann on 9/28/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +import PassKit +@_spi(STP) import StripeCore +import UIKit + +/// This class is a piece of fake UI that is intended to mimic `PKAddPaymentPassViewController`. That class is restricted to apps with a special entitlement from Apple, and as such can be difficult to build and test against. This class implements the same public API as `PKAddPaymentPassViewController`, and can be used to develop against the Stripe API in *testmode only*. (Obviously it will not actually place cards into the user's Apple Pay wallet either.) When it's time to go to production, you may simply replace all references to `STPFakeAddPaymentPassViewController` in your app with `PKAddPaymentPassViewController` and it will continue to function. For more information on developing against this API, please see https://stripe.com/docs/issuing/cards/digital-wallets . +public class STPFakeAddPaymentPassViewController: UIViewController { + /// @see PKAddPaymentPassViewController + @objc + public class func canAddPaymentPass() -> Bool { + return true + } + + /// @see PKAddPaymentPassViewController + @objc(initWithRequestConfiguration:delegate:) + public required init?( + requestConfiguration configuration: PKAddPaymentPassRequestConfiguration, + delegate: PKAddPaymentPassViewControllerDelegate? + ) { + super.init(nibName: nil, bundle: nil) + assert(delegate != nil, "Invalid parameter not satisfying: delegate != nil") + state = .initial + self.delegate = delegate + self.configuration = configuration + if configuration.primaryAccountSuffix == nil && configuration.cardholderName == nil { + assert( + false, + "Your PKAddPaymentPassRequestConfiguration must provide either a cardholderName or a primaryAccountSuffix." + ) + } + } + + /// @see PKAddPaymentPassViewController + @objc public weak var delegate: PKAddPaymentPassViewControllerDelegate? + private var configuration: PKAddPaymentPassRequestConfiguration? + + private var _state: STPFakeAddPaymentPassViewControllerState = .initial + private var state: STPFakeAddPaymentPassViewControllerState { + get { + _state + } + set(state) { + _state = state + let cancelItem = UIBarButtonItem( + barButtonSystemItem: .cancel, + target: self, + action: #selector(cancel(_:)) + ) + let nextButton = UIButton(type: .system) + nextButton.addTarget(self, action: #selector(next(_:)), for: .touchUpInside) + var indicatorView: UIActivityIndicatorView? + if #available(iOS 13.0, *) { + indicatorView = UIActivityIndicatorView(style: .medium) + } else { + indicatorView = UIActivityIndicatorView(style: .gray) + } + indicatorView?.startAnimating() + var loadingItem: UIBarButtonItem? + if let indicatorView = indicatorView { + loadingItem = UIBarButtonItem(customView: indicatorView) + } + nextButton.setTitle(STPNonLocalizedString("Next"), for: .normal) + let nextItem = UIBarButtonItem(customView: nextButton) + let doneItem = UIBarButtonItem( + barButtonSystemItem: .done, + target: self, + action: #selector(done(_:)) + ) + + switch state { + case .initial: + contentLabel?.text = STPNonLocalizedString( + "This class simulates the delegate methods that PKAddPaymentPassViewController will call in your app. Press next to continue." + ) + navigationItem.leftBarButtonItem = cancelItem + navigationItem.rightBarButtonItem = nextItem + case .loading: + contentLabel?.text = STPNonLocalizedString("Fetching encrypted card details...") + cancelItem.isEnabled = false + navigationItem.leftBarButtonItem = cancelItem + navigationItem.rightBarButtonItem = loadingItem + case .error: + contentLabel?.text = STPNonLocalizedString("Error: " + (errorText ?? "")) + doneItem.isEnabled = false + navigationItem.leftBarButtonItem = cancelItem + navigationItem.rightBarButtonItem = doneItem + case .success: + contentLabel?.text = STPNonLocalizedString( + "Success! In production, your card would now have been added to your Apple Pay wallet. Your app's success callback will be triggered when the user presses 'Done'." + ) + cancelItem.isEnabled = false + navigationItem.leftBarButtonItem = cancelItem + navigationItem.rightBarButtonItem = doneItem + } + } + } + private var contentLabel: UILabel? + private var errorText: String? + + /// :nodoc: + @objc public convenience override init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + self.init(requestConfiguration: PKAddPaymentPassRequestConfiguration(), delegate: nil)! + } + + /// :nodoc: + @objc public required convenience init?( + coder aDecoder: NSCoder + ) { + self.init(requestConfiguration: PKAddPaymentPassRequestConfiguration(), delegate: nil) + } + + /// :nodoc: + @objc + public override func viewDidLoad() { + super.viewDidLoad() + view.backgroundColor = UIColor.white + + let navBar = UINavigationBar() + view.addSubview(navBar) + navBar.isTranslucent = false + navBar.backgroundColor = UIColor.white + navBar.items = [navigationItem] + navBar.translatesAutoresizingMaskIntoConstraints = false + navBar.leftAnchor.constraint(equalTo: view.leftAnchor).isActive = true + navBar.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true + navBar.topAnchor.constraint(equalTo: view.safeAreaLayoutGuide.topAnchor).isActive = true + + let contentLabel = UILabel(frame: CGRect.zero) + self.contentLabel = contentLabel + contentLabel.textAlignment = .center + contentLabel.textColor = UIColor.black + contentLabel.numberOfLines = 0 + contentLabel.font = UIFont.systemFont(ofSize: 18) + contentLabel.translatesAutoresizingMaskIntoConstraints = false + view.addSubview(contentLabel) + contentLabel.topAnchor.constraint(equalTo: navBar.bottomAnchor).isActive = true + contentLabel.leftAnchor.constraint(equalTo: view.leftAnchor, constant: 10.0).isActive = true + contentLabel.rightAnchor.constraint(equalTo: view.rightAnchor, constant: -10.0).isActive = + true + contentLabel.heightAnchor.constraint(equalToConstant: 150).isActive = true + + var pairs: [AnyHashable] = [] + if configuration?.cardholderName != nil { + pairs.append(["Name", configuration?.cardholderName]) + } + if configuration?.primaryAccountSuffix != nil { + pairs.append([ + "Card Number", + "···· \(configuration?.primaryAccountSuffix ?? "")", + ]) + } + var rows: [AnyHashable] = [] + for pair in pairs { + guard let pair = pair as? [AnyHashable] else { + continue + } + let left = UILabel() + left.text = pair[0] as? String + left.textAlignment = .left + left.font = UIFont.boldSystemFont(ofSize: 16) + let right = UILabel() + right.text = pair[1] as? String + right.textAlignment = .left + right.textColor = UIColor.lightGray + let row = UIStackView(arrangedSubviews: [left, right]) + row.axis = .horizontal + row.distribution = .fillEqually + row.alignment = .fill + row.translatesAutoresizingMaskIntoConstraints = false + rows.append(row) + } + var pairsTable: UIStackView? + if let rows = rows as? [UIView] { + pairsTable = UIStackView(arrangedSubviews: rows) + } + pairsTable?.isLayoutMarginsRelativeArrangement = true + pairsTable?.layoutMargins = UIEdgeInsets(top: 20, left: 20, bottom: 20, right: 20) + pairsTable?.axis = .vertical + pairsTable?.translatesAutoresizingMaskIntoConstraints = false + if let pairsTable = pairsTable { + view.addSubview(pairsTable) + } + + pairsTable?.leftAnchor.constraint(equalTo: view.leftAnchor).isActive = true + pairsTable?.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true + pairsTable?.topAnchor.constraint(equalTo: contentLabel.bottomAnchor).isActive = true + pairsTable?.heightAnchor.constraint(equalToConstant: CGFloat((rows.count * 50))).isActive = + true + state = .initial + } + + @objc func cancel(_ sender: Any?) { + let castedVC = unsafeBitCast(self, to: PKAddPaymentPassViewController.self) + delegate?.addPaymentPassViewController( + castedVC, + didFinishAdding: nil, + error: NSError( + domain: PKPassKitErrorDomain, + code: PKAddPaymentPassError.userCancelled.rawValue, + userInfo: nil + ) + ) + } + + @objc func next(_ sender: Any?) { + state = .loading + let certificates = [ + "cert1".data(using: .utf8), + "cert2".data(using: .utf8), + ] + let nonce = "nonce".data(using: .utf8) + let nonceSignature = "nonceSignature".data(using: .utf8) + + DispatchQueue.main.asyncAfter( + deadline: DispatchTime.now() + Double(Int64(10 * Double(NSEC_PER_SEC))) + / Double(NSEC_PER_SEC), + execute: { + if self.state == .loading { + self.errorText = + "You exceeded the timeout of 10 seconds to call the request completion handler. Please check your PKAddPaymentPassViewControllerDelegate implementation, and make sure you are calling the `completionHandler` in `addPaymentPassViewController:generateRequestWithCertificateChain:nonce:nonceSignature:completionHandler`." + self.state = .error + } + } + ) + if let nonce = nonce, let nonceSignature = nonceSignature { + let castedVC = unsafeBitCast(self, to: PKAddPaymentPassViewController.self) + delegate?.addPaymentPassViewController( + castedVC, + generateRequestWithCertificateChain: certificates.compactMap { $0 }, + nonce: nonce, + nonceSignature: nonceSignature, + completionHandler: { request in + if self.state == .loading { + var contents: String? + if request.encryptedPassData != nil { + if let encryptedPassData1 = request.encryptedPassData { + contents = String(data: encryptedPassData1, encoding: .utf8) + } + } + if request.stp_error != nil { + var error = + (request.stp_error as NSError?)?.userInfo[STPError.errorMessageKey] + as? String + if error == nil { + error = + (request.stp_error as NSError?)?.userInfo[ + NSLocalizedDescriptionKey + ] as? String + } + self.errorText = error + self.state = .error + } else if contents == "TESTMODE_CONTENTS" { + self.state = .success + } else { + self.errorText = + "Your server response contained the wrong encrypted card details. Please ensure that you are not modifying the response from the Stripe API in any way, and that your request is in testmode." + self.state = .error + } + } + } + ) + } + } + + @objc func done(_ sender: Any?) { + let pass = PKPaymentPass() + let castedVC = unsafeBitCast(self, to: PKAddPaymentPassViewController.self) + delegate?.addPaymentPassViewController(castedVC, didFinishAdding: pass, error: nil) + } +} + +enum STPFakeAddPaymentPassViewControllerState: Int { + case initial + case loading + case error + case success +} diff --git a/Stripe/StripeiOS/Source/STPImageLibrary.swift b/Stripe/StripeiOS/Source/STPImageLibrary.swift new file mode 100644 index 00000000..dac95afb --- /dev/null +++ b/Stripe/StripeiOS/Source/STPImageLibrary.swift @@ -0,0 +1,94 @@ +// +// STPImageLibrary.swift +// StripeiOS +// +// Created by Jack Flintermann on 6/30/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripePaymentsUI +@_spi(STP) import StripeUICore +import UIKit + +/// This class lets you access card icons used by the Stripe SDK. All icons are 32 x 20 points. +@objc class STPLegacyImageLibrary: NSObject { + + /// This returns the appropriate icon for the specified bank brand. + @objc(brandImageForFPXBankBrand:) public class func fpxBrandImage( + for brand: STPFPXBankBrand + ) + -> UIImage + { + let imageName = "stp_bank_fpx_\(STPFPXBank.identifierFrom(brand) ?? "")" + let image = self.safeImageNamed( + imageName, + templateIfAvailable: false + ) + return image + } + + /// An icon representing FPX. + @objc + public class func fpxLogo() -> UIImage { + return self.safeImageNamed("stp_fpx_logo", templateIfAvailable: false) + } + + /// A large branding image for FPX. + @objc + public class func largeFpxLogo() -> UIImage { + return self.safeImageNamed("stp_fpx_big_logo", templateIfAvailable: false) + } + + @objc class func addIcon() -> UIImage { + return self.safeImageNamed("stp_icon_add", templateIfAvailable: true) + } + + @objc class func checkmarkIcon() -> UIImage { + return self.safeImageNamed("stp_icon_checkmark", templateIfAvailable: true) + } + + @objc class func largeCardFrontImage() -> UIImage { + return self.safeImageNamed("stp_card_form_front", templateIfAvailable: true) + } + + @objc class func largeCardBackImage() -> UIImage { + return self.safeImageNamed("stp_card_form_back", templateIfAvailable: true) + } + + @objc class func largeCardAmexCVCImage() -> UIImage { + return self.safeImageNamed("stp_card_form_amex_cvc", templateIfAvailable: true) + } + + @objc class func largeShippingImage() -> UIImage { + return self.safeImageNamed("stp_shipping_form", templateIfAvailable: true) + } + + class func image( + withTintColor color: UIColor, + for image: UIImage + ) -> UIImage { + var newImage: UIImage? + UIGraphicsBeginImageContextWithOptions(image.size, false, image.scale) + color.set() + let templateImage = image.withRenderingMode(.alwaysTemplate) + templateImage.draw( + in: CGRect( + x: 0, + y: 0, + width: templateImage.size.width, + height: templateImage.size.height + ) + ) + newImage = UIGraphicsGetImageFromCurrentImageContext() + UIGraphicsEndImageContext() + return newImage ?? image + } +} + +// MARK: - ImageMaker + +// :nodoc: +@_spi(STP) extension STPLegacyImageLibrary: ImageMaker { + @_spi(STP) public typealias BundleLocator = StripeBundleLocator +} diff --git a/Stripe/StripeiOS/Source/STPIntentActionLinkAuthenticateAccount.swift b/Stripe/StripeiOS/Source/STPIntentActionLinkAuthenticateAccount.swift new file mode 100644 index 00000000..be029dbe --- /dev/null +++ b/Stripe/StripeiOS/Source/STPIntentActionLinkAuthenticateAccount.swift @@ -0,0 +1,33 @@ +// +// STPIntentActionLinkAuthenticateAccount.swift +// StripeiOS +// +// Created by Cameron Sabol on 7/6/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import UIKit + +class STPIntentActionLinkAuthenticateAccount: NSObject { + + let allResponseFields: [AnyHashable: Any] + + required init( + _ allResponseFields: [AnyHashable: Any] + ) { + self.allResponseFields = allResponseFields + super.init() + } + +} + +/// :nodoc: +extension STPIntentActionLinkAuthenticateAccount: STPAPIResponseDecodable { + class func decodedObject(fromAPIResponse response: [AnyHashable: Any]?) -> Self? { + guard let response = response else { + return nil + } + + return STPIntentActionLinkAuthenticateAccount(response) as? Self + } +} diff --git a/Stripe/StripeiOS/Source/STPLocalizedString.swift b/Stripe/StripeiOS/Source/STPLocalizedString.swift new file mode 100644 index 00000000..76afd007 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPLocalizedString.swift @@ -0,0 +1,16 @@ +// +// STPLocalizedString.swift +// StripeiOS +// +// Created by David Estes on 10/19/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore + +@inline(__always) func STPLocalizedString(_ key: String, _ comment: String?) -> String { + return STPLocalizationUtils.localizedStripeString( + forKey: key, + bundleLocator: StripeBundleLocator.self + ) +} diff --git a/Stripe/StripeiOS/Source/STPPaymentActivityIndicatorView.swift b/Stripe/StripeiOS/Source/STPPaymentActivityIndicatorView.swift new file mode 100644 index 00000000..68c03b6e --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentActivityIndicatorView.swift @@ -0,0 +1,135 @@ +// +// STPPaymentActivityIndicatorView.swift +// StripeiOS +// +// Created by Jack Flintermann on 5/12/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +/// This class can be used wherever you'd use a `UIActivityIndicatorView` and is intended to have a similar API. It renders as a spinning circle with a gap in it, similar to what you see in the App Store app or in the Apple Pay dialog when making a purchase. To change its color, set the `tintColor` property. +public class STPPaymentActivityIndicatorView: UIView { + /// Tell the view to start or stop spinning. If `hidesWhenStopped` is true, it will fade in/out if animated is true. + @objc + public func setAnimating( + _ animating: Bool, + animated: Bool + ) { + if animating == _animating { + return + } + _animating = animating + if animating { + if hidesWhenStopped { + UIView.animate( + withDuration: animated ? 0.2 : 0, + animations: { + self.alpha = 1.0 + } + ) + } + var currentRotation = Double(0) + if let currentLayer = layer.presentation() { + currentRotation = Double( + truncating: (currentLayer.value(forKeyPath: "transform.rotation.z") as! NSNumber) + ) + } + let animation = CABasicAnimation(keyPath: "transform.rotation.z") + animation.fromValue = NSNumber(value: Float(currentRotation)) + let toValue = NSNumber(value: currentRotation + 2 * Double.pi) + animation.toValue = toValue + animation.duration = 1.0 + animation.repeatCount = Float.infinity + layer.add(animation, forKey: "rotation") + } else { + if hidesWhenStopped { + UIView.animate( + withDuration: animated ? 0.2 : 0, + animations: { + self.alpha = 0.0 + } + ) + } + } + } + + private var _animating = false + /// Whether or not the view is animating. + @objc public var animating: Bool { + get { + _animating + } + set(animating) { + setAnimating(animating, animated: false) + } + } + + private var _hidesWhenStopped = true + /// If true, the view will hide when it is not spinning. Default is true. + @objc public var hidesWhenStopped: Bool { + get { + _hidesWhenStopped + } + set(hidesWhenStopped) { + _hidesWhenStopped = hidesWhenStopped + if !animating && hidesWhenStopped { + alpha = 0 + } else { + alpha = 1 + } + } + } + private weak var indicatorLayer: CAShapeLayer? + + /// :nodoc: + @objc override init( + frame: CGRect + ) { + var initialFrame = frame + if initialFrame.isEmpty { + initialFrame = CGRect(x: frame.origin.x, y: frame.origin.y, width: 40, height: 40) + } + super.init(frame: initialFrame) + backgroundColor = UIColor.clear + let layer = CAShapeLayer() + layer.backgroundColor = UIColor.clear.cgColor + layer.fillColor = UIColor.clear.cgColor + layer.strokeColor = tintColor.cgColor + layer.strokeStart = 0 + layer.lineCap = .round + layer.strokeEnd = 0.75 + layer.lineWidth = 2.0 + indicatorLayer = layer + self.layer.addSublayer(layer) + alpha = 0 + } + + /// :nodoc: + @objc public override var tintColor: UIColor! { + get { + return super.tintColor + } + set(tintColor) { + super.tintColor = tintColor + indicatorLayer?.strokeColor = tintColor.cgColor + } + } + + /// :nodoc: + @objc + public override func layoutSubviews() { + super.layoutSubviews() + var bounds = self.bounds + bounds.size.width = CGFloat(min(bounds.size.width, bounds.size.height)) + bounds.size.height = bounds.size.width + let path = UIBezierPath(ovalIn: bounds) + indicatorLayer?.path = path.cgPath + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentCardTextFieldCell.swift b/Stripe/StripeiOS/Source/STPPaymentCardTextFieldCell.swift new file mode 100644 index 00000000..40112d5d --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentCardTextFieldCell.swift @@ -0,0 +1,91 @@ +// +// STPPaymentCardTextFieldCell.swift +// StripeiOS +// +// Created by Jack Flintermann on 6/16/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripePaymentsUI +import UIKit + +class STPPaymentCardTextFieldCell: UITableViewCell { + private(set) weak var paymentField: STPPaymentCardTextField? + + var theme: STPTheme = STPTheme.defaultTheme { + didSet { + updateAppearance() + } + } + + private var _inputAccessoryView: UIView? + override var inputAccessoryView: UIView? { + get { + _inputAccessoryView + } + set(inputAccessoryView) { + _inputAccessoryView = inputAccessoryView + paymentField?.inputAccessoryView = inputAccessoryView + } + } + + func isEmpty() -> Bool { + return (paymentField?.cardNumber?.count ?? 0) == 0 + } + + override init( + style: UITableViewCell.CellStyle, + reuseIdentifier: String? + ) { + super.init(style: style, reuseIdentifier: reuseIdentifier) + let paymentField = STPPaymentCardTextField(frame: bounds) + paymentField.postalCodeEntryEnabled = false + contentView.addSubview(paymentField) + self.paymentField = paymentField + theme = STPTheme.defaultTheme + updateAppearance() + } + + override func layoutSubviews() { + super.layoutSubviews() + paymentField?.frame = contentView.bounds + } + + @objc func updateAppearance() { + paymentField?.backgroundColor = UIColor.clear + paymentField?.placeholderColor = theme.tertiaryForegroundColor + paymentField?.borderColor = UIColor.clear + paymentField?.textColor = theme.primaryForegroundColor + paymentField?.textErrorColor = theme.errorColor + paymentField?.font = theme.font + backgroundColor = theme.secondaryBackgroundColor + } + + @objc override func becomeFirstResponder() -> Bool { + return paymentField?.becomeFirstResponder() ?? false + } + + override func accessibilityElementCount() -> Int { + return paymentField?.allFields.count ?? 0 + } + + override func accessibilityElement(at index: Int) -> Any? { + return paymentField?.allFields[index] + } + + override func index(ofAccessibilityElement element: Any) -> Int { + let fields = paymentField?.allFields + for i in 0..<(fields?.count ?? 0) { + if (element as? AnyHashable) == fields?[i] { + return i + } + } + return 0 + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentConfiguration.swift b/Stripe/StripeiOS/Source/STPPaymentConfiguration.swift new file mode 100644 index 00000000..12719a49 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentConfiguration.swift @@ -0,0 +1,244 @@ +// +// STPPaymentConfiguration.swift +// StripeiOS +// +// Created by Jack Flintermann on 5/18/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +/// An `STPPaymentConfiguration` represents all the options you can set or change +/// around a payment. +/// You provide an `STPPaymentConfiguration` object to your `STPPaymentContext` +/// when making a charge. The configuration generally has settings that +/// will not change from payment to payment and thus is reusable, while the context +/// is specific to a single particular payment instance. +public class STPPaymentConfiguration: NSObject, NSCopying { + /// This is a convenience singleton configuration that uses the default values for + /// every property + @objc(sharedConfiguration) public static var shared = STPPaymentConfiguration() + + private var _applePayEnabled = true + /// The user is allowed to pay with Apple Pay if it's configured and available on their device. + @objc public var applePayEnabled: Bool { + get { + return appleMerchantIdentifier != nil && _applePayEnabled + && StripeAPI.deviceSupportsApplePay() + } + set { + _applePayEnabled = newValue + } + } + + /// The user is allowed to pay with FPX. + @objc public var fpxEnabled = false + + /// The billing address fields the user must fill out when prompted for their + /// payment details. These fields will all be present on the returned PaymentMethod from + /// Stripe. + /// The default value is `STPBillingAddressFieldsPostalCode`. + /// - seealso: https://stripe.com/docs/api/payment_methods/create#create_payment_method-billing_details + @objc public var requiredBillingAddressFields = STPBillingAddressFields.postalCode + /// The shipping address fields the user must fill out when prompted for their + /// shipping info. Set to nil if shipping address is not required. + /// The default value is nil. + @objc public var requiredShippingAddressFields: Set? + /// Whether the user should be prompted to verify prefilled shipping information. + /// The default value is YES. + @objc public var verifyPrefilledShippingAddress = true + /// The type of shipping for this purchase. This property sets the labels displayed + /// when the user is prompted for shipping info, and whether they should also be + /// asked to select a shipping method. + /// The default value is STPShippingTypeShipping. + @objc public var shippingType = STPShippingType.shipping + /// The set of countries supported when entering an address. This property accepts + /// a set of ISO 2-character country codes. + /// The default value is all known countries. Setting this property will limit + /// the available countries to your selected set. + @objc public var availableCountries: Set = Set(NSLocale.isoCountryCodes) + + /// The name of your company, for displaying to the user during payment flows. For + /// example, when using Apple Pay, the payment sheet's final line item will read + /// "PAY {companyName}". + /// The default value is the name of your iOS application which is derived from the + /// `kCFBundleNameKey` of `Bundle.main`. + @objc public var companyName = Bundle.stp_applicationName() ?? "" + /// The Apple Merchant Identifier to use during Apple Pay transactions. To create + /// one of these, see our guide at https://stripe.com/docs/apple-pay . You + /// must set this to a valid identifier in order to automatically enable Apple Pay. + @objc public var appleMerchantIdentifier: String? + /// Determines whether or not the user is able to delete payment options + /// This is only relevant to the `STPPaymentOptionsViewController` which, if + /// enabled, will allow the user to delete payment options by tapping the "Edit" + /// button in the navigation bar or by swiping left on a payment option and tapping + /// "Delete". Currently, the user is not allowed to delete the selected payment + /// option but this may change in the future. + /// Default value is YES but will only work if `STPPaymentOptionsViewController` is + /// initialized with a `STPCustomerContext` either through the `STPPaymentContext` + /// or directly as an init parameter. + @objc public var canDeletePaymentOptions = true + /// Determines whether STPAddCardViewController allows the user to + /// scan cards using the camera on devices running iOS 13 or later. + /// To use this feature, you must also set the `NSCameraUsageDescription` + /// value in your app's Info.plist. + /// @note This feature is currently in beta. Please file bugs at + /// https://github.com/stripe/stripe-ios/issues + /// The default value is currently NO. This will be changed in a future update. + @objc public var cardScanningEnabled = false + // MARK: - Deprecated + + /// An enum value representing which payment options you will accept from your user + /// in addition to credit cards. + @available( + *, + deprecated, + message: + "additionalPaymentOptions has been removed. Set applePayEnabled and fpxEnabled on STPPaymentConfiguration instead." + ) + @objc public var additionalPaymentOptions: Int = 0 + + private var _publishableKey: String? + /// If you used STPPaymentConfiguration.shared.publishableKey, use STPAPIClient.shared.publishableKey instead. The SDK uses STPAPIClient.shared to make API requests by default. + /// Your Stripe publishable key + /// - seealso: https://dashboard.stripe.com/account/apikeys + @available( + *, + deprecated, + message: + "If you used STPPaymentConfiguration.shared.publishableKey, use STPAPIClient.shared.publishableKey instead. If you passed a STPPaymentConfiguration instance to an SDK component, create an STPAPIClient, set publishableKey on it, and set the SDK component's APIClient property." + ) + @objc public var publishableKey: String? { + get { + if self == STPPaymentConfiguration.shared { + return STPAPIClient.shared.publishableKey + } + return _publishableKey ?? "" + } + set(publishableKey) { + if self == STPPaymentConfiguration.shared { + STPAPIClient.shared.publishableKey = publishableKey + } else { + _publishableKey = publishableKey + } + } + } + + private var _stripeAccount: String? + /// If you used STPPaymentConfiguration.shared.stripeAccount, use STPAPIClient.shared.stripeAccount instead. The SDK uses STPAPIClient.shared to make API requests by default. + /// In order to perform API requests on behalf of a connected account, e.g. to + /// create charges for a connected account, set this property to the ID of the + /// account for which this request is being made. + /// - seealso: https://stripe.com/docs/payments#connected-accounts + @available( + *, + deprecated, + message: + "If you used STPPaymentConfiguration.shared.stripeAccount, use STPAPIClient.shared.stripeAccount instead. If you passed a STPPaymentConfiguration instance to an SDK component, create an STPAPIClient, set stripeAccount on it, and set the SDK component's APIClient property." + ) + @objc public var stripeAccount: String? { + get { + if self == STPPaymentConfiguration.shared { + return STPAPIClient.shared.stripeAccount + } + return _stripeAccount ?? "" + } + set(stripeAccount) { + if self == STPPaymentConfiguration.shared { + STPAPIClient.shared.stripeAccount = stripeAccount + } else { + _stripeAccount = stripeAccount + } + } + } + + // MARK: - Description + /// :nodoc: + @objc public override var description: String { + var additionalPaymentOptionsDescription: String? + + var paymentOptions: [String] = [] + + if _applePayEnabled { + paymentOptions.append("STPPaymentOptionTypeApplePay") + } + + if fpxEnabled { + paymentOptions.append("STPPaymentOptionTypeFPX") + } + + additionalPaymentOptionsDescription = paymentOptions.joined(separator: "|") + + var requiredBillingAddressFieldsDescription: String? + + switch requiredBillingAddressFields { + case .none: + requiredBillingAddressFieldsDescription = "STPBillingAddressFieldsNone" + case .postalCode: + requiredBillingAddressFieldsDescription = "STPBillingAddressFieldsPostalCode" + case .full: + requiredBillingAddressFieldsDescription = "STPBillingAddressFieldsFull" + case .name: + requiredBillingAddressFieldsDescription = "STPBillingAddressFieldsName" + default: + break + } + + let requiredShippingAddressFieldsDescription = requiredShippingAddressFields?.map({ + $0.rawValue + }).joined(separator: "|") + + var shippingTypeDescription: String? + + switch shippingType { + case .shipping: + shippingTypeDescription = "STPShippingTypeShipping" + case .delivery: + shippingTypeDescription = "STPShippingTypeDelivery" + default: + break + } + + let props = [ + // Object + String(format: "%@: %p", NSStringFromClass(STPPaymentConfiguration.self), self), + // Basic configuration + "additionalPaymentOptions = \(additionalPaymentOptionsDescription ?? "")", + // Billing and shipping + "requiredBillingAddressFields = \(requiredBillingAddressFieldsDescription ?? "")", + "requiredShippingAddressFields = \(requiredShippingAddressFieldsDescription ?? "")", + "verifyPrefilledShippingAddress = \((verifyPrefilledShippingAddress) ? "YES" : "NO")", + "shippingType = \(shippingTypeDescription ?? "")", + "availableCountries = \(availableCountries )", + // Additional configuration + "companyName = \(companyName )", + "appleMerchantIdentifier = \(appleMerchantIdentifier ?? "")", + "canDeletePaymentOptions = \((canDeletePaymentOptions) ? "YES" : "NO")", + "cardScanningEnabled = \((cardScanningEnabled) ? "YES" : "NO")", + ] + + return "<\(props.joined(separator: "; "))>" + } + + // MARK: - NSCopying + /// :nodoc: + @objc + public func copy(with zone: NSZone? = nil) -> Any { + let copy = STPPaymentConfiguration() + copy.applePayEnabled = _applePayEnabled + copy.fpxEnabled = fpxEnabled + copy.requiredBillingAddressFields = requiredBillingAddressFields + copy.requiredShippingAddressFields = requiredShippingAddressFields + copy.verifyPrefilledShippingAddress = verifyPrefilledShippingAddress + copy.shippingType = shippingType + copy.companyName = companyName + copy.appleMerchantIdentifier = appleMerchantIdentifier + copy.canDeletePaymentOptions = canDeletePaymentOptions + copy.cardScanningEnabled = cardScanningEnabled + copy.availableCountries = availableCountries + copy._publishableKey = _publishableKey + copy._stripeAccount = _stripeAccount + return copy + } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentContext.swift b/Stripe/StripeiOS/Source/STPPaymentContext.swift new file mode 100644 index 00000000..4be7e0bd --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentContext.swift @@ -0,0 +1,1206 @@ +// +// STPPaymentContext.swift +// StripeiOS +// +// Created by Jack Flintermann on 4/20/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation +import ObjectiveC +import PassKit +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI + +/// An `STPPaymentContext` keeps track of all of the state around a payment. It will manage fetching a user's saved payment methods, tracking any information they select, and prompting them for required additional information before completing their purchase. It can be used to power your application's "payment confirmation" page with just a few lines of code. +/// `STPPaymentContext` also provides a unified interface to multiple payment methods - for example, you can write a single integration to accept both credit card payments and Apple Pay. +/// `STPPaymentContext` saves information about a user's payment methods to a Stripe customer object, and requires an `STPCustomerContext` to manage retrieving and modifying the customer. +@objc(STPPaymentContext) +public class STPPaymentContext: NSObject, STPAuthenticationContext, + STPPaymentOptionsViewControllerDelegate, STPShippingAddressViewControllerDelegate +{ + /// This is a convenience initializer; it is equivalent to calling + /// `init(customerContext:customerContext + /// configuration:STPPaymentConfiguration.shared + /// theme:STPTheme.defaultTheme`. + /// - Parameter customerContext: The customer context the payment context will use to fetch + /// and modify its Stripe customer. - seealso: STPCustomerContext.h + /// - Returns: the newly-instantiated payment context + @objc + public convenience init( + customerContext: STPCustomerContext + ) { + self.init(apiAdapter: customerContext) + } + + /// Initializes a new Payment Context with the provided customer context, configuration, + /// and theme. After this class is initialized, you should also make sure to set its + /// `delegate` and `hostViewController` properties. + /// - Parameters: + /// - customerContext: The customer context the payment context will use to fetch + /// and modify its Stripe customer. - seealso: STPCustomerContext.h + /// - configuration: The configuration for the payment context to use. This + /// lets you set your Stripe publishable API key, required billing address fields, etc. + /// - seealso: STPPaymentConfiguration.h + /// - theme: The theme describing the visual appearance of all UI + /// that the payment context automatically creates for you. - seealso: STPTheme.h + /// - Returns: the newly-instantiated payment context + @objc + public convenience init( + customerContext: STPCustomerContext, + configuration: STPPaymentConfiguration, + theme: STPTheme + ) { + self.init( + apiAdapter: customerContext, + configuration: configuration, + theme: theme + ) + } + + /// Note: Instead of providing your own backend API adapter, we recommend using + /// `STPCustomerContext`, which will manage retrieving and updating a + /// Stripe customer for you. - seealso: STPCustomerContext.h + /// This is a convenience initializer; it is equivalent to calling + /// `init(apiAdapter:apiAdapter configuration:STPPaymentConfiguration.shared theme:STPTheme.defaultTheme)`. + @objc + public convenience init( + apiAdapter: STPBackendAPIAdapter + ) { + self.init( + apiAdapter: apiAdapter, + configuration: STPPaymentConfiguration.shared, + theme: STPTheme.defaultTheme + ) + } + + /// Note: Instead of providing your own backend API adapter, we recommend using + /// `STPCustomerContext`, which will manage retrieving and updating a + /// Stripe customer for you. - seealso: STPCustomerContext.h + /// Initializes a new Payment Context with the provided API adapter and configuration. + /// After this class is initialized, you should also make sure to set its `delegate` + /// and `hostViewController` properties. + /// - Parameters: + /// - apiAdapter: The API adapter the payment context will use to fetch and + /// modify its contents. You need to make a class conforming to this protocol that + /// talks to your server. - seealso: STPBackendAPIAdapter.h + /// - configuration: The configuration for the payment context to use. This lets + /// you set your Stripe publishable API key, required billing address fields, etc. + /// - seealso: STPPaymentConfiguration.h + /// - theme: The theme describing the visual appearance of all UI that + /// the payment context automatically creates for you. - seealso: STPTheme.h + /// - Returns: the newly-instantiated payment context + @objc + public init( + apiAdapter: STPBackendAPIAdapter, + configuration: STPPaymentConfiguration, + theme: STPTheme + ) { + STPAnalyticsClient.sharedClient.addClass(toProductUsageIfNecessary: STPPaymentContext.self) + self.configuration = configuration + self.apiAdapter = apiAdapter + self.theme = theme + paymentCurrency = "USD" + paymentCountry = "US" + apiClient = STPAPIClient.shared + modalPresentationStyle = .fullScreen + state = STPPaymentContextState.none + super.init() + retryLoading() + } + + /// Note: Instead of providing your own backend API adapter, we recommend using + /// `STPCustomerContext`, which will manage retrieving and updating a + /// Stripe customer for you. - seealso: STPCustomerContext.h + /// The API adapter the payment context will use to fetch and modify its contents. + /// You need to make a class conforming to this protocol that talks to your server. + /// - seealso: STPBackendAPIAdapter.h + @objc public private(set) var apiAdapter: STPBackendAPIAdapter + /// The configuration for the payment context to use internally. - seealso: STPPaymentConfiguration.h + @objc public private(set) var configuration: STPPaymentConfiguration + /// The visual appearance that will be used by any views that the context generates. - seealso: STPTheme.h + @objc public private(set) var theme: STPTheme + + private var _prefilledInformation: STPUserInformation? + /// If you've already collected some information from your user, you can set it here and it'll be automatically filled out when possible/appropriate in any UI that the payment context creates. + @objc public var prefilledInformation: STPUserInformation? { + get { + _prefilledInformation + } + set(prefilledInformation) { + _prefilledInformation = prefilledInformation + if prefilledInformation?.shippingAddress != nil && shippingAddress == nil { + shippingAddress = prefilledInformation?.shippingAddress + shippingAddressNeedsVerification = true + } + } + } + + private weak var _hostViewController: UIViewController? + /// The view controller that any additional UI will be presented on. If you have a "checkout view controller" in your app, that should be used as the host view controller. + @objc public weak var hostViewController: UIViewController? { + get { + _hostViewController + } + set(hostViewController) { + assert( + _hostViewController == nil, + "You cannot change the hostViewController on an STPPaymentContext after it's already been set." + ) + _hostViewController = hostViewController + if hostViewController is UINavigationController { + originalTopViewController = + (hostViewController as? UINavigationController)?.topViewController + } + if let hostViewController = hostViewController { + artificiallyRetain(hostViewController) + } + } + } + + private weak var _delegate: STPPaymentContextDelegate? + /// This delegate will be notified when the payment context's contents change. - seealso: STPPaymentContextDelegate + @objc public weak var delegate: STPPaymentContextDelegate? { + get { + _delegate + } + set(delegate) { + _delegate = delegate + DispatchQueue.main.async(execute: { + self.delegate?.paymentContextDidChange(self) + }) + } + } + /// Whether or not the payment context is currently loading information from the network. + + @objc public var loading: Bool { + return !(loadingPromise?.completed)! + } + /// @note This is no longer recommended as of v18.3.0 - the SDK automatically saves the Stripe ID of the last selected + /// payment method using NSUserDefaults and displays it as the default pre-selected option. You can override this behavior + /// by setting this property. + /// The Stripe ID of a payment method to display as the default pre-selected option. + /// @note Set this property immediately after initializing STPPaymentContext, or call `retryLoading` afterwards. + @objc public var defaultPaymentMethod: String? + + private var _selectedPaymentOption: STPPaymentOption? + /// The user's currently selected payment option. May be nil. + @objc public private(set) var selectedPaymentOption: STPPaymentOption? { + get { + _selectedPaymentOption + } + set { + if let newValue = newValue, let paymentOptions = self.paymentOptions { + if !paymentOptions.contains(where: { (option) -> Bool in + newValue.isEqual(option) + }) { + if newValue.isReusable { + self.paymentOptions = paymentOptions + [newValue] + } + } + } + if !(_selectedPaymentOption?.isEqual(newValue) ?? false) { + _selectedPaymentOption = newValue + stpDispatchToMainThreadIfNecessary({ + self.delegate?.paymentContextDidChange(self) + }) + } + + } + } + + private var _paymentOptions: [STPPaymentOption]? + /// The available payment options the user can choose between. May be nil. + @objc public private(set) var paymentOptions: [STPPaymentOption]? { + get { + _paymentOptions + } + set { + _paymentOptions = newValue?.sorted(by: { (obj1, obj2) -> Bool in + let applePayKlass = STPApplePayPaymentOption.self + let paymentMethodKlass = STPPaymentMethod.self + if obj1.isKind(of: applePayKlass) { + return true + } else if obj2.isKind(of: applePayKlass) { + return false + } + if obj1.isKind(of: paymentMethodKlass) && obj2.isKind(of: paymentMethodKlass) { + return (obj1.label.compare(obj2.label) == .orderedAscending) + } + return false + }) + } + } + + /// The user's currently selected shipping method. May be nil. + @objc public internal(set) var selectedShippingMethod: PKShippingMethod? + + private var _shippingMethods: [PKShippingMethod]? + /// An array of STPShippingMethod objects that describe the supported shipping methods. May be nil. + @objc public private(set) var shippingMethods: [PKShippingMethod]? { + get { + _shippingMethods + } + set { + _shippingMethods = newValue + if let shippingMethods = newValue, + let selectedShippingMethod = self.selectedShippingMethod + { + if shippingMethods.count == 0 { + self.selectedShippingMethod = nil + } else if shippingMethods.contains(selectedShippingMethod) { + self.selectedShippingMethod = shippingMethods.first + } + } + } + } + + /// The user's shipping address. May be nil. + /// If you've already collected a shipping address from your user, you may + /// prefill it by setting a shippingAddress in PaymentContext's prefilledInformation. + /// When your user enters a new shipping address, PaymentContext will save it to + /// the current customer object. When PaymentContext loads, if you haven't + /// manually set a prefilled value, any shipping information saved on the customer + /// will be used to prefill the shipping address form. Note that because your + /// customer's email may not be the same as the email provided with their shipping + /// info, PaymentContext will not prefill the shipping form's email using your + /// customer's email. + /// You should not rely on the shipping information stored on the Stripe customer + /// for order fulfillment, as your user may change this information if they make + /// multiple purchases. We recommend adding shipping information when you create + /// a charge (which can also help prevent fraud), or saving it to your own + /// database. https://stripe.com/docs/api/payment_intents/create#create_payment_intent-shipping + /// Note: by default, your user will still be prompted to verify a prefilled + /// shipping address. To change this behavior, you can set + /// `verifyPrefilledShippingAddress` to NO in your `STPPaymentConfiguration`. + @objc public private(set) var shippingAddress: STPAddress? + /// The amount of money you're requesting from the user, in the smallest currency + /// unit for the selected currency. For example, to indicate $10 USD, use 1000 + /// (i.e. 1000 cents). For more information, see https://stripe.com/docs/api/payment_intents/create#create_payment_intent-amount + /// @note This value must be present and greater than zero in order for Apple Pay + /// to be automatically enabled. + /// @note You should only set either this or `paymentSummaryItems`, not both. + /// The other will be automatically calculated on demand using your `paymentCurrency`. + + @objc public var paymentAmount: Int { + get { + return paymentAmountModel.paymentAmount( + withCurrency: paymentCurrency, + shippingMethod: selectedShippingMethod + ) + } + set(paymentAmount) { + paymentAmountModel = STPPaymentContextAmountModel(amount: paymentAmount) + } + } + /// The three-letter currency code for the currency of the payment (i.e. USD, GBP, + /// JPY, etc). Defaults to "USD". + /// @note Changing this property may change the return value of `paymentAmount` + /// or `paymentSummaryItems` (whichever one you didn't directly set yourself). + @objc public var paymentCurrency: String + /// The two-letter country code for the country where the payment will be processed. + /// You should set this to the country your Stripe account is in. Defaults to "US". + /// @note Changing this property will change the `countryCode` of your Apple Pay + /// payment requests. + /// - seealso: PKPaymentRequest for more information. + @objc public var paymentCountry: String + /// If you support Apple Pay, you can optionally set the PKPaymentSummaryItems + /// you want to display here instead of using `paymentAmount`. Note that the + /// grand total (the amount of the last summary item) must be greater than zero. + /// If not set, a single summary item will be automatically generated using + /// `paymentAmount` and your configuration's `companyName`. + /// - seealso: PKPaymentRequest for more information + /// @note You should only set either this or `paymentAmount`, not both. + /// The other will be automatically calculated on demand using your `paymentCurrency.` + + @objc public var paymentSummaryItems: [PKPaymentSummaryItem] { + get { + return paymentAmountModel.paymentSummaryItems( + withCurrency: paymentCurrency, + companyName: configuration.companyName, + shippingMethod: selectedShippingMethod + ) ?? [] + } + set(paymentSummaryItems) { + paymentAmountModel = STPPaymentContextAmountModel( + paymentSummaryItems: paymentSummaryItems + ) + } + } + /// The presentation style used for all view controllers presented modally by the context. + /// Since custom transition styles are not supported, you should set this to either + /// `UIModalPresentationFullScreen`, `UIModalPresentationPageSheet`, or `UIModalPresentationFormSheet`. + /// The default value is `UIModalPresentationFullScreen`. + @objc public var modalPresentationStyle: UIModalPresentationStyle = .fullScreen + /// The mode to use when displaying the title of the navigation bar in all view + /// controllers presented by the context. The default value is `automatic`, + /// which causes the title to use the same styling as the previously displayed + /// navigation item (if the view controller is pushed onto the `hostViewController`). + /// If the `prefersLargeTitles` property of the `hostViewController`'s navigation bar + /// is false, this property has no effect and the navigation item's title is always + /// displayed as a small title. + /// If the view controller is presented modally, `automatic` and + /// `never` always result in a navigation bar with a small title. + @objc public var largeTitleDisplayMode = UINavigationItem.LargeTitleDisplayMode.automatic + /// A view that will be placed as the footer of the payment options selection + /// view controller. + /// When the footer view needs to be resized, it will be sent a + /// `sizeThatFits:` call. The view should respond correctly to this method in order + /// to be sized and positioned properly. + @objc public var paymentOptionsViewControllerFooterView: UIView? + /// A view that will be placed as the footer of the add card view controller. + /// When the footer view needs to be resized, it will be sent a + /// `sizeThatFits:` call. The view should respond correctly to this method in order + /// to be sized and positioned properly. + @objc public var addCardViewControllerFooterView: UIView? + /// The API Client to use to make requests. + /// Defaults to STPAPIClient.shared + public var apiClient: STPAPIClient = .shared + + /// If `paymentContext:didFailToLoadWithError:` is called on your delegate, you + /// can in turn call this method to try loading again (if that hasn't been called, + /// calling this will do nothing). If retrying in turn fails, `paymentContext:didFailToLoadWithError:` + /// will be called again (and you can again call this to keep retrying, etc). + @objc + public func retryLoading() { + // Clear any cached customer object and attached payment methods before refetching + if apiAdapter is STPCustomerContext { + let customerContext = apiAdapter as? STPCustomerContext + customerContext?.clearCache() + } + weak var weakSelf = self + loadingPromise = STPPromise.init().onSuccess({ tuple in + guard let strongSelf = weakSelf else { + return + } + strongSelf.paymentOptions = tuple.paymentOptions + strongSelf.selectedPaymentOption = tuple.selectedPaymentOption + }).onFailure({ error in + guard let strongSelf = weakSelf else { + return + } + if strongSelf.hostViewController != nil { + if strongSelf.paymentOptionsViewController != nil + && strongSelf.paymentOptionsViewController?.viewIfLoaded?.window != nil + { + if let paymentOptionsViewController1 = strongSelf.paymentOptionsViewController { + strongSelf.appropriatelyDismiss(paymentOptionsViewController1) { + strongSelf.delegate?.paymentContext( + strongSelf, + didFailToLoadWithError: error + ) + } + } + } else { + strongSelf.delegate?.paymentContext(strongSelf, didFailToLoadWithError: error) + } + } + }) + apiAdapter.retrieveCustomer({ customer, retrieveCustomerError in + stpDispatchToMainThreadIfNecessary({ + guard let strongSelf = weakSelf else { + return + } + if let retrieveCustomerError = retrieveCustomerError { + strongSelf.loadingPromise?.fail(retrieveCustomerError) + return + } + if strongSelf.shippingAddress == nil && customer?.shippingAddress != nil { + strongSelf.shippingAddress = customer?.shippingAddress + strongSelf.shippingAddressNeedsVerification = true + } + + strongSelf.apiAdapter.listPaymentMethodsForCustomer(completion: { + paymentMethods, + error in + guard let strongSelf2 = weakSelf else { + return + } + stpDispatchToMainThreadIfNecessary({ + if let error = error { + strongSelf2.loadingPromise?.fail(error) + return + } + + if self.defaultPaymentMethod == nil + && (strongSelf2.apiAdapter is STPCustomerContext) + { + // Retrieve the last selected payment method saved by STPCustomerContext + (strongSelf2.apiAdapter as? STPCustomerContext)? + .retrieveLastSelectedPaymentMethodIDForCustomer(completion: { + paymentMethodID, + _ in + guard let strongSelf3 = weakSelf else { + return + } + if let paymentMethods = paymentMethods { + let paymentTuple = STPPaymentOptionTuple( + filteredForUIWith: paymentMethods, + selectedPaymentMethod: paymentMethodID, + configuration: strongSelf3.configuration + ) + strongSelf3.loadingPromise?.succeed(paymentTuple) + } else { + strongSelf3.loadingPromise?.fail( + STPErrorCode.invalidRequestError as! Error + ) + } + }) + } else { + if let paymentMethods = paymentMethods { + let paymentTuple = STPPaymentOptionTuple( + filteredForUIWith: paymentMethods, + selectedPaymentMethod: self.defaultPaymentMethod, + configuration: strongSelf2.configuration + ) + strongSelf2.loadingPromise?.succeed(paymentTuple) + } + } + }) + }) + }) + }) + } + + /// This creates, configures, and appropriately presents an `STPPaymentOptionsViewController` + /// on top of the payment context's `hostViewController`. It'll be dismissed automatically + /// when the user is done selecting their payment method. + /// @note This method will do nothing if it is called while STPPaymentContext is + /// already showing a view controller or in the middle of requesting a payment. + @objc + public func presentPaymentOptionsViewController() { + presentPaymentOptionsViewController(withNewState: .showingRequestedViewController) + } + + /// This creates, configures, and appropriately pushes an `STPPaymentOptionsViewController` + /// onto the navigation stack of the context's `hostViewController`. It'll be popped + /// automatically when the user is done selecting their payment method. + /// @note This method will do nothing if it is called while STPPaymentContext is + /// already showing a view controller or in the middle of requesting a payment. + @objc + public func pushPaymentOptionsViewController() { + assert( + hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, + "hostViewController must not be nil on STPPaymentContext when calling pushPaymentOptionsViewController on it. Next time, set the hostViewController property first!" + ) + var navigationController: UINavigationController? + if hostViewController is UINavigationController { + navigationController = hostViewController as? UINavigationController + } else { + navigationController = hostViewController?.navigationController + } + assert( + navigationController != nil, + "The payment context's hostViewController is not a navigation controller, or is not contained in one. Either make sure it is inside a navigation controller before calling pushPaymentOptionsViewController, or call presentPaymentOptionsViewController instead." + ) + if state == STPPaymentContextState.none { + state = .showingRequestedViewController + + let paymentOptionsViewController = STPPaymentOptionsViewController(paymentContext: self) + self.paymentOptionsViewController = paymentOptionsViewController + paymentOptionsViewController.prefilledInformation = prefilledInformation + paymentOptionsViewController.defaultPaymentMethod = defaultPaymentMethod + paymentOptionsViewController.paymentOptionsViewControllerFooterView = + paymentOptionsViewControllerFooterView + paymentOptionsViewController.addCardViewControllerFooterView = + addCardViewControllerFooterView + paymentOptionsViewController.navigationItem.largeTitleDisplayMode = + largeTitleDisplayMode + + navigationController?.pushViewController( + paymentOptionsViewController, + animated: transitionAnimationsEnabled() + ) + } + } + + /// This creates, configures, and appropriately presents a view controller for + /// collecting shipping address and shipping method on top of the payment context's + /// `hostViewController`. It'll be dismissed automatically when the user is done + /// entering their shipping info. + /// @note This method will do nothing if it is called while STPPaymentContext is + /// already showing a view controller or in the middle of requesting a payment. + @objc + public func presentShippingViewController() { + presentShippingViewController(withNewState: .showingRequestedViewController) + } + + /// This creates, configures, and appropriately pushes a view controller for + /// collecting shipping address and shipping method onto the navigation stack of + /// the context's `hostViewController`. It'll be popped automatically when the + /// user is done entering their shipping info. + /// @note This method will do nothing if it is called while STPPaymentContext is + /// already showing a view controller, or in the middle of requesting a payment. + @objc + public func pushShippingViewController() { + assert( + hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, + "hostViewController must not be nil on STPPaymentContext when calling pushShippingViewController on it. Next time, set the hostViewController property first!" + ) + var navigationController: UINavigationController? + if hostViewController is UINavigationController { + navigationController = hostViewController as? UINavigationController + } else { + navigationController = hostViewController?.navigationController + } + assert( + navigationController != nil, + "The payment context's hostViewController is not a navigation controller, or is not contained in one. Either make sure it is inside a navigation controller before calling pushShippingInfoViewController, or call presentShippingInfoViewController instead." + ) + if state == STPPaymentContextState.none { + state = .showingRequestedViewController + + let addressViewController = STPShippingAddressViewController(paymentContext: self) + addressViewController.navigationItem.largeTitleDisplayMode = largeTitleDisplayMode + navigationController?.pushViewController( + addressViewController, + animated: transitionAnimationsEnabled() + ) + } + } + + /// Requests payment from the user. This may need to present some supplemental UI + /// to the user, in which case it will be presented on the payment context's + /// `hostViewController`. For instance, if they've selected Apple Pay as their + /// payment method, calling this method will show the payment sheet. If the user + /// has a card on file, this will use that without presenting any additional UI. + /// After this is called, the `paymentContext:didCreatePaymentResult:completion:` + /// and `paymentContext:didFinishWithStatus:error:` methods will be called on the + /// context's `delegate`. + /// @note This method will do nothing if it is called while STPPaymentContext is + /// already showing a view controller, or in the middle of requesting a payment. + @objc + public func requestPayment() { + weak var weakSelf = self + loadingPromise?.onSuccess({ _ in + guard let strongSelf = weakSelf else { + return + } + + if strongSelf.state != STPPaymentContextState.none { + return + } + + if strongSelf.selectedPaymentOption == nil { + strongSelf.presentPaymentOptionsViewController(withNewState: .requestingPayment) + } else if strongSelf.requestPaymentShouldPresentShippingViewController() { + strongSelf.presentShippingViewController(withNewState: .requestingPayment) + } else if (strongSelf.selectedPaymentOption is STPPaymentMethod) + || (self.selectedPaymentOption is STPPaymentMethodParams) + { + strongSelf.state = .requestingPayment + let result = STPPaymentResult(paymentOption: strongSelf.selectedPaymentOption!) + strongSelf.delegate?.paymentContext(self, didCreatePaymentResult: result) { + status, + error in + stpDispatchToMainThreadIfNecessary({ + strongSelf.didFinish(with: status, error: error) + }) + } + } else if strongSelf.selectedPaymentOption is STPApplePayPaymentOption { + assert( + strongSelf.hostViewController != nil, + "hostViewController must not be nil on STPPaymentContext. Next time, set the hostViewController property first!" + ) + strongSelf.state = .requestingPayment + let paymentRequest = strongSelf.buildPaymentRequest() + let shippingAddressHandler: STPShippingAddressSelectionBlock = { + shippingAddress, + completion in + // Apple Pay always returns a partial address here, so we won't + // update self.shippingAddress or self.shippingMethods + if strongSelf.delegate?.responds( + to: #selector( + STPPaymentContextDelegate.paymentContext( + _: + didUpdateShippingAddress: + completion: + )) + ) + ?? false + { + strongSelf.delegate?.paymentContext?( + strongSelf, + didUpdateShippingAddress: shippingAddress + ) { status, _, shippingMethods, _ in + completion( + status, + shippingMethods ?? [], + strongSelf.paymentSummaryItems + ) + } + } else { + completion( + .valid, + strongSelf.shippingMethods ?? [], + strongSelf.paymentSummaryItems + ) + } + } + let shippingMethodHandler: STPShippingMethodSelectionBlock = { + shippingMethod, + completion in + strongSelf.selectedShippingMethod = shippingMethod + strongSelf.delegate?.paymentContextDidChange(strongSelf) + completion(self.paymentSummaryItems) + } + let paymentHandler: STPPaymentAuthorizationBlock = { payment in + strongSelf.selectedShippingMethod = payment.shippingMethod + if let shippingContact = payment.shippingContact { + strongSelf.shippingAddress = STPAddress(pkContact: shippingContact) + } + strongSelf.shippingAddressNeedsVerification = false + strongSelf.delegate?.paymentContextDidChange(strongSelf) + if strongSelf.apiAdapter is STPCustomerContext { + let customerContext = strongSelf.apiAdapter as? STPCustomerContext + if let shippingAddress1 = strongSelf.shippingAddress { + customerContext?.updateCustomer( + withShippingAddress: shippingAddress1, + completion: nil + ) + } + } + } + let applePayPaymentMethodHandler: STPApplePayPaymentMethodHandlerBlock = { + paymentMethod, + completion in + strongSelf.apiAdapter.attachPaymentMethod(toCustomer: paymentMethod) { + attachPaymentMethodError in + stpDispatchToMainThreadIfNecessary({ + if attachPaymentMethodError != nil { + completion(.error, attachPaymentMethodError) + } else { + let result = STPPaymentResult(paymentOption: paymentMethod) + strongSelf.delegate?.paymentContext( + strongSelf, + didCreatePaymentResult: result + ) { + status, + error in + // for Apple Pay, the didFinishWithStatus callback is fired later when Apple Pay VC finishes + completion(status, error) + } + } + }) + } + } + if let paymentRequest = paymentRequest { + strongSelf.applePayVC = PKPaymentAuthorizationViewController.stp_controller( + with: paymentRequest, + apiClient: strongSelf.apiClient, + onShippingAddressSelection: shippingAddressHandler, + onShippingMethodSelection: shippingMethodHandler, + onPaymentAuthorization: paymentHandler, + onPaymentMethodCreation: applePayPaymentMethodHandler, + onFinish: { status, error in + if strongSelf.applePayVC?.presentingViewController != nil { + strongSelf.hostViewController?.dismiss( + animated: strongSelf.transitionAnimationsEnabled() + ) { + strongSelf.didFinish(with: status, error: error) + } + } else { + strongSelf.didFinish(with: status, error: error) + } + strongSelf.applePayVC = nil + } + ) + } + if let applePayVC1 = strongSelf.applePayVC { + strongSelf.hostViewController?.present( + applePayVC1, + animated: strongSelf.transitionAnimationsEnabled() + ) + } + } + }).onFailure({ error in + guard let strongSelf = weakSelf else { + return + } + strongSelf.didFinish(with: .error, error: error) + }) + } + private var loadingPromise: STPPromise? + private weak var paymentOptionsViewController: STPPaymentOptionsViewController? + private var state: STPPaymentContextState = .none + private var paymentAmountModel = STPPaymentContextAmountModel(amount: 0) + private var shippingAddressNeedsVerification = false + // If hostViewController was set to a nav controller, the original VC on top of the stack + private weak var originalTopViewController: UIViewController? + private var applePayVC: PKPaymentAuthorizationViewController? + + // Disable transition animations in tests + func transitionAnimationsEnabled() -> Bool { + return NSClassFromString("XCTest") == nil + } + + var currentValuePromise: STPPromise { + weak var weakSelf = self + return + (loadingPromise?.map({ _ in + guard let strongSelf = weakSelf, let paymentOptions = strongSelf.paymentOptions + else { + return STPPaymentOptionTuple() + } + return STPPaymentOptionTuple( + paymentOptions: paymentOptions, + selectedPaymentOption: strongSelf.selectedPaymentOption + ) + }))! + } + + func remove(_ paymentOptionToRemove: STPPaymentOption?) { + // Remove payment method from cached representation + var paymentOptions = self.paymentOptions + paymentOptions?.removeAll { $0 as AnyObject === paymentOptionToRemove as AnyObject } + self.paymentOptions = paymentOptions + + // Elect new selected payment method if needed + if let selectedPaymentOption = selectedPaymentOption, + selectedPaymentOption.isEqual(paymentOptionToRemove) + { + self.selectedPaymentOption = self.paymentOptions?.first + } + } + + // MARK: - Payment Methods + + func presentPaymentOptionsViewController(withNewState state: STPPaymentContextState) { + assert( + hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, + "hostViewController must not be nil on STPPaymentContext when calling pushPaymentOptionsViewController on it. Next time, set the hostViewController property first!" + ) + if self.state == STPPaymentContextState.none { + self.state = state + let paymentOptionsViewController = STPPaymentOptionsViewController(paymentContext: self) + self.paymentOptionsViewController = paymentOptionsViewController + paymentOptionsViewController.prefilledInformation = prefilledInformation + paymentOptionsViewController.defaultPaymentMethod = defaultPaymentMethod + paymentOptionsViewController.paymentOptionsViewControllerFooterView = + paymentOptionsViewControllerFooterView + paymentOptionsViewController.addCardViewControllerFooterView = + addCardViewControllerFooterView + paymentOptionsViewController.navigationItem.largeTitleDisplayMode = + largeTitleDisplayMode + + let navigationController = UINavigationController( + rootViewController: paymentOptionsViewController + ) + navigationController.navigationBar.stp_theme = theme + navigationController.navigationBar.prefersLargeTitles = true + navigationController.modalPresentationStyle = modalPresentationStyle + hostViewController?.present( + navigationController, + animated: transitionAnimationsEnabled() + ) + } + } + + @objc + public func paymentOptionsViewController( + _ paymentOptionsViewController: STPPaymentOptionsViewController, + didSelect paymentOption: STPPaymentOption + ) { + selectedPaymentOption = paymentOption + } + + @objc + public func paymentOptionsViewControllerDidFinish( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) { + appropriatelyDismiss(paymentOptionsViewController) { + if self.state == .requestingPayment { + self.state = STPPaymentContextState.none + self.requestPayment() + } else { + self.state = STPPaymentContextState.none + } + } + } + + @objc + public func paymentOptionsViewControllerDidCancel( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) { + appropriatelyDismiss(paymentOptionsViewController) { + if self.state == .requestingPayment { + self.didFinish( + with: .userCancellation, + error: nil + ) + } else { + self.state = STPPaymentContextState.none + } + } + } + + @objc + public func paymentOptionsViewController( + _ paymentOptionsViewController: STPPaymentOptionsViewController, + didFailToLoadWithError error: Error + ) { + // we'll handle this ourselves when the loading promise fails. + } + + @objc(appropriatelyDismissPaymentOptionsViewController:completion:) func appropriatelyDismiss( + _ viewController: STPPaymentOptionsViewController, + completion: @escaping STPVoidBlock + ) { + if viewController.stp_isAtRootOfNavigationController() { + // if we're the root of the navigation controller, we've been presented modally. + viewController.presentingViewController?.dismiss( + animated: transitionAnimationsEnabled() + ) { + self.paymentOptionsViewController = nil + completion() + } + } else { + // otherwise, we've been pushed onto the stack. + var destinationViewController = hostViewController + // If hostViewController is a nav controller, pop to the original VC on top of the stack. + if hostViewController is UINavigationController { + destinationViewController = originalTopViewController + } + viewController.navigationController?.stp_pop( + to: destinationViewController, + animated: transitionAnimationsEnabled() + ) { + self.paymentOptionsViewController = nil + completion() + } + } + } + + // MARK: - Shipping Info + + func presentShippingViewController(withNewState state: STPPaymentContextState) { + assert( + hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, + "hostViewController must not be nil on STPPaymentContext when calling presentShippingViewController on it. Next time, set the hostViewController property first!" + ) + + if self.state == STPPaymentContextState.none { + self.state = state + + let addressViewController = STPShippingAddressViewController(paymentContext: self) + addressViewController.navigationItem.largeTitleDisplayMode = largeTitleDisplayMode + let navigationController = UINavigationController( + rootViewController: addressViewController + ) + navigationController.navigationBar.stp_theme = theme + navigationController.navigationBar.prefersLargeTitles = true + navigationController.modalPresentationStyle = modalPresentationStyle + hostViewController?.present( + navigationController, + animated: transitionAnimationsEnabled() + ) + } + } + + @objc + public func shippingAddressViewControllerDidCancel( + _ addressViewController: STPShippingAddressViewController + ) { + appropriatelyDismiss(addressViewController) { + if self.state == .requestingPayment { + self.didFinish( + with: .userCancellation, + error: nil + ) + } else { + self.state = STPPaymentContextState.none + } + } + } + + @objc + public func shippingAddressViewController( + _ addressViewController: STPShippingAddressViewController, + didEnter address: STPAddress, + completion: @escaping STPShippingMethodsCompletionBlock + ) { + if delegate?.responds( + to: #selector( + STPPaymentContextDelegate.paymentContext(_:didUpdateShippingAddress:completion:)) + ) + ?? false + { + delegate?.paymentContext?(self, didUpdateShippingAddress: address) { + status, + shippingValidationError, + shippingMethods, + selectedMethod in + self.shippingMethods = shippingMethods + completion(status, shippingValidationError, shippingMethods, selectedMethod) + } + } else { + completion(.valid, nil, nil, nil) + } + } + + @objc + public func shippingAddressViewController( + _ addressViewController: STPShippingAddressViewController, + didFinishWith address: STPAddress, + shippingMethod method: PKShippingMethod? + ) { + shippingAddress = address + shippingAddressNeedsVerification = false + selectedShippingMethod = method + delegate?.paymentContextDidChange(self) + if apiAdapter.responds( + to: #selector(STPCustomerContext.updateCustomer(withShippingAddress:completion:)) + ) { + if let shippingAddress = shippingAddress { + apiAdapter.updateCustomer?(withShippingAddress: shippingAddress, completion: nil) + } + } + appropriatelyDismiss(addressViewController) { + if self.state == .requestingPayment { + self.state = STPPaymentContextState.none + self.requestPayment() + } else { + self.state = STPPaymentContextState.none + } + } + } + + @objc(appropriatelyDismissViewController:completion:) func appropriatelyDismiss( + _ viewController: UIViewController, + completion: @escaping STPVoidBlock + ) { + if viewController.stp_isAtRootOfNavigationController() { + // if we're the root of the navigation controller, we've been presented modally. + viewController.presentingViewController?.dismiss( + animated: transitionAnimationsEnabled() + ) { + completion() + } + } else { + // otherwise, we've been pushed onto the stack. + var destinationViewController = hostViewController + // If hostViewController is a nav controller, pop to the original VC on top of the stack. + if hostViewController is UINavigationController { + destinationViewController = originalTopViewController + } + viewController.navigationController?.stp_pop( + to: destinationViewController, + animated: transitionAnimationsEnabled() + ) { + completion() + } + } + } + + // MARK: - Request Payment + func requestPaymentShouldPresentShippingViewController() -> Bool { + let shippingAddressRequired = (configuration.requiredShippingAddressFields?.count ?? 0) > 0 + var shippingAddressIncomplete: Bool? + if let requiredShippingAddressFields1 = configuration.requiredShippingAddressFields { + shippingAddressIncomplete = + !(shippingAddress?.containsRequiredShippingAddressFields( + requiredShippingAddressFields1 + ) + ?? false) + } + let shippingMethodRequired = + configuration.shippingType == .shipping + && delegate?.responds( + to: #selector( + STPPaymentContextDelegate.paymentContext(_:didUpdateShippingAddress:completion:) + ) + ) + ?? false + && selectedShippingMethod == nil + let verificationRequired = + configuration.verifyPrefilledShippingAddress && shippingAddressNeedsVerification + // true if STPShippingVC should be presented to collect or verify a shipping address + let shouldPresentShippingAddress = + shippingAddressRequired && (shippingAddressIncomplete ?? false || verificationRequired) + // this handles a corner case where STPShippingVC should be presented because: + // - shipping address has been pre-filled + // - no verification is required, but the user still needs to enter a shipping method + let shouldPresentShippingMethods = + shippingAddressRequired && !(shippingAddressIncomplete ?? false) + && !verificationRequired + && shippingMethodRequired + return shouldPresentShippingAddress || shouldPresentShippingMethods + } + + func didFinish( + with status: STPPaymentStatus, + error: Error? + ) { + state = STPPaymentContextState.none + delegate?.paymentContext( + self, + didFinishWith: status, + error: error + ) + } + + func buildPaymentRequest() -> PKPaymentRequest? { + guard let appleMerchantIdentifier = configuration.appleMerchantIdentifier, paymentAmount > 0 + else { + return nil + } + let paymentRequest = StripeAPI.paymentRequest( + withMerchantIdentifier: appleMerchantIdentifier, + country: paymentCountry, + currency: paymentCurrency + ) + + let summaryItems = paymentSummaryItems + paymentRequest.paymentSummaryItems = summaryItems + + let requiredFields = STPAddress.applePayContactFields( + from: configuration.requiredBillingAddressFields + ) + paymentRequest.requiredBillingContactFields = requiredFields + + var shippingRequiredFields: Set? + if let requiredShippingAddressFields1 = configuration.requiredShippingAddressFields { + shippingRequiredFields = STPAddress.pkContactFields( + fromStripeContactFields: requiredShippingAddressFields1 + ) + } + if let shippingRequiredFields = shippingRequiredFields { + paymentRequest.requiredShippingContactFields = shippingRequiredFields + } + + paymentRequest.currencyCode = paymentCurrency.uppercased() + if let selectedShippingMethod = selectedShippingMethod { + var orderedShippingMethods = shippingMethods + orderedShippingMethods?.removeAll { + $0 as AnyObject === selectedShippingMethod as AnyObject + } + orderedShippingMethods?.insert(selectedShippingMethod, at: 0) + paymentRequest.shippingMethods = orderedShippingMethods + } else { + paymentRequest.shippingMethods = shippingMethods + } + + paymentRequest.shippingType = STPPaymentContext.pkShippingType(configuration.shippingType) + + if let shippingAddress = shippingAddress { + paymentRequest.shippingContact = shippingAddress.pkContactValue() + } + return paymentRequest + } + + class func pkShippingType(_ shippingType: STPShippingType) -> PKShippingType { + switch shippingType { + case .shipping: + return .shipping + case .delivery: + return .delivery + @unknown default: + fatalError() + } + } + + func artificiallyRetain(_ host: NSObject) { + objc_setAssociatedObject( + host, + UnsafeRawPointer(&kSTPPaymentCoordinatorAssociatedObjectKey), + self, + .OBJC_ASSOCIATION_RETAIN_NONATOMIC + ) + } + + // MARK: - STPAuthenticationContext + @objc + public func authenticationPresentingViewController() -> UIViewController { + return hostViewController! + } + + @objc + public func prepare(forPresentation completion: @escaping STPVoidBlock) { + if applePayVC != nil && applePayVC?.presentingViewController != nil { + hostViewController?.dismiss( + animated: transitionAnimationsEnabled() + ) { + completion() + } + } else { + completion() + } + } +} + +/// Implement `STPPaymentContextDelegate` to get notified when a payment context changes, finishes, encounters errors, etc. In practice, if your app has a "checkout screen view controller", that is a good candidate to implement this protocol. +@objc public protocol STPPaymentContextDelegate: NSObjectProtocol { + /// Called when the payment context encounters an error when fetching its initial set of data. A few ways to handle this are: + /// - If you're showing the user a checkout page, dismiss the checkout page when this is called and present the error to the user. + /// - Present the error to the user using a `UIAlertController` with two buttons: Retry and Cancel. If they cancel, dismiss your UI. If they Retry, call `retryLoading` on the payment context. + /// To make it harder to get your UI into a bad state, this won't be called until the context's `hostViewController` has finished appearing. + /// - Parameters: + /// - paymentContext: the payment context that encountered the error + /// - error: the error that was encountered + func paymentContext(_ paymentContext: STPPaymentContext, didFailToLoadWithError error: Error) + /// This is called every time the contents of the payment context change. When this is called, you should update your app's UI to reflect the current state of the payment context. For example, if you have a checkout page with a "selected payment method" row, you should update its payment method with `paymentContext.selectedPaymentOption.label`. If that checkout page has a "buy" button, you should enable/disable it depending on the result of `paymentContext.isReadyForPayment`. + /// - Parameter paymentContext: the payment context that changed + func paymentContextDidChange(_ paymentContext: STPPaymentContext) + /// Inside this method, you should make a call to your backend API to make a PaymentIntent with that Customer + payment method, and invoke the `completion` block when that is done. + /// - Parameters: + /// - paymentContext: The context that succeeded + /// - paymentResult: Information associated with the payment that you can pass to your server. You should go to your backend API with this payment result and use the PaymentIntent API to complete the payment. See https://stripe.com/docs/mobile/ios/basic#submit-payment-intents Once that's done call the `completion` block with any error that occurred (or none, if the payment succeeded). - seealso: STPPaymentResult.h + /// - completion: Call this block when you're done creating a payment intent (or subscription, etc) on your backend. If it succeeded, call `completion(STPPaymentStatusSuccess, nil)`. If it failed with an error, call `completion(STPPaymentStatusError, error)`. If the user canceled, call `completion(STPPaymentStatusUserCancellation, nil)`. + func paymentContext( + _ paymentContext: STPPaymentContext, + didCreatePaymentResult paymentResult: STPPaymentResult, + completion: @escaping STPPaymentStatusBlock + ) + /// This is invoked by an `STPPaymentContext` when it is finished. This will be called after the payment is done and all necessary UI has been dismissed. You should inspect the returned `status` and behave appropriately. For example: if it's `STPPaymentStatusSuccess`, show the user a receipt. If it's `STPPaymentStatusError`, inform the user of the error. If it's `STPPaymentStatusUserCancellation`, do nothing. + /// - Parameters: + /// - paymentContext: The payment context that finished + /// - status: The status of the payment - `STPPaymentStatusSuccess` if it succeeded, `STPPaymentStatusError` if it failed with an error (in which case the `error` parameter will be non-nil), `STPPaymentStatusUserCancellation` if the user canceled the payment. + /// - error: An error that occurred, if any. + func paymentContext( + _ paymentContext: STPPaymentContext, + didFinishWith status: STPPaymentStatus, + error: Error? + ) + + /// Inside this method, you should verify that you can ship to the given address. + /// You should call the completion block with the results of your validation + /// and the available shipping methods for the given address. If you don't implement + /// this method, the user won't be prompted to select a shipping method and all + /// addresses will be valid. If you call the completion block with nil or an + /// empty array of shipping methods, the user won't be prompted to select a + /// shipping method. + /// @note If a user updates their shipping address within the Apple Pay dialog, + /// this address will be anonymized. For example, in the US, it will only include the + /// city, state, and zip code. The payment context will have the user's complete + /// shipping address by the time `paymentContext:didFinishWithStatus:error` is + /// called. + /// - Parameters: + /// - paymentContext: The context that updated its shipping address + /// - address: The current shipping address + /// - completion: Call this block when you're done validating the shipping + /// address and calculating available shipping methods. If you call the completion + /// block with nil or an empty array of shipping methods, the user won't be prompted + /// to select a shipping method. + @objc optional func paymentContext( + _ paymentContext: STPPaymentContext, + didUpdateShippingAddress address: STPAddress, + completion: @escaping STPShippingMethodsCompletionBlock + ) +} + +/// The current state of the payment context +/// - STPPaymentContextStateNone: No view controllers are currently being shown. The payment may or may not have already been completed +/// - STPPaymentContextStateShowingRequestedViewController: The view controller that you requested the context show is being shown (via the push or present payment methods or shipping view controller methods) +/// - STPPaymentContextStateRequestingPayment: The payment context is in the middle of requesting payment. It may be showing some other UI or view controller if more information is necessary to complete the payment. +enum STPPaymentContextState: Int { + case none + case showingRequestedViewController + case requestingPayment +} + +private var kSTPPaymentCoordinatorAssociatedObjectKey = 0 + +/// :nodoc: +@_spi(STP) extension STPPaymentContext: STPAnalyticsProtocol { + @_spi(STP) public static var stp_analyticsIdentifier = "STPPaymentContext" +} diff --git a/Stripe/StripeiOS/Source/STPPaymentContextAmountModel.swift b/Stripe/StripeiOS/Source/STPPaymentContextAmountModel.swift new file mode 100644 index 00000000..c79c13e2 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentContextAmountModel.swift @@ -0,0 +1,96 @@ +// +// STPPaymentContextAmountModel.swift +// StripeiOS +// +// Created by Brian Dorfman on 8/16/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripePayments + +/// Internal model for STPPaymentContext's `paymentAmount` and +/// `paymentSummaryItems` properties. +class STPPaymentContextAmountModel: NSObject { + private var paymentAmount = 0 + private var paymentSummaryItems: [PKPaymentSummaryItem]? + + init( + amount paymentAmount: Int + ) { + super.init() + self.paymentAmount = paymentAmount + paymentSummaryItems = nil + } + + init( + paymentSummaryItems: [PKPaymentSummaryItem]? + ) { + super.init() + paymentAmount = 0 + self.paymentSummaryItems = paymentSummaryItems + } + + func paymentAmount(withCurrency currency: String?, shippingMethod: PKShippingMethod?) -> Int { + let shippingAmount = + ((shippingMethod != nil) + ? shippingMethod?.amount.stp_amount(withCurrency: currency) : 0) ?? 0 + if paymentSummaryItems == nil { + return paymentAmount + shippingAmount + } else { + let lastItem = paymentSummaryItems?.last + return (lastItem?.amount.stp_amount(withCurrency: currency) ?? 0) + shippingAmount + } + } + + func paymentSummaryItems( + withCurrency currency: String?, + companyName: String?, + shippingMethod: PKShippingMethod? + ) -> [PKPaymentSummaryItem]? { + var shippingItem: PKPaymentSummaryItem? + if let shippingMethod = shippingMethod { + shippingItem = PKPaymentSummaryItem( + label: shippingMethod.label, + amount: shippingMethod.amount + ) + } + if paymentSummaryItems == nil { + let shippingAmount = shippingMethod?.amount.stp_amount(withCurrency: currency) ?? 0 + let total = NSDecimalNumber.stp_decimalNumber( + withAmount: paymentAmount + shippingAmount, + currency: currency + ) + let totalItem = PKPaymentSummaryItem(label: companyName ?? "", amount: total) + var items = [totalItem] + if let shippingItem = shippingItem { + items.insert(shippingItem, at: 0) + } + return items.compactMap { $0 } + } else { + if (paymentSummaryItems?.count ?? 0) > 0 && shippingItem != nil { + var items = paymentSummaryItems + let origTotalItem = items?.last + var newTotal: NSDecimalNumber? + if let amount1 = shippingItem?.amount { + newTotal = origTotalItem?.amount.adding(amount1) + } + var totalItem: PKPaymentSummaryItem? + if let newTotal = newTotal { + totalItem = PKPaymentSummaryItem( + label: origTotalItem?.label ?? "", + amount: newTotal + ) + } + items?.removeLast() + if let items = items { + return items + [shippingItem, totalItem].compactMap { $0 } + } + return nil + } else { + return paymentSummaryItems + } + } + } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentIntentParams+BasicUI.swift b/Stripe/StripeiOS/Source/STPPaymentIntentParams+BasicUI.swift new file mode 100644 index 00000000..007367db --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentIntentParams+BasicUI.swift @@ -0,0 +1,22 @@ +// +// STPPaymentIntentParams+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation + +extension STPPaymentIntentParams { + /// Provide an STPPaymentResult from STPPaymentContext, and this will populate + /// the proper field (either paymentMethodId or paymentMethodParams) for your PaymentMethod. + @objc + public func configure(with paymentResult: STPPaymentResult) { + if let paymentMethod = paymentResult.paymentMethod { + paymentMethodId = paymentMethod.stripeId + } else if let params = paymentResult.paymentMethodParams { + paymentMethodParams = params + } + } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentMethod+BasicUI.swift b/Stripe/StripeiOS/Source/STPPaymentMethod+BasicUI.swift new file mode 100644 index 00000000..c514f1ca --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentMethod+BasicUI.swift @@ -0,0 +1,79 @@ +// +// STPPaymentMethod+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments +@_spi(STP) import StripePaymentsUI +import UIKit + +extension STPPaymentMethod: STPPaymentOption { + // MARK: - STPPaymentOption + @objc public var image: UIImage { + if type == .card, let card = card { + return STPImageLibrary.cardBrandImage(for: card.brand) + } else { + return STPImageLibrary.cardBrandImage(for: .unknown) + } + } + + @objc public var templateImage: UIImage { + if type == .card, let card = card { + return STPImageLibrary.templatedBrandImage(for: card.brand) + } else { + return STPImageLibrary.templatedBrandImage(for: .unknown) + } + } + + @objc public var label: String { + switch type { + case .card: + if let card = card { + let brand = STPCardBrandUtilities.stringFrom(card.brand) + return "\(brand ?? "") \(card.last4 ?? "")" + } else { + return STPCardBrandUtilities.stringFrom(.unknown) ?? "" + } + case .FPX: + if let fpx = fpx { + return STPFPXBank.stringFrom(STPFPXBank.brandFrom(fpx.bankIdentifierCode)) ?? "" + } else { + fallthrough + } + case .USBankAccount: + if let usBankAccount = usBankAccount { + return String( + format: String.Localized.bank_name_account_ending_in_last_4, + usBankAccount.bankName, + usBankAccount.last4 + ) + } else { + fallthrough + } + default: + return type.displayName + } + } + + @objc public var isReusable: Bool { + switch type { + case .card, .link, .USBankAccount: + return true + case .alipay, // Careful! Revisit this if/when we support recurring Alipay + .AUBECSDebit, + .bacsDebit, .SEPADebit, .iDEAL, .FPX, .cardPresent, .giropay, .EPS, .payPal, + .przelewy24, .bancontact, + .OXXO, .sofort, .grabPay, .netBanking, .UPI, .afterpayClearpay, .blik, + .weChatPay, .boleto, .klarna, .linkInstantDebit, .affirm, .cashApp, // fall through + .unknown: + return false + @unknown default: + return false + } + } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentMethodParams+BasicUI.swift b/Stripe/StripeiOS/Source/STPPaymentMethodParams+BasicUI.swift new file mode 100644 index 00000000..2efd25be --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentMethodParams+BasicUI.swift @@ -0,0 +1,48 @@ +// +// STPPaymentMethodParams+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripePaymentsUI +import UIKit + +extension STPPaymentMethodParams: STPPaymentOption { + // MARK: - STPPaymentOption + @objc public var image: UIImage { + if type == .card && card != nil { + let brand = STPCardValidator.brand(forNumber: card?.number ?? "") + return STPImageLibrary.cardBrandImage(for: brand) + } else { + return STPImageLibrary.cardBrandImage(for: .unknown) + } + } + + @objc public var templateImage: UIImage { + if type == .card && card != nil { + let brand = STPCardValidator.brand(forNumber: card?.number ?? "") + return STPImageLibrary.templatedBrandImage(for: brand) + } else if type == .FPX { + return STPImageLibrary.bankIcon() + } else { + return STPImageLibrary.templatedBrandImage(for: .unknown) + } + } + + @objc public var isReusable: Bool { + switch type { + case .card, .link, .USBankAccount: + return true + case .alipay, .AUBECSDebit, .bacsDebit, .SEPADebit, .iDEAL, .FPX, .cardPresent, .giropay, + .grabPay, .EPS, .przelewy24, .bancontact, .netBanking, .OXXO, .payPal, .sofort, .UPI, + .afterpayClearpay, .blik, .weChatPay, .boleto, .klarna, .linkInstantDebit, .affirm, .cashApp, + .unknown: + return false + @unknown default: + return false + } + } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentOption.swift b/Stripe/StripeiOS/Source/STPPaymentOption.swift new file mode 100644 index 00000000..278c3d38 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentOption.swift @@ -0,0 +1,36 @@ +// +// STPPaymentOption.swift +// StripeiOS +// +// Created by Ben Guo on 4/19/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +/// This protocol represents a payment method that a user can select and use to +/// pay. +/// The classes that conform to it and are supported by the UI: +/// - `STPApplePay`, which represents that the user wants to pay with +/// Apple Pay +/// - `STPPaymentMethod`. Only `STPPaymentMethod.type == STPPaymentMethodTypeCard` and +/// `STPPaymentMethod.type == STPPaymentMethodTypeFPX` are supported by `STPPaymentContext` +/// and `STPPaymentOptionsViewController` +/// - `STPPaymentMethodParams`. This should be used with non-reusable payment method, such +/// as FPX and iDEAL. Instead of reaching out to Stripe to create a PaymentMethod, you can +/// pass an STPPaymentMethodParams directly to Stripe when confirming a PaymentIntent. +/// @note card-based Sources, Cards, and FPX support this protocol for use +/// in a custom integration. +@objc public protocol STPPaymentOption: NSObjectProtocol { + /// A small (32 x 20 points) logo image representing the payment method. For + /// example, the Visa logo for a Visa card, or the Apple Pay logo. + var image: UIImage { get } + /// A small (32 x 20 points) logo image representing the payment method that can be + /// used as template for tinted icons. + var templateImage: UIImage { get } + /// A string describing the payment method, such as "Apple Pay" or "Visa 4242". + var label: String { get } + /// Describes whether this payment option may be used multiple times. If it is not reusable, + /// the payment method must be discarded after use. + var isReusable: Bool { get } +} diff --git a/Stripe/StripeiOS/Source/STPPaymentOptionTableViewCell.swift b/Stripe/StripeiOS/Source/STPPaymentOptionTableViewCell.swift new file mode 100644 index 00000000..fd01cf7d --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentOptionTableViewCell.swift @@ -0,0 +1,338 @@ +// +// STPPaymentOptionTableViewCell.swift +// StripeiOS +// +// Created by Ben Guo on 8/30/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +import UIKit + +class STPPaymentOptionTableViewCell: UITableViewCell { + @objc(configureForNewCardRowWithTheme:) func configureForNewCardRow(with theme: STPTheme) { + paymentOption = nil + self.theme = theme + + backgroundColor = theme.secondaryBackgroundColor + + // Left icon + leftIcon.image = STPLegacyImageLibrary.addIcon() + leftIcon.tintColor = theme.accentColor + + // Title label + titleLabel.font = theme.font + titleLabel.textColor = theme.accentColor + titleLabel.text = STPLocalizedString("Add New Card…", "Button to add a new credit card.") + + // Checkmark icon + checkmarkIcon.isHidden = true + + setNeedsLayout() + } + + @objc(configureWithPaymentOption:theme:selected:) func configure( + with paymentOption: STPPaymentOption?, + theme: STPTheme, + selected: Bool + ) { + self.paymentOption = paymentOption + self.theme = theme + + backgroundColor = theme.secondaryBackgroundColor + + // Left icon + leftIcon.image = paymentOption?.templateImage + leftIcon.tintColor = primaryColorForPaymentOption(withSelected: selected) + + // Title label + titleLabel.font = theme.font + titleLabel.attributedText = buildAttributedString(with: paymentOption, selected: selected) + + // Checkmark icon + checkmarkIcon.tintColor = theme.accentColor + checkmarkIcon.isHidden = !selected + + // Accessibility + if selected { + accessibilityTraits.insert(.selected) + } else { + accessibilityTraits.remove(.selected) + } + + setNeedsLayout() + } + + @objc(configureForFPXRowWithTheme:) func configureForFPXRow(with theme: STPTheme) { + paymentOption = nil + self.theme = theme + + backgroundColor = theme.secondaryBackgroundColor + + // Left icon + leftIcon.image = STPImageLibrary.bankIcon() + leftIcon.tintColor = primaryColorForPaymentOption(withSelected: false) + + // Title label + titleLabel.font = theme.font + titleLabel.textColor = self.theme.primaryForegroundColor + titleLabel.text = STPLocalizedString( + "Online Banking (FPX)", + "Button to pay with a Bank Account (using FPX)." + ) + + // Checkmark icon + checkmarkIcon.isHidden = true + accessoryType = .disclosureIndicator + setNeedsLayout() + } + + private var paymentOption: STPPaymentOption? + private var theme: STPTheme = .defaultTheme + private var leftIcon = UIImageView() + private var titleLabel = UILabel() + private var checkmarkIcon = UIImageView(image: STPLegacyImageLibrary.checkmarkIcon()) + + override init( + style: UITableViewCell.CellStyle, + reuseIdentifier: String? + ) { + super.init(style: style, reuseIdentifier: reuseIdentifier) + // Left icon + leftIcon.translatesAutoresizingMaskIntoConstraints = false + contentView.addSubview(leftIcon) + + // Title label + titleLabel.translatesAutoresizingMaskIntoConstraints = false + contentView.addSubview(titleLabel) + + // Checkmark icon + checkmarkIcon.translatesAutoresizingMaskIntoConstraints = false + contentView.addSubview(checkmarkIcon) + + NSLayoutConstraint.activate( + [ + self.leftIcon.centerXAnchor.constraint( + equalTo: contentView.leadingAnchor, + constant: kPadding + 0.5 * kDefaultIconWidth + ), + self.leftIcon.centerYAnchor.constraint( + lessThanOrEqualTo: contentView.centerYAnchor + ), + self.checkmarkIcon.widthAnchor.constraint(equalToConstant: kCheckmarkWidth), + self.checkmarkIcon.heightAnchor.constraint( + equalTo: self.checkmarkIcon.widthAnchor, + multiplier: 1.0 + ), + self.checkmarkIcon.centerXAnchor.constraint( + equalTo: contentView.trailingAnchor, + constant: -kPadding + ), + self.checkmarkIcon.centerYAnchor.constraint(equalTo: contentView.centerYAnchor), + // Constrain label to leadingAnchor with the default + // icon width so that the text always aligns vertically + // even if the icond widths differ + self.titleLabel.leadingAnchor.constraint( + equalTo: contentView.leadingAnchor, + constant: 2.0 * kPadding + kDefaultIconWidth + ), + self.titleLabel.trailingAnchor.constraint( + equalTo: self.checkmarkIcon.leadingAnchor, + constant: -kPadding + ), + self.titleLabel.centerYAnchor.constraint(equalTo: contentView.centerYAnchor), + ]) + accessibilityTraits.insert(.button) + isAccessibilityElement = true + } + + func primaryColorForPaymentOption(withSelected selected: Bool) -> UIColor { + let fadedColor: UIColor = { + if #available(iOS 13.0, *) { + return UIColor(dynamicProvider: { _ in + return self.theme.primaryForegroundColor.withAlphaComponent(0.6) + }) + } else { + return theme.primaryForegroundColor.withAlphaComponent(0.6) + } + }() + + return (selected ? theme.accentColor : fadedColor) + } + + func buildAttributedString( + with paymentOption: STPPaymentOption?, + selected: Bool + ) + -> NSAttributedString + { + if let paymentOption = paymentOption as? STPCard { + return buildAttributedString(with: paymentOption, selected: selected) + } else if let source = paymentOption as? STPSource { + if source.type == .card && source.cardDetails != nil { + return buildAttributedString(withCardSource: source, selected: selected) + } + } else if let paymentMethod = paymentOption as? STPPaymentMethod { + if paymentMethod.type == .card && paymentMethod.card != nil { + return buildAttributedString( + withCardPaymentMethod: paymentMethod, + selected: selected + ) + } + if paymentMethod.type == .FPX && paymentMethod.fpx != nil { + return buildAttributedString( + with: STPFPXBank.brandFrom(paymentMethod.fpx?.bankIdentifierCode), + selected: selected + ) + } + } else if paymentOption is STPApplePayPaymentOption { + let label = String.Localized.apple_pay + let primaryColor = primaryColorForPaymentOption(withSelected: selected) + return NSAttributedString( + string: label, + attributes: [ + NSAttributedString.Key.foregroundColor: primaryColor + ] + ) + } else if let paymentMethodParams = paymentOption as? STPPaymentMethodParams { + if paymentMethodParams.type == .card && paymentMethodParams.card != nil { + return buildAttributedString( + withCardPaymentMethodParams: paymentMethodParams, + selected: selected + ) + } + if paymentMethodParams.type == .FPX && paymentMethodParams.fpx != nil { + return buildAttributedString( + with: paymentMethodParams.fpx?.bank ?? STPFPXBankBrand.unknown, + selected: selected + ) + } + } + + // Unrecognized payment method + return NSAttributedString(string: "") + } + + func buildAttributedString(with card: STPCard, selected: Bool) -> NSAttributedString { + return buildAttributedString( + with: card.brand, + last4: card.last4, + selected: selected + ) + } + + func buildAttributedString(withCardSource card: STPSource, selected: Bool) -> NSAttributedString + { + return buildAttributedString( + with: card.cardDetails?.brand ?? .unknown, + last4: card.cardDetails?.last4 ?? "", + selected: selected + ) + } + + func buildAttributedString( + withCardPaymentMethod paymentMethod: STPPaymentMethod, + selected: Bool + ) + -> NSAttributedString + { + return buildAttributedString( + with: paymentMethod.card?.brand ?? .unknown, + last4: paymentMethod.card?.last4 ?? "", + selected: selected + ) + } + + func buildAttributedString( + withCardPaymentMethodParams paymentMethodParams: STPPaymentMethodParams, + selected: Bool + ) -> NSAttributedString { + let brand = STPCardValidator.brand(forNumber: paymentMethodParams.card?.number ?? "") + return buildAttributedString( + with: brand, + last4: paymentMethodParams.card?.last4 ?? "", + selected: selected + ) + } + + func buildAttributedString( + with bankBrand: STPFPXBankBrand, + selected: Bool + ) + -> NSAttributedString + { + let label = (STPFPXBank.stringFrom(bankBrand) ?? "") + " (FPX)" + let primaryColor = primaryColorForPaymentOption(withSelected: selected) + return NSAttributedString( + string: label, + attributes: [ + NSAttributedString.Key.foregroundColor: primaryColor + ] + ) + } + + func buildAttributedString( + with brand: STPCardBrand, + last4: String, + selected: Bool + ) -> NSAttributedString { + let format = String.Localized.card_brand_ending_in_last_4 + let brandString = STPCard.string(from: brand) + let label = String(format: format, brandString, last4) + + let primaryColor = selected ? theme.accentColor : theme.primaryForegroundColor + + let secondaryColor: UIColor = { + if #available(iOS 13.0, *) { + return UIColor(dynamicProvider: { _ in + return primaryColor.withAlphaComponent(0.6) + }) + } else { + return primaryColor.withAlphaComponent(0.6) + } + }() + + let attributes: [NSAttributedString.Key: Any] = [ + NSAttributedString.Key.foregroundColor: secondaryColor, + NSAttributedString.Key.font: self.theme.font, + ] + + let attributedString = NSMutableAttributedString( + string: label, + attributes: attributes as [NSAttributedString.Key: Any] + ) + attributedString.addAttribute( + .foregroundColor, + value: primaryColor, + range: (label as NSString).range(of: brandString) + ) + attributedString.addAttribute( + .foregroundColor, + value: primaryColor, + range: (label as NSString).range(of: last4) + ) + attributedString.addAttribute( + .font, + value: theme.emphasisFont, + range: (label as NSString).range(of: brandString) + ) + attributedString.addAttribute( + .font, + value: theme.emphasisFont, + range: (label as NSString).range(of: last4) + ) + + return attributedString + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} + +private let kDefaultIconWidth: CGFloat = 26.0 +private let kPadding: CGFloat = 15.0 +private let kCheckmarkWidth: CGFloat = 14.0 diff --git a/Stripe/StripeiOS/Source/STPPaymentOptionTuple.swift b/Stripe/StripeiOS/Source/STPPaymentOptionTuple.swift new file mode 100644 index 00000000..6084d132 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentOptionTuple.swift @@ -0,0 +1,88 @@ +// +// STPPaymentOptionTuple.swift +// StripeiOS +// +// Created by Jack Flintermann on 5/17/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation + +class STPPaymentOptionTuple: NSObject { + @objc + public convenience init( + paymentOptions: [STPPaymentOption], + selectedPaymentOption: STPPaymentOption? + ) { + self.init() + self.paymentOptions = paymentOptions + self.selectedPaymentOption = selectedPaymentOption + } + + @objc + public convenience init( + paymentOptions: [STPPaymentOption], + selectedPaymentOption: STPPaymentOption?, + addApplePayOption applePayEnabled: Bool, + addFPXOption fpxEnabled: Bool + ) { + var mutablePaymentOptions = paymentOptions + weak var selected = selectedPaymentOption + + if applePayEnabled { + let applePay = STPApplePayPaymentOption() + mutablePaymentOptions.append(applePay) + + if selected == nil { + selected = applePay + } + } + + if fpxEnabled { + let fpx = STPPaymentMethodFPXParams() + let fpxPaymentOption = STPPaymentMethodParams( + fpx: fpx, + billingDetails: nil, + metadata: nil + ) + mutablePaymentOptions.append(fpxPaymentOption) + } + + self.init( + paymentOptions: mutablePaymentOptions, + selectedPaymentOption: selected + ) + } + + /// Returns a tuple for the given array of STPPaymentMethod, filtered to only include the + /// the types supported by STPPaymentContext/STPPaymentOptionsViewController and adding + /// Apple Pay as a method if appropriate. + /// - Returns: A new tuple ready to be used by the SDK's UI elements + @objc(tupleFilteredForUIWithPaymentMethods:selectedPaymentMethod:configuration:) + public convenience init( + filteredForUIWith paymentMethods: [STPPaymentMethod], + selectedPaymentMethod selectedPaymentMethodID: String?, + configuration: STPPaymentConfiguration + ) { + var paymentOptions: [STPPaymentOption] = [] + var selectedPaymentMethod: STPPaymentMethod? + for paymentMethod in paymentMethods { + if paymentMethod.type == .card { + paymentOptions.append(paymentMethod) + if paymentMethod.stripeId == selectedPaymentMethodID { + selectedPaymentMethod = paymentMethod + } + } + } + + self.init( + paymentOptions: paymentOptions, + selectedPaymentOption: selectedPaymentMethod, + addApplePayOption: configuration.applePayEnabled, + addFPXOption: configuration.fpxEnabled + ) + } + + private(set) weak var selectedPaymentOption: STPPaymentOption? + private(set) var paymentOptions: [STPPaymentOption] = [] +} diff --git a/Stripe/StripeiOS/Source/STPPaymentOptionsInternalViewController.swift b/Stripe/StripeiOS/Source/STPPaymentOptionsInternalViewController.swift new file mode 100644 index 00000000..6a59e1b6 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentOptionsInternalViewController.swift @@ -0,0 +1,584 @@ +// +// STPPaymentOptionsInternalViewController.swift +// StripeiOS +// +// Created by Jack Flintermann on 6/9/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +import UIKit + +@objc protocol STPPaymentOptionsInternalViewControllerDelegate: AnyObject { + func internalViewControllerDidSelect(_ paymentOption: STPPaymentOption?) + func internalViewControllerDidDelete(_ paymentOption: STPPaymentOption?) + func internalViewControllerDidCreatePaymentOption( + _ paymentOption: STPPaymentOption?, + completion: @escaping STPErrorBlock + ) + func internalViewControllerDidCancel() +} + +class STPPaymentOptionsInternalViewController: STPCoreTableViewController, UITableViewDataSource, + UITableViewDelegate, STPAddCardViewControllerDelegate, STPBankSelectionViewControllerDelegate +{ + init( + configuration: STPPaymentConfiguration, + customerContext: STPCustomerContext?, + apiClient: STPAPIClient, + theme: STPTheme, + prefilledInformation: STPUserInformation?, + shippingAddress: STPAddress?, + paymentOptionTuple tuple: STPPaymentOptionTuple, + delegate: STPPaymentOptionsInternalViewControllerDelegate? + ) { + super.init(theme: theme) + self.configuration = configuration + // This parameter may be a custom API adapter, and not a CustomerContext. + apiAdapter = customerContext + self.apiClient = apiClient + self.prefilledInformation = prefilledInformation + self.shippingAddress = shippingAddress + paymentOptions = tuple.paymentOptions + selectedPaymentOption = tuple.selectedPaymentOption + self.delegate = delegate + + title = STPLocalizedString("Payment Method", "Title for Payment Method screen") + } + + func update(with tuple: STPPaymentOptionTuple) { + if let selectedPaymentOption = selectedPaymentOption, + selectedPaymentOption.isEqual(tuple.selectedPaymentOption) + { + return + } + + paymentOptions = tuple.paymentOptions + selectedPaymentOption = tuple.selectedPaymentOption + + // Reload card list section + let sections = NSMutableIndexSet(index: PaymentOptionSectionCardList) + tableView?.reloadSections(sections as IndexSet, with: .automatic) + } + + private var _customFooterView: UIView? + var customFooterView: UIView? { + get { + _customFooterView + } + set(footerView) { + _customFooterView = footerView + _didSetCustomFooterView() + } + } + func _didSetCustomFooterView() { + if isViewLoaded { + if let size = _customFooterView?.sizeThatFits( + CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) + ) { + _customFooterView?.frame = CGRect( + x: 0, + y: 0, + width: size.width, + height: size.height + ) + } + + tableView?.tableFooterView = _customFooterView + } + } + + var addCardViewControllerCustomFooterView: UIView? + var prefilledInformation: STPUserInformation? + private var configuration: STPPaymentConfiguration? + private var apiAdapter: STPBackendAPIAdapter? + private var shippingAddress: STPAddress? + private var paymentOptions: [STPPaymentOption]? + private var apiClient: STPAPIClient = .shared + private var selectedPaymentOption: STPPaymentOption? + private weak var delegate: STPPaymentOptionsInternalViewControllerDelegate? + private var cardImageView: UIImageView? + + override func createAndSetupViews() { + super.createAndSetupViews() + + // Table view + tableView?.register( + STPPaymentOptionTableViewCell.self, + forCellReuseIdentifier: PaymentOptionCellReuseIdentifier + ) + + tableView?.dataSource = self + tableView?.delegate = self + tableView?.reloadData() + + // Table header view + let cardImageView = UIImageView(image: STPLegacyImageLibrary.largeCardFrontImage()) + cardImageView.contentMode = .center + cardImageView.frame = CGRect( + x: 0.0, + y: 0.0, + width: view.bounds.size.width, + height: cardImageView.bounds.size.height + (57.0 * 2.0) + ) + cardImageView.image = STPLegacyImageLibrary.largeCardFrontImage() + cardImageView.tintColor = theme.accentColor + self.cardImageView = cardImageView + + tableView?.tableHeaderView = cardImageView + + // Table view editing state + tableView?.setEditing(false, animated: false) + reloadRightBarButtonItem( + withTableViewIsEditing: tableView?.isEditing ?? false, + animated: false + ) + + stp_navigationItemProxy?.leftBarButtonItem?.accessibilityIdentifier = + "PaymentOptionsViewControllerCancelButtonIdentifier" + // re-set the custom footer view if it was added before we loaded + _didSetCustomFooterView() + } + + override func viewDidLayoutSubviews() { + super.viewDidLayoutSubviews() + + // Resetting it re-calculates the size based on new view width + // UITableView requires us to call setter again to actually pick up frame + // change on footers + if tableView?.tableFooterView != nil { + customFooterView = tableView?.tableFooterView + } + } + + func reloadRightBarButtonItem(withTableViewIsEditing tableViewIsEditing: Bool, animated: Bool) { + var barButtonItem: UIBarButtonItem? + + if !tableViewIsEditing { + if isAnyPaymentOptionDetachable() { + // Show edit button + barButtonItem = UIBarButtonItem( + barButtonSystemItem: .edit, + target: self, + action: #selector(handleEditButtonTapped(_:)) + ) + } else { + // Show no button + barButtonItem = nil + } + } else { + // Show done button + barButtonItem = UIBarButtonItem( + barButtonSystemItem: .done, + target: self, + action: #selector(handleDoneButtonTapped(_:)) + ) + } + + barButtonItem?.stp_setTheme(theme) + + stp_navigationItemProxy?.setRightBarButton(barButtonItem, animated: animated) + } + + func isAnyPaymentOptionDetachable() -> Bool { + for paymentOption in cardPaymentOptions() { + if isPaymentOptionDetachable(paymentOption) { + return true + } + } + + return false + } + + func isPaymentOptionDetachable(_ paymentOption: STPPaymentOption?) -> Bool { + if !(configuration?.canDeletePaymentOptions ?? false) { + // Feature is disabled + return false + } + + if apiAdapter == nil { + // Cannot detach payment methods without customer context + return false + } + + if !(apiAdapter?.responds( + to: #selector(STPCustomerContext.detachPaymentMethod(fromCustomer:completion:)) + ) + ?? false) + { + // Cannot detach payment methods if customerContext is an apiAdapter + // that doesn't implement detachPaymentMethod + return false + } + + if paymentOption == nil { + // Cannot detach non-existent payment method + return false + } + + if !(paymentOption is STPPaymentMethod) { + // Cannot detach non-payment method + return false + } + + // Payment method can be deleted from customer + return true + } + + func cardPaymentOptions() -> [STPPaymentOption] { + guard let paymentOptions = paymentOptions else { + return [] + } + + return paymentOptions.filter({ (o) -> Bool in + if o is STPPaymentMethodParams { + let paymentMethodParams = o as? STPPaymentMethodParams + if paymentMethodParams?.type != .card { + return false + } + } + return true + }) + } + + func apmPaymentOptions() -> [STPPaymentOption] { + guard let paymentOptions = paymentOptions else { + return [] + } + return paymentOptions.filter({ (o) -> Bool in + if (o) is STPPaymentMethodParams { + let paymentMethodParams = o as? STPPaymentMethodParams + if paymentMethodParams?.type == .FPX { + // Add other APMs as we gain support for them in Basic Integration + return true + } + } + return false + }) + } + + // MARK: - Button Handlers + @objc override func handleCancelTapped(_ sender: Any?) { + delegate?.internalViewControllerDidCancel() + } + + @objc func handleEditButtonTapped(_ sender: Any?) { + tableView?.setEditing(true, animated: true) + reloadRightBarButtonItem( + withTableViewIsEditing: tableView?.isEditing ?? false, + animated: true + ) + } + + @objc func handleDoneButtonTapped(_ sender: Any?) { + _endTableViewEditing() + reloadRightBarButtonItem( + withTableViewIsEditing: tableView?.isEditing ?? false, + animated: true + ) + } + + func _endTableViewEditing() { + tableView?.setEditing(false, animated: true) + } + + // MARK: - UITableViewDataSource + func numberOfSections(in tableView: UITableView) -> Int { + if apmPaymentOptions().count > 0 { + return 3 + } else { + return 2 + } + } + + func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { + if section == PaymentOptionSectionCardList { + return cardPaymentOptions().count + } + + if section == PaymentOptionSectionAddCard { + return 1 + } + + if section == PaymentOptionSectionAPM { + return apmPaymentOptions().count + } + + return 0 + } + + func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { + let cell = + tableView.dequeueReusableCell( + withIdentifier: PaymentOptionCellReuseIdentifier, + for: indexPath + ) + as? STPPaymentOptionTableViewCell + + if indexPath.section == PaymentOptionSectionCardList { + weak var paymentOption = + cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) + let selected = paymentOption!.isEqual(selectedPaymentOption) + + cell?.configure(with: paymentOption!, theme: theme, selected: selected) + } else if indexPath.section == PaymentOptionSectionAddCard { + cell?.configureForNewCardRow(with: theme) + cell?.accessibilityIdentifier = "PaymentOptionsTableViewAddNewCardButtonIdentifier" + } else if indexPath.section == PaymentOptionSectionAPM { + weak var paymentOption = + apmPaymentOptions().stp_boundSafeObject(at: indexPath.row) + if paymentOption is STPPaymentMethodParams { + let paymentMethodParams = paymentOption as? STPPaymentMethodParams + if paymentMethodParams?.type == .FPX { + cell?.configureForFPXRow(with: theme) + cell?.accessibilityIdentifier = "PaymentOptionsTableViewFPXButtonIdentifier" + } + } + } + + return cell! + } + + func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool { + if indexPath.section == PaymentOptionSectionCardList { + weak var paymentOption = + cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) + + if isPaymentOptionDetachable(paymentOption) { + return true + } + } + + return false + } + + func tableView( + _ tableView: UITableView, + commit editingStyle: UITableViewCell.EditingStyle, + forRowAt indexPath: IndexPath + ) { + if indexPath.section == PaymentOptionSectionCardList { + if editingStyle != .delete { + // Showed the user a non-delete option when we shouldn't have + tableView.reloadData() + return + } + + if !(indexPath.row < cardPaymentOptions().count) { + // Data source and table view out of sync for some reason + tableView.reloadData() + return + } + + weak var paymentOptionToDelete = + cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) + + if !isPaymentOptionDetachable(paymentOptionToDelete) { + // Showed the user a delete option for a payment method when we shouldn't have + tableView.reloadData() + return + } + + let paymentMethod = paymentOptionToDelete as? STPPaymentMethod + + // Kickoff request to delete payment method from customer + if let paymentMethod = paymentMethod { + apiAdapter?.detachPaymentMethod?(fromCustomer: paymentMethod, completion: nil) + } + + // Optimistically remove payment method from data source + var paymentOptions = self.paymentOptions + paymentOptions?.removeAll { $0 as AnyObject === paymentOptionToDelete as AnyObject } + self.paymentOptions = paymentOptions + + // Perform deletion animation for single row + tableView.deleteRows(at: [indexPath], with: .automatic) + + var tableViewIsEditing = tableView.isEditing + if !isAnyPaymentOptionDetachable() { + // we deleted the last available payment option, stop editing + // (but delay to next runloop because calling tableView setEditing:animated: + // in this function is not allowed) + DispatchQueue.main.async(execute: { + self._endTableViewEditing() + }) + // manually set the value passed to reloadRightBarButtonItemWithTableViewIsEditing + // below + tableViewIsEditing = false + } + + // Reload right bar button item text + reloadRightBarButtonItem(withTableViewIsEditing: tableViewIsEditing, animated: true) + + // Notify delegate + delegate?.internalViewControllerDidDelete(paymentOptionToDelete) + } + } + + // MARK: - UITableViewDelegate + func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { + if indexPath.section == PaymentOptionSectionCardList { + // Update data source + weak var paymentOption = + cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) + selectedPaymentOption = paymentOption + + // Perform selection animation + tableView.reloadSections( + NSIndexSet(index: PaymentOptionSectionCardList) as IndexSet, + with: .fade + ) + + // Notify delegate + delegate?.internalViewControllerDidSelect(paymentOption) + } else if indexPath.section == PaymentOptionSectionAddCard { + var paymentCardViewController: STPAddCardViewController? + if let configuration = configuration { + paymentCardViewController = STPAddCardViewController( + configuration: configuration, + theme: theme + ) + } + paymentCardViewController?.apiClient = apiClient + paymentCardViewController?.delegate = self + paymentCardViewController?.prefilledInformation = prefilledInformation + paymentCardViewController?.shippingAddress = shippingAddress + paymentCardViewController?.customFooterView = addCardViewControllerCustomFooterView + + if let paymentCardViewController = paymentCardViewController { + navigationController?.pushViewController(paymentCardViewController, animated: true) + } + } else if indexPath.section == PaymentOptionSectionAPM { + weak var paymentOption = + apmPaymentOptions().stp_boundSafeObject(at: indexPath.row) + if paymentOption is STPPaymentMethodParams { + if let paymentMethodParams = paymentOption as? STPPaymentMethodParams, + paymentMethodParams.type == .FPX + { + var bankSelectionViewController: STPBankSelectionViewController? + if let configuration = configuration { + bankSelectionViewController = STPBankSelectionViewController( + bankMethod: .FPX, + configuration: configuration, + theme: theme + ) + } + bankSelectionViewController?.apiClient = apiClient + bankSelectionViewController?.delegate = self + + if let bankSelectionViewController = bankSelectionViewController { + navigationController?.pushViewController( + bankSelectionViewController, + animated: true + ) + } + } + } + } + + tableView.deselectRow(at: indexPath, animated: true) + } + + func tableView( + _ tableView: UITableView, + willDisplay cell: UITableViewCell, + forRowAt indexPath: IndexPath + ) { + let isTopRow = indexPath.row == 0 + let isBottomRow = + self.tableView(tableView, numberOfRowsInSection: indexPath.section) - 1 == indexPath.row + + cell.stp_setBorderColor(theme.tertiaryBackgroundColor) + cell.stp_setTopBorderHidden(!isTopRow) + cell.stp_setBottomBorderHidden(!isBottomRow) + cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) + cell.stp_setFakeSeparatorLeftInset(15.0) + } + + func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat { + if self.tableView(tableView, numberOfRowsInSection: section) == 0 { + return 0.01 + } + + return 27.0 + } + + override func tableView( + _ tableView: UITableView, + heightForHeaderInSection section: Int + ) + -> CGFloat + { + return 0.01 + } + + func tableView( + _ tableView: UITableView, + editingStyleForRowAt indexPath: IndexPath + ) + -> UITableViewCell.EditingStyle + { + if indexPath.section == PaymentOptionSectionCardList { + return .delete + } + + return .none + } + + func tableView(_ tableView: UITableView, willBeginEditingRowAt indexPath: IndexPath) { + reloadRightBarButtonItem(withTableViewIsEditing: true, animated: true) + } + + func tableView(_ tableView: UITableView, didEndEditingRowAt indexPath: IndexPath?) { + reloadRightBarButtonItem(withTableViewIsEditing: tableView.isEditing, animated: true) + } + + // MARK: - STPAddCardViewControllerDelegate + func addCardViewControllerDidCancel(_ addCardViewController: STPAddCardViewController) { + navigationController?.popViewController(animated: true) + } + + @objc func addCardViewController( + _ addCardViewController: STPAddCardViewController, + didCreatePaymentMethod paymentMethod: STPPaymentMethod, + completion: @escaping STPErrorBlock + ) { + delegate?.internalViewControllerDidCreatePaymentOption( + paymentMethod, + completion: completion + ) + } + + @objc func bankSelectionViewController( + _ bankViewController: STPBankSelectionViewController, + didCreatePaymentMethodParams paymentMethodParams: STPPaymentMethodParams + ) { + delegate?.internalViewControllerDidCreatePaymentOption(paymentMethodParams) { _ in + } + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } + + required init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + fatalError("init(nibName:bundle:) has not been implemented") + } + + required init( + theme: STPTheme? + ) { + fatalError("init(theme:) has not been implemented") + } +} + +private let PaymentOptionCellReuseIdentifier = "PaymentOptionCellReuseIdentifier" +private let PaymentOptionSectionCardList = 0 +private let PaymentOptionSectionAddCard = 1 +private let PaymentOptionSectionAPM = 2 diff --git a/Stripe/StripeiOS/Source/STPPaymentOptionsViewController.swift b/Stripe/StripeiOS/Source/STPPaymentOptionsViewController.swift new file mode 100644 index 00000000..9476a3bd --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentOptionsViewController.swift @@ -0,0 +1,655 @@ +// +// STPPaymentOptionsViewController.swift +// StripeiOS +// +// Created by Jack Flintermann on 1/12/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +import UIKit + +/// This view controller presents a list of payment method options to the user, +/// which they can select between. They can also add credit cards to the list. +/// It must be displayed inside a `UINavigationController`, so you can either +/// create a `UINavigationController` with an `STPPaymentOptionsViewController` +/// as the `rootViewController` and then present the `UINavigationController`, +/// or push a new `STPPaymentOptionsViewController` onto an existing +/// `UINavigationController`'s stack. You can also have `STPPaymentContext` do this +/// for you automatically, by calling `presentPaymentOptionsViewController` +/// or `pushPaymentOptionsViewController` on it. +public class STPPaymentOptionsViewController: STPCoreViewController, + STPPaymentOptionsInternalViewControllerDelegate, STPAddCardViewControllerDelegate +{ + + /// The delegate for the view controller. + /// The delegate receives callbacks when the user selects a method or cancels, + /// and is responsible for dismissing the payments methods view controller when + /// it is finished. + @objc private(set) weak var delegate: STPPaymentOptionsViewControllerDelegate? + + /// Creates a new payment methods view controller. + /// - Parameter paymentContext: A payment context to power the view controller's view. + /// The payment context will in turn use its backend API adapter to fetch the + /// information it needs from your application. + /// - Returns: an initialized view controller. + @objc(initWithPaymentContext:) + public convenience init( + paymentContext: STPPaymentContext + ) { + self.init( + configuration: paymentContext.configuration, + apiAdapter: paymentContext.apiAdapter, + apiClient: paymentContext.apiClient, + loadingPromise: paymentContext.currentValuePromise, + theme: paymentContext.theme, + shippingAddress: paymentContext.shippingAddress, + delegate: paymentContext + ) + } + + init( + configuration: STPPaymentConfiguration?, + apiAdapter: STPBackendAPIAdapter, + apiClient: STPAPIClient?, + loadingPromise: STPPromise?, + theme: STPTheme?, + shippingAddress: STPAddress?, + delegate: STPPaymentOptionsViewControllerDelegate + ) { + self.apiAdapter = apiAdapter + super.init(theme: theme) + commonInit( + configuration: configuration, + apiAdapter: apiAdapter, + apiClient: apiClient, + loadingPromise: loadingPromise, + shippingAddress: shippingAddress, + delegate: delegate + ) + } + + func commonInit( + configuration: STPPaymentConfiguration?, + apiAdapter: STPBackendAPIAdapter, + apiClient: STPAPIClient?, + loadingPromise: STPPromise?, + shippingAddress: STPAddress?, + delegate: STPPaymentOptionsViewControllerDelegate + ) { + STPAnalyticsClient.sharedClient.addClass( + toProductUsageIfNecessary: STPPaymentOptionsViewController.self + ) + + self.configuration = configuration + self.apiClient = apiClient ?? .shared + self.shippingAddress = shippingAddress + self.apiAdapter = apiAdapter + self.loadingPromise = loadingPromise + self.delegate = delegate + + navigationItem.title = STPLocalizedString( + "Loading…", + "Title for screen when data is still loading from the network." + ) + + weak var weakSelf = self + loadingPromise?.onSuccess({ tuple in + guard let strongSelf = weakSelf else { + return + } + var `internal`: UIViewController? + if (tuple.paymentOptions.count) > 0 { + let customerContext = strongSelf.apiAdapter as? STPCustomerContext + + var payMethodsInternal: STPPaymentOptionsInternalViewController? + if let configuration1 = strongSelf.configuration { + payMethodsInternal = STPPaymentOptionsInternalViewController( + configuration: configuration1, + customerContext: customerContext, + apiClient: strongSelf.apiClient, + theme: strongSelf.theme, + prefilledInformation: strongSelf.prefilledInformation, + shippingAddress: strongSelf.shippingAddress, + paymentOptionTuple: tuple, + delegate: strongSelf + ) + } + if strongSelf.paymentOptionsViewControllerFooterView != nil { + payMethodsInternal?.customFooterView = + strongSelf.paymentOptionsViewControllerFooterView + } + if strongSelf.addCardViewControllerFooterView != nil { + payMethodsInternal?.addCardViewControllerCustomFooterView = + strongSelf.addCardViewControllerFooterView + } + `internal` = payMethodsInternal + } else { + var addCardViewController: STPAddCardViewController? + if let configuration1 = strongSelf.configuration { + addCardViewController = STPAddCardViewController( + configuration: configuration1, + theme: strongSelf.theme + ) + } + addCardViewController?.apiClient = strongSelf.apiClient + addCardViewController?.delegate = strongSelf + addCardViewController?.prefilledInformation = strongSelf.prefilledInformation + addCardViewController?.shippingAddress = strongSelf.shippingAddress + `internal` = addCardViewController + + if strongSelf.addCardViewControllerFooterView != nil { + addCardViewController?.customFooterView = + strongSelf.addCardViewControllerFooterView + } + } + + `internal`?.stp_navigationItemProxy = strongSelf.navigationItem + if let controller = `internal` { + strongSelf.addChild(controller) + } + `internal`?.view.alpha = 0 + if let view = `internal`?.view, let activityIndicator1 = strongSelf.activityIndicator { + strongSelf.view.insertSubview(view, belowSubview: activityIndicator1) + } + if let view = `internal`?.view { + strongSelf.view.addSubview(view) + } + `internal`?.view.frame = strongSelf.view.bounds + `internal`?.didMove(toParent: strongSelf) + UIView.animate( + withDuration: 0.2, + animations: { + strongSelf.activityIndicator?.alpha = 0 + `internal`?.view.alpha = 1 + } + ) { _ in + strongSelf.activityIndicator?.animating = false + } + strongSelf.navigationItem.setRightBarButton( + `internal`?.stp_navigationItemProxy?.rightBarButtonItem, + animated: true + ) + strongSelf.internalViewController = `internal` + }) + } + + /// Initializes a new payment methods view controller without using a + /// payment context. + /// - Parameters: + /// - configuration: The configuration to use to determine what types of + /// payment method to offer your user. - seealso: STPPaymentConfiguration.h + /// - theme: The theme to inform the appearance of the UI. + /// - customerContext: The customer context the view controller will use to + /// fetch and modify its Stripe customer + /// - delegate: A delegate that will be notified when the payment + /// methods view controller's selection changes. + /// - Returns: an initialized view controller. + @objc(initWithConfiguration:theme:customerContext:delegate:) + public convenience init( + configuration: STPPaymentConfiguration, + theme: STPTheme, + customerContext: STPCustomerContext, + delegate: STPPaymentOptionsViewControllerDelegate + ) { + self.init( + configuration: configuration, + theme: theme, + apiAdapter: customerContext, + delegate: delegate + ) + } + + /// Note: Instead of providing your own backend API adapter, we recommend using + /// `STPCustomerContext`, which will manage retrieving and updating a + /// Stripe customer for you. - seealso: STPCustomerContext.h + /// Initializes a new payment methods view controller without using + /// a payment context. + /// - Parameters: + /// - configuration: The configuration to use to determine what types of + /// payment method to offer your user. + /// - theme: The theme to inform the appearance of the UI. + /// - apiAdapter: The API adapter to use to retrieve a customer's stored + /// payment methods and save new ones. + /// - delegate: A delegate that will be notified when the payment methods + /// view controller's selection changes. + @objc(initWithConfiguration:theme:apiAdapter:delegate:) + public init( + configuration: STPPaymentConfiguration, + theme: STPTheme, + apiAdapter: STPBackendAPIAdapter, + delegate: STPPaymentOptionsViewControllerDelegate + ) { + self.apiAdapter = apiAdapter + super.init(theme: theme) + let promise = retrievePaymentMethods(with: configuration, apiAdapter: apiAdapter) + + commonInit( + configuration: configuration, + apiAdapter: apiAdapter, + apiClient: STPAPIClient.shared, + loadingPromise: promise, + shippingAddress: nil, + delegate: delegate + ) + } + + /// If you've already collected some information from your user, you can set it + /// here and it'll be automatically filled out when possible/appropriate in any UI + /// that the payment context creates. + @objc public var prefilledInformation: STPUserInformation? { + didSet { + if let payMethodsInternal = internalViewController as? STPPaymentOptionsInternalViewController { + payMethodsInternal.prefilledInformation = prefilledInformation + } else if let payMethodsInternal = internalViewController as? STPAddCardViewController { + payMethodsInternal.prefilledInformation = prefilledInformation + } + } + } + /// @note This is no longer recommended as of v18.3.0 - the SDK automatically saves the Stripe ID of the last selected + /// payment method using NSUserDefaults and displays it as the default pre-selected option. You can override this behavior + /// by setting this property. + /// The Stripe ID of a payment method to display as the default pre-selected option. + /// @note Setting this after the view controller's view has loaded has no effect. + @objc public var defaultPaymentMethod: String? + /// A view that will be placed as the footer of the view controller when it is + /// showing a list of saved payment methods to select from. + /// When the footer view needs to be resized, it will be sent a + /// `sizeThatFits:` call. The view should respond correctly to this method in order + /// to be sized and positioned properly. + @objc public var paymentOptionsViewControllerFooterView: UIView? { + didSet { + if let payMethodsInternal = internalViewController as? STPPaymentOptionsInternalViewController { + payMethodsInternal.customFooterView = paymentOptionsViewControllerFooterView + } + } + } + + /// A view that will be placed as the footer of the view controller when it is + /// showing the add card view. + /// When the footer view needs to be resized, it will be sent a + /// `sizeThatFits:` call. The view should respond correctly to this method in order + /// to be sized and positioned properly. + @objc public var addCardViewControllerFooterView: UIView? { + didSet { + if let payMethodsInternal = internalViewController as? STPPaymentOptionsInternalViewController { + payMethodsInternal.addCardViewControllerCustomFooterView = addCardViewControllerFooterView + } else if let payMethodsInternal = internalViewController as? STPAddCardViewController { + payMethodsInternal.customFooterView = addCardViewControllerFooterView + } + } + } + + /// The API Client to use to make requests. + /// Defaults to STPAPIClient.shared + public var apiClient: STPAPIClient = .shared + + /// If you're pushing `STPPaymentOptionsViewController` onto an existing + /// `UINavigationController`'s stack, you should use this method to dismiss it, + /// since it may have pushed an additional add card view controller onto the + /// navigation controller's stack. + /// - Parameter completion: The callback to run after the view controller is dismissed. + /// You may specify nil for this parameter. + @objc(dismissWithCompletion:) + public func dismiss(withCompletion completion: STPVoidBlock?) { + if stp_isAtRootOfNavigationController() { + presentingViewController?.dismiss(animated: true, completion: completion) + } else { + var previous = navigationController?.viewControllers.first + for viewController in navigationController?.viewControllers ?? [] { + if viewController == self { + break + } + previous = viewController + } + navigationController?.stp_pop( + to: previous, + animated: true, + completion: completion ?? {} + ) + } + } + + /// Use one of the initializers declared in this interface. + @available( + *, + unavailable, + message: "Use one of the initializers declared in this interface instead." + ) + @objc public required init( + theme: STPTheme? + ) { + fatalError("init(theme:) has not been implemented") + } + + /// Use one of the initializers declared in this interface. + @available( + *, + unavailable, + message: "Use one of the initializers declared in this interface instead." + ) + @objc public required init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + fatalError("init(nibName:bundle:) has not been implemented") + } + + /// Use one of the initializers declared in this interface. + @available( + *, + unavailable, + message: "Use one of the initializers declared in this interface instead." + ) + @objc public required init?( + coder aDecoder: NSCoder + ) { + fatalError("init(coder:) has not been implemented") + } + + private var configuration: STPPaymentConfiguration? + private var shippingAddress: STPAddress? + private var apiAdapter: STPBackendAPIAdapter + var loadingPromise: STPPromise? + private var activityIndicator: STPPaymentActivityIndicatorView? + internal var internalViewController: UIViewController? + + func retrievePaymentMethods( + with configuration: STPPaymentConfiguration, + apiAdapter: STPBackendAPIAdapter? + ) -> STPPromise { + let promise = STPPromise() + apiAdapter?.listPaymentMethodsForCustomer(completion: { paymentMethods, error in + // We don't use stpDispatchToMainThreadIfNecessary here because we want this completion block to always be called asynchronously, so that users can set self.defaultPaymentMethod in time. + DispatchQueue.main.async(execute: { + if let error = error { + promise.fail(error) + } else { + let defaultPaymentMethod = self.defaultPaymentMethod + if defaultPaymentMethod == nil && (apiAdapter is STPCustomerContext) { + // Retrieve the last selected payment method saved by STPCustomerContext + (apiAdapter as? STPCustomerContext)? + .retrieveLastSelectedPaymentMethodIDForCustomer( + completion: { paymentMethodID, _ in + var paymentTuple: STPPaymentOptionTuple? + if let paymentMethods = paymentMethods { + paymentTuple = STPPaymentOptionTuple.init( + filteredForUIWith: paymentMethods, + selectedPaymentMethod: paymentMethodID, + configuration: configuration + ) + } + promise.succeed(paymentTuple!) + }) + } + var paymentTuple: STPPaymentOptionTuple? + if let paymentMethods = paymentMethods { + paymentTuple = STPPaymentOptionTuple.init( + filteredForUIWith: paymentMethods, + selectedPaymentMethod: defaultPaymentMethod, + configuration: configuration + ) + } + promise.succeed(paymentTuple!) + } + }) + }) + return promise + } + + override func createAndSetupViews() { + super.createAndSetupViews() + + let activityIndicator = STPPaymentActivityIndicatorView() + activityIndicator.animating = true + view.addSubview(activityIndicator) + self.activityIndicator = activityIndicator + } + + /// :nodoc: + @objc + public override func viewDidLayoutSubviews() { + super.viewDidLayoutSubviews() + let centerX = (view.frame.size.width - (activityIndicator?.frame.size.width ?? 0.0)) / 2 + let centerY = (view.frame.size.height - (activityIndicator?.frame.size.height ?? 0.0)) / 2 + activityIndicator?.frame = CGRect( + x: centerX, + y: centerY, + width: activityIndicator?.frame.size.width ?? 0.0, + height: activityIndicator?.frame.size.height ?? 0.0 + ) + internalViewController?.view.frame = view.bounds + } + + /// :nodoc: + @objc + public override func viewDidAppear(_ animated: Bool) { + super.viewDidAppear(animated) + weak var weakSelf = self + loadingPromise?.onSuccess({ tuple in + let strongSelf = weakSelf + if strongSelf == nil { + return + } + + if tuple.selectedPaymentOption != nil { + if strongSelf?.delegate?.responds( + to: #selector( + STPPaymentOptionsViewControllerDelegate.paymentOptionsViewController( + _: + didSelect: + )) + ) + ?? false + { + if let strongSelf = strongSelf, + let selectedPaymentOption = tuple.selectedPaymentOption + { + strongSelf.delegate?.paymentOptionsViewController?( + strongSelf, + didSelect: selectedPaymentOption + ) + } + } + } + }).onFailure({ error in + let strongSelf = weakSelf + if strongSelf == nil { + return + } + + if let strongSelf = strongSelf { + strongSelf.delegate?.paymentOptionsViewController( + strongSelf, + didFailToLoadWithError: error + ) + } + }) + } + + @objc override func updateAppearance() { + super.updateAppearance() + + activityIndicator?.tintColor = theme.accentColor + } + + func finish(with paymentOption: STPPaymentOption?) { + let isReusablePaymentMethod = + (paymentOption is STPPaymentMethod) + && (paymentOption as? STPPaymentMethod)?.isReusable ?? false + + if apiAdapter is STPCustomerContext { + if isReusablePaymentMethod { + // Save the payment method + let paymentMethod = paymentOption as? STPPaymentMethod + (apiAdapter as? STPCustomerContext)?.saveLastSelectedPaymentMethodID( + forCustomer: paymentMethod?.stripeId ?? "", + completion: nil + ) + } else { + // The customer selected something else (like Apple Pay) + (apiAdapter as? STPCustomerContext)?.saveLastSelectedPaymentMethodID( + forCustomer: nil, + completion: nil + ) + } + } + + if delegate?.responds( + to: #selector( + STPPaymentOptionsViewControllerDelegate.paymentOptionsViewController(_:didSelect:)) + ) + ?? false + { + if let paymentOption = paymentOption { + delegate?.paymentOptionsViewController?(self, didSelect: paymentOption) + } + } + delegate?.paymentOptionsViewControllerDidFinish(self) + } + + func internalViewControllerDidSelect(_ paymentOption: STPPaymentOption?) { + finish(with: paymentOption) + } + + func internalViewControllerDidDelete(_ paymentOption: STPPaymentOption?) { + if delegate is STPPaymentContext { + // Notify payment context to update its copy of payment methods + if let paymentContext = delegate as? STPPaymentContext, + let paymentOption = paymentOption + { + paymentContext.remove(paymentOption) + } + } + } + + func internalViewControllerDidCreatePaymentOption( + _ paymentOption: STPPaymentOption?, + completion: @escaping STPErrorBlock + ) { + if !(paymentOption?.isReusable ?? false) { + // Don't save a non-reusable payment option + finish(with: paymentOption) + return + } + let paymentMethod = paymentOption as? STPPaymentMethod + if let paymentMethod = paymentMethod { + apiAdapter.attachPaymentMethod(toCustomer: paymentMethod) { error in + stpDispatchToMainThreadIfNecessary({ + completion(error) + if error == nil { + var promise: STPPromise? + if let configuration = self.configuration { + promise = self.retrievePaymentMethods( + with: configuration, + apiAdapter: self.apiAdapter + ) + } + weak var weakSelf = self + promise?.onSuccess({ tuple in + let strongSelf = weakSelf + if strongSelf == nil { + return + } + let paymentTuple = STPPaymentOptionTuple( + paymentOptions: tuple.paymentOptions, + selectedPaymentOption: paymentMethod + ) + if strongSelf?.internalViewController + is STPPaymentOptionsInternalViewController + { + let paymentOptionsVC = + strongSelf?.internalViewController + as? STPPaymentOptionsInternalViewController + paymentOptionsVC?.update(with: paymentTuple) + } + }) + self.finish(with: paymentMethod) + } + }) + } + } + } + + func internalViewControllerDidCancel() { + delegate?.paymentOptionsViewControllerDidCancel(self) + } + + @objc override func handleCancelTapped(_ sender: Any?) { + delegate?.paymentOptionsViewControllerDidCancel(self) + } + + @objc + public func addCardViewControllerDidCancel( + _ addCardViewController: STPAddCardViewController + ) { + // Add card is only our direct delegate if there are no other payment methods possible + // and we skipped directly to this screen. In this case, a cancel from it is the same as a cancel to us. + delegate?.paymentOptionsViewControllerDidCancel(self) + } + + @objc + public func addCardViewController( + _ addCardViewController: STPAddCardViewController, + didCreatePaymentMethod paymentMethod: STPPaymentMethod, + completion: @escaping STPErrorBlock + ) { + internalViewControllerDidCreatePaymentOption(paymentMethod, completion: completion) + } +} + +// MARK: - STPPaymentOptionsViewControllerDelegate + +/// An `STPPaymentOptionsViewControllerDelegate` responds when a user selects a +/// payment option from (or cancels) an `STPPaymentOptionsViewController`. In both +/// of these instances, you should dismiss the view controller (either by popping +/// it off the navigation stack, or dismissing it). +@objc public protocol STPPaymentOptionsViewControllerDelegate: NSObjectProtocol { + /// This is called when the view controller encounters an error fetching the user's + /// payment options from its API adapter. You should dismiss the view controller + /// when this is called. + /// - Parameters: + /// - paymentOptionsViewController: the view controller in question + /// - error: the error that occurred + func paymentOptionsViewController( + _ paymentOptionsViewController: STPPaymentOptionsViewController, + didFailToLoadWithError error: Error + ) + /// This is called when the user selects or adds a payment method, so it will often + /// be called immediately after calling `paymentOptionsViewController:didSelectPaymentOption:`. + /// You should dismiss the view controller when this is called. + /// - Parameter paymentOptionsViewController: the view controller that has finished + func paymentOptionsViewControllerDidFinish( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) + /// This is called when the user taps "cancel". + /// You should dismiss the view controller when this is called. + /// - Parameter paymentOptionsViewController: the view controller that has finished + func paymentOptionsViewControllerDidCancel( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) + + /// This is called when the user either makes a selection, or adds a new card. + /// This will be triggered after the view controller loads with the user's current + /// selection (if they have one) and then subsequently when they change their + /// choice. You should use this callback to update any necessary UI in your app + /// that displays the user's currently selected payment method. You should *not* + /// dismiss the view controller at this point, instead do this in + /// `paymentOptionsViewControllerDidFinish:`. `STPPaymentOptionsViewController` + /// will also call the necessary methods on your API adapter, so you don't need to + /// call them directly during this method. + /// - Parameters: + /// - paymentOptionsViewController: the view controller in question + /// - paymentOption: the selected payment method + @objc(paymentOptionsViewController:didSelectPaymentOption:) + optional func paymentOptionsViewController( + _ paymentOptionsViewController: STPPaymentOptionsViewController, + didSelect paymentOption: STPPaymentOption + ) +} + +/// :nodoc: +@_spi(STP) extension STPPaymentOptionsViewController: STPAnalyticsProtocol { + @_spi(STP) public static var stp_analyticsIdentifier = "STPPaymentOptionsViewController" +} diff --git a/Stripe/StripeiOS/Source/STPPaymentResult.swift b/Stripe/StripeiOS/Source/STPPaymentResult.swift new file mode 100644 index 00000000..0a170738 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPaymentResult.swift @@ -0,0 +1,43 @@ +// +// STPPaymentResult.swift +// StripeiOS +// +// Created by Jack Flintermann on 1/15/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation + +/// When you're using `STPPaymentContext` to request your user's payment details, this is the object that will be returned to your application when they've successfully made a payment. +/// See https://stripe.com/docs/mobile/ios/basic#submit-payment-intents +public class STPPaymentResult: NSObject { + /// The payment method that the user has selected. This may come from a variety of different payment methods, such as an Apple Pay payment or a stored credit card. - seealso: STPPaymentMethod.h + /// If paymentMethod is nil, paymentMethodParams will be populated instead. + @objc public private(set) var paymentMethod: STPPaymentMethod? + /// The parameters for a payment method that the user has selected. This is + /// populated for non-reusable payment methods, such as FPX and iDEAL. - seealso: STPPaymentMethodParams.h + /// If paymentMethodParams is nil, paymentMethod will be populated instead. + @objc public private(set) var paymentMethodParams: STPPaymentMethodParams? + /// The STPPaymentOption that was used to initialize this STPPaymentResult, either an STPPaymentMethod or an STPPaymentMethodParams. + + @objc public weak var paymentOption: STPPaymentOption? { + if paymentMethod != nil { + return paymentMethod + } else { + return paymentMethodParams + } + } + + /// Initializes the payment result with a given payment option. This is invoked by `STPPaymentContext` internally; you shouldn't have to call it directly. + @objc + public init( + paymentOption: STPPaymentOption? + ) { + super.init() + if paymentOption is STPPaymentMethod { + paymentMethod = paymentOption as? STPPaymentMethod + } else if paymentOption is STPPaymentMethodParams { + paymentMethodParams = paymentOption as? STPPaymentMethodParams + } + } +} diff --git a/Stripe/StripeiOS/Source/STPPinManagementService.swift b/Stripe/StripeiOS/Source/STPPinManagementService.swift new file mode 100644 index 00000000..bbf471b6 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPinManagementService.swift @@ -0,0 +1,143 @@ +// +// STPPinManagementService.swift +// StripeiOS +// +// Created by Arnaud Cavailhez on 4/29/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePayments +@_spi(STP) import StripePaymentsUI +import UIKit + +/// STPAPIClient extensions to manage PIN on Stripe Issuing cards +public class STPPinManagementService: NSObject { + /// The API Client to use to make requests. + /// Defaults to STPAPIClient.shared + public var apiClient: STPAPIClient = STPAPIClient.shared + + /// Create a STPPinManagementService, you must provide an implementation of STPIssuingCardEphemeralKeyProvider + @objc + public init( + keyProvider: STPIssuingCardEphemeralKeyProvider + ) { + super.init() + keyManager = STPEphemeralKeyManager( + keyProvider: keyProvider as Any, + apiVersion: STPAPIClient.apiVersion, + performsEagerFetching: false + ) + } + + /// Retrieves a PIN number for a given card, + /// this call is asynchronous, implement the completion block to receive the updates + @objc + public func retrievePin( + _ cardId: String, + verificationId: String, + oneTimeCode: String, + completion: @escaping STPPinCompletionBlock + ) { + let endpoint = "issuing/cards/\(cardId)/pin" + let parameters = [ + "verification": [ + "id": verificationId, + "one_time_code": oneTimeCode, + ], + ] + keyManager?.getOrCreateKey({ ephemeralKey, keyError in + if ephemeralKey == nil { + completion(nil, .ephemeralKeyError, keyError) + return + } + + if let ephemeralKey = ephemeralKey { + APIRequest.getWith( + self.apiClient, + endpoint: endpoint, + additionalHeaders: self.apiClient.authorizationHeader(using: ephemeralKey), + parameters: parameters + ) { details, _, error in + // Find if there were errors + if details?.error != nil { + let code = details?.error?["code"] as? String + if "api_key_expired" == code { + completion(nil, .ephemeralKeyError, error) + } else if "expired" == code { + completion(nil, .errorVerificationExpired, nil) + } else if "incorrect_code" == code { + completion(nil, .errorVerificationCodeIncorrect, nil) + } else if "too_many_attempts" == code { + completion(nil, .errorVerificationTooManyAttempts, nil) + } else if "already_redeemed" == code { + completion(nil, .errorVerificationAlreadyRedeemed, nil) + } else { + completion(nil, .unknownError, error) + } + return + } + completion(details, .success, nil) + } + } + }) + } + + /// Updates a PIN number for a given card, + /// this call is asynchronous, implement the completion block to receive the updates + @objc + public func updatePin( + _ cardId: String, + newPin: String, + verificationId: String, + oneTimeCode: String, + completion: @escaping STPPinCompletionBlock + ) { + let endpoint = "issuing/cards/\(cardId)/pin" + let parameters = + [ + "verification": [ + "id": verificationId, + "one_time_code": oneTimeCode, + ], + "pin": newPin, + ] as [String: Any] + keyManager?.getOrCreateKey({ ephemeralKey, keyError in + if ephemeralKey == nil { + completion(nil, .ephemeralKeyError, keyError) + return + } + if let ephemeralKey = ephemeralKey { + APIRequest.post( + with: self.apiClient, + endpoint: endpoint, + additionalHeaders: self.apiClient.authorizationHeader(using: ephemeralKey), + parameters: parameters + ) { details, _, error in + // Find if there were errors + if details?.error != nil { + let code = details?.error?["code"] as? String + if "api_key_expired" == code { + completion(nil, .ephemeralKeyError, error) + } else if "expired" == code { + completion(nil, .errorVerificationExpired, nil) + } else if "incorrect_code" == code { + completion(nil, .errorVerificationCodeIncorrect, nil) + } else if "too_many_attempts" == code { + completion(nil, .errorVerificationTooManyAttempts, nil) + } else if "already_redeemed" == code { + completion(nil, .errorVerificationAlreadyRedeemed, nil) + } else { + completion(nil, .unknownError, error) + } + return + } + completion(details, .success, nil) + } + } + }) + } + + private var keyManager: STPEphemeralKeyManagerProtocol? +} diff --git a/Stripe/StripeiOS/Source/STPPushProvisioningContext.swift b/Stripe/StripeiOS/Source/STPPushProvisioningContext.swift new file mode 100644 index 00000000..62973701 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPushProvisioningContext.swift @@ -0,0 +1,144 @@ +// +// STPPushProvisioningContext.swift +// StripeiOS +// +// Created by Jack Flintermann on 9/27/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI + +/// This class makes it easier to implement "Push Provisioning", the process by which an end-user can add a card to their Apple Pay wallet without having to type their number. This process is mediated by an Apple class called `PKAddPaymentPassViewController`; this class will help you implement that class' delegate methods. Note that this flow requires a special entitlement from Apple; for more information please see https://stripe.com/docs/issuing/cards/digital-wallets . +public class STPPushProvisioningContext: NSObject { + /// The API Client to use to make requests. + /// Defaults to STPAPIClient.shared + public var apiClient: STPAPIClient = .shared + + /// This is a helper method to generate a PKAddPaymentPassRequestConfiguration that will work with + /// Stripe's Issuing APIs. Pass the returned configuration object to `PKAddPaymentPassViewController`'s `initWithRequestConfiguration:delegate:` initializer. + /// @deprecated Use requestConfiguration(withName:description:last4:brand:primaryAccountIdentifier:) instead. + /// - Parameters: + /// - name: Your cardholder's name. Example: John Appleseed + /// - description: A localized description of your card's name. This will appear in Apple's UI as "{description} will be available in Wallet". Example: Platinum Rewards Card + /// - last4: The last 4 of the card to be added to the user's Apple Pay wallet. Example: 4242 + /// - brand: The brand of the card. Example: `STPCardBrandVisa` + @objc + @available( + *, + deprecated, + message: + "Use `requestConfiguration(withName:description:last4:brand:primaryAccountIdentifier:)` instead.", + renamed: "requestConfiguration(withName:description:last4:brand:primaryAccountIdentifier:)" + ) + public class func requestConfiguration( + withName name: String, + description: String?, + last4: String?, + brand: STPCardBrand + ) -> PKAddPaymentPassRequestConfiguration { + return self.requestConfiguration( + withName: name, + description: description, + last4: last4, + brand: brand, + primaryAccountIdentifier: nil + ) + } + + /// This is a helper method to generate a PKAddPaymentPassRequestConfiguration that will work with + /// Stripe's Issuing APIs. Pass the returned configuration object to `PKAddPaymentPassViewController`'s `initWithRequestConfiguration:delegate:` initializer. + /// - Parameters: + /// - name: Your cardholder's name. Example: John Appleseed + /// - description: A localized description of your card's name. This will appear in Apple's UI as "{description} will be available in Wallet". Example: Platinum Rewards Card + /// - last4: The last 4 of the card to be added to the user's Apple Pay wallet. Example: 4242 + /// - brand: The brand of the card. Example: `STPCardBrandVisa` + /// - primaryAccountIdentifier: The `primary_account_identifier` value from the issued card. + @objc + public class func requestConfiguration( + withName name: String, + description: String?, + last4: String?, + brand: STPCardBrand, + primaryAccountIdentifier: String? + ) -> PKAddPaymentPassRequestConfiguration { + let config = PKAddPaymentPassRequestConfiguration(encryptionScheme: .ECC_V2) + config?.cardholderName = name + config?.primaryAccountSuffix = last4 + config?.localizedDescription = description + config?.style = .payment + if brand == .visa { + config?.paymentNetwork = .visa + } + if brand == .mastercard { + config?.paymentNetwork = .masterCard + } + config?.primaryAccountIdentifier = primaryAccountIdentifier + return config! + } + + /// In order to retreive the encrypted payload that PKAddPaymentPassViewController expects, the Stripe SDK must talk to the Stripe API. As this requires privileged access, you must write a "key provider" that generates an Ephemeral Key on your backend and provides it to the SDK when requested. For more information, see https://stripe.com/docs/mobile/ios/basic#ephemeral-key + @objc + public init( + keyProvider: STPIssuingCardEphemeralKeyProvider + ) { + apiClient = STPAPIClient.shared + keyManager = STPEphemeralKeyManager( + keyProvider: keyProvider, + apiVersion: STPAPIClient.apiVersion, + performsEagerFetching: false + ) + super.init() + } + + /// This method lines up with the method of the same name on `PKAddPaymentPassViewControllerDelegate`. You should implement that protocol in your own app, and when that method is called, call this method on your `STPPushProvisioningContext`. This in turn will first initiate a call to your `keyProvider` (see above) to obtain an Ephemeral Key, then make a call to the Stripe Issuing API to fetch an encrypted payload for the card in question, then return that payload to iOS. + @objc + public func addPaymentPassViewController( + _ controller: PKAddPaymentPassViewController, + generateRequestWithCertificateChain certificates: [Data], + nonce: Data, + nonceSignature: Data, + completionHandler handler: @escaping (PKAddPaymentPassRequest) -> Void + ) { + keyManager.getOrCreateKey({ ephemeralKey, keyError in + if let keyError = keyError { + let request = PKAddPaymentPassRequest() + request.stp_error = keyError as NSError + // handler, bizarrely, cannot take an NSError, but passing an empty PKAddPaymentPassRequest causes roughly equivalent behavior. + handler(request) + return + } + let params = STPPushProvisioningDetailsParams( + cardId: ephemeralKey?.issuingCardID ?? "", + certificates: certificates, + nonce: nonce, + nonceSignature: nonceSignature + ) + if let ephemeralKey = ephemeralKey { + self.apiClient.retrievePushProvisioningDetails( + with: params, + ephemeralKey: ephemeralKey + ) { + details, + error in + if let error = error { + let request = PKAddPaymentPassRequest() + request.stp_error = error as NSError + handler(request) + return + } + let request = PKAddPaymentPassRequest() + request.activationData = details?.activationData + request.encryptedPassData = details?.encryptedPassData + request.ephemeralPublicKey = details?.ephemeralPublicKey + handler(request) + } + } + }) + } + + private var keyManager: STPEphemeralKeyManagerProtocol + private var ephemeralKey: STPEphemeralKey? +} diff --git a/Stripe/StripeiOS/Source/STPPushProvisioningDetails.swift b/Stripe/StripeiOS/Source/STPPushProvisioningDetails.swift new file mode 100644 index 00000000..ae260456 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPushProvisioningDetails.swift @@ -0,0 +1,97 @@ +// +// STPPushProvisioningDetails.swift +// StripeiOS +// +// Created by Jack Flintermann on 9/26/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripePayments + +class STPPushProvisioningDetails: NSObject, STPAPIResponseDecodable { + let cardId: String + let livemode: Bool + let encryptedPassData: Data + let activationData: Data + let ephemeralPublicKey: Data + + required init( + cardId: String, + livemode: Bool, + encryptedPass encryptedPassData: Data, + activationData: Data, + ephemeralPublicKey: Data + ) { + self.cardId = cardId + self.livemode = livemode + self.encryptedPassData = encryptedPassData + self.activationData = activationData + self.ephemeralPublicKey = ephemeralPublicKey + super.init() + } + private(set) var allResponseFields: [AnyHashable: Any] = [:] + + // MARK: - STPAPIResponseDecodable + class func decodedObject(fromAPIResponse response: [AnyHashable: Any]?) -> Self? { + guard + let dict = response?.stp_dictionaryByRemovingNulls() + else { + return nil + } + + // required fields + let cardId = dict.stp_string(forKey: "card") + let livemode = dict.stp_bool(forKey: "livemode", or: false) + let encryptedPassString = dict.stp_string(forKey: "contents") + let encryptedPassData = + encryptedPassString != nil + ? Data(base64Encoded: encryptedPassString ?? "", options: []) : nil + + let activationString = dict.stp_string(forKey: "activation_data") + let activationData = + activationString != nil ? Data(base64Encoded: activationString ?? "", options: []) : nil + + let ephemeralPublicKeyString = dict.stp_string(forKey: "ephemeral_public_key") + let ephemeralPublicKeyData = + ephemeralPublicKeyString != nil + ? Data(base64Encoded: ephemeralPublicKeyString ?? "", options: []) : nil + + if cardId == nil || encryptedPassData == nil || activationData == nil + || ephemeralPublicKeyData == nil + { + return nil + } + + if let encryptedPassData = encryptedPassData, let activationData = activationData, + let ephemeralPublicKeyData = ephemeralPublicKeyData + { + let details = self.init( + cardId: cardId ?? "", + livemode: livemode, + encryptedPass: encryptedPassData, + activationData: activationData, + ephemeralPublicKey: ephemeralPublicKeyData + ) + details.allResponseFields = dict + return details + } + return nil + } + + // MARK: - Equality + override func isEqual(_ object: Any?) -> Bool { + if let details = object as? STPPushProvisioningDetails { + return isEqual(to: details) + } + return false + } + + override var hash: Int { + return activationData.hashValue + } + + func isEqual(to details: STPPushProvisioningDetails) -> Bool { + return details.activationData == self.activationData + } +} diff --git a/Stripe/StripeiOS/Source/STPPushProvisioningDetailsParams.swift b/Stripe/StripeiOS/Source/STPPushProvisioningDetailsParams.swift new file mode 100644 index 00000000..c7fbcccb --- /dev/null +++ b/Stripe/StripeiOS/Source/STPPushProvisioningDetailsParams.swift @@ -0,0 +1,73 @@ +// +// STPPushProvisioningDetailsParams.swift +// StripeiOS +// +// Created by Jack Flintermann on 9/26/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +import Foundation + +/// A helper class for turning the raw certificate array, nonce, and nonce signature emitted by PKAddPaymentPassViewController into a format that is understandable by the Stripe API. +/// If you are using STPPushProvisioningContext to implement your integration, you do not need to use this class. +public class STPPushProvisioningDetailsParams: NSObject { + /// The Stripe ID of the Issuing card object to retrieve details for. + @objc public private(set) var cardId: String + /// An array of certificates that should be used to encrypt the card details. + @objc public private(set) var certificates: [Data] + /// A nonce that should be used during the encryption of the card details. + @objc public private(set) var nonce: Data + /// A nonce signature that should be used during the encryption of the card details. + @objc public private(set) var nonceSignature: Data + /// Implemented for convenience - the Stripe API expects the certificate chain as an array of base64-encoded strings. + + @objc public var certificatesBase64: [String] { + var base64Certificates: [AnyHashable] = [] + for certificate in certificates { + base64Certificates.append(certificate.base64EncodedString(options: [])) + } + return base64Certificates as? [String] ?? [] + } + /// Implemented for convenience - the Stripe API expects the nonce as a hex-encoded string. + + @objc public var nonceHex: String { + STPPushProvisioningDetailsParams.hexadecimalString(for: nonce) + } + /// Implemented for convenience - the Stripe API expects the nonce signature as a hex-encoded string. + + @objc public var nonceSignatureHex: String { + STPPushProvisioningDetailsParams.hexadecimalString(for: nonceSignature) + } + + /// Instantiates a new params object with the provided attributes. + @objc public required init( + cardId: String, + certificates: [Data], + nonce: Data, + nonceSignature: Data + ) { + self.cardId = cardId + self.certificates = certificates + self.nonce = nonce + self.nonceSignature = nonceSignature + } + + @objc(paramsWithCardId:certificates:nonce:nonceSignature:) class func paramsWithCardId( + cardId: String, + certificates: [Data], + nonce: Data, + nonceSignature: Data + ) -> Self { + return self.init( + cardId: cardId, + certificates: certificates, + nonce: nonce, + nonceSignature: nonceSignature + ) + } + + // Adapted from https://stackoverflow.com/questions/39075043/how-to-convert-data-to-hex-string-in-swift + class func hexadecimalString(for data: Data) -> String { + return data.map { String(format: "%02hhx", $0) }.joined() + } +} diff --git a/Stripe/StripeiOS/Source/STPSectionHeaderView.swift b/Stripe/StripeiOS/Source/STPSectionHeaderView.swift new file mode 100644 index 00000000..2b45b2af --- /dev/null +++ b/Stripe/StripeiOS/Source/STPSectionHeaderView.swift @@ -0,0 +1,161 @@ +// +// STPSectionHeaderView.swift +// StripeiOS +// +// Created by Ben Guo on 1/3/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import UIKit + +class STPSectionHeaderView: UIView { + private var _theme: STPTheme = STPTheme.defaultTheme + var theme: STPTheme { + get { + _theme + } + set(theme) { + _theme = theme + updateAppearance() + } + } + + private var _title: String? + var title: String? { + get { + _title + } + set(title) { + _title = title + if let title = title { + let style = NSMutableParagraphStyle() + style.firstLineHeadIndent = 15 + style.headIndent = style.firstLineHeadIndent + let attributes = [ + NSAttributedString.Key.paragraphStyle: style + ] + label?.attributedText = NSAttributedString( + string: title, + attributes: attributes + ) + } else { + label?.attributedText = nil + } + setNeedsLayout() + } + } + weak var button: UIButton? + + private var _buttonHidden = false + var buttonHidden: Bool { + get { + _buttonHidden + } + set(buttonHidden) { + _buttonHidden = buttonHidden + button?.alpha = buttonHidden ? 0 : 1 + } + } + private weak var label: UILabel? + private let buttonInsets = UIEdgeInsets(top: 5, left: 5, bottom: 5, right: 15) + + override init( + frame: CGRect + ) { + super.init(frame: frame) + let label = UILabel() + label.numberOfLines = 0 + label.lineBreakMode = .byWordWrapping + label.accessibilityTraits.insert(.header) + addSubview(label) + self.label = label + let button = UIButton(type: .system) + button.contentHorizontalAlignment = .right + button.titleLabel?.numberOfLines = 0 + button.titleLabel?.lineBreakMode = .byWordWrapping + button.titleEdgeInsets = UIEdgeInsets(top: 0, left: 0, bottom: 0, right: 15) + button.contentEdgeInsets = .zero + addSubview(button) + self.button = button + backgroundColor = UIColor.clear + updateAppearance() + } + + @objc func updateAppearance() { + label?.font = theme.smallFont + label?.textColor = theme.secondaryForegroundColor + button?.titleLabel?.font = theme.smallFont + button?.tintColor = theme.accentColor + } + + override func layoutSubviews() { + super.layoutSubviews() + let bounds = stp_boundsWithHorizontalSafeAreaInsets() + if buttonHidden { + label?.frame = bounds + } else { + let halfWidth = bounds.size.width / 2 + let heightThatFits = self.heightThatFits(bounds.size) + label?.frame = CGRect( + x: bounds.origin.x, + y: bounds.origin.y, + width: halfWidth, + height: heightThatFits + ) + button?.frame = CGRect( + x: bounds.origin.x + halfWidth, + y: bounds.origin.y, + width: halfWidth, + height: heightThatFits + ) + } + } + + func heightThatFits(_ size: CGSize) -> CGFloat { + let labelPadding: CGFloat = 16 + if buttonHidden { + let labelHeight = label?.sizeThatFits(size).height ?? 0.0 + return labelHeight + labelPadding + } else { + let halfSize = CGSize(width: size.width / 2, height: size.height) + let labelHeight = (label?.sizeThatFits(halfSize).height ?? 0.0) + labelPadding + let buttonHeight = height( + forButtonText: button?.titleLabel?.text, + width: halfSize.width + ) + return CGFloat(max(buttonHeight, labelHeight)) + } + } + + private func height(forButtonText text: String?, width: CGFloat) -> CGFloat { + let insets = buttonInsets + let textSize = CGSize( + width: width - insets.left - insets.right, + height: CGFloat.greatestFiniteMagnitude + ) + var attributes: [NSAttributedString.Key: Any]? + if let font1 = button?.titleLabel?.font { + attributes = [ + NSAttributedString.Key.font: font1 + ] + } + let buttonSize = + text?.boundingRect( + with: textSize, + options: .usesLineFragmentOrigin, + attributes: attributes, + context: nil + ).size ?? .zero + return buttonSize.height + insets.top + insets.bottom + } + + override func sizeThatFits(_ size: CGSize) -> CGSize { + return CGSize(width: size.width, height: heightThatFits(size)) + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/STPShippingAddressViewController.swift b/Stripe/StripeiOS/Source/STPShippingAddressViewController.swift new file mode 100644 index 00000000..ac63189d --- /dev/null +++ b/Stripe/StripeiOS/Source/STPShippingAddressViewController.swift @@ -0,0 +1,674 @@ +// +// STPShippingAddressViewController.swift +// StripeiOS +// +// Created by Ben Guo on 8/29/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import PassKit +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +@_spi(STP) import StripeUICore +import UIKit + +/// This view controller contains a shipping address collection form. It renders a right bar button item that submits the form, so it must be shown inside a `UINavigationController`. Depending on your configuration's shippingType, the view controller may present a shipping method selection form after the user enters an address. +public class STPShippingAddressViewController: STPCoreTableViewController { + + /// A convenience initializer; equivalent to calling `init(configuration: STPPaymentConfiguration.shared theme: STPTheme.defaultTheme currency:"" shippingAddress:nil selectedShippingMethod:nil prefilledInformation:nil)`. + @objc + public convenience init() { + self.init( + configuration: STPPaymentConfiguration.shared, + theme: STPTheme.defaultTheme, + currency: "", + shippingAddress: nil, + selectedShippingMethod: nil, + prefilledInformation: nil + ) + } + + /// Initializes a new `STPShippingAddressViewController` with the given payment context and sets the payment context as its delegate. + /// - Parameter paymentContext: The payment context to use. + @objc(initWithPaymentContext:) + public convenience init( + paymentContext: STPPaymentContext + ) { + STPAnalyticsClient.sharedClient.addClass( + toProductUsageIfNecessary: STPShippingAddressViewController.self + ) + + var billingAddress: STPAddress? + weak var paymentOption = paymentContext.selectedPaymentOption + if paymentOption is STPCard { + let card = paymentOption as? STPCard + billingAddress = card?.address + } else if paymentOption is STPPaymentMethod { + let paymentMethod = paymentOption as? STPPaymentMethod + if let billingDetails1 = paymentMethod?.billingDetails { + billingAddress = STPAddress(paymentMethodBillingDetails: billingDetails1) + } + } + var prefilledInformation: STPUserInformation? + if paymentContext.prefilledInformation != nil { + prefilledInformation = paymentContext.prefilledInformation + } else { + prefilledInformation = STPUserInformation() + } + prefilledInformation?.billingAddress = billingAddress + self.init( + configuration: paymentContext.configuration, + theme: paymentContext.theme, + currency: paymentContext.paymentCurrency, + shippingAddress: paymentContext.shippingAddress, + selectedShippingMethod: paymentContext.selectedShippingMethod, + prefilledInformation: prefilledInformation + ) + + self.delegate = paymentContext + } + + /// Initializes a new `STPShippingAddressCardViewController` with the provided parameters. + /// - Parameters: + /// - configuration: The configuration to use (this determines the required shipping address fields and shipping type). - seealso: STPPaymentConfiguration + /// - theme: The theme to use to inform the view controller's visual appearance. - seealso: STPTheme + /// - currency: The currency to use when displaying amounts for shipping methods. The default is USD. + /// - shippingAddress: If set, the shipping address view controller will be pre-filled with this address. - seealso: STPAddress + /// - selectedShippingMethod: If set, the shipping methods view controller will use this method as the selected shipping method. If `selectedShippingMethod` is nil, the first shipping method in the array of methods returned by your delegate will be selected. + /// - prefilledInformation: If set, the shipping address view controller will be pre-filled with this information. - seealso: STPUserInformation + @objc( + initWithConfiguration: + theme: + currency: + shippingAddress: + selectedShippingMethod: + prefilledInformation: + ) + public init( + configuration: STPPaymentConfiguration, + theme: STPTheme, + currency: String?, + shippingAddress: STPAddress?, + selectedShippingMethod: PKShippingMethod?, + prefilledInformation: STPUserInformation? + ) { + STPAnalyticsClient.sharedClient.addClass( + toProductUsageIfNecessary: STPShippingAddressViewController.self + ) + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + super.init(theme: theme) + assert( + (configuration.requiredShippingAddressFields?.count ?? 0) > 0, + "`requiredShippingAddressFields` must not be empty when initializing an STPShippingAddressViewController." + ) + self.configuration = configuration + self.currency = currency + self.selectedShippingMethod = selectedShippingMethod + billingAddress = prefilledInformation?.billingAddress + hasUsedBillingAddress = false + addressViewModel = STPAddressViewModel( + requiredShippingFields: configuration.requiredShippingAddressFields ?? [], + availableCountries: configuration.availableCountries + ) + addressViewModel.delegate = self + if let shippingAddress = shippingAddress { + addressViewModel.address = shippingAddress + } else if prefilledInformation?.shippingAddress != nil { + addressViewModel.address = prefilledInformation?.shippingAddress ?? STPAddress() + } + title = title(for: self.configuration?.shippingType ?? .shipping) + } + + /// The view controller's delegate. This must be set before showing the view controller in order for it to work properly. - seealso: STPShippingAddressViewControllerDelegate + @objc public weak var delegate: STPShippingAddressViewControllerDelegate? + + /// If you're pushing `STPShippingAddressViewController` onto an existing `UINavigationController`'s stack, you should use this method to dismiss it, since it may have pushed an additional shipping method view controller onto the navigation controller's stack. + /// - Parameter completion: The callback to run after the view controller is dismissed. You may specify nil for this parameter. + @objc(dismissWithCompletion:) + public func dismiss(withCompletion completion: STPVoidBlock?) { + if stp_isAtRootOfNavigationController() { + presentingViewController?.dismiss(animated: true, completion: completion ?? {}) + } else { + var previous = navigationController?.viewControllers.first + for viewController in navigationController?.viewControllers ?? [] { + if viewController == self { + break + } + previous = viewController + } + navigationController?.stp_pop( + to: previous, + animated: true, + completion: completion ?? {} + ) + } + } + + /// Use one of the initializers declared in this interface. + @available( + *, + unavailable, + message: "Use one of the initializers declared in this interface instead." + ) + @objc public required init( + theme: STPTheme? + ) { + let configuration = STPPaymentConfiguration.shared + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + + super.init(theme: theme) + } + + /// Use one of the initializers declared in this interface. + @available( + *, + unavailable, + message: "Use one of the initializers declared in this interface instead." + ) + @objc public required init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + let configuration = STPPaymentConfiguration.shared + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil) + } + + /// Use one of the initializers declared in this interface. + @available( + *, + unavailable, + message: "Use one of the initializers declared in this interface instead." + ) + required init?( + coder aDecoder: NSCoder + ) { + let configuration = STPPaymentConfiguration.shared + addressViewModel = STPAddressViewModel( + requiredBillingFields: configuration.requiredBillingAddressFields, + availableCountries: configuration.availableCountries + ) + super.init(coder: aDecoder) + } + + private var configuration: STPPaymentConfiguration? + private var currency: String? + private var selectedShippingMethod: PKShippingMethod? + private weak var imageView: UIImageView? + private var nextItem: UIBarButtonItem? + + private var _loading = false + private var loading: Bool { + get { + _loading + } + set(loading) { + if loading == _loading { + return + } + _loading = loading + stp_navigationItemProxy?.setHidesBackButton(loading, animated: true) + stp_navigationItemProxy?.leftBarButtonItem?.isEnabled = !loading + activityIndicator?.animating = loading + if loading { + tableView?.endEditing(true) + var loadingItem: UIBarButtonItem? + if let activityIndicator = activityIndicator { + loadingItem = UIBarButtonItem(customView: activityIndicator) + } + stp_navigationItemProxy?.setRightBarButton(loadingItem, animated: true) + } else { + stp_navigationItemProxy?.setRightBarButton(nextItem, animated: true) + } + for cell in addressViewModel.addressCells { + cell.isUserInteractionEnabled = !loading + UIView.animate( + withDuration: 0.1, + animations: { + cell.alpha = loading ? 0.7 : 1.0 + } + ) + } + } + } + private var activityIndicator: STPPaymentActivityIndicatorView? + internal var addressViewModel: STPAddressViewModel + private var billingAddress: STPAddress? + private var hasUsedBillingAddress = false + private var addressHeaderView: STPSectionHeaderView? + + override func createAndSetupViews() { + super.createAndSetupViews() + + var nextItem: UIBarButtonItem? + switch configuration?.shippingType { + case .shipping: + nextItem = UIBarButtonItem( + title: STPLocalizedString("Next", "Button to move to the next text entry field"), + style: .done, + target: self, + action: #selector(next(_:)) + ) + case .delivery, .none, .some: + nextItem = UIBarButtonItem( + barButtonSystemItem: .done, + target: self, + action: #selector(next(_:)) + ) + } + self.nextItem = nextItem + stp_navigationItemProxy?.rightBarButtonItem = nextItem + stp_navigationItemProxy?.rightBarButtonItem?.isEnabled = false + stp_navigationItemProxy?.rightBarButtonItem?.accessibilityIdentifier = + "ShippingViewControllerNextButtonIdentifier" + + let imageView = UIImageView(image: STPLegacyImageLibrary.largeShippingImage()) + imageView.contentMode = .center + imageView.frame = CGRect( + x: 0, + y: 0, + width: view.bounds.size.width, + height: imageView.bounds.size.height + (57 * 2) + ) + self.imageView = imageView + tableView?.tableHeaderView = imageView + + activityIndicator = STPPaymentActivityIndicatorView( + frame: CGRect(x: 0, y: 0, width: 20.0, height: 20.0) + ) + + tableView?.dataSource = self + tableView?.delegate = self + tableView?.reloadData() + view.addGestureRecognizer( + UITapGestureRecognizer( + target: self, + action: #selector(NSMutableAttributedString.endEditing) + ) + ) + + let headerView = STPSectionHeaderView() + headerView.theme = theme + if let shippingType1 = configuration?.shippingType { + headerView.title = headerTitle(for: shippingType1) + } + headerView.button?.setTitle( + STPLocalizedString( + "Use Billing", + "Button to fill shipping address from billing address." + ), + for: .normal + ) + headerView.button?.addTarget( + self, + action: #selector(useBillingAddress(_:)), + for: .touchUpInside + ) + headerView.button?.accessibilityIdentifier = "ShippingAddressViewControllerUseBillingButton" + var buttonVisible = false + if let requiredFields = configuration?.requiredShippingAddressFields { + let needsAddress = + requiredFields.contains(.postalAddress) && !(addressViewModel.isValid) + buttonVisible = + needsAddress + && billingAddress?.containsContent(forShippingAddressFields: requiredFields) + ?? false + && !hasUsedBillingAddress + } + headerView.button?.alpha = buttonVisible ? 1 : 0 + headerView.setNeedsLayout() + addressHeaderView = headerView + + updateDoneButton() + } + + @objc func endEditing() { + view.endEditing(false) + } + + @objc override func updateAppearance() { + super.updateAppearance() + let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme + nextItem?.stp_setTheme(navBarTheme) + + tableView?.allowsSelection = false + + imageView?.tintColor = theme.accentColor + activityIndicator?.tintColor = theme.accentColor + for cell in addressViewModel.addressCells { + cell.theme = theme + } + addressHeaderView?.theme = theme + } + + /// :nodoc: + @objc + public override func viewDidAppear(_ animated: Bool) { + super.viewDidAppear(animated) + stp_beginObservingKeyboardAndInsettingScrollView( + tableView, + onChange: nil + ) + firstEmptyField()?.becomeFirstResponder() + } + + func firstEmptyField() -> UIResponder? { + for cell in addressViewModel.addressCells { + if (cell.contents?.count ?? 0) == 0 { + return cell + } + } + return nil + } + + @objc override func handleCancelTapped(_ sender: Any?) { + delegate?.shippingAddressViewControllerDidCancel(self) + } + + @objc func next(_ sender: Any?) { + let address = addressViewModel.address + switch configuration?.shippingType { + case .shipping: + loading = true + delegate?.shippingAddressViewController(self, didEnter: address) { + status, + shippingValidationError, + shippingMethods, + selectedShippingMethod in + self.loading = false + if status == .valid { + if (shippingMethods?.count ?? 0) > 0 { + var nextViewController: STPShippingMethodsViewController? + if let shippingMethods = shippingMethods, + let selectedShippingMethod = selectedShippingMethod + { + nextViewController = STPShippingMethodsViewController( + shippingMethods: shippingMethods, + selectedShippingMethod: selectedShippingMethod, + currency: self.currency ?? "", + theme: self.theme + ) + } + nextViewController?.delegate = self + if let nextViewController = nextViewController { + self.navigationController?.pushViewController( + nextViewController, + animated: true + ) + } + } else { + self.delegate?.shippingAddressViewController( + self, + didFinishWith: address, + shippingMethod: nil + ) + } + } else { + self.handleShippingValidationError(shippingValidationError) + } + } + case .delivery, .none, .some: + delegate?.shippingAddressViewController( + self, + didFinishWith: address, + shippingMethod: nil + ) + } + } + + func updateDoneButton() { + stp_navigationItemProxy?.rightBarButtonItem?.isEnabled = addressViewModel.isValid + } + + func handleShippingValidationError(_ error: Error?) { + firstEmptyField()?.becomeFirstResponder() + var title = STPLocalizedString("Invalid Shipping Address", "Shipping form error message") + var message: String? + if let error = error { + title = error.localizedDescription + message = (error as NSError).localizedFailureReason + } + let alertController = UIAlertController( + title: title, + message: message, + preferredStyle: .alert + ) + alertController.addAction( + UIAlertAction( + title: String.Localized.ok, + style: .cancel, + handler: nil + ) + ) + present(alertController, animated: true) + } + + /// :nodoc: + @objc + public override func tableView( + _ tableView: UITableView, + heightForHeaderInSection section: Int + ) -> CGFloat { + let size = addressHeaderView?.sizeThatFits( + CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) + ) + return size?.height ?? 0.0 + } + + @objc func useBillingAddress(_ sender: UIButton) { + guard let billingAddress = billingAddress else { + return + } + tableView?.beginUpdates() + addressViewModel.address = billingAddress + hasUsedBillingAddress = true + firstEmptyField()?.becomeFirstResponder() + UIView.animate( + withDuration: 0.2, + animations: { + self.addressHeaderView?.buttonHidden = true + } + ) + tableView?.endUpdates() + } + + func title(for type: STPShippingType) -> String { + if let shippingAddressFields = configuration?.requiredShippingAddressFields, + shippingAddressFields.contains(.postalAddress) + { + switch type { + case .shipping: + return STPLocalizedString("Shipping", "Title for shipping info form") + case .delivery: + return STPLocalizedString("Delivery", "Title for delivery info form") + } + } else { + return STPLocalizedString("Contact", "Title for contact info form") + } + } + + func headerTitle(for type: STPShippingType) -> String { + if let shippingAddressFields = configuration?.requiredShippingAddressFields, + shippingAddressFields.contains(.postalAddress) + { + switch type { + case .shipping: + return String.Localized.shipping_address + case .delivery: + return STPLocalizedString( + "Delivery Address", + "Title for delivery address entry section" + ) + } + } else { + return STPLocalizedString("Contact", "Title for contact info form") + } + } +} + +/// An `STPShippingAddressViewControllerDelegate` is notified when an `STPShippingAddressViewController` receives an address, completes with an address, or is cancelled. +@objc public protocol STPShippingAddressViewControllerDelegate: NSObjectProtocol { + /// Called when the user cancels entering a shipping address. You should dismiss (or pop) the view controller at this point. + /// - Parameter addressViewController: the view controller that has been cancelled + func shippingAddressViewControllerDidCancel( + _ addressViewController: STPShippingAddressViewController + ) + /// This is called when the user enters a shipping address and taps next. You + /// should validate the address and determine what shipping methods are available, + /// and call the `completion` block when finished. If an error occurrs, call + /// the `completion` block with the error. Otherwise, call the `completion` + /// block with a nil error and an array of available shipping methods. If you don't + /// need to collect a shipping method, you may pass an empty array or nil. + /// - Parameters: + /// - addressViewController: the view controller where the address was entered + /// - address: the address that was entered. - seealso: STPAddress + /// - completion: call this callback when you're done validating the address and determining available shipping methods. + + @objc(shippingAddressViewController:didEnterAddress:completion:) + func shippingAddressViewController( + _ addressViewController: STPShippingAddressViewController, + didEnter address: STPAddress, + completion: @escaping STPShippingMethodsCompletionBlock + ) + /// This is called when the user selects a shipping method. If no shipping methods are given, or if the shipping type doesn't require a shipping method, this will be called after the user has a shipping address and your validation has succeeded. After updating your app with the user's shipping info, you should dismiss (or pop) the view controller. Note that if `shippingMethod` is non-nil, there will be an additional shipping methods view controller on the navigation controller's stack. + /// - Parameters: + /// - addressViewController: the view controller where the address was entered + /// - address: the address that was entered. - seealso: STPAddress + /// - method: the shipping method that was selected. + @objc(shippingAddressViewController:didFinishWithAddress:shippingMethod:) + func shippingAddressViewController( + _ addressViewController: STPShippingAddressViewController, + didFinishWith address: STPAddress, + shippingMethod method: PKShippingMethod? + ) +} + +extension STPShippingAddressViewController: + STPAddressViewModelDelegate, UITableViewDelegate, UITableViewDataSource, + STPShippingMethodsViewControllerDelegate +{ + + // MARK: - UITableView + /// :nodoc: + @objc + public func numberOfSections(in tableView: UITableView) -> Int { + return 1 + } + + /// :nodoc: + @objc + public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { + return addressViewModel.addressCells.count + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + cellForRowAt indexPath: IndexPath + ) -> UITableViewCell { + let cell = + addressViewModel.addressCells.stp_boundSafeObject(at: indexPath.row) + cell?.backgroundColor = theme.secondaryBackgroundColor + cell?.contentView.backgroundColor = UIColor.clear + return cell! + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + willDisplay cell: UITableViewCell, + forRowAt indexPath: IndexPath + ) { + let topRow = indexPath.row == 0 + let bottomRow = tableView.numberOfRows(inSection: indexPath.section) - 1 == indexPath.row + cell.stp_setBorderColor(theme.tertiaryBackgroundColor) + cell.stp_setTopBorderHidden(!topRow) + cell.stp_setBottomBorderHidden(!bottomRow) + cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) + cell.stp_setFakeSeparatorLeftInset(15.0) + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + heightForFooterInSection section: Int + ) + -> CGFloat + { + return 0.01 + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + viewForFooterInSection section: Int + ) + -> UIView? + { + return UIView() + } + + /// :nodoc: + @objc + public func tableView( + _ tableView: UITableView, + viewForHeaderInSection section: Int + ) + -> UIView? + { + return addressHeaderView + } + + // MARK: - STPShippingMethodsViewControllerDelegate + func shippingMethodsViewController( + _ methodsViewController: STPShippingMethodsViewController, + didFinishWith method: PKShippingMethod + ) { + delegate?.shippingAddressViewController( + self, + didFinishWith: addressViewModel.address, + shippingMethod: method + ) + } + + // MARK: - STPAddressViewModelDelegate + func addressViewModel(_ addressViewModel: STPAddressViewModel, addedCellAt index: Int) { + let indexPath = IndexPath(row: index, section: 0) + tableView?.insertRows(at: [indexPath], with: .automatic) + } + + func addressViewModel(_ addressViewModel: STPAddressViewModel, removedCellAt index: Int) { + let indexPath = IndexPath(row: index, section: 0) + tableView?.deleteRows(at: [indexPath], with: .automatic) + } + + func addressViewModelDidChange(_ addressViewModel: STPAddressViewModel) { + updateDoneButton() + } + + func addressViewModelWillUpdate(_ addressViewModel: STPAddressViewModel) { + tableView?.beginUpdates() + } + + func addressViewModelDidUpdate(_ addressViewModel: STPAddressViewModel) { + tableView?.endUpdates() + } +} + +/// :nodoc: +@_spi(STP) extension STPShippingAddressViewController: STPAnalyticsProtocol { + @_spi(STP) public static var stp_analyticsIdentifier = "STPShippingAddressViewController" +} diff --git a/Stripe/StripeiOS/Source/STPShippingMethodTableViewCell.swift b/Stripe/StripeiOS/Source/STPShippingMethodTableViewCell.swift new file mode 100644 index 00000000..f40c493c --- /dev/null +++ b/Stripe/StripeiOS/Source/STPShippingMethodTableViewCell.swift @@ -0,0 +1,147 @@ +// +// STPShippingMethodTableViewCell.swift +// StripeiOS +// +// Created by Ben Guo on 8/30/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import PassKit +@_spi(STP) import StripePayments +import UIKit + +class STPShippingMethodTableViewCell: UITableViewCell { + private var _theme: STPTheme? + var theme: STPTheme? { + get { + _theme + } + set(theme) { + _theme = theme + updateAppearance() + } + } + + func setShippingMethod(_ method: PKShippingMethod, currency: String) { + shippingMethod = method + titleLabel?.text = method.label + subtitleLabel?.text = method.detail + var localeInfo = [ + NSLocale.Key.currencyCode.rawValue: currency + ] + localeInfo[NSLocale.Key.languageCode.rawValue] = NSLocale.preferredLanguages.first ?? "" + let localeID = NSLocale.localeIdentifier(fromComponents: localeInfo) + let locale = NSLocale(localeIdentifier: localeID) + numberFormatter?.locale = locale as Locale + let amount = method.amount.stp_amount(withCurrency: currency) + if amount == 0 { + amountLabel?.text = STPLocalizedString("Free", "Label for free shipping method") + } else { + let number = NSDecimalNumber.stp_decimalNumber( + withAmount: amount, + currency: currency + ) + amountLabel?.text = numberFormatter?.string(from: number) + } + setNeedsLayout() + } + + private weak var titleLabel: UILabel? + private weak var subtitleLabel: UILabel? + private weak var amountLabel: UILabel? + private weak var checkmarkIcon: UIImageView? + private var shippingMethod: PKShippingMethod? + private var numberFormatter: NumberFormatter? + + override init( + style: UITableViewCell.CellStyle, + reuseIdentifier: String? + ) { + super.init(style: style, reuseIdentifier: reuseIdentifier) + theme = STPTheme() + let titleLabel = UILabel() + self.titleLabel = titleLabel + let subtitleLabel = UILabel() + self.subtitleLabel = subtitleLabel + let amountLabel = UILabel() + self.amountLabel = amountLabel + let checkmarkIcon = UIImageView(image: STPLegacyImageLibrary.checkmarkIcon()) + self.checkmarkIcon = checkmarkIcon + let formatter = NumberFormatter() + formatter.numberStyle = .currency + formatter.usesGroupingSeparator = true + numberFormatter = formatter + contentView.addSubview(titleLabel) + contentView.addSubview(subtitleLabel) + contentView.addSubview(amountLabel) + contentView.addSubview(checkmarkIcon) + updateAppearance() + } + + override var isSelected: Bool { + get { + return super.isSelected + } + set(selected) { + super.isSelected = selected + updateAppearance() + } + } + + @objc func updateAppearance() { + contentView.backgroundColor = theme?.secondaryBackgroundColor + backgroundColor = UIColor.clear + titleLabel?.font = theme?.font + subtitleLabel?.font = theme?.smallFont + amountLabel?.font = theme?.font + titleLabel?.textColor = isSelected ? theme?.accentColor : theme?.primaryForegroundColor + amountLabel?.textColor = titleLabel?.textColor + var subduedAccentColor: UIColor? + if #available(iOS 13.0, *) { + subduedAccentColor = UIColor(dynamicProvider: { _ in + return self.theme?.accentColor.withAlphaComponent(0.6) ?? UIColor.clear + }) + } else { + subduedAccentColor = theme?.accentColor.withAlphaComponent(0.6) + } + subtitleLabel?.textColor = isSelected ? subduedAccentColor : theme?.secondaryForegroundColor + checkmarkIcon?.tintColor = theme?.accentColor + checkmarkIcon?.isHidden = !isSelected + } + + override func layoutSubviews() { + super.layoutSubviews() + let midY = bounds.midY + checkmarkIcon?.frame = CGRect(x: 0, y: 0, width: 14, height: 14) + checkmarkIcon?.center = CGPoint( + x: bounds.width - 15 - (checkmarkIcon?.bounds.midX ?? 0.0), + y: midY + ) + amountLabel?.sizeToFit() + amountLabel?.center = CGPoint( + x: (checkmarkIcon?.frame.minX ?? 0.0) - 15 - (amountLabel?.bounds.midX ?? 0.0), + y: midY + ) + let labelWidth = (amountLabel?.frame.minX ?? 0.0) - 30 + titleLabel?.sizeToFit() + titleLabel?.frame = CGRect( + x: 15, + y: 8, + width: labelWidth, + height: titleLabel?.frame.size.height ?? 0.0 + ) + subtitleLabel?.sizeToFit() + subtitleLabel?.frame = CGRect( + x: 15, + y: bounds.size.height - 8 - (subtitleLabel?.frame.size.height ?? 0.0), + width: labelWidth, + height: subtitleLabel?.frame.size.height ?? 0.0 + ) + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/STPShippingMethodsViewController.swift b/Stripe/StripeiOS/Source/STPShippingMethodsViewController.swift new file mode 100644 index 00000000..8d915855 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPShippingMethodsViewController.swift @@ -0,0 +1,211 @@ +// +// STPShippingMethodsViewController.swift +// StripeiOS +// +// Created by Ben Guo on 8/29/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import PassKit +@_spi(STP) import StripeCore +import UIKit + +class STPShippingMethodsViewController: STPCoreTableViewController, UITableViewDataSource, + UITableViewDelegate +{ + init( + shippingMethods methods: [PKShippingMethod], + selectedShippingMethod selectedMethod: PKShippingMethod, + currency: String, + theme: STPTheme + ) { + super.init(theme: theme) + shippingMethods = methods + if (methods.firstIndex(of: selectedMethod) ?? NSNotFound) != NSNotFound { + selectedShippingMethod = selectedMethod + } else { + selectedShippingMethod = methods.stp_boundSafeObject(at: 0) + } + + self.currency = currency + title = STPLocalizedString("Shipping", "Title for shipping info form") + } + + weak var delegate: STPShippingMethodsViewControllerDelegate? + private var shippingMethods: [PKShippingMethod]? + private var selectedShippingMethod: PKShippingMethod? + private var currency: String? + private weak var imageView: UIImageView? + private var doneItem: UIBarButtonItem? + private var headerView: STPSectionHeaderView? + + override func createAndSetupViews() { + super.createAndSetupViews() + + tableView?.register( + STPShippingMethodTableViewCell.self, + forCellReuseIdentifier: STPShippingMethodCellReuseIdentifier + ) + + let doneItem = UIBarButtonItem( + barButtonSystemItem: .done, + target: self, + action: #selector(done(_:)) + ) + self.doneItem = doneItem + stp_navigationItemProxy?.rightBarButtonItem = doneItem + stp_navigationItemProxy?.rightBarButtonItem?.accessibilityIdentifier = + "ShippingMethodsViewControllerDoneButtonIdentifier" + + let imageView = UIImageView(image: STPLegacyImageLibrary.largeShippingImage()) + imageView.contentMode = .center + imageView.frame = CGRect( + x: 0, + y: 0, + width: view.bounds.size.width, + height: imageView.bounds.size.height + (57 * 2) + ) + self.imageView = imageView + + tableView?.tableHeaderView = imageView + tableView?.dataSource = self + tableView?.delegate = self + tableView?.reloadData() + + let headerView = STPSectionHeaderView() + headerView.theme = theme + headerView.buttonHidden = true + headerView.title = STPLocalizedString("Shipping Method", "Label for shipping method form") + headerView.setNeedsLayout() + self.headerView = headerView + } + + @objc override func updateAppearance() { + super.updateAppearance() + + let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme + doneItem?.stp_setTheme(navBarTheme) + + imageView?.tintColor = theme.accentColor + for cell in tableView?.visibleCells ?? [] { + let shippingCell = cell as? STPShippingMethodTableViewCell + shippingCell?.theme = theme + } + } + + @objc func done(_ sender: Any?) { + if let selectedShippingMethod = selectedShippingMethod { + delegate?.shippingMethodsViewController(self, didFinishWith: selectedShippingMethod) + } + } + + override func useSystemBackButton() -> Bool { + return true + } + + // MARK: - UITableView + func numberOfSections(in tableView: UITableView) -> Int { + return 1 + } + + func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { + return shippingMethods?.count ?? 0 + } + + func tableView( + _ tableView: UITableView, + cellForRowAt indexPath: IndexPath + ) -> UITableViewCell { + let cell = + tableView.dequeueReusableCell( + withIdentifier: STPShippingMethodCellReuseIdentifier, + for: indexPath + ) + as? STPShippingMethodTableViewCell + let method = + shippingMethods?.stp_boundSafeObject(at: indexPath.row) + cell?.theme = theme + if let method = method { + cell?.setShippingMethod(method, currency: currency ?? "") + } + cell?.isSelected = method?.identifier == selectedShippingMethod?.identifier + return cell! + } + + func tableView( + _ tableView: UITableView, + willDisplay cell: UITableViewCell, + forRowAt indexPath: IndexPath + ) { + let topRow = indexPath.row == 0 + let bottomRow = + self.tableView(tableView, numberOfRowsInSection: indexPath.section) - 1 == indexPath.row + cell.stp_setBorderColor(theme.tertiaryBackgroundColor) + cell.stp_setTopBorderHidden(!topRow) + cell.stp_setBottomBorderHidden(!bottomRow) + cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) + cell.stp_setFakeSeparatorLeftInset(15.0) + } + + func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat { + return 57 + } + + func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat { + return 27.0 + } + + override func tableView( + _ tableView: UITableView, + heightForHeaderInSection section: Int + ) + -> CGFloat + { + let size = headerView?.sizeThatFits( + CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) + ) + return size?.height ?? 0.0 + } + + func tableView(_ tableView: UITableView, viewForHeaderInSection section: Int) -> UIView? { + return headerView + } + + func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { + tableView.deselectRow(at: indexPath, animated: true) + selectedShippingMethod = + shippingMethods?.stp_boundSafeObject(at: indexPath.row) + tableView.reloadSections( + NSIndexSet(index: indexPath.section) as IndexSet, + with: .fade + ) + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } + + required init( + nibName nibNameOrNil: String?, + bundle nibBundleOrNil: Bundle? + ) { + fatalError("init(nibName:bundle:) has not been implemented") + } + + required init( + theme: STPTheme? + ) { + fatalError("init(theme:) has not been implemented") + } +} + +@objc protocol STPShippingMethodsViewControllerDelegate: NSObjectProtocol { + func shippingMethodsViewController( + _ methodsViewController: STPShippingMethodsViewController, + didFinishWith method: PKShippingMethod + ) +} + +private let STPShippingMethodCellReuseIdentifier = "STPShippingMethodCellReuseIdentifier" diff --git a/Stripe/StripeiOS/Source/STPSource+BasicUI.swift b/Stripe/StripeiOS/Source/STPSource+BasicUI.swift new file mode 100644 index 00000000..67510e70 --- /dev/null +++ b/Stripe/StripeiOS/Source/STPSource+BasicUI.swift @@ -0,0 +1,74 @@ +// +// STPSource+BasicUI.swift +// StripeiOS +// +// Created by David Estes on 6/30/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripePayments +import UIKit + +extension STPSource: STPPaymentOption { + // MARK: - STPPaymentOption + @objc public var image: UIImage { + if type == .card, let cardDetails = cardDetails { + return STPImageLibrary.cardBrandImage(for: cardDetails.brand) + } else { + return STPImageLibrary.cardBrandImage(for: .unknown) + } + } + + @objc public var templateImage: UIImage { + if type == .card, let cardDetails = cardDetails { + return STPImageLibrary.templatedBrandImage(for: cardDetails.brand) + } else { + return STPImageLibrary.templatedBrandImage(for: .unknown) + } + } + + @objc public var label: String { + switch type { + case .bancontact: + return STPPaymentMethodType.bancontact.displayName + case .card: + if let cardDetails = cardDetails { + let brand = STPCard.string(from: cardDetails.brand) + return "\(brand) \(cardDetails.last4 ?? "")" + } else { + return STPCard.string(from: .unknown) + } + case .giropay: + return STPPaymentMethodType.giropay.displayName + case .iDEAL: + return STPPaymentMethodType.iDEAL.displayName + case .SEPADebit: + return STPPaymentMethodType.SEPADebit.displayName + case .sofort: + return STPPaymentMethodType.sofort.displayName + case .threeDSecure: + return STPLocalizedString("3D Secure", "Source type brand name") + case .alipay: + return STPPaymentMethodType.alipay.displayName + case .P24: + return STPPaymentMethodType.przelewy24.displayName + case .EPS: + return STPPaymentMethodType.EPS.displayName + case .multibanco: + return STPLocalizedString("Multibanco", "Source type brand name") + case .weChatPay: + return STPPaymentMethodType.weChatPay.displayName + case .klarna: + return STPPaymentMethodType.klarna.displayName + case .unknown: + return STPPaymentMethodType.unknown.displayName + @unknown default: + return STPPaymentMethodType.unknown.displayName + } + } + + @objc public var isReusable: Bool { + return usage != .singleUse + } +} diff --git a/Stripe/StripeiOS/Source/STPTheme.swift b/Stripe/StripeiOS/Source/STPTheme.swift new file mode 100644 index 00000000..b26d3a5a --- /dev/null +++ b/Stripe/StripeiOS/Source/STPTheme.swift @@ -0,0 +1,260 @@ +// +// STPTheme.swift +// StripeiOS +// +// Created by Jack Flintermann on 5/3/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeUICore +import UIKit + +// swift-format-ignore: DontRepeatTypeInStaticProperties +/// STPTheme objects can be used to visually style Stripe-provided UI. See https://stripe.com/docs/mobile/ios/basic#theming for more information. +final public class STPTheme: NSObject { + + /// The default theme used by all Stripe UI. All themable UI classes, such as `STPAddCardViewController`, have one initializer that takes a `theme` and one that does not. If you use the one that does not, the default theme will be used to customize that view controller's appearance. + @objc public static let defaultTheme = STPTheme() + + /// :nodoc: + @available(*, deprecated, message: "Use defaultTheme instead", renamed: "defaultTheme") + public static func `default`() -> STPTheme { + return STPTheme.defaultTheme + } + + /// The primary background color of the theme. This will be used as the `backgroundColor` for any views with this theme. + @objc public var primaryBackgroundColor: UIColor = STPThemeDefaultPrimaryBackgroundColor + + /// The secondary background color of this theme. This will be used as the `backgroundColor` for any supplemental views inside a view with this theme - for example, a `UITableView` will set it's cells' background color to this value. + @objc public var secondaryBackgroundColor: UIColor = STPThemeDefaultSecondaryBackgroundColor + + /// This color is automatically derived by reducing the alpha of the `primaryBackgroundColor` and is used as a section border color in table view cells. + @objc public var tertiaryBackgroundColor: UIColor { + let colorBlock: STPColorBlock = { + var hue: CGFloat = 0 + var saturation: CGFloat = 0 + var brightness: CGFloat = 0 + var alpha: CGFloat = 0 + self.primaryBackgroundColor.getHue( + &hue, + saturation: &saturation, + brightness: &brightness, + alpha: &alpha + ) + + return UIColor( + hue: hue, + saturation: saturation, + brightness: brightness - 0.09, + alpha: alpha + ) + } + if #available(iOS 13.0, *) { + return UIColor(dynamicProvider: { _ in + return colorBlock() + }) + } else { + return colorBlock() + } + } + + /// This color is automatically derived by reducing the brightness of the `primaryBackgroundColor` and is used as a separator color in table view cells. + @objc public var quaternaryBackgroundColor: UIColor { + let colorBlock: STPColorBlock = { + var hue: CGFloat = 0 + var saturation: CGFloat = 0 + var brightness: CGFloat = 0 + var alpha: CGFloat = 0 + self.primaryBackgroundColor.getHue( + &hue, + saturation: &saturation, + brightness: &brightness, + alpha: &alpha + ) + + return UIColor( + hue: hue, + saturation: saturation, + brightness: brightness - 0.03, + alpha: alpha + ) + } + if #available(iOS 13.0, *) { + return UIColor(dynamicProvider: { _ in + return colorBlock() + }) + } else { + return colorBlock() + } + } + + /// The primary foreground color of this theme. This will be used as the text color for any important labels in a view with this theme (such as the text color for a text field that the user needs to fill out). + @objc public var primaryForegroundColor: UIColor = STPThemeDefaultPrimaryForegroundColor + + /// The secondary foreground color of this theme. This will be used as the text color for any supplementary labels in a view with this theme (such as the placeholder color for a text field that the user needs to fill out). + @objc public var secondaryForegroundColor: UIColor = STPThemeDefaultSecondaryForegroundColor + + /// This color is automatically derived from the `secondaryForegroundColor` with a lower alpha component, used for disabled text. + @objc public var tertiaryForegroundColor: UIColor { + if #available(iOS 13.0, *) { + return UIColor(dynamicProvider: { _ in + return self.primaryForegroundColor.withAlphaComponent(0.25) + }) + } else { + return primaryForegroundColor.withAlphaComponent(0.25) + } + } + + /// The accent color of this theme - it will be used for any buttons and other elements on a view that are important to highlight. + @objc public var accentColor: UIColor = STPThemeDefaultAccentColor + + /// The error color of this theme - it will be used for rendering any error messages or views. + @objc public var errorColor: UIColor = STPThemeDefaultErrorColor + + /// The font to be used for all views using this theme. Make sure to select an appropriate size. + @objc public var font: UIFont { + get { + if let _font = _font { + return _font + } else { + let fontMetrics = UIFontMetrics(forTextStyle: .body) + return fontMetrics.scaledFont(for: STPThemeDefaultFont) + } + } + set { + _font = newValue + } + } + private var _font: UIFont? + + /// The medium-weight font to be used for all bold text in views using this theme. Make sure to select an appropriate size. + @objc public var emphasisFont: UIFont { + get { + if let _emphasisFont = _emphasisFont { + return _emphasisFont + } else { + let fontMetrics = UIFontMetrics(forTextStyle: .body) + return fontMetrics.scaledFont(for: STPThemeDefaultMediumFont) + } + } + set { + _emphasisFont = newValue + } + } + private var _emphasisFont: UIFont? + + /// The navigation bar style to use for any view controllers presented modally + /// by the SDK. The default value will be determined based on the brightness + /// of the theme's `secondaryBackgroundColor`. + @objc public var barStyle: UIBarStyle { + get { + if let _barStyle = _barStyle { + return _barStyle + } else { + return barStyle(for: secondaryBackgroundColor) + } + } + set { + _barStyle = newValue + } + } + private var _barStyle: UIBarStyle? + + /// A Boolean value indicating whether the navigation bar for any view controllers + /// presented modally by the SDK should be translucent. The default value is YES. + @objc public var translucentNavigationBar = true + + /// This font is automatically derived from the font, with a slightly lower point size, and will be used for supplementary labels. + @objc public var smallFont: UIFont { + return font.withSize(max(font.pointSize - 2, 1)) + } + + /// This font is automatically derived from the font, with a larger point size, and will be used for large labels such as SMS code entry. + @objc public var largeFont: UIFont { + return font.withSize(font.pointSize + 15) + } + + private func barStyle(for color: UIColor) -> UIBarStyle { + if color.isBright { + return .default + } else { + return .black + } + } +} + +extension STPTheme: NSCopying { + /// :nodoc: + @objc + public func copy(with zone: NSZone? = nil) -> Any { + let otherTheme = STPTheme() + otherTheme.primaryBackgroundColor = primaryBackgroundColor + otherTheme.secondaryBackgroundColor = secondaryBackgroundColor + otherTheme.primaryForegroundColor = primaryForegroundColor + otherTheme.secondaryForegroundColor = secondaryForegroundColor + otherTheme.accentColor = accentColor + otherTheme.errorColor = errorColor + otherTheme.translucentNavigationBar = translucentNavigationBar + otherTheme._font = _font + otherTheme._emphasisFont = _emphasisFont + otherTheme._barStyle = _barStyle + + return otherTheme + } +} + +private typealias STPColorBlock = () -> UIColor + +// MARK: Default Colors + +private var STPThemeDefaultPrimaryBackgroundColor: UIColor { + if #available(iOS 13.0, *) { + return .secondarySystemBackground + } else { + return UIColor(red: 242.0 / 255.0, green: 242.0 / 255.0, blue: 245.0 / 255.0, alpha: 1) + } +} + +private var STPThemeDefaultSecondaryBackgroundColor: UIColor { + if #available(iOS 13.0, *) { + return .systemBackground + } else { + return .white + } +} + +private var STPThemeDefaultPrimaryForegroundColor: UIColor { + if #available(iOS 13.0, *) { + return .label + } else { + return UIColor(red: 43.0 / 255.0, green: 43.0 / 255.0, blue: 45.0 / 255.0, alpha: 1) + } +} + +private var STPThemeDefaultSecondaryForegroundColor: UIColor { + if #available(iOS 13.0, *) { + return .secondaryLabel + } else { + return UIColor(red: 142.0 / 255.0, green: 142.0 / 255.0, blue: 147.0 / 255.0, alpha: 1) + } +} + +private var STPThemeDefaultAccentColor: UIColor { + if #available(iOS 13.0, *) { + return .systemBlue + } else { + return UIColor(red: 0.0 / 255.0, green: 122.0 / 255.0, blue: 255.0 / 255.0, alpha: 1) + } +} + +private var STPThemeDefaultErrorColor: UIColor { + if #available(iOS 13.0, *) { + return .systemRed + } else { + return UIColor(red: 255.0 / 255.0, green: 72.0 / 255.0, blue: 68.0 / 255.0, alpha: 1) + } +} + +// MARK: Default Fonts +private let STPThemeDefaultFont = UIFont.systemFont(ofSize: 17) +private let STPThemeDefaultMediumFont = UIFont.systemFont(ofSize: 17, weight: .medium) diff --git a/Stripe/StripeiOS/Source/STPUserInformation.swift b/Stripe/StripeiOS/Source/STPUserInformation.swift new file mode 100644 index 00000000..5e6eb2cd --- /dev/null +++ b/Stripe/StripeiOS/Source/STPUserInformation.swift @@ -0,0 +1,42 @@ +// +// STPUserInformation.swift +// StripeiOS +// +// Created by Jack Flintermann on 6/15/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import Foundation + +/// You can use this class to specify information that you've already collected +/// from your user. You can then set the `prefilledInformation` property on +/// `STPPaymentContext`, `STPAddCardViewController`, etc and it will pre-fill +/// this information whenever possible. +public class STPUserInformation: NSObject, NSCopying { + /// The user's billing address. When set, the add card form will be filled with + /// this address. The user will also have the option to fill their shipping address + /// using this address. + /// @note Set this using `setBillingAddressWithBillingDetails:` to use the billing + /// details from an `STPPaymentMethod` or `STPPaymentMethodParams` instance. + @objc public var billingAddress: STPAddress? + /// The user's shipping address. When set, the shipping address form will be filled + /// with this address. The user will also have the option to fill their billing + /// address using this address. + @objc public var shippingAddress: STPAddress? + + /// A convenience method to populate `billingAddress` with a PaymentMethod's billing details. + /// @note Calling this overwrites the value of `billingAddress`. + @objc(setBillingAddressWithBillingDetails:) + public func setBillingAddress(with billingDetails: STPPaymentMethodBillingDetails) { + billingAddress = STPAddress(paymentMethodBillingDetails: billingDetails) + } + + /// :nodoc: + @objc + public func copy(with zone: NSZone? = nil) -> Any { + let copy = STPUserInformation() + copy.billingAddress = billingAddress + copy.shippingAddress = shippingAddress + return copy + } +} diff --git a/Stripe/StripeiOS/Source/String+Localized.swift b/Stripe/StripeiOS/Source/String+Localized.swift new file mode 100644 index 00000000..55bd59dd --- /dev/null +++ b/Stripe/StripeiOS/Source/String+Localized.swift @@ -0,0 +1,22 @@ +// +// String+Localized.swift +// StripeiOS +// +// Created by Mel Ludowise on 7/6/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore +@_spi(STP) import StripePaymentsUI +@_spi(STP) import StripeUICore + +// MARK: - Legacy strings + +/// Legacy strings +extension StripeSharedStrings { + static func localizedPostalCodeString(for countryCode: String?) -> String { + return countryCode == "US" + ? String.Localized.zip : String.Localized.postal_code + } +} diff --git a/Stripe/StripeiOS/Source/Stripe+Exports.swift b/Stripe/StripeiOS/Source/Stripe+Exports.swift new file mode 100644 index 00000000..16160f1e --- /dev/null +++ b/Stripe/StripeiOS/Source/Stripe+Exports.swift @@ -0,0 +1,13 @@ +// +// Stripe+Exports.swift +// StripeiOS +// +// This file re-exports classes from Stripe's dependent SDKs, enabling users to use classes from all +// Stripe Payments SDKs with one `import Stripe` declaration. +// + +import Foundation +@_exported import StripeApplePay +@_exported import StripeCore +@_exported import StripePayments +@_exported import StripePaymentsUI diff --git a/Stripe/StripeiOS/Source/StripeBundleLocator.swift b/Stripe/StripeiOS/Source/StripeBundleLocator.swift new file mode 100644 index 00000000..2ffd6423 --- /dev/null +++ b/Stripe/StripeiOS/Source/StripeBundleLocator.swift @@ -0,0 +1,20 @@ +// +// StripeBundleLocator.swift +// StripeiOS +// +// Created by Mel Ludowise on 7/6/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +/// :nodoc: +@_spi(STP) public final class StripeBundleLocator: BundleLocatorProtocol { + public static let internalClass: AnyClass = StripeBundleLocator.self + public static let bundleName = "Stripe" + #if SWIFT_PACKAGE + public static let spmResourcesBundle = Bundle.module + #endif + public static let resourcesBundle = StripeBundleLocator.computeResourcesBundle() +} diff --git a/Stripe/StripeiOS/Source/UIBarButtonItem+Stripe.swift b/Stripe/StripeiOS/Source/UIBarButtonItem+Stripe.swift new file mode 100644 index 00000000..2c792c1b --- /dev/null +++ b/Stripe/StripeiOS/Source/UIBarButtonItem+Stripe.swift @@ -0,0 +1,54 @@ +// +// UIBarButtonItem+Stripe.swift +// StripeiOS +// +// Created by Jack Flintermann on 5/18/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeUICore +import UIKit + +extension UIBarButtonItem { + @objc(stp_setTheme:) func stp_setTheme(_ theme: STPTheme) { + let image = backgroundImage(for: .normal, barMetrics: .default) + if let image = image { + let enabledImage: UIImage = STPLegacyImageLibrary.image( + withTintColor: theme.accentColor, + for: image + ) + let disabledImage: UIImage = STPLegacyImageLibrary.image( + withTintColor: theme.secondaryForegroundColor, + for: image + ) + setBackgroundImage(enabledImage, for: .normal, barMetrics: .default) + setBackgroundImage(disabledImage, for: .disabled, barMetrics: .default) + } + + tintColor = isEnabled ? theme.accentColor : theme.secondaryForegroundColor + + setTitleTextAttributes( + [ + NSAttributedString.Key.font: style == .plain ? theme.font : theme.emphasisFont, + NSAttributedString.Key.foregroundColor: theme.accentColor, + ], + for: .normal + ) + + setTitleTextAttributes( + [ + NSAttributedString.Key.font: style == .plain ? theme.font : theme.emphasisFont, + NSAttributedString.Key.foregroundColor: theme.secondaryForegroundColor, + ], + for: .disabled + ) + + setTitleTextAttributes( + [ + NSAttributedString.Key.font: style == .plain ? theme.font : theme.emphasisFont, + NSAttributedString.Key.foregroundColor: theme.accentColor, + ], + for: .highlighted + ) + } +} diff --git a/Stripe/StripeiOS/Source/UINavigationBar+Stripe_Theme.swift b/Stripe/StripeiOS/Source/UINavigationBar+Stripe_Theme.swift new file mode 100644 index 00000000..21fcd646 --- /dev/null +++ b/Stripe/StripeiOS/Source/UINavigationBar+Stripe_Theme.swift @@ -0,0 +1,144 @@ +// +// UINavigationBar+Stripe_Theme.swift +// StripeiOS +// +// Created by Jack Flintermann on 5/17/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import ObjectiveC +import UIKit + +/// This allows quickly setting the appearance of a `UINavigationBar` to match your +/// application. This is useful if you're presenting an `STPAddCardViewController` +/// or `STPPaymentOptionsViewController` inside a `UINavigationController`. +extension UINavigationBar { + /// Sets the navigation bar's appearance to the desired theme. This will affect the + /// bar's `tintColor` and `barTintColor` properties, as well as the color of the + /// single-pixel line at the bottom of the navbar. + /// - Parameter theme: the theme to use to style the navigation bar. - seealso: STPTheme.h + /// @deprecated Use the `stp_theme` property instead + @available(*, deprecated, message: "Use the `stp_theme` property instead") + @objc + public func stp_setTheme(_ theme: STPTheme) { + stp_theme = theme + } + + /// Sets the navigation bar's appearance to the desired theme. This will affect the bar's `tintColor` and `barTintColor` properties, as well as the color of the single-pixel line at the bottom of the navbar. + /// Stripe view controllers will use their navigation bar's theme for their UIBarButtonItems instead of their own theme if it is not nil. + /// - seealso: STPTheme.h + + @objc public var stp_theme: STPTheme? { + get { + return objc_getAssociatedObject( + self, + UnsafeRawPointer(&kUINavigationBarSTPThemeObjectKey) + ) + as? STPTheme + } + set(theme) { + objc_setAssociatedObject( + self, + UnsafeRawPointer(&kUINavigationBarSTPThemeObjectKey), + theme, + .OBJC_ASSOCIATION_RETAIN_NONATOMIC + ) + + if let hairlineImageView = stp_hairlineImageView() { + hairlineImageView.isHidden = theme != nil + } + + guard let theme = theme else { + return + } + + stp_artificialHairlineView().backgroundColor = theme.tertiaryBackgroundColor + barTintColor = theme.secondaryBackgroundColor + tintColor = theme.accentColor + barStyle = theme.barStyle + isTranslucent = theme.translucentNavigationBar + + titleTextAttributes = [ + NSAttributedString.Key.font: theme.emphasisFont, + NSAttributedString.Key.foregroundColor: theme.primaryForegroundColor, + ] + + largeTitleTextAttributes = [ + NSAttributedString.Key.foregroundColor: theme.primaryForegroundColor + ] + + if #available(iOS 13.0, *) { + standardAppearance.backgroundColor = theme.secondaryBackgroundColor + if let titleTextAttributes = titleTextAttributes { + standardAppearance.titleTextAttributes = titleTextAttributes + } + if let largeTitleTextAttributes = largeTitleTextAttributes { + standardAppearance.largeTitleTextAttributes = largeTitleTextAttributes + } + standardAppearance.buttonAppearance.normal.titleTextAttributes = [ + NSAttributedString.Key.font: theme.font, + NSAttributedString.Key.foregroundColor: theme.accentColor, + ] + + standardAppearance.buttonAppearance.highlighted.titleTextAttributes = [ + NSAttributedString.Key.font: theme.font, + NSAttributedString.Key.foregroundColor: theme.accentColor, + ] + + standardAppearance.buttonAppearance.disabled.titleTextAttributes = [ + NSAttributedString.Key.font: theme.font, + NSAttributedString.Key.foregroundColor: theme.secondaryForegroundColor, + ] + + standardAppearance.doneButtonAppearance.normal.titleTextAttributes = [ + NSAttributedString.Key.font: theme.emphasisFont, + NSAttributedString.Key.foregroundColor: theme.accentColor, + ] + + standardAppearance.doneButtonAppearance.highlighted.titleTextAttributes = [ + NSAttributedString.Key.font: theme.emphasisFont, + NSAttributedString.Key.foregroundColor: theme.accentColor, + ] + + standardAppearance.doneButtonAppearance.disabled.titleTextAttributes = [ + NSAttributedString.Key.font: theme.emphasisFont, + NSAttributedString.Key.foregroundColor: theme.secondaryForegroundColor, + ] + scrollEdgeAppearance = standardAppearance + compactAppearance = standardAppearance + } + } + } + + func stp_artificialHairlineView() -> UIView { + var view = viewWithTag(STPNavigationBarHairlineViewTag) + if view == nil { + view = UIView(frame: CGRect(x: 0, y: bounds.maxY, width: bounds.width, height: 0.5)) + view?.autoresizingMask = [.flexibleWidth, .flexibleTopMargin] + view?.tag = STPNavigationBarHairlineViewTag + if let view = view { + addSubview(view) + } + } + return view! + } + + func stp_hairlineImageView() -> UIImageView? { + return stp_hairlineImageView(self) + } + + func stp_hairlineImageView(_ view: UIView) -> UIImageView? { + if (view is UIImageView) && view.bounds.size.height <= 1.0 { + return (view as? UIImageView)! + } + for subview in view.subviews { + if let imageView = stp_hairlineImageView(subview) { + return imageView + } + } + return nil + } +} + +private let STPNavigationBarHairlineViewTag = 787473 +private var kUINavigationBarSTPThemeObjectKey = 0 diff --git a/Stripe/StripeiOS/Source/UINavigationController+Stripe_Completion.swift b/Stripe/StripeiOS/Source/UINavigationController+Stripe_Completion.swift new file mode 100644 index 00000000..f0ed8296 --- /dev/null +++ b/Stripe/StripeiOS/Source/UINavigationController+Stripe_Completion.swift @@ -0,0 +1,61 @@ +// +// UINavigationController+Stripe_Completion.swift +// StripeiOS +// +// Created by Jack Flintermann on 3/23/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +// See http://stackoverflow.com/questions/9906966/completion-handler-for-uinavigationcontroller-pushviewcontrolleranimated/33767837#33767837 for some discussion around why using CATransaction is unreliable here. + +extension UINavigationController { + func stp_push( + _ viewController: UIViewController?, + animated: Bool, + completion: @escaping STPVoidBlock + ) { + if let viewController = viewController { + pushViewController(viewController, animated: animated) + } + if transitionCoordinator != nil && animated { + transitionCoordinator?.animate(alongsideTransition: nil) { _ in + completion() + } + } else { + completion() + } + } + + func stp_popViewController( + animated: Bool, + completion: @escaping STPVoidBlock + ) { + popViewController(animated: animated) + if transitionCoordinator != nil && animated { + transitionCoordinator?.animate(alongsideTransition: nil) { _ in + completion() + } + } else { + completion() + } + } + + @objc(stp_popToViewController:animated:completion:) func stp_pop( + to viewController: UIViewController?, + animated: Bool, + completion: @escaping STPVoidBlock + ) { + if let viewController = viewController { + popToViewController(viewController, animated: animated) + } + if transitionCoordinator != nil && animated { + transitionCoordinator?.animate(alongsideTransition: nil) { _ in + completion() + } + } else { + completion() + } + } +} diff --git a/Stripe/StripeiOS/Source/UITableViewCell+Stripe_Borders.swift b/Stripe/StripeiOS/Source/UITableViewCell+Stripe_Borders.swift new file mode 100644 index 00000000..c70200a0 --- /dev/null +++ b/Stripe/StripeiOS/Source/UITableViewCell+Stripe_Borders.swift @@ -0,0 +1,103 @@ +// +// UITableViewCell+Stripe_Borders.swift +// StripeiOS +// +// Created by Jack Flintermann on 5/16/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +private let STPTableViewCellTopBorderTag = 787473 +private let STPTableViewCellBottomBorderTag = 787474 +private let STPTableViewCellFakeSeparatorTag = 787475 + +extension UITableViewCell { + @objc(stp_setBorderColor:) func stp_setBorderColor(_ color: UIColor?) { + stp_topBorderView()?.backgroundColor = color + stp_bottomBorderView()?.backgroundColor = color + } + + @objc(stp_setTopBorderHidden:) func stp_setTopBorderHidden(_ hidden: Bool) { + stp_topBorderView()?.isHidden = hidden + } + + @objc(stp_setBottomBorderHidden:) func stp_setBottomBorderHidden(_ hidden: Bool) { + stp_bottomBorderView()?.isHidden = hidden + stp_fakeSeparatorView()?.isHidden = !hidden + } + + @objc(stp_setFakeSeparatorLeftInset:) func stp_setFakeSeparatorLeftInset(_ leftInset: CGFloat) { + stp_fakeSeparatorView()?.frame = CGRect( + x: leftInset, + y: bounds.size.height - 0.5, + width: bounds.size.width - leftInset, + height: 0.5 + ) + } + + @objc(stp_setFakeSeparatorColor:) func stp_setFakeSeparatorColor(_ color: UIColor?) { + stp_fakeSeparatorView()?.backgroundColor = color + } + + func stp_topBorderView() -> UIView? { + var view = viewWithTag(STPTableViewCellTopBorderTag) + if view == nil { + view = UIView(frame: CGRect(x: 0, y: 0, width: bounds.size.width, height: 0.5)) + view?.autoresizingMask = [.flexibleWidth, .flexibleBottomMargin] + view?.tag = STPTableViewCellTopBorderTag + view?.backgroundColor = backgroundColor + view?.isHidden = true + view?.accessibilityIdentifier = "stp_topBorderView" + if let view = view { + addSubview(view) + } + } + return view + } + + func stp_bottomBorderView() -> UIView? { + var view = viewWithTag(STPTableViewCellBottomBorderTag) + if view == nil { + view = UIView( + frame: CGRect( + x: 0, + y: bounds.size.height - 0.5, + width: bounds.size.width, + height: 0.5 + ) + ) + view?.autoresizingMask = [.flexibleWidth, .flexibleTopMargin] + view?.tag = STPTableViewCellBottomBorderTag + view?.backgroundColor = backgroundColor + view?.isHidden = true + view?.accessibilityIdentifier = "stp_bottomBorderView" + if let view = view { + addSubview(view) + } + } + return view + } + + func stp_fakeSeparatorView() -> UIView? { + var view = viewWithTag(STPTableViewCellFakeSeparatorTag) + if view == nil { + view = UIView( + frame: CGRect( + x: 0, + y: bounds.size.height - 0.5, + width: bounds.size.width, + height: 0.5 + ) + ) + view?.autoresizingMask = [.flexibleWidth, .flexibleTopMargin] + view?.tag = STPTableViewCellFakeSeparatorTag + view?.backgroundColor = backgroundColor + view?.accessibilityIdentifier = "stp_fakeSeparatorView" + if let view = view { + addSubview(view) + } + } + return view + } +} diff --git a/Stripe/StripeiOS/Source/UIToolbar+Stripe_InputAccessory.swift b/Stripe/StripeiOS/Source/UIToolbar+Stripe_InputAccessory.swift new file mode 100644 index 00000000..4194764e --- /dev/null +++ b/Stripe/StripeiOS/Source/UIToolbar+Stripe_InputAccessory.swift @@ -0,0 +1,38 @@ +// +// UIToolbar+Stripe_InputAccessory.swift +// StripeiOS +// +// Created by Jack Flintermann on 4/22/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +extension UIToolbar { + @objc(stp_inputAccessoryToolbarWithTarget:action:) class func stp_inputAccessoryToolbar( + withTarget target: Any?, + action: Selector + ) -> Self { + let toolbar = self.init() + let flexibleItem = UIBarButtonItem( + barButtonSystemItem: .flexibleSpace, + target: nil, + action: nil + ) + let nextItem = UIBarButtonItem( + title: STPLocalizedString("Next", "Button to move to the next text entry field"), + style: .done, + target: target, + action: action + ) + toolbar.items = [flexibleItem, nextItem] + toolbar.autoresizingMask = .flexibleHeight + return toolbar + } + + @objc(stp_setEnabled:) func stp_setEnabled(_ enabled: Bool) { + for barButtonItem in items ?? [] { + barButtonItem.isEnabled = enabled + } + } +} diff --git a/Stripe/StripeiOS/Source/UIView+Helpers.swift b/Stripe/StripeiOS/Source/UIView+Helpers.swift new file mode 100644 index 00000000..5ab99d68 --- /dev/null +++ b/Stripe/StripeiOS/Source/UIView+Helpers.swift @@ -0,0 +1,35 @@ +// +// UIView+Helpers.swift +// StripeiOS +// +// Created by Yuki Tokuhiro on 11/4/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import UIKit + +protocol SafeAreaLayoutGuide { + var leadingAnchor: NSLayoutXAxisAnchor { get } + var trailingAnchor: NSLayoutXAxisAnchor { get } + var leftAnchor: NSLayoutXAxisAnchor { get } + var rightAnchor: NSLayoutXAxisAnchor { get } + var topAnchor: NSLayoutYAxisAnchor { get } + var bottomAnchor: NSLayoutYAxisAnchor { get } + var widthAnchor: NSLayoutDimension { get } + var heightAnchor: NSLayoutDimension { get } + var centerXAnchor: NSLayoutXAxisAnchor { get } + var centerYAnchor: NSLayoutYAxisAnchor { get } +} + +extension UIView: SafeAreaLayoutGuide {} +extension UILayoutGuide: SafeAreaLayoutGuide {} + +extension UIView { + var _safeAreaLayoutGuide: SafeAreaLayoutGuide { + if #available(iOSApplicationExtension 11.0, *) { + return safeAreaLayoutGuide + } else { + return self + } + } +} diff --git a/Stripe/StripeiOS/Source/UIView+Stripe_FirstResponder.swift b/Stripe/StripeiOS/Source/UIView+Stripe_FirstResponder.swift new file mode 100644 index 00000000..40267b0a --- /dev/null +++ b/Stripe/StripeiOS/Source/UIView+Stripe_FirstResponder.swift @@ -0,0 +1,24 @@ +// +// UIView+Stripe_FirstResponder.swift +// StripeiOS +// +// Created by Jack Flintermann on 4/15/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +extension UIView { + @objc func stp_findFirstResponder() -> UIView? { + if isFirstResponder { + return self + } + for subView in subviews { + let responder = subView.stp_findFirstResponder() + if let responder = responder { + return responder + } + } + return nil + } +} diff --git a/Stripe/StripeiOS/Source/UIView+Stripe_SafeAreaBounds.swift b/Stripe/StripeiOS/Source/UIView+Stripe_SafeAreaBounds.swift new file mode 100644 index 00000000..d29b49b8 --- /dev/null +++ b/Stripe/StripeiOS/Source/UIView+Stripe_SafeAreaBounds.swift @@ -0,0 +1,24 @@ +// +// UIView+Stripe_SafeAreaBounds.swift +// StripeiOS +// +// Created by Ben Guo on 12/12/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import UIKit + +extension UIView { + /// Returns this view's bounds inset to `safeAreaInsets.left` and `safeAreaInsets.right`. + /// Top and bottom safe area insets are ignored. On iOS <11, this returns self.bounds. + @objc func stp_boundsWithHorizontalSafeAreaInsets() -> CGRect { + let insets = safeAreaInsets + let safeBounds = CGRect( + x: bounds.origin.x + insets.left, + y: bounds.origin.y, + width: bounds.size.width - (insets.left + insets.right), + height: bounds.size.height + ) + return safeBounds + } +} diff --git a/Stripe/StripeiOS/Source/UIViewController+Stripe_KeyboardAvoiding.swift b/Stripe/StripeiOS/Source/UIViewController+Stripe_KeyboardAvoiding.swift new file mode 100644 index 00000000..ce6408dc --- /dev/null +++ b/Stripe/StripeiOS/Source/UIViewController+Stripe_KeyboardAvoiding.swift @@ -0,0 +1,166 @@ +// +// UIViewController+Stripe_KeyboardAvoiding.swift +// StripeiOS +// +// Created by Jack Flintermann on 4/15/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +typealias STPKeyboardFrameBlock = (CGRect, UIView?) -> Void +extension UIViewController { + @objc(stp_beginObservingKeyboardAndInsettingScrollView:onChangeBlock:) + func stp_beginObservingKeyboardAndInsettingScrollView( + _ scrollView: UIScrollView?, + onChange block: STPKeyboardFrameBlock? + ) { + if let existing = stp_keyboardDetectingViewController() { + existing.removeFromParent() + existing.view.removeFromSuperview() + existing.didMove(toParent: nil) + } + let keyboardAvoiding = STPKeyboardDetectingViewController( + keyboardFrameBlock: block, + scrollView: scrollView + ) + addChild(keyboardAvoiding) + view.addSubview(keyboardAvoiding.view) + keyboardAvoiding.didMove(toParent: self) + } + + @objc func stp_keyboardDetectingViewController() -> STPKeyboardDetectingViewController? { + return + (children as NSArray).filtered( + using: NSPredicate(block: { viewController, _ in + return viewController is STPKeyboardDetectingViewController + }) + ).first as? STPKeyboardDetectingViewController + } +} + +// This is a private class that is only a UIViewController subclass by virtue of the fact +// that that makes it easier to attach to another UIViewController as a child. +class STPKeyboardDetectingViewController: UIViewController { + var lastKeyboardFrame = CGRect.zero + weak var lastResponder: UIView? + var keyboardFrameBlock: STPKeyboardFrameBlock? + weak var managedScrollView: UIScrollView? + var currentBottomInsetChange: CGFloat = 0.0 + + init( + keyboardFrameBlock block: STPKeyboardFrameBlock?, + scrollView: UIScrollView? + ) { + keyboardFrameBlock = block + super.init(nibName: nil, bundle: nil) + NotificationCenter.default.addObserver( + self, + selector: #selector(keyboardWillChangeFrame(_:)), + name: UIResponder.keyboardWillChangeFrameNotification, + object: nil + ) + NotificationCenter.default.addObserver( + self, + selector: #selector(textFieldWillBeginEditing(_:)), + name: UITextField.textDidBeginEditingNotification, + object: nil + ) + managedScrollView = scrollView + currentBottomInsetChange = 0 + } + + deinit { + NotificationCenter.default.removeObserver(self) + } + + override func loadView() { + let view = UIView() + view.backgroundColor = UIColor.clear + view.autoresizingMask = [] + self.view = view + } + + @objc func textFieldWillBeginEditing(_ notification: Notification) { + guard let textField = notification.object as? UITextField, let parentView = parent?.view, + textField.isDescendant(of: parentView) + else { + return + } + if let keyboardFrameBlock = keyboardFrameBlock, + textField != lastResponder && !lastKeyboardFrame.isEmpty + { + UIView.animate( + withDuration: 0.3, + delay: 0, + options: .curveEaseOut, + animations: { + keyboardFrameBlock(self.lastKeyboardFrame, textField) + } + ) + } + } + + @objc func keyboardWillChangeFrame(_ notification: Notification) { + // As of iOS 8, this all takes place inside the necessary animation block + // https://twitter.com/SmileyKeith/status/684100833823174656 + guard + var keyboardFrame = + (notification.userInfo?[UIResponder.keyboardFrameEndUserInfoKey] as? NSValue)? + .cgRectValue, + let window = view.window + else { + return + } + keyboardFrame = window.convert(keyboardFrame, from: nil) + + if managedScrollView != nil { + if !lastKeyboardFrame.equalTo(keyboardFrame) { + let responder = parent?.view.stp_findFirstResponder() + lastResponder = responder + doKeyboardChangeAnimation(withNewFrame: keyboardFrame) + } + } + } + + func doKeyboardChangeAnimation(withNewFrame keyboardFrame: CGRect) { + lastKeyboardFrame = keyboardFrame + + if let managedScrollView = managedScrollView { + let scrollView = managedScrollView + let scrollViewSuperView = managedScrollView.superview + + var contentInsets = scrollView.contentInset + var scrollIndicatorInsets: UIEdgeInsets = .zero + #if !TARGET_OS_MACCATALYST + scrollIndicatorInsets = scrollView.verticalScrollIndicatorInsets + #else + scrollIndicatorInsets = scrollView.scrollIndicatorInsets + #endif + + let windowFrame = scrollViewSuperView?.convert( + scrollViewSuperView?.frame ?? CGRect.zero, + to: nil + ) + + let bottomIntersection = windowFrame?.intersection(keyboardFrame) + let bottomInsetDelta = + (bottomIntersection?.size.height ?? 0.0) - currentBottomInsetChange + contentInsets.bottom += bottomInsetDelta + scrollIndicatorInsets.bottom += bottomInsetDelta + currentBottomInsetChange += bottomInsetDelta + scrollView.contentInset = contentInsets + scrollView.scrollIndicatorInsets = scrollIndicatorInsets + } + + if let keyboardFrameBlock = keyboardFrameBlock { + keyboardFrameBlock(keyboardFrame, lastResponder) + } + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } +} diff --git a/Stripe/StripeiOS/Source/UIViewController+Stripe_NavigationItemProxy.swift b/Stripe/StripeiOS/Source/UIViewController+Stripe_NavigationItemProxy.swift new file mode 100644 index 00000000..0b5fd9ee --- /dev/null +++ b/Stripe/StripeiOS/Source/UIViewController+Stripe_NavigationItemProxy.swift @@ -0,0 +1,38 @@ +// +// UIViewController+Stripe_NavigationItemProxy.swift +// StripeiOS +// +// Created by Jack Flintermann on 6/9/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import ObjectiveC +import UIKit + +extension UIViewController { + @objc var stp_navigationItemProxy: UINavigationItem? { + get { + return objc_getAssociatedObject(self, UnsafeRawPointer(&kSTPNavigationItemProxyKey)) + as? UINavigationItem ?? self.navigationItem + } + set(stp_navigationItemProxy) { + objc_setAssociatedObject( + self, + UnsafeRawPointer(&kSTPNavigationItemProxyKey), + stp_navigationItemProxy, + .OBJC_ASSOCIATION_RETAIN_NONATOMIC + ) + if navigationItem.leftBarButtonItem != nil { + stp_navigationItemProxy?.leftBarButtonItem = navigationItem.leftBarButtonItem + } + if navigationItem.rightBarButtonItem != nil { + stp_navigationItemProxy?.rightBarButtonItem = navigationItem.rightBarButtonItem + } + if navigationItem.title != nil { + stp_navigationItemProxy?.title = navigationItem.title + } + } + } +} + +private var kSTPNavigationItemProxyKey = 0 diff --git a/Stripe/StripeiOS/Source/UIViewController+Stripe_ParentViewController.swift b/Stripe/StripeiOS/Source/UIViewController+Stripe_ParentViewController.swift new file mode 100644 index 00000000..534172cb --- /dev/null +++ b/Stripe/StripeiOS/Source/UIViewController+Stripe_ParentViewController.swift @@ -0,0 +1,50 @@ +// +// UIViewController+Stripe_ParentViewController.swift +// StripeiOS +// +// Created by Jack Flintermann on 1/12/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import UIKit + +extension UIViewController { + @objc(stp_parentViewControllerOfClass:) func stp_parentViewControllerOf( + _ klass: AnyClass + ) + -> UIViewController? + { + if let parent = parent, parent.isKind(of: klass) { + return parent + } + return parent?.stp_parentViewControllerOf(klass) + } + + @objc func stp_isTopNavigationController() -> Bool { + return navigationController?.topViewController == self + } + + @objc func stp_isAtRootOfNavigationController() -> Bool { + let viewController = navigationController?.viewControllers.first + var tested: UIViewController? = self + while tested != nil { + if tested == viewController { + return true + } + if let parent = tested?.parent { + tested = parent + } else { + return false + } + } + return false + } + + @objc func stp_previousViewControllerInNavigation() -> UIViewController? { + let index = navigationController?.viewControllers.firstIndex(of: self) ?? NSNotFound + if index == NSNotFound || index <= 0 { + return nil + } + return navigationController?.viewControllers[index - 1] + } +} diff --git a/Stripe/StripeiOS/Stripe-umbrella.h b/Stripe/StripeiOS/Stripe-umbrella.h new file mode 100644 index 00000000..b062ed0c --- /dev/null +++ b/Stripe/StripeiOS/Stripe-umbrella.h @@ -0,0 +1,11 @@ +// +// Stripe-umbrella.h +// StripeiOS +// +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#ifndef Stripe_umbrella_h +#define Stripe_umbrella_h + +#endif /* Stripe_umbrella_h */ diff --git a/Stripe/StripeiOS/Stripe.modulemap b/Stripe/StripeiOS/Stripe.modulemap new file mode 100644 index 00000000..35dad19e --- /dev/null +++ b/Stripe/StripeiOS/Stripe.modulemap @@ -0,0 +1,6 @@ +framework module Stripe { + umbrella header "Stripe-umbrella.h" + + export * + module * { export * } +} diff --git a/Stripe/StripeiOSAppHostedTests/Info.plist b/Stripe/StripeiOSAppHostedTests/Info.plist new file mode 100644 index 00000000..64d65ca4 --- /dev/null +++ b/Stripe/StripeiOSAppHostedTests/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + + diff --git a/Stripe/StripeiOSAppHostedTests/LinkSecureCookieStoreTests.swift b/Stripe/StripeiOSAppHostedTests/LinkSecureCookieStoreTests.swift new file mode 100644 index 00000000..225d3b7f --- /dev/null +++ b/Stripe/StripeiOSAppHostedTests/LinkSecureCookieStoreTests.swift @@ -0,0 +1,76 @@ +// +// LinkSecureCookieStoreTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 12/22/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +@testable import Stripe +@testable import StripePaymentSheet +import XCTest + +class LinkSecureCookieStoreTests: XCTestCase { + + static let testKey: LinkCookieKey = .session + + let cookieStore: LinkSecureCookieStore = .shared + + func testWrite() { + cookieStore.write(key: Self.testKey, value: "cookie_value") + + XCTAssertEqual(cookieStore.read(key: Self.testKey), "cookie_value") + } + + func testWrite_allowSyncTrue() { + cookieStore.write(key: Self.testKey, value: "cookie_value", allowSync: true) + + XCTAssertEqual(cookieStore.read(key: Self.testKey), "cookie_value") + } + + func testWrite_overwriting() { + cookieStore.write(key: Self.testKey, value: "cookie_value") + cookieStore.write(key: Self.testKey, value: "new_cookie_value") + + XCTAssertEqual(cookieStore.read(key: Self.testKey), "new_cookie_value") + } + + func testDelete() { + cookieStore.write(key: Self.testKey, value: "cookie_value") + cookieStore.delete(key: Self.testKey) + + XCTAssertNil(cookieStore.read(key: Self.testKey)) + } + + func testDelete_allowSyncTrue() { + cookieStore.write(key: Self.testKey, value: "cookie_value", allowSync: true) + cookieStore.delete(key: Self.testKey) + + XCTAssertNil(cookieStore.read(key: Self.testKey)) + } + + // MARK: Session cookies + + func testFormattedSessionCookies() { + cookieStore.write(key: .session, value: "cookie_value") + XCTAssertEqual(cookieStore.formattedSessionCookies(), [ + "verification_session_client_secrets": ["cookie_value"] + ]) + + cookieStore.delete(key: .session) + XCTAssertNil(cookieStore.formattedSessionCookies()) + } + + func testUpdateSessionCookie() { + cookieStore.updateSessionCookie(with: "top_secret") + XCTAssertEqual(cookieStore.read(key: .session), "top_secret") + + // Updating with a `nil` client secret should be a no-op. + cookieStore.updateSessionCookie(with: nil) + XCTAssertEqual(cookieStore.read(key: .session), "top_secret") + + cookieStore.updateSessionCookie(with: "") + XCTAssertNil(cookieStore.read(key: .session)) + } + +} diff --git a/Stripe/StripeiOSTestHostApp/AppDelegate.swift b/Stripe/StripeiOSTestHostApp/AppDelegate.swift new file mode 100644 index 00000000..729e9852 --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/AppDelegate.swift @@ -0,0 +1,18 @@ +// +// AppDelegate.swift +// StripeiOSTestHostApp +// +// Created by Cameron Sabol on 11/4/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import UIKit + +@main +class AppDelegate: UIResponder, UIApplicationDelegate { + + func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool { + // Override point for customization after application launch. + return true + } +} diff --git a/Stripe/StripeiOSTestHostApp/Info.plist b/Stripe/StripeiOSTestHostApp/Info.plist new file mode 100644 index 00000000..5b531f7b --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/Info.plist @@ -0,0 +1,66 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + LSRequiresIPhoneOS + + UIApplicationSceneManifest + + UIApplicationSupportsMultipleScenes + + UISceneConfigurations + + UIWindowSceneSessionRoleApplication + + + UISceneConfigurationName + Default Configuration + UISceneDelegateClassName + $(PRODUCT_MODULE_NAME).SceneDelegate + UISceneStoryboardFile + Main + + + + + UIApplicationSupportsIndirectInputEvents + + UILaunchStoryboardName + LaunchScreen + UIMainStoryboardFile + Main + UIRequiredDeviceCapabilities + + armv7 + + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + + diff --git a/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AccentColor.colorset/Contents.json b/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AccentColor.colorset/Contents.json new file mode 100644 index 00000000..eb878970 --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AccentColor.colorset/Contents.json @@ -0,0 +1,11 @@ +{ + "colors" : [ + { + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AppIcon.appiconset/Contents.json b/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 00000000..9221b9bb --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -0,0 +1,98 @@ +{ + "images" : [ + { + "idiom" : "iphone", + "scale" : "2x", + "size" : "20x20" + }, + { + "idiom" : "iphone", + "scale" : "3x", + "size" : "20x20" + }, + { + "idiom" : "iphone", + "scale" : "2x", + "size" : "29x29" + }, + { + "idiom" : "iphone", + "scale" : "3x", + "size" : "29x29" + }, + { + "idiom" : "iphone", + "scale" : "2x", + "size" : "40x40" + }, + { + "idiom" : "iphone", + "scale" : "3x", + "size" : "40x40" + }, + { + "idiom" : "iphone", + "scale" : "2x", + "size" : "60x60" + }, + { + "idiom" : "iphone", + "scale" : "3x", + "size" : "60x60" + }, + { + "idiom" : "ipad", + "scale" : "1x", + "size" : "20x20" + }, + { + "idiom" : "ipad", + "scale" : "2x", + "size" : "20x20" + }, + { + "idiom" : "ipad", + "scale" : "1x", + "size" : "29x29" + }, + { + "idiom" : "ipad", + "scale" : "2x", + "size" : "29x29" + }, + { + "idiom" : "ipad", + "scale" : "1x", + "size" : "40x40" + }, + { + "idiom" : "ipad", + "scale" : "2x", + "size" : "40x40" + }, + { + "idiom" : "ipad", + "scale" : "1x", + "size" : "76x76" + }, + { + "idiom" : "ipad", + "scale" : "2x", + "size" : "76x76" + }, + { + "idiom" : "ipad", + "scale" : "2x", + "size" : "83.5x83.5" + }, + { + "idiom" : "ios-marketing", + "scale" : "1x", + "size" : "1024x1024" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/Contents.json b/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/Resources/Assets.xcassets/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Stripe/StripeiOSTestHostApp/Resources/Base.lproj/LaunchScreen.storyboard b/Stripe/StripeiOSTestHostApp/Resources/Base.lproj/LaunchScreen.storyboard new file mode 100644 index 00000000..865e9329 --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/Resources/Base.lproj/LaunchScreen.storyboard @@ -0,0 +1,25 @@ + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Stripe/StripeiOSTestHostApp/Resources/Base.lproj/Main.storyboard b/Stripe/StripeiOSTestHostApp/Resources/Base.lproj/Main.storyboard new file mode 100644 index 00000000..25a76385 --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/Resources/Base.lproj/Main.storyboard @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Stripe/StripeiOSTestHostApp/ViewController.swift b/Stripe/StripeiOSTestHostApp/ViewController.swift new file mode 100644 index 00000000..a8daad08 --- /dev/null +++ b/Stripe/StripeiOSTestHostApp/ViewController.swift @@ -0,0 +1,18 @@ +// +// ViewController.swift +// StripeiOSTestHostApp +// +// Created by Cameron Sabol on 11/4/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import UIKit + +class ViewController: UIViewController { + + override func viewDidLoad() { + super.viewDidLoad() + // Do any additional setup after loading the view. + } + +} diff --git a/Stripe/StripeiOSTests.xctestplan b/Stripe/StripeiOSTests.xctestplan new file mode 100644 index 00000000..2f5591e1 --- /dev/null +++ b/Stripe/StripeiOSTests.xctestplan @@ -0,0 +1,40 @@ +{ + "configurations" : [ + { + "id" : "E4E61B3B-0FEC-4C2A-BE82-E8558946FFBC", + "name" : "Configuration 1", + "options" : { + + } + } + ], + "defaultOptions" : { + "codeCoverage" : false, + "environmentVariableEntries" : [ + { + "key" : "FB_REFERENCE_IMAGE_DIR", + "value" : "$(SOURCE_ROOT)\/..\/Tests\/ReferenceImages" + } + ], + "targetForVariableExpansion" : { + "containerPath" : "container:Stripe.xcodeproj", + "identifier" : "ADF894AA8F6022D9BED17346", + "name" : "StripeiOS" + } + }, + "testTargets" : [ + { + "skippedTests" : [ + "PaymentMethodMessagingViewSnapshotTests", + "STPCardBINMetadataTests", + "TextFieldElementCardTest\/testBINRangeThatRequiresNetworkCallToValidate()" + ], + "target" : { + "containerPath" : "container:Stripe.xcodeproj", + "identifier" : "8BE23AD5D9A3D939AF46F31E", + "name" : "StripeiOSTests" + } + } + ], + "version" : 1 +} diff --git a/Stripe/StripeiOSTests/APIRequestTest.swift b/Stripe/StripeiOSTests/APIRequestTest.swift new file mode 100644 index 00000000..9a141484 --- /dev/null +++ b/Stripe/StripeiOSTests/APIRequestTest.swift @@ -0,0 +1,287 @@ +// +// APIRequestTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 9/23/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class AnyAPIResponse: NSObject, STPAPIResponseDecodable { + override required init() { + super.init() + } + + static func decodedObject(fromAPIResponse response: [AnyHashable: Any]?) -> Self? { + guard let response = response else { + return nil + } + let a = Self() + a.allResponseFields = response + return a + } + + var allResponseFields: [AnyHashable: Any] = [:] + +} + +class APIRequestTest: XCTestCase { + let apiClient = STPAPIClient() + override func setUp() { + apiClient.apiURL = URL(string: "https://httpbin.org/") + } + + func testPublishableKeyAuthorization() { + let e = expectation(description: "Request completed") + apiClient.publishableKey = "pk_foo" + APIRequest.getWith( + apiClient, + endpoint: "bearer", + parameters: ["foo": "bar"] + ) { + (obj, _, error) in + guard let obj = obj, + let token = obj.allResponseFields["token"] as? String + else { + XCTFail() + XCTAssertNil(error) + return + } + XCTAssertEqual(token, self.apiClient.publishableKey) + e.fulfill() + } + waitForExpectations(timeout: 2) + } + + func testStripeAccountAuthorization() { + + } + + func testGet() { + let e = expectation(description: "Request completed") + APIRequest.getWith(apiClient, endpoint: "get", parameters: [:]) { + (obj, response, error) in + XCTAssertNil(error) + XCTAssertEqual(response?.statusCode, 200) + XCTAssertNotNil(obj) + e.fulfill() + } + waitForExpectations(timeout: 2) + } + + func testPost() { + let parameters = ["foo": "bar"] + let e = expectation(description: "Request completed") + APIRequest.post( + with: apiClient, + endpoint: "post", + parameters: ["foo": "bar"] + ) { + (obj, response, error) in + XCTAssertNil(error) + XCTAssertEqual(response?.statusCode, 200) + guard let obj = obj, + let form = obj.allResponseFields["form"] as? [String: String], + let headers = obj.allResponseFields["headers"] as? [String: String] + else { + XCTFail() + return + } + XCTAssertNotNil(headers["X-Stripe-User-Agent"]) + XCTAssertEqual(headers["Stripe-Version"], STPAPIClient.apiVersion) + XCTAssertEqual(form, parameters) + e.fulfill() + } + waitForExpectations(timeout: 2) + } + + func testDelete() { + let e = expectation(description: "Request completed") + APIRequest.delete(with: apiClient, endpoint: "delete", parameters: [:]) { + (obj, response, error) in + XCTAssertNil(error) + XCTAssertEqual(response?.statusCode, 200) + guard let obj = obj, + let headers = obj.allResponseFields["headers"] as? [String: String] + else { + XCTFail() + return + } + XCTAssertNotNil(headers["X-Stripe-User-Agent"]) + XCTAssertEqual(headers["Stripe-Version"], STPAPIClient.apiVersion) + + e.fulfill() + } + waitForExpectations(timeout: 2) + } + + func testParseResponseWithConnectionError() { + let expectation = self.expectation(description: "parseResponse") + + let httpURLResponse = HTTPURLResponse() + let json: [AnyHashable: Any] = [:] + let body = try? JSONSerialization.data(withJSONObject: json, options: []) + let errorParameter = NSError( + domain: NSURLErrorDomain, + code: NSURLErrorNotConnectedToInternet, + userInfo: nil + ) + + APIRequest.parseResponse( + httpURLResponse, + body: body, + error: errorParameter + ) { (object: STPCard?, response, error) in + guard let error = error else { + XCTFail() + return + } + XCTAssertNil(object) + XCTAssertEqual(response, httpURLResponse) + XCTAssertEqual(error as NSError, errorParameter) + expectation.fulfill() + } + + waitForExpectations(timeout: 2.0, handler: nil) + } + + func testParseResponseWithReturnedError() { + let expectation = self.expectation(description: "parseResponse") + + let httpURLResponse = HTTPURLResponse() + let json = [ + "error": [ + "type": "invalid_request_error", + "message": "Your card number is incorrect.", + "code": "incorrect_number", + ], + ] + let body = try? JSONSerialization.data(withJSONObject: json, options: []) + let errorParameter: NSError? = nil + let expectedError = NSError.stp_error(fromStripeResponse: json) + + APIRequest.parseResponse( + httpURLResponse, + body: body, + error: errorParameter + ) { (object: STPCard?, response, error) in + guard let error = error, let expectedError = expectedError else { + XCTFail() + return + } + XCTAssertNil(object) + XCTAssertEqual(response, httpURLResponse) + XCTAssertEqual(error as NSError, expectedError as NSError) + expectation.fulfill() + } + + waitForExpectations(timeout: 2.0, handler: nil) + } + + func testParseResponseWithMissingError() { + let expectation = self.expectation(description: "parseResponse") + + let httpURLResponse = HTTPURLResponse() + let json: [AnyHashable: Any] = [:] + let body = try? JSONSerialization.data(withJSONObject: json, options: []) + let errorParameter: NSError? = nil + let expectedError = NSError.stp_genericFailedToParseResponseError() + + APIRequest.parseResponse( + httpURLResponse, + body: body, + error: errorParameter + ) { (object: STPCard?, response, error) in + guard let error = error else { + XCTFail() + return + } + XCTAssertNil(object) + XCTAssertEqual(response, httpURLResponse) + XCTAssertEqual(error as NSError, expectedError as NSError) + expectation.fulfill() + } + + waitForExpectations(timeout: 2.0, handler: nil) + } + + func testParseResponseWithResponseObjectAndReturnedError() { + let expectation = self.expectation(description: "parseResponse") + + let httpURLResponse = HTTPURLResponse() + let json: [AnyHashable: Any] = STPTestUtils.jsonNamed("CardSource")! + let body = try? JSONSerialization.data(withJSONObject: json, options: []) + let errorParameter: NSError? = NSError( + domain: NSURLErrorDomain, + code: NSURLErrorUnknown, + userInfo: nil + ) + + APIRequest.parseResponse( + httpURLResponse, + body: body, + error: errorParameter + ) { (object: STPCard?, response, error) in + guard let error = error else { + XCTFail() + return + } + XCTAssertNil(object) + XCTAssertEqual(response, httpURLResponse) + XCTAssertEqual(error as NSError, errorParameter) + expectation.fulfill() + } + + waitForExpectations(timeout: 2.0, handler: nil) + } + + func test429Backoff() { + var inProgress = true + + let e = expectation(description: "Request completed") + // We expect this request to retry a few times with exponential backoff before calling the completion handler. + APIRequest.getWith(apiClient, endpoint: "status/429", parameters: [:]) { + (_, response, _) in + XCTAssertEqual(response?.statusCode, 429) + inProgress = false + e.fulfill() + } + + let checkedStillInProgress = expectation( + description: "Checked that we're still in progress after 2s" + ) + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 2.0) { + // Make sure we're still in progress after 2 seconds + // This shows that we're retrying the request a few times + // while applying an appropriate amount of backoff. + XCTAssertEqual(inProgress, true) + checkedStillInProgress.fulfill() + } + + wait(for: [e, checkedStillInProgress], timeout: 30) + } + + func test429NoBackoff() { + let oldMaxRetries = StripeAPI.maxRetries + StripeAPI.maxRetries = 0 + + let e = expectation(description: "Request completed") + APIRequest.getWith(apiClient, endpoint: "status/429", parameters: [:]) { + (_, response, _) in + XCTAssertEqual(response?.statusCode, 429) + e.fulfill() + } + + // We expect this request to return ~immediately, so we set a timeout lower than the highest + // amount of backoff. + wait(for: [e], timeout: 5.0) + StripeAPI.maxRetries = oldMaxRetries + } +} diff --git a/Stripe/StripeiOSTests/AddPaymentMethodViewControllerSnapshotTests.swift b/Stripe/StripeiOSTests/AddPaymentMethodViewControllerSnapshotTests.swift new file mode 100644 index 00000000..81d6fdee --- /dev/null +++ b/Stripe/StripeiOSTests/AddPaymentMethodViewControllerSnapshotTests.swift @@ -0,0 +1,44 @@ +// +// AddPaymentMethodViewControllerSnapshotTests.swift +// StripeiOSTests +// +// Created by Yuki Tokuhiro on 3/22/23. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +@_spi(STP) @testable import StripePaymentSheet +@_spi(STP) @testable import StripeUICore +import XCTest + +final class AddPaymentMethodViewControllerSnapshotTests: FBSnapshotTestCase { + override func setUp() { + super.setUp() + let expectation = expectation(description: "Load specs") + AddressSpecProvider.shared.loadAddressSpecs { + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + } + waitForExpectations(timeout: 1) +// recordMode = true + } + + func test_with_previous_customer_card_details_and_checkbox() { + // Given the customer previously entered card details... + let previousCustomerInput = IntentConfirmParams.init( + params: .paramsWith(card: STPFixtures.paymentMethodCardParams(), billingDetails: STPFixtures.paymentMethodBillingDetails(), metadata: nil), + type: .card + ) + previousCustomerInput.saveForFutureUseCheckboxState = .selected + // ...and the card doesn't show up *first* in the list (so we can exercise the code that switches to the previously entered pm form)... + let intent = Intent.paymentIntent(STPFixtures.paymentIntent(paymentMethodTypes: ["paypal", "card", "cashApp"])) + var config = PaymentSheet.Configuration._testValue_MostPermissive() + // ...and a "Save this card" checkbox... + config.customer = .init(id: "id", ephemeralKeySecret: "ek") + // ...the AddPMVC should show the card type selected with the form pre-filled with the previous input + let sut = AddPaymentMethodViewController(intent: intent, configuration: config, previousCustomerInput: previousCustomerInput) + sut.view.autosizeHeight(width: 375) + STPSnapshotVerifyView(sut.view) + } +} diff --git a/Stripe/StripeiOSTests/AddressViewControllerSnapshotTests.swift b/Stripe/StripeiOSTests/AddressViewControllerSnapshotTests.swift new file mode 100644 index 00000000..0f3ee472 --- /dev/null +++ b/Stripe/StripeiOSTests/AddressViewControllerSnapshotTests.swift @@ -0,0 +1,138 @@ +// +// AddressViewControllerSnapshotTests.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/15/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +@_spi(STP)@testable import Stripe +@_spi(STP)@testable import StripeCore +@_spi(STP)@testable import StripePaymentSheet +@_spi(STP)@testable import StripeUICore + +class AddressViewControllerSnapshotTests: FBSnapshotTestCase { + private let addressSpecProvider: AddressSpecProvider = { + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + return specProvider + }() + var configuration: AddressViewController.Configuration { + var config = AddressViewController.Configuration() + config.apiClient = .init(publishableKey: "pk_test_1234") + return config + } + + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func testShippingAddressViewController() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + let vc = AddressViewController( + addressSpecProvider: addressSpecProvider, + configuration: configuration, + delegate: self + ) + let navVC = UINavigationController(rootViewController: vc) + testWindow.rootViewController = navVC + verify(navVC.view) + } + + @available(iOS 13.0, *) + func testShippingAddressViewController_darkMode() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + testWindow.overrideUserInterfaceStyle = .dark + let vc = AddressViewController( + addressSpecProvider: addressSpecProvider, + configuration: configuration, + delegate: self + ) + let navVC = UINavigationController(rootViewController: vc) + testWindow.rootViewController = navVC + verify(navVC.view) + } + + func testShippingAddressViewController_appearance() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + var configuration = configuration + configuration.appearance = PaymentSheetTestUtils.snapshotTestTheme + let vc = AddressViewController( + addressSpecProvider: addressSpecProvider, + configuration: configuration, + delegate: self + ) + let navVC = UINavigationController(rootViewController: vc) + testWindow.rootViewController = navVC + verify(navVC.view) + } + + func testShippingAddressViewController_customText() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + var configuration = configuration + configuration.title = "Custom title" + configuration.buttonTitle = "Custom button title" + let vc = AddressViewController( + addressSpecProvider: addressSpecProvider, + configuration: configuration, + delegate: self + ) + let navVC = UINavigationController(rootViewController: vc) + testWindow.rootViewController = navVC + verify(navVC.view) + } + + func testShippingAddressViewController_checkbox() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + var configuration = configuration + configuration.additionalFields.checkboxLabel = "Test checkbox text" + configuration.defaultValues = .init( + address: .init(), + name: nil, + phone: nil, + isCheckboxSelected: true + ) + let vc = AddressViewController( + addressSpecProvider: addressSpecProvider, + configuration: configuration, + delegate: self + ) + let navVC = UINavigationController(rootViewController: vc) + testWindow.rootViewController = navVC + verify(navVC.view) + } + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } +} + +extension AddressViewControllerSnapshotTests: AddressViewControllerDelegate { + func addressViewControllerDidFinish( + _ addressViewController: AddressViewController, + with address: AddressViewController.AddressDetails? + ) { + // no-op + } +} diff --git a/Stripe/StripeiOSTests/AfterpayPriceBreakdownViewSnapshotTests.swift b/Stripe/StripeiOSTests/AfterpayPriceBreakdownViewSnapshotTests.swift new file mode 100644 index 00000000..1a21e82a --- /dev/null +++ b/Stripe/StripeiOSTests/AfterpayPriceBreakdownViewSnapshotTests.swift @@ -0,0 +1,56 @@ +// +// AfterpayPriceBreakdownViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Jaime Park on 6/15/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class AfterpayPriceBreakdownViewSnapshotTests: FBSnapshotTestCase { + override func setUp() { + super.setUp() +// recordMode = true + } + + func embedInRenderableView( + _ priceBreakdownView: AfterpayPriceBreakdownView, + width: Int, + height: Int + ) -> UIView { + let containingView = UIView(frame: CGRect(x: 0, y: 0, width: width, height: height)) + containingView.addSubview(priceBreakdownView) + priceBreakdownView.translatesAutoresizingMaskIntoConstraints = false + NSLayoutConstraint.activate([ + priceBreakdownView.leadingAnchor.constraint(equalTo: containingView.leadingAnchor), + containingView.trailingAnchor.constraint(equalTo: priceBreakdownView.trailingAnchor), + priceBreakdownView.topAnchor.constraint(equalTo: containingView.topAnchor), + containingView.bottomAnchor.constraint(equalTo: priceBreakdownView.bottomAnchor), + ]) + + return containingView + } + + func testClearpayInMultiRow() { + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_GB") as Locale) { [self] in + let priceBreakdownView = AfterpayPriceBreakdownView(amount: 1000, currency: "GBP") + let containingView = embedInRenderableView(priceBreakdownView, width: 320, height: 50) + + STPSnapshotVerifyView(containingView) + } + } + + func testAfterpayInSingleRow() { + let priceBreakdownView = AfterpayPriceBreakdownView(amount: 1000, currency: "USD") + let containingView = embedInRenderableView(priceBreakdownView, width: 500, height: 30) + + STPSnapshotVerifyView(containingView) + } +} diff --git a/Stripe/StripeiOSTests/AnalyticsHelperTests.swift b/Stripe/StripeiOSTests/AnalyticsHelperTests.swift new file mode 100644 index 00000000..2c011fcf --- /dev/null +++ b/Stripe/StripeiOSTests/AnalyticsHelperTests.swift @@ -0,0 +1,60 @@ +// +// AnalyticsHelperTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 2/22/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class AnalyticsHelperTests: XCTestCase { + + func test_getDuration() { + let (sut, timeReference) = makeSUT() + + sut.startTimeMeasurement(.checkout) + + // Advance the clock by 10 seconds. + timeReference.advanceBy(10) + XCTAssertEqual(sut.getDuration(for: .checkout), 10) + + // Advance the clock by 5 seconds. + timeReference.advanceBy(5) + XCTAssertEqual(sut.getDuration(for: .checkout), 15) + } + + func test_getDuration_returnsNilWhenNotStarted() { + let (sut, _) = makeSUT() + XCTAssertNil(sut.getDuration(for: .checkout)) + } + +} + +extension AnalyticsHelperTests { + + class MockTimeReference { + var date = Date() + + func advanceBy(_ timeInterval: TimeInterval) { + date = date.addingTimeInterval(timeInterval) + } + + func now() -> Date { + return date + } + } + + func makeSUT() -> (AnalyticsHelper, MockTimeReference) { + let timeReference = MockTimeReference() + let helper = AnalyticsHelper(timeProvider: timeReference.now) + return (helper, timeReference) + } + +} diff --git a/Stripe/StripeiOSTests/AutoCompleteViewControllerSnapshotTests.swift b/Stripe/StripeiOSTests/AutoCompleteViewControllerSnapshotTests.swift new file mode 100644 index 00000000..9998bbac --- /dev/null +++ b/Stripe/StripeiOSTests/AutoCompleteViewControllerSnapshotTests.swift @@ -0,0 +1,150 @@ +// +// AutoCompleteViewControllerSnapshotTests.swift +// StripeiOS Tests +// +// Created by Nick Porter on 6/7/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class AutoCompleteViewControllerSnapshotTests: FBSnapshotTestCase { + + private var configuration: AddressViewController.Configuration { + return AddressViewController.Configuration() + } + + private let addressSpecProvider: AddressSpecProvider = { + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "ACSZP", + require: "AZ", + cityNameType: .post_town, + stateNameType: .state, + zip: "", + zipNameType: .pin + ), + ] + return specProvider + }() + + private let mockSearchResults: [AddressSearchResult] = [ + MockAddressSearchResult( + title: "199 Water Street", + subtitle: "New York, NY 10038 United States", + titleHighlightRanges: [NSValue(range: NSRange(location: 0, length: 6))], + subtitleHighlightRanges: [NSValue(range: NSRange(location: 2, length: 4))] + ), + MockAddressSearchResult( + title: "354 Oyster Point Blvd", + subtitle: "San Francisco, CA 94080 United States", + titleHighlightRanges: [NSValue(range: NSRange(location: 2, length: 4))], + subtitleHighlightRanges: [NSValue(range: NSRange(location: 4, length: 2))] + ), + MockAddressSearchResult( + title: "10 Boulevard", + subtitle: "Haussmann Paris 75009 France", + titleHighlightRanges: [NSValue(range: NSRange(location: 4, length: 2))], + subtitleHighlightRanges: [NSValue(range: NSRange(location: 0, length: 4))] + ), + ] + + override func setUp() { + super.setUp() + + // self.recordMode = true + } + + func testAutoCompleteViewController() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + let vc = AutoCompleteViewController( + configuration: configuration, + initialLine1Text: nil, + addressSpecProvider: addressSpecProvider + ) + vc.results = mockSearchResults + testWindow.rootViewController = vc + + verify(vc.view) + } + + @available(iOS 13.0, *) + func testAutoCompleteViewController_darkMode() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + testWindow.overrideUserInterfaceStyle = .dark + let vc = AutoCompleteViewController( + configuration: configuration, + initialLine1Text: nil, + addressSpecProvider: addressSpecProvider + ) + + vc.results = mockSearchResults + testWindow.rootViewController = vc + + verify(vc.view) + } + + func testAutoCompleteViewController_appearance() { + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 428, height: 500)) + testWindow.isHidden = false + + var config = configuration + config.appearance.colors.background = .blue + config.appearance.colors.text = .yellow + config.appearance.colors.textSecondary = .red + config.appearance.colors.componentPlaceholderText = .cyan + config.appearance.colors.componentBackground = .red + config.appearance.colors.componentDivider = .green + config.appearance.cornerRadius = 0.0 + config.appearance.borderWidth = 2.0 + config.appearance.font.base = UIFont(name: "AmericanTypeWriter", size: 12)! + config.appearance.colors.primary = .red + + let vc = AutoCompleteViewController( + configuration: config, + initialLine1Text: nil, + addressSpecProvider: addressSpecProvider + ) + vc.results = mockSearchResults + testWindow.rootViewController = vc + + verify(vc.view) + } + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + STPSnapshotVerifyView( + view, + identifier: identifier, + suffixes: FBSnapshotTestCaseDefaultSuffixes(), + file: file, + line: line + ) + } +} + +private struct MockAddressSearchResult: AddressSearchResult { + let title: String + let subtitle: String + let titleHighlightRanges: [NSValue] + let subtitleHighlightRanges: [NSValue] + + func asAddress(completion: @escaping (PaymentSheet.Address?) -> Void) { + completion(nil) + } +} diff --git a/Stripe/StripeiOSTests/ButtonLinkSnapshotTests.swift b/Stripe/StripeiOSTests/ButtonLinkSnapshotTests.swift new file mode 100644 index 00000000..0b59fe4e --- /dev/null +++ b/Stripe/StripeiOSTests/ButtonLinkSnapshotTests.swift @@ -0,0 +1,79 @@ +// +// ButtonLinkSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 2/14/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripeUICore + +class ButtonLinkSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testPrimary() { + let sut = makeSUT(configuration: .linkPrimary(), title: "Primary Button") + verify(sut) + } + + func testSecondary() { + let sut = makeSUT(configuration: .linkSecondary(), title: "Secondary Button") + verify(sut) + } + + func testBordered() { + let sut = makeSUT(configuration: .linkBordered(), title: "Bordered Button") + verify(sut) + } + + func testPlain() { + let sut = makeSUT(configuration: .linkPlain(), title: "Plain Button") + verify(sut) + } + + func verify( + _ sut: Button, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + let size = sut.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + sut.bounds = CGRect(origin: .zero, size: size) + STPSnapshotVerifyView(sut, file: file, line: line) + + sut.isHighlighted = true + STPSnapshotVerifyView(sut, identifier: "Highlighted", file: file, line: line) + + sut.isHighlighted = false + sut.isEnabled = false + STPSnapshotVerifyView(sut, identifier: "Disabled", file: file, line: line) + + sut.isHighlighted = false + sut.isEnabled = true + sut.isLoading = true + STPSnapshotVerifyView(sut, identifier: "Loading", file: file, line: line) + } + +} + +extension ButtonLinkSnapshotTests { + + func makeSUT( + configuration: Button.Configuration, + title: String + ) -> Button { + return Button(configuration: configuration, title: title) + } + +} diff --git a/Stripe/StripeiOSTests/CardExpiryDateTests.swift b/Stripe/StripeiOSTests/CardExpiryDateTests.swift new file mode 100644 index 00000000..c9fa5255 --- /dev/null +++ b/Stripe/StripeiOSTests/CardExpiryDateTests.swift @@ -0,0 +1,73 @@ +// +// CardExpiryDateTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 4/15/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class CardExpiryDateTests: XCTestCase { + + func test_init() { + let sut = CardExpiryDate(month: 2, year: 2026) + XCTAssertEqual(sut.month, 2) + XCTAssertEqual(sut.year, 2026) + } + + func test_init_shouldNormalizeTheYear() { + let sut = CardExpiryDate(month: 2, year: 50) + XCTAssertEqual(sut.year, 2050) + } + + func test_initFromString() { + let sut = CardExpiryDate("0226") + XCTAssertEqual(sut?.month, 2) + XCTAssertEqual(sut?.year, 2026) + } + + func test_initFromString_withInvalidString() { + XCTAssertNil(CardExpiryDate("")) // empty + XCTAssertNil(CardExpiryDate("0")) // missing 3 digits + XCTAssertNil(CardExpiryDate("023")) // missing a digit + XCTAssertNil(CardExpiryDate("1234567890")) // too many digits + XCTAssertNil(CardExpiryDate("abcd")) // alpha + + // month out of range + XCTAssertNil(CardExpiryDate("1326")) + XCTAssertNil(CardExpiryDate("0026")) + XCTAssertNil(CardExpiryDate("-126")) + + // year out of range + XCTAssertNil(CardExpiryDate("02-1")) + } + + func test_displayString() { + let sut = CardExpiryDate(month: 2, year: 2026) + XCTAssertEqual(sut.displayString, "0226") + } + + func test_expired() throws { + let calendar = Calendar(identifier: .gregorian) + + let sut = CardExpiryDate(month: 2, year: 2026) + + let aDayBefore = try XCTUnwrap(calendar.date(from: .init(year: 2026, month: 2, day: 28))) + let aMonthBefore = try XCTUnwrap(calendar.date(from: .init(year: 2026, month: 1, day: 31))) + let aDayAfter = try XCTUnwrap(calendar.date(from: .init(year: 2026, month: 3, day: 1))) + let aMonthAfter = try XCTUnwrap(calendar.date(from: .init(year: 2026, month: 3, day: 30))) + + XCTAssertFalse(sut.expired(now: aDayBefore)) + XCTAssertFalse(sut.expired(now: aMonthBefore)) + XCTAssertTrue(sut.expired(now: aDayAfter)) + XCTAssertTrue(sut.expired(now: aMonthAfter)) + } + +} diff --git a/Stripe/StripeiOSTests/CircularButtonSnapshotTests.swift b/Stripe/StripeiOSTests/CircularButtonSnapshotTests.swift new file mode 100644 index 00000000..7c336ccb --- /dev/null +++ b/Stripe/StripeiOSTests/CircularButtonSnapshotTests.swift @@ -0,0 +1,66 @@ +// +// CircularButtonSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 1/7/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +@_spi(STP) import StripeUICore + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class CircularButtonSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testNormal() { + let button = CircularButton(style: .close) + verify(button) + } + + func testDisabled() { + let button = CircularButton(style: .close) + button.isEnabled = false + verify(button) + } + + func verify( + _ button: CircularButton, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + // Ensures that the button shadow gets captured + let wrapper = UIView() + wrapper.addAndPinSubview( + button, + insets: .insets(top: 10, leading: 10, bottom: 10, trailing: 10) + ) + + // Adding the view to a window updates the traits + let window = UIWindow() + window.addSubview(wrapper) + + let size = wrapper.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + wrapper.bounds = CGRect(origin: .zero, size: size) + + // Test light mode + wrapper.overrideUserInterfaceStyle = .light + STPSnapshotVerifyView(wrapper, identifier: identifier, file: file, line: line) + + // Test dark mode + wrapper.overrideUserInterfaceStyle = .dark + let updatedIdentifier = (identifier ?? "").appending("darkMode") + STPSnapshotVerifyView(wrapper, identifier: updatedIdentifier, file: file, line: line) + } + +} diff --git a/Stripe/StripeiOSTests/ConfirmButtonSnapshotTests.swift b/Stripe/StripeiOSTests/ConfirmButtonSnapshotTests.swift new file mode 100644 index 00000000..ab87d3a0 --- /dev/null +++ b/Stripe/StripeiOSTests/ConfirmButtonSnapshotTests.swift @@ -0,0 +1,86 @@ +// +// ConfirmButtonSnapshotTests.swift +// StripeiOS Tests +// +// Created by Nick Porter on 3/11/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import iOSSnapshotTestCase +import StripeCoreTestUtils +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet + +class ConfirmButtonSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func testConfirmButton() { + let confirmButton = ConfirmButton(style: .stripe, callToAction: .setup, didTap: {}) + + verify(confirmButton) + } + + // Tests that `primaryButton` appearance is used over standard variables + func testConfirmButtonBackgroundColor() { + var appearance = PaymentSheet.Appearance.default + var button = PaymentSheet.Appearance.PrimaryButton() + button.backgroundColor = .red + appearance.primaryButton = button + + let confirmButton = ConfirmButton( + style: .stripe, + callToAction: .setup, + appearance: appearance, + didTap: {} + ) + + verify(confirmButton) + } + + func testConfirmButtonCustomFont() throws { + var appearance = PaymentSheet.Appearance.default + appearance.font.base = try XCTUnwrap(UIFont(name: "AmericanTypewriter", size: 12.0)) + + let confirmButton = ConfirmButton( + style: .stripe, + callToAction: .custom(title: "Custom Title"), + appearance: appearance, + didTap: {} + ) + + verify(confirmButton) + } + + func testConfirmButtonCustomFontScales() throws { + var appearance = PaymentSheet.Appearance.default + appearance.font.base = try XCTUnwrap(UIFont(name: "AmericanTypewriter", size: 12.0)) + appearance.font.sizeScaleFactor = 0.85 + + let confirmButton = ConfirmButton( + style: .stripe, + callToAction: .custom(title: "Custom Title"), + appearance: appearance, + didTap: {} + ) + + verify(confirmButton) + } + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 300) + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } +} diff --git a/Stripe/StripeiOSTests/ConfirmButtonTests.swift b/Stripe/StripeiOSTests/ConfirmButtonTests.swift new file mode 100644 index 00000000..57b559a9 --- /dev/null +++ b/Stripe/StripeiOSTests/ConfirmButtonTests.swift @@ -0,0 +1,91 @@ +// +// ConfirmButtonTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 10/6/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class ConfirmButtonTests: XCTestCase { + + func testBuyButtonShouldAutomaticallyAdjustItsForegroundColor() { + let testCases: [(background: UIColor, foreground: UIColor)] = [ + // Dark backgrounds + (background: .systemBlue, foreground: .white), + (background: .black, foreground: .white), + // Light backgrounds + ( + background: UIColor(red: 1.0, green: 0.87, blue: 0.98, alpha: 1.0), + foreground: .black + ), + ( + background: UIColor(red: 1.0, green: 0.89, blue: 0.35, alpha: 1.0), + foreground: .black + ), + ] + + for (backgroundColor, expectedForeground) in testCases { + let button = ConfirmButton.BuyButton() + button.tintColor = backgroundColor + button.update( + status: .enabled, + callToAction: .pay(amount: 900, currency: "usd"), + animated: false + ) + + XCTAssertEqual( + // Test against `.cgColor` because any color set as `.backgroundColor` + // will be automatically wrapped in `UIDynamicModifiedColor` (private subclass) by iOS. + button.backgroundColor?.cgColor, + backgroundColor.cgColor + ) + + XCTAssertEqual( + button.foregroundColor, + expectedForeground, + "The foreground color should contrast with the background color" + ) + } + } + + func testUpdateShouldCallTheCompletionBlock() { + let sut = ConfirmButton( + style: .stripe, + callToAction: .pay(amount: 1000, currency: "usd"), + didTap: {} + ) + + let expectation = XCTestExpectation(description: "Should call the completion block") + + sut.update(state: .disabled, animated: false) { + expectation.fulfill() + } + + wait(for: [expectation], timeout: 1) + } + + func testUpdateShouldCallTheCompletionBlockWhenAnimated() { + let sut = ConfirmButton( + style: .stripe, + callToAction: .pay(amount: 1000, currency: "usd"), + didTap: {} + ) + + let expectation = XCTestExpectation(description: "Should call the completion block") + + sut.update(state: .disabled, animated: true) { + expectation.fulfill() + } + + wait(for: [expectation], timeout: 3) + } + +} diff --git a/Stripe/StripeiOSTests/ConsumerSessionTests.swift b/Stripe/StripeiOSTests/ConsumerSessionTests.swift new file mode 100644 index 00000000..dc8ddf6a --- /dev/null +++ b/Stripe/StripeiOSTests/ConsumerSessionTests.swift @@ -0,0 +1,490 @@ +// +// ConsumerSessionTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 4/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class ConsumerSessionTests: XCTestCase { + +// Disable Consumer Session integration tests +/* + let apiClient: STPAPIClient = { + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + return apiClient + }() + + let cookieStore = LinkInMemoryCookieStore() + + func testLookupSession_noParams() { + let expectation = self.expectation(description: "Lookup ConsumerSession") + + ConsumerSession.lookupSession(for: nil, with: apiClient, cookieStore: cookieStore) { + result in + switch result { + case .success(let lookupResponse): + switch lookupResponse.responseType { + case .found: + XCTFail("Got a response without any params") + + case .notFound(let errorMessage): + XCTFail("Got not found response with \(errorMessage)") + + case .noAvailableLookupParams: + break // Pass + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + expectation.fulfill() + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testLookupSession_shouldDeleteInvalidSessionCookies() { + let expectation = self.expectation(description: "Lookup ConsumerSession") + + cookieStore.write(key: .session, value: "bad_session_cookie", allowSync: false) + + ConsumerSession.lookupSession(for: nil, with: apiClient, cookieStore: cookieStore) { + result in + switch result { + case .success(let lookupResponse): + switch lookupResponse.responseType { + case .notFound: + // Expected response type. + break + + case .noAvailableLookupParams, .found: + XCTFail("Unexpected response type: \(lookupResponse.responseType)") + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + expectation.fulfill() + } + + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + XCTAssertNil(cookieStore.read(key: .session), "Invalid cookie not deleted") + } + + func testLookupSession_cookieOnly() { + _ = createVerifiedConsumerSession() + let expectation = self.expectation(description: "Lookup ConsumerSession") + ConsumerSession.lookupSession(for: nil, with: apiClient, cookieStore: cookieStore) { + result in + switch result { + case .success(let lookupResponse): + switch lookupResponse.responseType { + case .found: + break // Pass + + case .notFound(let errorMessage): + XCTFail("Got not found response with \(errorMessage)") + + case .noAvailableLookupParams: + XCTFail("Got no avilable lookup params") + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + expectation.fulfill() + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testLookupSession_existingConsumer() { + let expectation = self.expectation(description: "Lookup ConsumerSession") + + ConsumerSession.lookupSession( + for: "mobile-payments-sdk-ci+a-consumer@stripe.com", + with: apiClient, + cookieStore: cookieStore + ) { result in + switch result { + case .success(let lookupResponse): + switch lookupResponse.responseType { + case .found: + break // Pass + + case .notFound(let errorMessage): + XCTFail("Got not found response with \(errorMessage)") + + case .noAvailableLookupParams: + XCTFail("Got no available lookup params") + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + expectation.fulfill() + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testLookupSession_newConsumer() { + let expectation = self.expectation(description: "Lookup ConsumerSession") + + ConsumerSession.lookupSession( + for: "mobile-payments-sdk-ci+not-a-consumer@stripe.com", + with: apiClient, + cookieStore: cookieStore + ) { result in + switch result { + case .success(let lookupResponse): + switch lookupResponse.responseType { + case .found(let consumerSession): + XCTFail("Got unexpected found response with \(consumerSession)") + + case .notFound: + break // Pass + + case .noAvailableLookupParams: + XCTFail("Got no available lookup params") + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + expectation.fulfill() + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + // tests signup, createPaymentDetails + func testSignUpAndCreateDetails() { + let expectation = self.expectation(description: "consumer sign up") + let newAccountEmail = "mobile-payments-sdk-ci+\(UUID())@stripe.com" + + var sessionWithKey: ConsumerSession.SessionWithPublishableKey? + + ConsumerSession.signUp( + email: newAccountEmail, + phoneNumber: "+13105551234", + legalName: nil, + countryCode: "US", + consentAction: nil, + with: apiClient, + cookieStore: cookieStore + ) { result in + switch result { + case .success(let signupResponse): + XCTAssertTrue(signupResponse.consumerSession.isVerifiedForSignup) + XCTAssertTrue( + signupResponse.consumerSession.verificationSessions.isVerifiedForSignup + ) + XCTAssertTrue( + signupResponse.consumerSession.verificationSessions.contains(where: { + $0.type == .signup + }) + ) + + sessionWithKey = signupResponse + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + expectation.fulfill() + } + + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + + if let consumerSession = sessionWithKey?.consumerSession { + let cardParams = STPPaymentMethodCardParams() + cardParams.number = "4242424242424242" + cardParams.expMonth = 12 + cardParams.expYear = NSNumber( + value: Calendar.autoupdatingCurrent.component(.year, from: Date()) + 1 + ) + cardParams.cvc = "123" + + let billingParams = STPPaymentMethodBillingDetails() + billingParams.name = "Payments SDK CI" + let address = STPPaymentMethodAddress() + address.postalCode = "55555" + billingParams.address = address + + let paymentMethodParams = STPPaymentMethodParams.paramsWith( + card: cardParams, + billingDetails: billingParams, + metadata: nil + ) + + let createExpectation = self.expectation(description: "create payment details") + consumerSession.createPaymentDetails( + paymentMethodParams: paymentMethodParams, + with: apiClient, + consumerAccountPublishableKey: sessionWithKey?.publishableKey + ) { result in + switch result { + case .success(let createdPaymentDetails): + if case .card(let cardDetails) = createdPaymentDetails.details { + XCTAssertEqual(cardDetails.expiryMonth, cardParams.expMonth?.intValue) + XCTAssertEqual(cardDetails.expiryYear, cardParams.expYear?.intValue) + } else { + XCTAssert(false) + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + createExpectation.fulfill() + } + + wait(for: [createExpectation], timeout: STPTestingNetworkRequestTimeout) + } + } + + func testListPaymentDetails() { + let (consumerSession, publishableKey) = createVerifiedConsumerSession() + + let listExpectation = self.expectation(description: "list payment details") + + consumerSession.listPaymentDetails( + with: apiClient, + consumerAccountPublishableKey: publishableKey + ) { result in + switch result { + case .success(let paymentDetails): + XCTAssertFalse(paymentDetails.isEmpty) + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + listExpectation.fulfill() + } + + wait(for: [listExpectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testCreateLinkAccountSession() { + let createLinkAccountSessionExpectation = self.expectation( + description: "Create LinkAccountSession" + ) + + let (consumerSession, publishableKey) = createVerifiedConsumerSession() + consumerSession.createLinkAccountSession( + with: apiClient, + consumerAccountPublishableKey: publishableKey + ) { result in + switch result { + case .success: + // Pass + break + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + createLinkAccountSessionExpectation.fulfill() + } + + wait(for: [createLinkAccountSessionExpectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testUpdatePaymentDetails() { + let (consumerSession, publishableKey) = createVerifiedConsumerSession() + + let listExpectation = self.expectation(description: "list payment details") + var storedPaymentDetails = [ConsumerPaymentDetails]() + + consumerSession.listPaymentDetails( + with: apiClient, + consumerAccountPublishableKey: publishableKey + ) { result in + switch result { + case .success(let paymentDetails): + storedPaymentDetails = paymentDetails + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + listExpectation.fulfill() + } + + wait(for: [listExpectation], timeout: STPTestingNetworkRequestTimeout) + + let billingParams = STPPaymentMethodBillingDetails() + billingParams.name = "Payments SDK CI" + let address = STPPaymentMethodAddress() + address.postalCode = "55555" + billingParams.address = address + + let updateExpectation = self.expectation(description: "update payment details") + let paymentMethodToUpdate = try! XCTUnwrap(storedPaymentDetails.first) + + guard case .card(let card) = paymentMethodToUpdate.details else { + XCTFail("Payment method must be `card` type") + return + } + + let calendar = Calendar(identifier: .gregorian) + let yearOne = calendar.component(.year, from: Date()) + 1 + let yearTwo = calendar.component(.year, from: Date()) + 2 + + // toggle between expiry years/months + let newExpiryDate = CardExpiryDate( + month: card.expiryDate.month == 1 ? 2 : 1, + year: card.expiryDate.year == yearOne ? yearTwo : yearOne + ) + + let updateParams = UpdatePaymentDetailsParams( + isDefault: !paymentMethodToUpdate.isDefault, + details: .card( + expiryDate: newExpiryDate, + billingDetails: billingParams + ) + ) + + consumerSession.updatePaymentDetails( + with: apiClient, + id: paymentMethodToUpdate.stripeID, + updateParams: updateParams, + consumerAccountPublishableKey: publishableKey + ) { result in + switch result { + case .success(let paymentDetails): + XCTAssertNotEqual(paymentDetails.isDefault, paymentMethodToUpdate.isDefault) + switch paymentDetails.details { + case .card(let card): + XCTAssertEqual(newExpiryDate, card.expiryDate) + case .bankAccount, .unparsable: + XCTFail("Unexpected payment details type") + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + updateExpectation.fulfill() + } + + wait(for: [updateExpectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testLogout() { + let (consumerSession, publishableKey) = createVerifiedConsumerSession() + + XCTAssertNotNil(cookieStore.formattedSessionCookies()) + + let logoutExpectation = self.expectation(description: "Logout") + + consumerSession.logout( + with: apiClient, + cookieStore: cookieStore, + consumerAccountPublishableKey: publishableKey + ) { result in + switch result { + case .success: + XCTAssertNil(self.cookieStore.formattedSessionCookies()) + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + logoutExpectation.fulfill() + } + + wait(for: [logoutExpectation], timeout: STPTestingNetworkRequestTimeout) + } +*/ +} + +extension ConsumerSessionTests { +/* + fileprivate func lookupExistingConsumer() -> ConsumerSession.SessionWithPublishableKey { + var sessionWithKey: ConsumerSession.SessionWithPublishableKey! + + let lookupExpectation = self.expectation(description: "Lookup ConsumerSession") + + let email = "mobile-payments-sdk-ci+a-consumer@stripe.com" + + ConsumerSession.lookupSession( + for: email, + with: apiClient, + cookieStore: cookieStore + ) { result in + switch result { + case .success(let lookupResponse): + switch lookupResponse.responseType { + case .found(let session): + sessionWithKey = session + case .notFound(let errorMessage): + XCTFail("Got not found response with \(errorMessage)") + case .noAvailableLookupParams: + XCTFail("Got no avilable lookup params") + } + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + lookupExpectation.fulfill() + } + + wait(for: [lookupExpectation], timeout: STPTestingNetworkRequestTimeout) + + return sessionWithKey + } + + fileprivate func createVerifiedConsumerSession() -> (ConsumerSession, String) { + let sessionWithKey = lookupExistingConsumer() + var consumerSession = sessionWithKey.consumerSession + let publishableKey = sessionWithKey.publishableKey + + // Start verification + + let startVerificationExpectation = self.expectation(description: "Start verification") + + consumerSession.startVerification( + with: apiClient, + cookieStore: cookieStore, + consumerAccountPublishableKey: publishableKey + ) { result in + switch result { + case .success: + // Pass + break + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + startVerificationExpectation.fulfill() + } + + wait(for: [startVerificationExpectation], timeout: STPTestingNetworkRequestTimeout) + + // Verify via SMS + + let confirmVerificationExpectation = self.expectation(description: "Confirm verification") + + consumerSession.confirmSMSVerification( + with: "000000", + with: apiClient, + cookieStore: cookieStore, + consumerAccountPublishableKey: publishableKey + ) { result in + switch result { + case .success(let verifiedSession): + consumerSession = verifiedSession + case .failure(let error): + XCTFail("Received error: \(error.nonGenericDescription)") + } + + confirmVerificationExpectation.fulfill() + } + + wait(for: [confirmVerificationExpectation], timeout: STPTestingNetworkRequestTimeout) + + return (consumerSession, publishableKey) + } +*/ +} diff --git a/Stripe/StripeiOSTests/Error+PaymentSheetTests.swift b/Stripe/StripeiOSTests/Error+PaymentSheetTests.swift new file mode 100644 index 00000000..ebff9a26 --- /dev/null +++ b/Stripe/StripeiOSTests/Error+PaymentSheetTests.swift @@ -0,0 +1,54 @@ +// +// Error+PaymentSheetTests.swift +// StripeiOSTests +// +// Created by Nick Porter on 3/21/23. +// + +import Foundation +import XCTest + +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet + +class Error_PaymentSheetTests: XCTestCase { + + private enum TestableError: LocalizedError { + case generic + case custom(String) + + var errorDescription: String? { + switch self { + case .generic: + return NSError.stp_unexpectedErrorMessage() + case .custom(let errorMessage): + return errorMessage + } + } + } + + func testPaymentSheetError_UsesDebubDescription() { + let error = PaymentSheetError.unknown(debugDescription: "Test debugDescription") + + XCTAssertEqual("An error occured in PaymentSheet. Test debugDescription", error.nonGenericDescription) + } + + func testError_HasGenericLocalizedDescription_NoSeverError() { + XCTAssertEqual(NSError.stp_unexpectedErrorMessage(), TestableError.generic.nonGenericDescription) + } + + func testError_HasGenericLocalizedDescription_WithServerError() { + let serverErrorMessage = "Test failed server response error messasge" + let info = [NSLocalizedDescriptionKey: NSError.stp_unexpectedErrorMessage(), STPError.errorMessageKey: serverErrorMessage] + let error = NSError(domain: "Test error domain", code: 123, userInfo: info) + + XCTAssertEqual(serverErrorMessage, error.nonGenericDescription) + } + + func testError_HasLocalizedDescription() { + let errorMessage = "Test errorMessage" + let error = TestableError.custom(errorMessage) + + XCTAssertEqual(errorMessage, error.nonGenericDescription) + } +} diff --git a/Stripe/StripeiOSTests/FBSnapshotTestCase+STPViewControllerLoading.swift b/Stripe/StripeiOSTests/FBSnapshotTestCase+STPViewControllerLoading.swift new file mode 100644 index 00000000..3b75e218 --- /dev/null +++ b/Stripe/StripeiOSTests/FBSnapshotTestCase+STPViewControllerLoading.swift @@ -0,0 +1,79 @@ +// +// FBSnapshotTestCase+STPViewControllerLoading.swift +// StripeiOS Tests +// +// Created by Brian Dorfman on 12/11/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +extension FBSnapshotTestCase { + /// Embeds the given controller in a navigation controller, prepares it for + /// snapshot testing and returns the view controller's view. + @objc(stp_preparedAndSizedViewForSnapshotTestFromViewController:) + func stp_preparedAndSizedViewForSnapshotTest(from viewController: UIViewController?) -> UIView? + { + let navController = stp_navigationControllerForSnapshotTest(withRootVC: viewController) + return stp_preparedAndSizedViewForSnapshotTest(from: navController) + } + + /// Returns a navigation controller initialized with the given root view controller + /// and prepares it for snapshot testing (adding it to a UIWindow and loading views) + @objc func stp_navigationControllerForSnapshotTest( + withRootVC viewController: UIViewController? + ) + -> UINavigationController? + { + var navController: UINavigationController? + if let viewController = viewController { + navController = UINavigationController(rootViewController: viewController) + } + let testWindow = UIWindow(frame: CGRect(x: 0, y: 0, width: 320, height: 480)) + testWindow.rootViewController = navController + testWindow.isHidden = false + + // Test that views loaded properly + loads them on first call + XCTAssertNotNil(navController?.view) + XCTAssertNotNil(viewController?.view) + + return navController + } + + /// Returns a view for snapshot testing from the topViewController of the given + /// navigation controller, making necessary layout adjustments for + /// `STPCoreScrollViewController`. + @objc(stp_preparedAndSizedViewForSnapshotTestFromNavigationController:) + func stp_preparedAndSizedViewForSnapshotTest( + from navController: UINavigationController? + ) + -> UIView? + { + let viewController = navController?.topViewController + + // Test that views loaded properly + loads them on first call + XCTAssertNotNil(navController?.view) + XCTAssertNotNil(viewController?.view) + + if viewController is STPCoreScrollViewController { + guard let scrollView = (viewController as? STPCoreScrollViewController)?.scrollView, + let navController = navController + else { + return nil + } + navController.view.layoutIfNeeded() + + let topOffset = scrollView.convert(scrollView.frame.origin, to: navController.view).y + navController.view.frame = CGRect( + x: 0, + y: 0, + width: 320, + height: (topOffset) + (scrollView.contentSize.height) + + (scrollView.contentInset.top) + + (scrollView.contentInset.bottom) + ) + } + + return navController?.view + } +} diff --git a/Stripe/StripeiOSTests/FormSpecProviderTest.swift b/Stripe/StripeiOSTests/FormSpecProviderTest.swift new file mode 100644 index 00000000..e54b67e2 --- /dev/null +++ b/Stripe/StripeiOSTests/FormSpecProviderTest.swift @@ -0,0 +1,577 @@ +// +// FormSpecProviderTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/7/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class FormSpecProviderTest: XCTestCase { + func testLoadsJSON() throws { + let e = expectation(description: "Loads form specs file") + let sut = FormSpecProvider() + sut.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + + guard let eps = sut.formSpec(for: "eps") else { + XCTFail() + return + } + XCTAssertEqual(eps.fields.count, 5) + XCTAssertEqual( + eps.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + + // ...and iDEAL has the correct dropdown spec + guard let ideal = sut.formSpec(for: "ideal"), + case .name = ideal.fields[0], + case .selector(let selector) = ideal.fields[3] + else { + XCTFail() + return + } + XCTAssertEqual(selector.apiPath?["v1"], "ideal[bank]") + XCTAssertEqual(selector.items.count, 11) + } + + func testLoadJsonCanOverwriteLoadedSpecs() throws { + let e = expectation(description: "Loads form specs file") + let sut = FormSpecProvider() + let paymentMethodType = "eps" + sut.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + let eps = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(eps.fields.count, 5) + XCTAssertEqual( + eps.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + let updatedSpecJson = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + { + "type": "name", + "api_path": { + "v1": "billing_details[someOtherValue]" + } + } + ] + }] + """.data(using: .utf8)! + let formSpec = try! JSONSerialization.jsonObject(with: updatedSpecJson) + + let result = sut.loadFrom(formSpec) + XCTAssert(result) + + let epsUpdated = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(epsUpdated.fields.count, 1) + XCTAssertEqual( + epsUpdated.fields.first, + .name( + FormSpec.NameFieldSpec( + apiPath: ["v1": "billing_details[someOtherValue]"], + translationId: nil + ) + ) + ) + } + + func testLoadJsonFailsGracefully() throws { + let e = expectation(description: "Loads form specs file") + let sut = FormSpecProvider() + let paymentMethodType = "eps" + sut.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 20, handler: nil) + let eps = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(eps.fields.count, 5) + XCTAssertEqual( + eps.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + let updatedSpecJson = + """ + [{ + "INVALID_type": "eps", + "async": false, + "fields": [ + { + "type": "name", + "api_path": { + "v1": "billing_details[someOtherValue]" + } + } + ] + }] + """.data(using: .utf8)! + let formSpec = try! JSONSerialization.jsonObject(with: updatedSpecJson) + + let result = sut.loadFrom(formSpec) + XCTAssertFalse(result) + let epsUpdated = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(epsUpdated.fields.count, 5) + XCTAssertEqual( + epsUpdated.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + } + + func testLoadNotValidJsonFailsGracefully() throws { + let e = expectation(description: "Loads form specs file") + let sut = FormSpecProvider() + let paymentMethodType = "eps" + sut.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + + let eps = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(eps.fields.count, 5) + XCTAssertEqual( + eps.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + + let updatedSpecJson = + """ + NOT VALID JSON + """.data(using: .utf8)! + + let result = sut.loadFrom(updatedSpecJson) + XCTAssertFalse(result) + let epsUpdated = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(epsUpdated.fields.count, 5) + XCTAssertEqual( + epsUpdated.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + } + + func testLoadJsonDoesOverwrites() throws { + let e = expectation(description: "Loads form specs file") + let sut = FormSpecProvider() + let paymentMethodType = "eps" + sut.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + let eps = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(eps.fields.count, 5) + XCTAssertEqual( + eps.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + guard + case .redirect_to_url = eps.nextActionSpec?.confirmResponseStatusSpecs[ + "requires_action" + ]?.type, + case .finished = eps.nextActionSpec?.postConfirmHandlingPiStatusSpecs?["succeeded"]? + .type, + case .canceled = eps.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "requires_action" + ]?.type + else { + XCTFail() + return + } + + let updatedSpecJson = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + { + "type": "name", + "api_path": { + "v1": "billing_details[someOtherValue]" + } + } + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished" + }, + "requires_action": { + "type": "finished" + } + } + } + }] + """.data(using: .utf8)! + let formSpec = try! JSONSerialization.jsonObject(with: updatedSpecJson) + + let result = sut.loadFrom(formSpec) + XCTAssert(result) + + // Validate ability to override LPM behavior of next actions + let epsUpdated = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(epsUpdated.fields.count, 1) + XCTAssertEqual( + epsUpdated.fields.first, + .name( + FormSpec.NameFieldSpec( + apiPath: ["v1": "billing_details[someOtherValue]"], + translationId: nil + ) + ) + ) + guard + case .redirect_to_url = epsUpdated.nextActionSpec?.confirmResponseStatusSpecs[ + "requires_action" + ]?.type, + case .finished = epsUpdated.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "succeeded" + ]?.type, + case .finished = epsUpdated.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "requires_action" + ]?.type + else { + XCTFail() + return + } + } + + func testLoadJsonDoesOverwritesWithoutNextActionSpec() throws { + let e = expectation(description: "Loads form specs file") + let sut = FormSpecProvider() + let paymentMethodType = "affirm" + sut.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + let affirm = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(affirm.fields.count, 1) + XCTAssertEqual(affirm.fields.first, .affirm_header) + guard + case .redirect_to_url = affirm.nextActionSpec?.confirmResponseStatusSpecs[ + "requires_action" + ]?.type, + case .finished = affirm.nextActionSpec?.postConfirmHandlingPiStatusSpecs?["succeeded"]? + .type, + case .canceled = affirm.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "requires_action" + ]?.type + else { + XCTFail() + return + } + + let updatedSpecJson = + """ + [{ + "type": "affirm", + "async": false, + "fields": [ + { + "type": "name" + } + ] + }] + """.data(using: .utf8)! + let formSpec = try! JSONSerialization.jsonObject(with: updatedSpecJson) as! [NSDictionary] + + let result = sut.loadFrom(formSpec) + XCTAssert(result) + + guard let affirmUpdated = sut.formSpec(for: paymentMethodType), + affirmUpdated.fields.count == 1, + affirmUpdated.fields.first + == .name(FormSpec.NameFieldSpec(apiPath: nil, translationId: nil)), + affirmUpdated.nextActionSpec == nil + else { + XCTFail() + return + } + } + + func testLoadJsonDoesNotOverwriteWhenWithUnsupportedNextAction() throws { + let e = expectation(description: "Loads form specs file") + let sut = FormSpecProvider() + let paymentMethodType = "eps" + sut.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + let eps = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(eps.fields.count, 5) + XCTAssertEqual( + eps.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + guard + case .redirect_to_url(let redirectToURLDetails) = eps.nextActionSpec?.confirmResponseStatusSpecs[ + "requires_action" + ]?.type, + case .finished = eps.nextActionSpec?.postConfirmHandlingPiStatusSpecs?["succeeded"]? + .type, + case .canceled = eps.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "requires_action" + ]?.type + else { + XCTFail() + return + } + XCTAssertEqual(redirectToURLDetails.redirectStrategy, .none) + XCTAssertEqual(redirectToURLDetails.urlPath, "next_action[redirect_to_url][url]") + XCTAssertEqual(redirectToURLDetails.returnUrlPath, "next_action[redirect_to_url][return_url]") + + let updatedSpecJsonWithUnsupportedNextAction = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + { + "type": "name", + "api_path": { + "v1": "billing_details[someOtherValue]" + } + } + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url_v2_NotSupported" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished" + }, + "requires_action": { + "type": "canceled" + } + } + } + }] + """.data(using: .utf8)! + let formSpec = + try! JSONSerialization.jsonObject(with: updatedSpecJsonWithUnsupportedNextAction) + as! [NSDictionary] + let result = sut.loadFrom(formSpec) + XCTAssertFalse(result) + + // Validate that we were not able to override the spec read in from disk + let epsUpdated = try XCTUnwrap(sut.formSpec(for: paymentMethodType)) + XCTAssertEqual(epsUpdated.fields.count, 5) + XCTAssertEqual( + epsUpdated.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + guard + case .redirect_to_url = epsUpdated.nextActionSpec?.confirmResponseStatusSpecs[ + "requires_action" + ]?.type, + case .finished = epsUpdated.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "succeeded" + ]?.type, + case .canceled = epsUpdated.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "requires_action" + ]?.type + else { + XCTFail() + return + } + } + func testContainsKnownNextAction() throws { + let formSpec = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished" + }, + "requires_action": { + "type": "canceled" + } + } + } + }] + """.data(using: .utf8)! + let decoder = JSONDecoder() + decoder.keyDecodingStrategy = .convertFromSnakeCase + let decodedFormSpecs = try decoder.decode([FormSpec].self, from: formSpec) + + let sut = FormSpecProvider() + XCTAssertFalse(sut.containsUnknownNextActions(formSpecs: decodedFormSpecs)) + } + + func testContainsUnknownNextAction_confirm() throws { + let formSpec = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url_v2_NotSupported" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished" + }, + "requires_action": { + "type": "canceled" + } + } + } + }] + """.data(using: .utf8)! + let decoder = JSONDecoder() + decoder.keyDecodingStrategy = .convertFromSnakeCase + let decodedFormSpecs = try decoder.decode([FormSpec].self, from: formSpec) + + let sut = FormSpecProvider() + XCTAssert(sut.containsUnknownNextActions(formSpecs: decodedFormSpecs)) + } + + func testContainsUnknownNextAction_PostConfirm() throws { + let formSpec = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished_NotSupportedType" + }, + "requires_action": { + "type": "canceled" + } + } + } + }] + """.data(using: .utf8)! + let decoder = JSONDecoder() + decoder.keyDecodingStrategy = .convertFromSnakeCase + let decodedFormSpecs = try decoder.decode([FormSpec].self, from: formSpec) + + let sut = FormSpecProvider() + XCTAssert(sut.containsUnknownNextActions(formSpecs: decodedFormSpecs)) + } + func testRedirectToURLWithExternalBrowserStrategy() throws { + let formSpec = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url", + "native_mobile_redirect_strategy": "external_browser" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished" + } + } + } + }] + """.data(using: .utf8)! + + let decoder = JSONDecoder() + decoder.keyDecodingStrategy = .convertFromSnakeCase + let eps = try decoder.decode([FormSpec].self, from: formSpec).first + + guard case let .redirect_to_url(redirectToURL) = eps?.nextActionSpec?.confirmResponseStatusSpecs["requires_action"]?.type else { + XCTFail("Unable to parse requires_action") + return + } + XCTAssertEqual(redirectToURL.redirectStrategy, .external_browser) + XCTAssertEqual(redirectToURL.urlPath, "next_action[redirect_to_url][url]") + XCTAssertEqual(redirectToURL.returnUrlPath, "next_action[redirect_to_url][return_url]") + } + func testRedirectToURLWithFollowRedirectsStrategy() throws { + let formSpec = + """ + [{ + "type": "eps", + "async": false, + "fields": [ + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url", + "native_mobile_redirect_strategy": "follow_redirects" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished" + } + } + } + }] + """.data(using: .utf8)! + + let decoder = JSONDecoder() + decoder.keyDecodingStrategy = .convertFromSnakeCase + let eps = try decoder.decode([FormSpec].self, from: formSpec).first + + guard case let .redirect_to_url(redirectToURL) = eps?.nextActionSpec?.confirmResponseStatusSpecs["requires_action"]?.type else { + XCTFail("Unable to parse requires_action") + return + } + XCTAssertEqual(redirectToURL.redirectStrategy, .follow_redirects) + XCTAssertEqual(redirectToURL.urlPath, "next_action[redirect_to_url][url]") + XCTAssertEqual(redirectToURL.returnUrlPath, "next_action[redirect_to_url][return_url]") + } +} diff --git a/Stripe/StripeiOSTests/FraudDetectionDataTest.swift b/Stripe/StripeiOSTests/FraudDetectionDataTest.swift new file mode 100644 index 00000000..3b4ba93d --- /dev/null +++ b/Stripe/StripeiOSTests/FraudDetectionDataTest.swift @@ -0,0 +1,31 @@ +// +// FraudDetectionDataTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 5/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class FraudDetectionDataTest: XCTestCase { + func testResetsSIDIfExpired() { + FraudDetectionData.shared.sidCreationDate = Date(timeInterval: -30 * 60 - 1, since: Date()) + FraudDetectionData.shared.resetSIDIfExpired() + XCTAssertNil(FraudDetectionData.shared.sid) + } + + func testSIDNotExpired() { + // Test resets sid if expired + FraudDetectionData.shared.sid = "123" + FraudDetectionData.shared.sidCreationDate = Date() + FraudDetectionData.shared.resetSIDIfExpired() + XCTAssertNotNil(FraudDetectionData.shared.sid) + } +} diff --git a/Stripe/StripeiOSTests/ImageTest.swift b/Stripe/StripeiOSTests/ImageTest.swift new file mode 100644 index 00000000..3c31b62e --- /dev/null +++ b/Stripe/StripeiOSTests/ImageTest.swift @@ -0,0 +1,27 @@ +// +// ImageTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 5/19/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class ImageTest: XCTestCase { + func testAllImagesExist() throws { + for image in Image.allCases { + let image = UIImage( + named: image.rawValue, + in: StripePaymentSheetBundleLocator.resourcesBundle, + compatibleWith: nil + ) + XCTAssertNotNil(image) + } + } +} diff --git a/Stripe/StripeiOSTests/Info.plist b/Stripe/StripeiOSTests/Info.plist new file mode 100644 index 00000000..ba72822e --- /dev/null +++ b/Stripe/StripeiOSTests/Info.plist @@ -0,0 +1,24 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + BNDL + CFBundleShortVersionString + 1.0 + CFBundleSignature + ???? + CFBundleVersion + 1 + + diff --git a/Stripe/StripeiOSTests/KlarnaHelperTest.swift b/Stripe/StripeiOSTests/KlarnaHelperTest.swift new file mode 100644 index 00000000..052399f7 --- /dev/null +++ b/Stripe/StripeiOSTests/KlarnaHelperTest.swift @@ -0,0 +1,97 @@ +// +// KlarnaHelperTest.swift +// StripeiOS Tests +// +// Created by Nick Porter on 11/1/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class KlarnaHelperTest: XCTestCase { + + func testAvailableCountries_eur() { + let expected = ["AT", "FI", "DE", "NL", "BE", "ES", "IT", "FR", "GR", "IE", "PT"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "eur")) + } + + func testAvailableCountries_dkk() { + let expected = ["DK"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "dkk")) + } + + func testAvailableCountries_nok() { + let expected = ["NO"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "nok")) + } + + func testAvailableCountries_sek() { + let expected = ["SE"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "sek")) + } + + func testAvailableCountries_gbp() { + let expected = ["GB"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "gbp")) + } + + func testAvailableCountries_usd() { + let expected = ["US"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "usd")) + } + + func testAvailableCountries_aud() { + let expected = ["AU"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "aud")) + } + + func testAvailableCountries_cad() { + let expected = ["CA"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "cad")) + } + + func testAvailableCountries_czk() { + let expected = ["CZ"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "czk")) + } + + func testAvailableCountries_nzd() { + let expected = ["NZ"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "nzd")) + } + + func testAvailableCountries_pln() { + let expected = ["PL"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "pln")) + } + + func testAvailableCountries_chf() { + let expected = ["CH"] + XCTAssertEqual(expected, KlarnaHelper.availableCountries(currency: "chf")) + } + + func testCanBuyNow_shouldReturnTrue() { + // https://site-admin.stripe.com/docs/payments/klarna#payment-options + let canBuyNow = ["de_AT", "nl_BE", "de_DE", "it_IT", "nl_NL", "es_ES", "sv_SE", "en_CA", "en_AU", "pl_PL", "es_PT", "de_CH", "fr_CA"] + + for country in canBuyNow { + XCTAssertTrue(KlarnaHelper.canBuyNow(locale: Locale(identifier: country))) + } + } + + func testCanBuyNow_shouldReturnFalse() { + // https://site-admin.stripe.com/docs/payments/klarna#payment-options + let canNotBuyNow = ["da_DK", "fi_FI", "fr_FR", "no_NO", "en_GB", "en_US"] + + for country in canNotBuyNow { + XCTAssertFalse(KlarnaHelper.canBuyNow(locale: Locale(identifier: country))) + } + } + +} diff --git a/Stripe/StripeiOSTests/LinkAccountServiceTests.swift b/Stripe/StripeiOSTests/LinkAccountServiceTests.swift new file mode 100644 index 00000000..5fe54322 --- /dev/null +++ b/Stripe/StripeiOSTests/LinkAccountServiceTests.swift @@ -0,0 +1,42 @@ +// +// LinkAccountServiceTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 2/22/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkAccountServiceTests: XCTestCase { + + func testWrite() { + let sut = makeSUT() + XCTAssertTrue(sut.hasEmailLoggedOut(email: "user@example.com")) + XCTAssertTrue(sut.hasEmailLoggedOut(email: "USER@EXAMPLE.COM")) + XCTAssertFalse(sut.hasEmailLoggedOut(email: "user@example.net")) + } + +} + +extension LinkAccountServiceTests { + + func makeSUT() -> LinkAccountService { + let cookieStore = LinkInMemoryCookieStore() + + cookieStore.write( + key: .lastLogoutEmail, + // SHA-256 hash for `user@example.com` + value: "tMmiiTI7IaAcPpQPFQ65uMVCWH8av9jw4cwf/F5HVRQ=" + ) + + return LinkAccountService(cookieStore: cookieStore) + } + +} diff --git a/Stripe/StripeiOSTests/LinkBadgeViewSnapshotTest.swift b/Stripe/StripeiOSTests/LinkBadgeViewSnapshotTest.swift new file mode 100644 index 00000000..d5f9d717 --- /dev/null +++ b/Stripe/StripeiOSTests/LinkBadgeViewSnapshotTest.swift @@ -0,0 +1,59 @@ +// +// LinkBadgeViewSnapshotTest.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 4/29/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkBadgeViewSnapshotTest: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testNeutral() { + verify( + LinkBadgeView( + type: .neutral, + text: "Neutral message" + ) + ) + } + + func testError() { + verify( + LinkBadgeView( + type: .error, + text: "Error message" + ) + ) + } + + func verify( + _ sut: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + let size = sut.systemLayoutSizeFitting( + UIView.layoutFittingCompressedSize, + withHorizontalFittingPriority: .fittingSizeLevel, + verticalFittingPriority: .fittingSizeLevel + ) + + sut.bounds = CGRect(origin: .zero, size: size) + STPSnapshotVerifyView(sut, identifier: identifier, file: file, line: line) + } + +} diff --git a/Stripe/StripeiOSTests/LinkCardEditElementSnapshotTests.swift b/Stripe/StripeiOSTests/LinkCardEditElementSnapshotTests.swift new file mode 100644 index 00000000..c80772cf --- /dev/null +++ b/Stripe/StripeiOSTests/LinkCardEditElementSnapshotTests.swift @@ -0,0 +1,75 @@ +// +// LinkCardEditElementSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 10/3/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import UIKit + +@testable import Stripe +@testable@_spi(STP) import StripeCoreTestUtils +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripeUICore + +final class LinkCardEditElementSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // self.recordMode = true + + // `LinkCardEditElement` depends on `AddressSectionElement`, which requires + // address specs to be loaded in memory. + let expectation = expectation(description: "Load address specs") + AddressSpecProvider.shared.loadAddressSpecs { + expectation.fulfill() + } + + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testDefault() { + let sut = makeSUT(isDefault: true) + verify(sut) + } + + func testNonDefault() { + let sut = makeSUT(isDefault: false) + verify(sut) + } + + func verify( + _ element: LinkCardEditElement, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + element.view.autosizeHeight(width: 340) + STPSnapshotVerifyView(element.view, identifier: identifier, file: file, line: line) + } + +} + +extension LinkCardEditElementSnapshotTests { + + func makeSUT(isDefault: Bool) -> LinkCardEditElement { + let paymentMethod = ConsumerPaymentDetails( + stripeID: "1", + details: .card( + card: .init( + expiryYear: 2032, + expiryMonth: 1, + brand: "visa", + last4: "4242", + checks: nil + ) + ), + isDefault: isDefault + ) + + return LinkCardEditElement(paymentMethod: paymentMethod, configuration: .init()) + } + +} diff --git a/Stripe/StripeiOSTests/LinkInMemoryCookieStoreTests.swift b/Stripe/StripeiOSTests/LinkInMemoryCookieStoreTests.swift new file mode 100644 index 00000000..1eb64c5d --- /dev/null +++ b/Stripe/StripeiOSTests/LinkInMemoryCookieStoreTests.swift @@ -0,0 +1,81 @@ +// +// LinkInMemoryCookieStoreTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 12/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkInMemoryCookieStoreTests: XCTestCase { + + static let testKey: LinkCookieKey = .session + + func testWrite() { + let sut = makeSUT() + sut.write(key: Self.testKey, value: "cookie_value") + + XCTAssertEqual(sut.read(key: Self.testKey), "cookie_value") + } + + func testWrite_overwriting() { + let sut = makeSUT() + sut.write(key: Self.testKey, value: "cookie_value") + sut.write(key: Self.testKey, value: "new_cookie_value") + + XCTAssertEqual(sut.read(key: Self.testKey), "new_cookie_value") + } + + func testDelete() { + let sut = makeSUT() + sut.write(key: Self.testKey, value: "cookie_value") + sut.delete(key: Self.testKey) + + XCTAssertNil(sut.read(key: Self.testKey)) + } + + // MARK: Session cookies + + func testFormattedSessionCookies() { + let sut = makeSUT() + + sut.write(key: .session, value: "cookie_value") + XCTAssertEqual( + sut.formattedSessionCookies(), + [ + "verification_session_client_secrets": ["cookie_value"] + ] + ) + + sut.delete(key: .session) + XCTAssertNil(sut.formattedSessionCookies()) + } + + func testUpdateSessionCookie() { + let sut = makeSUT() + sut.updateSessionCookie(with: "top_secret") + XCTAssertEqual(sut.read(key: .session), "top_secret") + + sut.updateSessionCookie(with: nil) + XCTAssertEqual(sut.read(key: .session), "top_secret") + + sut.updateSessionCookie(with: "") + XCTAssertNil(sut.read(key: .session)) + } + +} + +extension LinkInMemoryCookieStoreTests { + + func makeSUT() -> LinkInMemoryCookieStore { + return LinkInMemoryCookieStore() + } + +} diff --git a/Stripe/StripeiOSTests/LinkInlineSignupElementSnapshotTests.swift b/Stripe/StripeiOSTests/LinkInlineSignupElementSnapshotTests.swift new file mode 100644 index 00000000..0d9e04bc --- /dev/null +++ b/Stripe/StripeiOSTests/LinkInlineSignupElementSnapshotTests.swift @@ -0,0 +1,123 @@ +// +// LinkInlineSignupElementSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 1/21/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +@_spi(STP) import StripeCoreTestUtils +@_spi(STP) import StripeUICore +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkInlineSignupElementSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testDefaultState() { + let sut = makeSUT() + verify(sut) + } + + func testExpandedState() { + let sut = makeSUT(saveCheckboxChecked: true, emailAddress: "user@example.com") + verify(sut) + } + + func testExpandedState_nonUS() { + let sut = makeSUT( + saveCheckboxChecked: true, + emailAddress: "user@example.com", + country: "CA" + ) + verify(sut) + } + + func testExpandedState_nonUS_preFilled() { + let sut = makeSUT( + saveCheckboxChecked: true, + emailAddress: "user@example.com", + country: "CA", + preFillName: "Jane Diaz", + preFillPhone: "+13105551234" + ) + verify(sut) + } + + func verify( + _ element: LinkInlineSignupElement, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + element.view.autosizeHeight(width: 340) + STPSnapshotVerifyView(element.view, identifier: identifier, file: file, line: line) + } + +} + +extension LinkInlineSignupElementSnapshotTests { + + struct MockAccountService: LinkAccountServiceProtocol { + func lookupAccount( + withEmail email: String?, + completion: @escaping (Result) -> Void + ) { + completion( + .success( + PaymentSheetLinkAccount( + email: "user@example.com", + session: nil, + publishableKey: nil + ) + ) + ) + } + + func hasEmailLoggedOut(email: String) -> Bool { + // TODO(porter): Determine if we want to implement this in tests + return false + } + } + + func makeSUT( + saveCheckboxChecked: Bool = false, + emailAddress: String? = nil, + country: String = "US", + preFillName: String? = nil, + preFillPhone: String? = nil + ) -> LinkInlineSignupElement { + var configuration = PaymentSheet.Configuration() + configuration.merchantDisplayName = "[Merchant]" + configuration.defaultBillingDetails.name = preFillName + configuration.defaultBillingDetails.phone = preFillPhone + + let viewModel = LinkInlineSignupViewModel( + configuration: configuration, + accountService: MockAccountService(), + country: country + ) + + viewModel.saveCheckboxChecked = saveCheckboxChecked + viewModel.emailAddress = emailAddress + + if emailAddress != nil { + // Wait for account to load + let expectation = notNullExpectation(for: viewModel, keyPath: \.linkAccount) + wait(for: [expectation], timeout: 10) + } + + return .init(viewModel: viewModel) + } + +} diff --git a/Stripe/StripeiOSTests/LinkInstantDebitMandateViewSnapshotTests.swift b/Stripe/StripeiOSTests/LinkInstantDebitMandateViewSnapshotTests.swift new file mode 100644 index 00000000..b0646cce --- /dev/null +++ b/Stripe/StripeiOSTests/LinkInstantDebitMandateViewSnapshotTests.swift @@ -0,0 +1,78 @@ +// +// LinkInstantDebitMandateViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 2/17/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class LinkInstantDebitMandateViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testDefault() { + let sut = makeSUT() + verify(sut) + } + + func testLocalization() { + performLocalizedSnapshotTest(forLanguage: "de") + performLocalizedSnapshotTest(forLanguage: "es") + performLocalizedSnapshotTest(forLanguage: "el-GR") + performLocalizedSnapshotTest(forLanguage: "it") + performLocalizedSnapshotTest(forLanguage: "ja") + performLocalizedSnapshotTest(forLanguage: "ko") + performLocalizedSnapshotTest(forLanguage: "zh-Hans") + } + +} + +// MARK: - Helpers + +extension LinkInstantDebitMandateViewSnapshotTests { + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 250) + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } + + func performLocalizedSnapshotTest( + forLanguage language: String, + file: StaticString = #filePath, + line: UInt = #line + ) { + STPLocalizationUtils.overrideLanguage(to: language) + let sut = makeSUT() + STPLocalizationUtils.overrideLanguage(to: nil) + verify(sut, identifier: language, file: file, line: line) + } + +} + +// MARK: - Factory + +extension LinkInstantDebitMandateViewSnapshotTests { + + func makeSUT() -> LinkInstantDebitMandateView { + return LinkInstantDebitMandateView() + } + +} diff --git a/Stripe/StripeiOSTests/LinkLegalTermsViewSnapshotTests.swift b/Stripe/StripeiOSTests/LinkLegalTermsViewSnapshotTests.swift new file mode 100644 index 00000000..82a55ac6 --- /dev/null +++ b/Stripe/StripeiOSTests/LinkLegalTermsViewSnapshotTests.swift @@ -0,0 +1,94 @@ +// +// LinkLegalTermsViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 1/26/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class LinkLegalTermsViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testDefault() { + let sut = makeSUT() + verify(sut) + } + + func testCentered() { + let sut = makeSUT(textAlignment: .center) + verify(sut) + } + + func testColorCustomization() { + let sut = makeSUT() + sut.textColor = .orange + sut.tintColor = .purple + verify(sut) + } + + func testLocalization() { + performLocalizedSnapshotTest(forLanguage: "de") + performLocalizedSnapshotTest(forLanguage: "es") + performLocalizedSnapshotTest(forLanguage: "el-GR") + performLocalizedSnapshotTest(forLanguage: "it") + performLocalizedSnapshotTest(forLanguage: "ja") + performLocalizedSnapshotTest(forLanguage: "ko") + performLocalizedSnapshotTest(forLanguage: "zh-Hans") + } + +} + +// MARK: - Helpers + +extension LinkLegalTermsViewSnapshotTests { + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 250) + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } + + func performLocalizedSnapshotTest( + forLanguage language: String, + file: StaticString = #filePath, + line: UInt = #line + ) { + STPLocalizationUtils.overrideLanguage(to: language) + let sut = makeSUT() + STPLocalizationUtils.overrideLanguage(to: nil) + verify(sut, identifier: language, file: file, line: line) + } + +} + +// MARK: - Factory + +extension LinkLegalTermsViewSnapshotTests { + + func makeSUT() -> LinkLegalTermsView { + return LinkLegalTermsView() + } + + func makeSUT(textAlignment: NSTextAlignment) -> LinkLegalTermsView { + return LinkLegalTermsView(textAlignment: textAlignment) + } + +} diff --git a/Stripe/StripeiOSTests/LinkNavigationBarSnapshotTests.swift b/Stripe/StripeiOSTests/LinkNavigationBarSnapshotTests.swift new file mode 100644 index 00000000..b5900c44 --- /dev/null +++ b/Stripe/StripeiOSTests/LinkNavigationBarSnapshotTests.swift @@ -0,0 +1,89 @@ +// +// LinkNavigationBarSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 4/27/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkNavigationBarSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testDefault() { + let sut = makeSUT() + verify(sut) + + sut.showBackButton = true + verify(sut, identifier: "BackButton") + } + + func testWithEmailAddress() { + let sut = makeSUT() + sut.linkAccount = makeAccountStub(email: "user@example.com") + verify(sut) + + sut.showBackButton = true + verify(sut, identifier: "BackButton") + } + + func testWithLongEmailAddress() { + let sut = makeSUT() + sut.linkAccount = makeAccountStub(email: "a.very.very.long.customer.name@example.com") + verify(sut) + + sut.showBackButton = true + verify(sut, identifier: "BackButton") + } + + func verify( + _ sut: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + let size = sut.systemLayoutSizeFitting( + CGSize(width: 375, height: UIView.layoutFittingCompressedSize.height), + withHorizontalFittingPriority: .required, + verticalFittingPriority: .fittingSizeLevel + ) + + sut.bounds = CGRect(origin: .zero, size: size) + STPSnapshotVerifyView(sut, identifier: identifier, file: file, line: line) + } + +} + +extension LinkNavigationBarSnapshotTests { + fileprivate struct LinkAccountStub: PaymentSheetLinkAccountInfoProtocol { + let email: String + let redactedPhoneNumber: String? + let isRegistered: Bool + let isLoggedIn: Bool + } + + fileprivate func makeAccountStub(email: String) -> LinkAccountStub { + return LinkAccountStub( + email: email, + redactedPhoneNumber: "+1********55", + isRegistered: true, + isLoggedIn: true + ) + } + + fileprivate func makeSUT() -> LinkNavigationBar { + return LinkNavigationBar() + } +} diff --git a/Stripe/StripeiOSTests/LinkNoticeViewSnapshotTests.swift b/Stripe/StripeiOSTests/LinkNoticeViewSnapshotTests.swift new file mode 100644 index 00000000..e62a1a1e --- /dev/null +++ b/Stripe/StripeiOSTests/LinkNoticeViewSnapshotTests.swift @@ -0,0 +1,51 @@ +// +// LinkNoticeViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 3/31/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkNoticeViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testError() { + verify( + LinkNoticeView( + type: .error, + text: + "This card has expired. Update it to keep using it or use a different payment method." + ) + ) + } + + func verify( + _ sut: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + let size = sut.systemLayoutSizeFitting( + CGSize(width: 340, height: UIView.layoutFittingCompressedSize.height), + withHorizontalFittingPriority: .required, + verticalFittingPriority: .fittingSizeLevel + ) + + sut.bounds = CGRect(origin: .zero, size: size) + STPSnapshotVerifyView(sut, identifier: identifier, file: file, line: line) + } + +} diff --git a/Stripe/StripeiOSTests/LinkPaymentMethodPickerSnapshotTests.swift b/Stripe/StripeiOSTests/LinkPaymentMethodPickerSnapshotTests.swift new file mode 100644 index 00000000..686715d5 --- /dev/null +++ b/Stripe/StripeiOSTests/LinkPaymentMethodPickerSnapshotTests.swift @@ -0,0 +1,108 @@ +// +// LinkPaymentMethodPickerSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 11/8/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkPaymentMethodPickerSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testNormal() { + let mockDataSource = MockDataSource() + + let picker = LinkPaymentMethodPicker() + picker.dataSource = mockDataSource + picker.layoutSubviews() + + verify(picker, identifier: "First Option") + + picker.selectedIndex = 1 + verify(picker, identifier: "Second Option") + } + + func testExpanded() { + let mockDataSource = MockDataSource() + + let picker = LinkPaymentMethodPicker() + picker.dataSource = mockDataSource + picker.layoutSubviews() + picker.setExpanded(true, animated: false) + + verify(picker) + } + + func testUnsupportedBankAccount() { + let mockDataSource = MockDataSource() + + let picker = LinkPaymentMethodPicker() + picker.dataSource = mockDataSource + picker.supportedPaymentMethodTypes = [.card] + picker.layoutSubviews() + picker.setExpanded(true, animated: false) + + verify(picker) + } + + func testEmpty() { + let mockDataSource = MockDataSource(empty: true) + + let picker = LinkPaymentMethodPicker() + picker.dataSource = mockDataSource + picker.layoutSubviews() + + verify(picker) + } + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 335) + view.backgroundColor = .white + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } + +} + +extension LinkPaymentMethodPickerSnapshotTests { + + fileprivate final class MockDataSource: LinkPaymentMethodPickerDataSource { + let paymentMethods: [ConsumerPaymentDetails] + + init( + empty: Bool = false + ) { + self.paymentMethods = empty ? [] : LinkStubs.paymentMethods() + } + + func numberOfPaymentMethods(in picker: LinkPaymentMethodPicker) -> Int { + return paymentMethods.count + } + + func paymentPicker( + _ picker: LinkPaymentMethodPicker, + paymentMethodAt index: Int + ) -> ConsumerPaymentDetails { + return paymentMethods[index] + } + } + +} diff --git a/Stripe/StripeiOSTests/LinkSignupViewModelTests.swift b/Stripe/StripeiOSTests/LinkSignupViewModelTests.swift new file mode 100644 index 00000000..d11f6a34 --- /dev/null +++ b/Stripe/StripeiOSTests/LinkSignupViewModelTests.swift @@ -0,0 +1,179 @@ +// +// LinkSignupViewModelTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 1/21/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +import StripeCoreTestUtils +@_spi(STP) import StripeUICore +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkInlineSignupViewModelTests: XCTestCase { + + // Should be ~4x the debounce time for best results. + let accountLookupTimeout: TimeInterval = 4 + + func test_defaults() { + let sut = makeSUT(country: "US") + + XCTAssertFalse(sut.shouldShowEmailField) + XCTAssertFalse(sut.shouldShowPhoneField) + XCTAssertFalse(sut.shouldShowNameField) + XCTAssertFalse(sut.shouldShowLegalTerms) + } + + func test_shouldShowEmailFieldWhenCheckboxIsChecked() { + let sut = makeSUT(country: "US") + + sut.saveCheckboxChecked = true + XCTAssertTrue(sut.shouldShowEmailField) + + sut.saveCheckboxChecked = false + XCTAssertFalse(sut.shouldShowEmailField) + } + + func test_shouldShowRegistrationFieldsWhenEmailIsProvided() { + let sut = makeSUT(country: "US") + + sut.saveCheckboxChecked = true + sut.emailAddress = "user@example.com" + + // Wait for async change on `shouldShowPhoneField`. + let showPhoneFieldExpectation = expectation( + for: sut, + keyPath: \.shouldShowPhoneField, + equalsToValue: true + ) + wait(for: [showPhoneFieldExpectation], timeout: accountLookupTimeout) + + XCTAssertFalse(sut.shouldShowNameField, "Should not show name field for US customers") + XCTAssertTrue( + sut.shouldShowLegalTerms, + "Should show legal terms when creating a new account" + ) + + sut.emailAddress = nil + + // Wait for async change on `shouldShowPhoneField`. + let hidePhoneFieldExpectation = expectation( + for: sut, + keyPath: \.shouldShowPhoneField, + equalsToValue: false + ) + wait(for: [hidePhoneFieldExpectation], timeout: accountLookupTimeout) + XCTAssertFalse(sut.shouldShowNameField) + XCTAssertFalse(sut.shouldShowLegalTerms) + } + + func test_shouldShowNameField_nonUSCustomers() { + let sut = makeSUT(country: "CA", hasAccount: true) + sut.saveCheckboxChecked = true + XCTAssertTrue(sut.shouldShowNameField, "Should show name field for non-US customers") + } + + func test_action_returnsNilUnlessPhoneRequirementIsFulfilled() { + let sut = makeSUT(country: "US", hasAccount: true) + + sut.saveCheckboxChecked = true + XCTAssertNil(sut.action) + + sut.phoneNumber = PhoneNumber(number: "5555555555", countryCode: "US") + XCTAssertNotNil(sut.action) + } + + func test_action_returnsNilUnlessNameRequirementIsFulfilled() { + // Non-US customers require providing a name + let sut = makeSUT(country: "CA", hasAccount: true) + + sut.saveCheckboxChecked = true + sut.phoneNumber = PhoneNumber(number: "5555555555", countryCode: "CA") + XCTAssertNil(sut.action, "`action` must be nil unless a name is provided") + + sut.legalName = "Jane Doe" + XCTAssertNotNil(sut.action) + } + + func test_action_returnsContinueWithoutLinkIfCheckboxIsNotChecked() { + let sut = makeSUT(country: "US") + + sut.saveCheckboxChecked = false + XCTAssertEqual(sut.action, .continueWithoutLink) + } + + func test_action_returnsContinueWithoutLinkIfLookupFails() { + let sut = makeSUT(country: "US", shouldFailLookup: true) + + sut.saveCheckboxChecked = true + sut.emailAddress = "user@example.com" + + // Wait for lookup to fail + let lookupFailedExpectation = expectation( + for: sut, + keyPath: \.lookupFailed, + equalsToValue: true + ) + wait(for: [lookupFailedExpectation], timeout: accountLookupTimeout) + + XCTAssertEqual(sut.action, .continueWithoutLink) + } + +} + +extension LinkInlineSignupViewModelTests { + + struct MockAccountService: LinkAccountServiceProtocol { + let shouldFailLookup: Bool + + func lookupAccount( + withEmail email: String?, + completion: @escaping (Result) -> Void + ) { + if shouldFailLookup { + completion(.failure(NSError.stp_genericConnectionError())) + } else { + completion( + .success( + PaymentSheetLinkAccount( + email: "user@example.com", + session: nil, + publishableKey: nil + ) + ) + ) + } + } + + func hasEmailLoggedOut(email: String) -> Bool { + // TODO(porter): Determine if we want to implement this in tests + return false + } + } + + func makeSUT( + country: String, + hasAccount: Bool = false, + shouldFailLookup: Bool = false + ) -> LinkInlineSignupViewModel { + let linkAccount: PaymentSheetLinkAccount? = + hasAccount + ? PaymentSheetLinkAccount(email: "user@example.com", session: nil, publishableKey: nil) + : nil + + return LinkInlineSignupViewModel( + configuration: .init(), + accountService: MockAccountService(shouldFailLookup: shouldFailLookup), + linkAccount: linkAccount, + country: country + ) + } + +} diff --git a/Stripe/StripeiOSTests/LinkStubs.swift b/Stripe/StripeiOSTests/LinkStubs.swift new file mode 100644 index 00000000..0e3693a0 --- /dev/null +++ b/Stripe/StripeiOSTests/LinkStubs.swift @@ -0,0 +1,99 @@ +// +// LinkStubs.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 3/31/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +struct LinkStubs { + private init() {} +} + +extension LinkStubs { + + struct PaymentMethodIndices { + static let card = 0 + static let cardWithFailingChecks = 1 + static let bankAccount = 2 + static let expiredCard = 3 + static let notExisting = -1 + } + + static func paymentMethods() -> [ConsumerPaymentDetails] { + let calendar = Calendar(identifier: .gregorian) + let nextYear = calendar.component(.year, from: Date()) + 1 + + return [ + ConsumerPaymentDetails( + stripeID: "1", + details: .card( + card: .init( + expiryYear: nextYear, + expiryMonth: 1, + brand: "visa", + last4: "4242", + checks: .init(cvcCheck: .pass) + ) + ), + isDefault: true + ), + ConsumerPaymentDetails( + stripeID: "2", + details: .card( + card: .init( + expiryYear: nextYear, + expiryMonth: 1, + brand: "mastercard", + last4: "4444", + checks: .init(cvcCheck: .fail) + ) + ), + isDefault: false + ), + ConsumerPaymentDetails( + stripeID: "3", + details: .bankAccount( + bankAccount: .init( + iconCode: "capitalone", + name: "Capital One", + last4: "4242" + ) + ), + isDefault: false + ), + ConsumerPaymentDetails( + stripeID: "4", + details: .card( + card: .init( + expiryYear: 2020, + expiryMonth: 1, + brand: "american_express", + last4: "0005", + checks: .init(cvcCheck: .fail) + ) + ), + isDefault: false + ), + ] + } + + static func consumerSession() -> ConsumerSession { + return ConsumerSession( + clientSecret: "client_secret", + emailAddress: "user@example.com", + redactedPhoneNumber: "1********55", + verificationSessions: [], + supportedPaymentDetailsTypes: [.card, .bankAccount] + ) + } + +} diff --git a/Stripe/StripeiOSTests/LinkToastSnapshotTests.swift b/Stripe/StripeiOSTests/LinkToastSnapshotTests.swift new file mode 100644 index 00000000..7f6d910e --- /dev/null +++ b/Stripe/StripeiOSTests/LinkToastSnapshotTests.swift @@ -0,0 +1,41 @@ +// +// LinkToastSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 12/7/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkToastSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testSuccess() { + let toast = LinkToast(type: .success, text: "Success message!") + verify(toast) + } + + func verify( + _ toast: LinkToast, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + let size = toast.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + toast.bounds = CGRect(origin: .zero, size: size) + STPSnapshotVerifyView(toast, identifier: identifier, file: file, line: line) + } + +} diff --git a/Stripe/StripeiOSTests/LinkVerificationViewSnapshotTests.swift b/Stripe/StripeiOSTests/LinkVerificationViewSnapshotTests.swift new file mode 100644 index 00000000..752ddecc --- /dev/null +++ b/Stripe/StripeiOSTests/LinkVerificationViewSnapshotTests.swift @@ -0,0 +1,84 @@ +// +// LinkVerificationViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 12/7/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class LinkVerificationViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testModal() { + let sut = makeSUT(mode: .modal) + verify(sut) + } + + func testModalWithErrorMessage() { + let sut = makeSUT(mode: .modal) + sut.errorMessage = "The provided verification code has expired." + verify(sut) + } + + func testInlineLogin() { + let sut = makeSUT(mode: .inlineLogin) + verify(sut) + } + + func testEmbedded() { + let sut = makeSUT(mode: .embedded) + verify(sut) + } + + func verify( + _ view: LinkVerificationView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 340) + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } + +} + +extension LinkVerificationViewSnapshotTests { + + struct LinkAccountStub: PaymentSheetLinkAccountInfoProtocol { + let email: String + let redactedPhoneNumber: String? + let isRegistered: Bool + let isLoggedIn: Bool + } + + func makeSUT(mode: LinkVerificationView.Mode) -> LinkVerificationView { + let sut = LinkVerificationView( + mode: mode, + linkAccount: LinkAccountStub( + email: "user@example.com", + redactedPhoneNumber: "+1********55", + isRegistered: true, + isLoggedIn: false + ) + ) + + sut.tintColor = .linkBrand + + return sut + } + +} diff --git a/Stripe/StripeiOSTests/MKPlacemark+PaymentSheetTests.swift b/Stripe/StripeiOSTests/MKPlacemark+PaymentSheetTests.swift new file mode 100644 index 00000000..711e3863 --- /dev/null +++ b/Stripe/StripeiOSTests/MKPlacemark+PaymentSheetTests.swift @@ -0,0 +1,209 @@ +// +// MKPlacemark+PaymentSheetTests.swift +// StripeiOS Tests +// +// Created by Nick Porter on 6/13/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Contacts +import MapKit +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class MKPlacemark_PaymentSheetTests: XCTestCase { + + // All address dictionaries are based on an actual placemark of an `MKLocalSearchCompletion` + + func testAsAddress_UnitedStates() { + // Search string used to generate address dictionary: "4 Pennsylvania Pl" + let addressDictionary = + [ + CNPostalAddressStreetKey: "4 Pennsylvania Plaza", + CNPostalAddressStateKey: "NY", + CNPostalAddressCountryKey: "United States", + CNPostalAddressISOCountryCodeKey: "US", + CNPostalAddressCityKey: "New York", + "SubThoroughfare": "4", + CNPostalAddressPostalCodeKey: "10001", + "Thoroughfare": "Pennsylvania Plaza", + CNPostalAddressSubLocalityKey: "Manhattan", + CNPostalAddressSubAdministrativeAreaKey: "New York County", + "Name": "4 Pennsylvania Plaza", + ] as [String: Any] + let placemark = MKPlacemark( + coordinate: CLLocationCoordinate2D(), + addressDictionary: addressDictionary + ) + let expectedAddress = PaymentSheet.Address( + city: "New York", + country: "US", + line1: "4 Pennsylvania Plaza", + line2: nil, + postalCode: "10001", + state: "NY" + ) + + XCTAssertEqual(placemark.asAddress, expectedAddress) + } + + func testAsAddress_Canada() { + // Search string used to generate address dictionary: "40 Bay St To" + let addressDictionary = + [ + CNPostalAddressStreetKey: "40 Bay St", + CNPostalAddressStateKey: "ON", + CNPostalAddressISOCountryCodeKey: "CA", + CNPostalAddressCountryKey: "Canada", + CNPostalAddressCityKey: "Toronto", + "SubThoroughfare": "40", + CNPostalAddressPostalCodeKey: "M5J 2X2", + "Thoroughfare": "Bay St", + CNPostalAddressSubLocalityKey: "Downtown Toronto", + CNPostalAddressSubAdministrativeAreaKey: "SubAdministrativeArea", + "Name": "40 Bay St", + ] as [String: Any] + let placemark = MKPlacemark( + coordinate: CLLocationCoordinate2D(), + addressDictionary: addressDictionary + ) + let expectedAddress = PaymentSheet.Address( + city: "Toronto", + country: "CA", + line1: "40 Bay St", + line2: nil, + postalCode: "M5J 2X2", + state: "ON" + ) + + XCTAssertEqual(placemark.asAddress, expectedAddress) + } + + func testAsAddress_Germany() { + // Search string used to generate address dictionary: "Rüsternallee 14" + let addressDictionary = + [ + CNPostalAddressStreetKey: "Rüsternallee 14", + CNPostalAddressISOCountryCodeKey: "DE", + CNPostalAddressCountryKey: "Germany", + CNPostalAddressCityKey: "Berlin", + CNPostalAddressPostalCodeKey: "14050", + "SubThoroughfare": "14", + "Thoroughfare": "Rüsternallee", + CNPostalAddressSubLocalityKey: "Charlottenburg-Wilmersdorf", + CNPostalAddressSubAdministrativeAreaKey: "Berlin", + "Name": "Rüsternallee 14", + ] as [String: Any] + let placemark = MKPlacemark( + coordinate: CLLocationCoordinate2D(), + addressDictionary: addressDictionary + ) + let expectedAddress = PaymentSheet.Address( + city: "Berlin", + country: "DE", + line1: "Rüsternallee 14", + line2: nil, + postalCode: "14050", + state: nil + ) + + XCTAssertEqual(placemark.asAddress, expectedAddress) + } + + func testAsAddress_Brazil() { + // Search string used to generate address dictionary: "Avenida Paulista 500" + let addressDictionary = + [ + CNPostalAddressStreetKey: "Avenida Paulista, 500", + CNPostalAddressStateKey: "SP", + CNPostalAddressISOCountryCodeKey: "BR", + CNPostalAddressCountryKey: "Brazil", + CNPostalAddressCityKey: "Paulínia", + "SubThoroughfare": "500", + CNPostalAddressPostalCodeKey: "13145-089", + "Thoroughfare": "Avenida Paulista", + CNPostalAddressSubLocalityKey: "Jardim Planalto", + "Name": "Avenida Paulista, 500", + ] as [String: Any] + let placemark = MKPlacemark( + coordinate: CLLocationCoordinate2D(), + addressDictionary: addressDictionary + ) + let expectedAddress = PaymentSheet.Address( + city: "Paulínia", + country: "BR", + line1: "Avenida Paulista, 500", + line2: nil, + postalCode: "13145-089", + state: "SP" + ) + + XCTAssertEqual(placemark.asAddress, expectedAddress) + } + + func testAsAddress_Japan() { + // Search string used to generate address dictionary: "Nagatacho 2" + let addressDictionary = + [ + CNPostalAddressStreetKey: "Nagatacho 2-Chōme", + CNPostalAddressStateKey: "Tokyo", + CNPostalAddressISOCountryCodeKey: "JP", + CNPostalAddressCountryKey: "Japan", + CNPostalAddressCityKey: "Chiyoda", + "Thoroughfare": "Nagatacho 2-Chōme", + CNPostalAddressSubLocalityKey: "Nagatacho", + "Name": "Nagatacho 2-Chōme", + ] as [String: Any] + let placemark = MKPlacemark( + coordinate: CLLocationCoordinate2D(), + addressDictionary: addressDictionary + ) + let expectedAddress = PaymentSheet.Address( + city: "Chiyoda", + country: "JP", + line1: "Nagatacho 2-Chōme", + line2: nil, + postalCode: nil, + state: "Tokyo" + ) + + XCTAssertEqual(placemark.asAddress, expectedAddress) + } + + func testAsAddress_Australia() { + // Search string used to generate address dictionary: "488 George St Syd" + let addressDictionary = + [ + CNPostalAddressStreetKey: "488 George St", + CNPostalAddressStateKey: "NSW", + CNPostalAddressISOCountryCodeKey: "AU", + CNPostalAddressCountryKey: "Australia", + CNPostalAddressCityKey: "Sydney", + CNPostalAddressPostalCodeKey: "2000", + "SubThoroughfare": "488", + "Thoroughfare": "George St", + CNPostalAddressSubAdministrativeAreaKey: "Sydney", + "Name": "488 George St", + ] as [String: Any] + let placemark = MKPlacemark( + coordinate: CLLocationCoordinate2D(), + addressDictionary: addressDictionary + ) + let expectedAddress = PaymentSheet.Address( + city: "Sydney", + country: "AU", + line1: "488 George St", + line2: nil, + postalCode: "2000", + state: "NSW" + ) + + XCTAssertEqual(placemark.asAddress, expectedAddress) + } + +} diff --git a/Stripe/StripeiOSTests/NSArray+StripeTest.swift b/Stripe/StripeiOSTests/NSArray+StripeTest.swift new file mode 100644 index 00000000..8e12190e --- /dev/null +++ b/Stripe/StripeiOSTests/NSArray+StripeTest.swift @@ -0,0 +1,67 @@ +// +// NSArray+StripeTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 1/19/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +import XCTest + +class Array_StripeTest: XCTestCase { + func test_arrayByRemovingNulls_removesNullsDeeply() { + let array: [Any] = [ + "id", + NSNull(), // null in root + [ + "user": "user_123", + "country": NSNull(), // null in dictionary + "nicknames": ["john", "johnny", NSNull()], + "profiles": [ + "facebook": "fb_123", + "twitter": NSNull(), + ], + ], + [ + NSNull(), // null in array + [ + "id": "fee_123", + "frequency": NSNull(), + ], + ["payment", NSNull()], + ], + ] + + let expected: [Any] = [ + "id", + [ + "user": "user_123", + "nicknames": ["john", "johnny"], + "profiles": [ + "facebook": "fb_123" + ], + ], + [ + [ + "id": "fee_123" + ], ["payment"], + ], + ] + + let result = array.stp_arrayByRemovingNulls() + + XCTAssertEqual(result as NSArray, expected as NSArray) + } + + func test_arrayByRemovingNulls_keepsEmptyLeaves() { + let array = [NSNull()] + let result = array.stp_arrayByRemovingNulls() + + XCTAssertEqual(result as NSArray, [] as NSArray) + } +} diff --git a/Stripe/StripeiOSTests/NSDecimalNumber+StripeTest.swift b/Stripe/StripeiOSTests/NSDecimalNumber+StripeTest.swift new file mode 100644 index 00000000..0f18bb84 --- /dev/null +++ b/Stripe/StripeiOSTests/NSDecimalNumber+StripeTest.swift @@ -0,0 +1,98 @@ +// +// NSDecimalNumber+StripeTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 4/19/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +import XCTest + +class NSDecimalNumberStripeTest: XCTestCase { + // an incomplete list of 2 decimal point currencies + private let twoDecimalPointCurrencies = [ + "usd", + "dkk", + "eur", + "aud", + "sek", + "sgd", + "cop", + ] + + private let noDecimalPointCurrencies = [ + "bif", + "clp", + "djf", + "gnf", + "jpy", + "kmf", + "krw", + "mga", + "pyg", + "rwf", + "vnd", + "vuv", + "xaf", + "xof", + "xpf", + ] + + private let threeDecimalCurrencies = [ + "bhd", + "jod", + "kwd", + "omr", + "tnd", + ] + + func testDecimalAmount_twoDecimal() { + for twoDecimalPointCurrency in twoDecimalPointCurrencies { + let decimalNumber = NSDecimalNumber.stp_decimalNumber(withAmount: 92123, currency: twoDecimalPointCurrency) + XCTAssertEqual(decimalNumber, NSDecimalNumber(string: "921.23")) + } + } + + func testDecimalAmount_noDecimal() { + for currency in noDecimalPointCurrencies { + let decimalNumber = NSDecimalNumber.stp_decimalNumber(withAmount: 92123, currency: currency) + XCTAssertEqual(decimalNumber, NSDecimalNumber(string: "92123")) + } + } + + func testDecimalAmount_threeDecimal() { + for currency in threeDecimalCurrencies { + let decimalNumber = NSDecimalNumber.stp_decimalNumber(withAmount: 92123, currency: currency) + XCTAssertEqual(decimalNumber, NSDecimalNumber(string: "92.123")) + } + } + + func testAmount_twoDecimal() { + for twoDecimalPointCurrency in twoDecimalPointCurrencies { + let amount = NSDecimalNumber(value: 1000.12) + let decimalNumber = amount.stp_amount(withCurrency: twoDecimalPointCurrency) + XCTAssertEqual(decimalNumber, 100012) + } + } + + func testAmount_noDecimal() { + for noDecimalPointCurrency in noDecimalPointCurrencies { + let amount = NSDecimalNumber(value: 1000.12) + let decimalNumber = amount.stp_amount(withCurrency: noDecimalPointCurrency) + XCTAssertEqual(decimalNumber, 1000) + } + } + + func testAmount_threeDecimal() { + for threeDecimalPointCurrency in threeDecimalCurrencies { + let amount = NSDecimalNumber(value: 1000.12) + let decimalNumber = amount.stp_amount(withCurrency: threeDecimalPointCurrency) + XCTAssertEqual(decimalNumber, 1000120) + } + } +} diff --git a/Stripe/StripeiOSTests/NSDictionary+StripeTest.swift b/Stripe/StripeiOSTests/NSDictionary+StripeTest.swift new file mode 100644 index 00000000..84287033 --- /dev/null +++ b/Stripe/StripeiOSTests/NSDictionary+StripeTest.swift @@ -0,0 +1,254 @@ +// +// Dictionary+StripeTest.swift +// StripeiOS Tests +// +// Created by Joey Dong on 7/24/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +import XCTest + +class Dictionary_StripeTest: XCTestCase { + // MARK: - dictionaryByRemovingNullsValidatingRequiredFields + func test_dictionaryByRemovingNulls_removesNullsDeeply() { + let dictionary = + [ + "id": "card_123", + "tokenization_method": NSNull(), // null in root + "metadata": [ + "user": "user_123", + "country": NSNull(), // null in dictionary + "nicknames": ["john", "johnny", NSNull()], + "profiles": [ + "facebook": "fb_123", + "twitter": NSNull(), + ], + ], + "fees": [ + NSNull(), // null in array + [ + "id": "fee_123", + "frequency": NSNull(), + ], + ["payment", NSNull()], + ], + ] as [AnyHashable: Any] + + let expected = + [ + "id": "card_123", + "metadata": [ + "user": "user_123", + "nicknames": ["john", "johnny"], + "profiles": [ + "facebook": "fb_123" + ], + ], + "fees": [ + [ + "id": "fee_123" + ], ["payment"], + ], + ] as [AnyHashable: Any] + + let result = dictionary.stp_dictionaryByRemovingNulls() + XCTAssertEqual(result as NSDictionary, expected as NSDictionary) + } + + func test_dictionaryByRemovingNullsValidatingRequiredFields_keepsEmptyLeaves() { + let dictionary = + [ + "id": NSNull() + ] as [AnyHashable: Any] + let result = dictionary.stp_dictionaryByRemovingNulls() + + XCTAssertEqual(result as NSDictionary, [:] as NSDictionary) + } + + // MARK: - dictionaryByRemovingNonStrings + func test_dictionaryByRemovingNonStrings_basicCases() { + // Empty dictionary + var dictionary = [:] as [AnyHashable: Any] + var expected = [:] as [AnyHashable: Any] + var result = dictionary.stp_dictionaryByRemovingNonStrings() + XCTAssertEqual(result as NSDictionary, expected as NSDictionary) + + // Regular case + dictionary = + [ + "user": "user_123", + "nicknames": "John, Johnny", + ] + expected = + [ + "user": "user_123", + "nicknames": "John, Johnny", + ] + result = dictionary.stp_dictionaryByRemovingNonStrings() + XCTAssertEqual(result as NSDictionary, expected as NSDictionary) + + // Strips non-NSString keys and values + dictionary = + [ + "user": "user_123", + "nicknames": "John, Johnny", + "profiles": NSNull(), + NSNull(): "San Francisco, CA", + "age": NSNumber(value: 21), + NSNumber(value: 21): "age", + "fees": [ + "plan": "monthly" + ], + "visits": ["january", "february"], + ] + expected = + [ + "user": "user_123", + "nicknames": "John, Johnny", + ] + result = dictionary.stp_dictionaryByRemovingNonStrings() + XCTAssertEqual(result as NSDictionary, expected as NSDictionary) + + // Strips non-NSString keys and values + dictionary = + [ + "user": "user_123", + "nicknames": "John, Johnny", + "profiles": NSNull(), + NSNull(): NSNull(), + "age": NSNumber(value: 21), + NSNumber(value: 21): NSNumber(value: 21), + "fees": [ + "plan": "monthly" + ], + "visits": ["january", "february"], + ] + expected = + [ + "user": "user_123", + "nicknames": "John, Johnny", + ] + result = dictionary.stp_dictionaryByRemovingNonStrings() + XCTAssertEqual(result as NSDictionary, expected as NSDictionary) + } + + // MARK: - Getters + func testArrayForKey() { + let dict = + [ + "a": ["foo"] + ] as [AnyHashable: Any] + + XCTAssertEqual(dict.stp_array(forKey: "a") as! [String], ["foo"]) + XCTAssertNil(dict.stp_array(forKey: "b")) + } + + func testBoolForKey() { + let dict = + [ + "a": NSNumber(value: 1), + "b": NSNumber(value: 0), + "c": "true", + "d": "false", + "e": "1", + "f": "foo", + ] as [AnyHashable: Any] + + XCTAssertTrue(dict.stp_bool(forKey: "a", or: false)) + XCTAssertFalse(dict.stp_bool(forKey: "b", or: true)) + XCTAssertTrue(dict.stp_bool(forKey: "c", or: false)) + XCTAssertFalse(dict.stp_bool(forKey: "d", or: true)) + XCTAssertTrue(dict.stp_bool(forKey: "e", or: false)) + XCTAssertFalse(dict.stp_bool(forKey: "f", or: false)) + } + + func testIntForKey() { + let dict = + [ + "a": NSNumber(value: 1), + "b": NSNumber(value: -1), + "c": "1", + "d": "-1", + "e": "10.0", + "f": "10.5", + "g": NSNumber(value: 10.0), + "h": NSNumber(value: 10.5), + "i": "foo", + ] as [AnyHashable: Any] + + XCTAssertEqual(dict.stp_int(forKey: "a", or: 0), 1) + XCTAssertEqual(dict.stp_int(forKey: "b", or: 0), -1) + XCTAssertEqual(dict.stp_int(forKey: "c", or: 0), 1) + XCTAssertEqual(dict.stp_int(forKey: "d", or: 0), -1) + XCTAssertEqual(dict.stp_int(forKey: "e", or: 0), 10) + XCTAssertEqual(dict.stp_int(forKey: "f", or: 0), 10) + XCTAssertEqual(dict.stp_int(forKey: "g", or: 0), 10) + XCTAssertEqual(dict.stp_int(forKey: "h", or: 0), 10) + XCTAssertEqual(dict.stp_int(forKey: "i", or: 0), 0) + } + + func testDateForKey() { + let dict = + [ + "a": NSNumber(value: 0), + "b": "0", + ] as [AnyHashable: Any] + let expectedDate = Date(timeIntervalSince1970: 0) + + XCTAssertEqual(dict.stp_date(forKey: "a"), expectedDate) + XCTAssertEqual(dict.stp_date(forKey: "b"), expectedDate) + XCTAssertNil(dict.stp_date(forKey: "c")) + } + + func testDictionaryForKey() { + let dict = + [ + "a": [ + "foo": "bar" + ], + ] as [AnyHashable: Any] + + XCTAssertEqual( + dict.stp_dictionary(forKey: "a")! as NSDictionary, + [ + "foo": "bar" + ] as NSDictionary + ) + XCTAssertNil(dict.stp_dictionary(forKey: "b")) + } + + func testNumberForKey() { + let dict = + [ + "a": NSNumber(value: 1) + ] as [AnyHashable: Any] + + XCTAssertEqual(dict.stp_number(forKey: "a"), NSNumber(value: 1)) + XCTAssertNil(dict.stp_number(forKey: "b")) + } + + func testStringForKey() { + let dict = + [ + "a": "foo" + ] as [AnyHashable: Any] + XCTAssertEqual(dict.stp_string(forKey: "a"), "foo") + XCTAssertNil(dict.stp_string(forKey: "b")) + } + + func testURLForKey() { + let dict = + [ + "a": "https://example.com", + "b": "not a url", + ] as [AnyHashable: Any] + XCTAssertEqual(dict.stp_url(forKey: "a"), URL(string: "https://example.com")) + XCTAssertNil(dict.stp_url(forKey: "b")) + XCTAssertNil(dict.stp_url(forKey: "c")) + } +} diff --git a/Stripe/StripeiOSTests/NSLocale+STPSwizzling.h b/Stripe/StripeiOSTests/NSLocale+STPSwizzling.h new file mode 100644 index 00000000..cda18337 --- /dev/null +++ b/Stripe/StripeiOSTests/NSLocale+STPSwizzling.h @@ -0,0 +1,15 @@ +// +// NSLocale+STPSwizzling.h +// StripeiOS Tests +// +// Created by Cameron Sabol on 7/17/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import + +@interface NSLocale (STPSwizzling) + ++ (void)stp_withLocaleAs:(NSLocale *)locale perform:(void (^)(void))block; + +@end diff --git a/Stripe/StripeiOSTests/NSLocale+STPSwizzling.m b/Stripe/StripeiOSTests/NSLocale+STPSwizzling.m new file mode 100644 index 00000000..2ed46028 --- /dev/null +++ b/Stripe/StripeiOSTests/NSLocale+STPSwizzling.m @@ -0,0 +1,87 @@ +// +// NSLocale+STPSwizzling.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 7/17/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import "NSLocale+STPSwizzling.h" + +#import + +@interface NSObject (STPSwizzling) + ++ (void)stp_swizzleClassMethod:(SEL)original withReplacement:(SEL)replacement; + +@end + +@implementation NSObject (STPSwizzling) + ++ (void)stp_swizzleClassMethod:(SEL)original withReplacement:(SEL)replacement +{ + Class class = object_getClass((id)self); + Method originalMethod = class_getClassMethod(self, original); + Method replacementMethod = class_getClassMethod(self, replacement); + + BOOL addedMethod = class_addMethod(class, + original, + method_getImplementation(replacementMethod), + method_getTypeEncoding(replacementMethod)); + if (addedMethod) { + class_replaceMethod(class, + replacement, + method_getImplementation(originalMethod), + method_getTypeEncoding(originalMethod)); + } else { + method_exchangeImplementations(originalMethod, replacementMethod); + } +} + +@end + +@implementation NSLocale (STPSwizzling) + +static NSLocale *_stpLocaleOverride = nil; + ++ (void)load { + static dispatch_once_t onceToken; + dispatch_once(&onceToken, ^{ + [self stp_swizzleClassMethod:@selector(currentLocale) withReplacement:@selector(stp_currentLocale)]; + [self stp_swizzleClassMethod:@selector(autoupdatingCurrentLocale) withReplacement:@selector(stp_autoUpdatingCurrentLocale)]; + [self stp_swizzleClassMethod:@selector(systemLocale) withReplacement:@selector(stp_systemLocale)]; + }); + +} + ++ (void)stp_withLocaleAs:(NSLocale *)locale perform:(void (^)(void))block { + NSLocale *currentLocale = NSLocale.currentLocale; + [self stp_setCurrentLocale:locale]; + block(); + [self stp_resetCurrentLocale]; + NSAssert([currentLocale isEqual:NSLocale.currentLocale], @"Failed to reset locale."); +} + ++ (void)stp_setCurrentLocale:(NSLocale *)locale +{ + _stpLocaleOverride = locale; +} + ++ (void)stp_resetCurrentLocale +{ + [self stp_setCurrentLocale:nil]; +} + ++ (instancetype)stp_currentLocale { + return _stpLocaleOverride ?: [self stp_currentLocale]; +} + ++ (instancetype)stp_autoUpdatingCurrentLocale { + return _stpLocaleOverride ?: [self stp_autoUpdatingCurrentLocale]; +} + ++ (instancetype)stp_systemLocale { + return _stpLocaleOverride ?: [self stp_systemLocale]; +} + +@end diff --git a/Stripe/StripeiOSTests/NSString+StripeTest.swift b/Stripe/StripeiOSTests/NSString+StripeTest.swift new file mode 100644 index 00000000..f0d1421d --- /dev/null +++ b/Stripe/StripeiOSTests/NSString+StripeTest.swift @@ -0,0 +1,95 @@ +// +// NSString+StripeTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 3/22/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +import XCTest + +class NSString_StripeTest: XCTestCase { + + func testIsBlank() { + XCTAssertTrue("".isBlank) + XCTAssertTrue(" ".isBlank) + XCTAssertTrue("\t\t\t".isBlank) + XCTAssertFalse("a".isBlank) + XCTAssertFalse(" a ".isBlank) + } + + func testSafeSubstringToIndex() { + XCTAssertEqual("foo".stp_safeSubstring(to: 0), "") + XCTAssertEqual("foo".stp_safeSubstring(to: 500), "foo") + XCTAssertEqual("foo".stp_safeSubstring(to: 1), "f") + XCTAssertEqual("foo".stp_safeSubstring(to: -1), "") + XCTAssertEqual("foo".stp_safeSubstring(to: -100), "") + XCTAssertEqual("".stp_safeSubstring(to: 0), "") + XCTAssertEqual("".stp_safeSubstring(to: 1), "") + } + + func testSafeSubstringFromIndex() { + XCTAssertEqual("foo".stp_safeSubstring(from: 0), "foo") + XCTAssertEqual("foo".stp_safeSubstring(from: 1), "oo") + XCTAssertEqual("foo".stp_safeSubstring(from: 3), "") + XCTAssertEqual("foo".stp_safeSubstring(from: -1), "foo") + XCTAssertEqual("foo".stp_safeSubstring(from: -100), "foo") + XCTAssertEqual("".stp_safeSubstring(from: 0), "") + XCTAssertEqual("".stp_safeSubstring(from: 1), "") + } + + func testStringByRemovingSuffix() { + XCTAssertEqual("foobar".stp_string(byRemovingSuffix: "bar"), "foo") + XCTAssertEqual("foobar".stp_string(byRemovingSuffix: "baz"), "foobar") + XCTAssertEqual("foobar".stp_string(byRemovingSuffix: nil), "foobar") + XCTAssertEqual("foobar".stp_string(byRemovingSuffix: "foobar"), "") + XCTAssertEqual("foobar".stp_string(byRemovingSuffix: ""), "foobar") + XCTAssertEqual("foobar".stp_string(byRemovingSuffix: "oba"), "foobar") + + XCTAssertEqual("foobar☺¿".stp_string(byRemovingSuffix: "bar☺¿"), "foo") + XCTAssertEqual("foobar☺¿".stp_string(byRemovingSuffix: "bar¿"), "foobar☺¿") + + XCTAssertEqual("foobar\u{202C}".stp_string(byRemovingSuffix: "bar"), "foobar\u{202C}") + XCTAssertEqual("foobar\u{202C}".stp_string(byRemovingSuffix: "bar\u{202C}"), "foo") + + // e + \u0041 => é + XCTAssertEqual("foobare\u{0301}".stp_string(byRemovingSuffix: "bare"), "foobare\u{0301}") + XCTAssertEqual("foobare\u{0301}".stp_string(byRemovingSuffix: "bare\u{0301}"), "foo") + XCTAssertEqual("foobare".stp_string(byRemovingSuffix: "bare\u{0301}"), "foobare") + + } + + func testLocalizedAmountDisplayString() { + XCTAssertEqual(String.localizedAmountDisplayString(for: 1099, currency: "USD"), "$10.99") + XCTAssertEqual( + String.localizedAmountDisplayString( + for: 1099, + currency: "USD", + locale: Locale(identifier: "fr_FR") + ), + "10,99 $US" + ) + XCTAssertEqual( + String.localizedAmountDisplayString( + for: 1099, + currency: "USD", + locale: Locale(identifier: "zh_HANT") + ), + "US$10.99" + ) + + XCTAssertEqual( + String.localizedAmountDisplayString( + for: 1099, + currency: "ZZZ", + locale: Locale(identifier: "z") + ), + "ZZZ 10.99" + ) + } +} diff --git a/Stripe/StripeiOSTests/NSURLComponents_StripeTest.swift b/Stripe/StripeiOSTests/NSURLComponents_StripeTest.swift new file mode 100644 index 00000000..86da11a4 --- /dev/null +++ b/Stripe/StripeiOSTests/NSURLComponents_StripeTest.swift @@ -0,0 +1,40 @@ +// +// NSURLComponents_StripeTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 5/24/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +import XCTest + +class NSURLComponents_StripeTest: XCTestCase { + func testCaseInsensitiveSchemeComparison() { + let lhs = NSURLComponents(string: "com.bar.foo://host")! + let rhs = NSURLComponents(string: "COM.BAR.FOO://HOST")! + XCTAssert(lhs.stp_matchesURLComponents(lhs)) // sanity + XCTAssert(lhs.stp_matchesURLComponents(rhs)) + XCTAssert(rhs.stp_matchesURLComponents(lhs)) + } + + func testMatchesURLsWithQueryString() { + // e.g. STPSourceFunctionalTest passes "https://shop.example.com/crtABC" for the return_url, + // but the Source object returned by the API comes has "https://shop.example.com/crtABC?redirect_merchant_name=xctest" + let expectedComponents = NSURLComponents( + string: "https://shop.example.com/crtABC?redirect_merchant_name=xctest" + )! + let components = NSURLComponents(string: "https://shop.example.com/crtABC")! + XCTAssertTrue(components.stp_matchesURLComponents(expectedComponents)) + } + + func testMatchesURLWithNilParameters() { + let nil1 = NSURLComponents(string: "")! + let nil2 = NSURLComponents(string: "")! + XCTAssert(nil1.stp_matchesURLComponents(nil2)) + } +} diff --git a/Stripe/StripeiOSTests/OneTimeCodeTextFieldSnapshotTests.swift b/Stripe/StripeiOSTests/OneTimeCodeTextFieldSnapshotTests.swift new file mode 100644 index 00000000..d5bc331a --- /dev/null +++ b/Stripe/StripeiOSTests/OneTimeCodeTextFieldSnapshotTests.swift @@ -0,0 +1,46 @@ +// +// OneTimeCodeTextFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 11/10/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class OneTimeCodeTextFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func testEmpty() { + let field = OneTimeCodeTextField(numberOfDigits: 6, theme: LinkUI.appearance.asElementsTheme) + verify(field) + } + + func testFilled() { + let field = OneTimeCodeTextField(numberOfDigits: 6, theme: LinkUI.appearance.asElementsTheme) + field.value = "123456" + verify(field) + } + + func verify( + _ view: UIView, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 300) + STPSnapshotVerifyView(view, file: file, line: line) + } +} diff --git a/Stripe/StripeiOSTests/OneTimeCodeTextFieldTests.swift b/Stripe/StripeiOSTests/OneTimeCodeTextFieldTests.swift new file mode 100644 index 00000000..17dd57dc --- /dev/null +++ b/Stripe/StripeiOSTests/OneTimeCodeTextFieldTests.swift @@ -0,0 +1,322 @@ +// +// OneTimeCodeTextFieldTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 11/5/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class OneTimeCodeTextFieldTests: XCTestCase { + + func test_isComplete() { + let field = makeSUT() + + field.value = "12345" + XCTAssertFalse(field.isComplete) + + field.value = "123456" + XCTAssertTrue(field.isComplete) + } + + func test_insertText() { + let field = makeSUT() + + field.insertText("1") + XCTAssertEqual(field.value, "1") + + field.insertText("2") + XCTAssertEqual(field.value, "12") + XCTAssertEqual(field.selectedTextRange?.start, field.endOfDocument) + XCTAssertEqual(field.selectedTextRange?.end, field.endOfDocument) + } + + func test_insertText_shouldNotInsertBeyondNumberOfDigits() { + let field = makeSUT(numberOfDigits: 4) + field.value = "123" + field.selectedTextRange = field.textRange( + from: field.endOfDocument, + to: field.endOfDocument + ) + field.insertText("45") + XCTAssertEqual(field.value, "1234") + } + + func test_insertText_shouldIgnoreInvalidCharacters() { + let field = makeSUT() + field.insertText("123-456") + XCTAssertEqual(field.value, "123456") + } + + func test_deleteBackward() throws { + let field = makeSUT(value: "12") + + field.selectedTextRange = field.textRange( + from: try XCTUnwrap(field.position(from: field.endOfDocument, in: .left, offset: 1)), + to: field.endOfDocument + ) + field.deleteBackward() + XCTAssertEqual(field.value, "1") + + field.selectedTextRange = field.textRange( + from: try XCTUnwrap(field.position(from: field.endOfDocument, in: .left, offset: 1)), + to: field.endOfDocument + ) + field.deleteBackward() + XCTAssertEqual(field.value, "") + + // Delete while empty + field.selectedTextRange = field.textRange( + from: field.beginningOfDocument, + to: field.endOfDocument + ) + field.deleteBackward() + XCTAssertEqual(field.value, "") + } + +// TODO(RUN_MOBILESDK-1848): This test is broken on iOS 16, as it invokes the pasteboard permission dialog +// func test_paste() { +// UIPasteboard.general.string = "123-456" +// +// let field = makeSUT() +// field.paste(nil) +// XCTAssertEqual(field.value, "123456") +// } + + // MARK: - UITextInput conformance + + func test_beginningOfDocument() throws { + let field = makeSUT(value: "123456") + + let position = try XCTUnwrap( + field.beginningOfDocument as? OneTimeCodeTextField.TextPosition + ) + XCTAssertEqual(position.index, 0) + } + + func test_endOfDocument() throws { + let field = makeSUT(value: "123456") + + let position = try XCTUnwrap(field.endOfDocument as? OneTimeCodeTextField.TextPosition) + XCTAssertEqual(position.index, 6) + } + + func test_textInRange() { + let field = makeSUT(value: "123456") + + let result = field.text( + in: OneTimeCodeTextField.TextRange( + start: OneTimeCodeTextField.TextPosition(0), + end: OneTimeCodeTextField.TextPosition(3) + ) + ) + + XCTAssertEqual(result, "123") + } + + func test_textInRange_emptyRange() { + let field = makeSUT(value: "123456") + + let result = field.text( + in: OneTimeCodeTextField.TextRange( + start: OneTimeCodeTextField.TextPosition(0), + end: OneTimeCodeTextField.TextPosition(0) + ) + ) + + XCTAssertNil(result) + } + + func test_positionFromOffset() { + let field = makeSUT(value: "123456") + + XCTAssertEqual( + field.position(from: field.beginningOfDocument, offset: 3), + OneTimeCodeTextField.TextPosition(3) + ) + + XCTAssertNil( + field.position(from: field.beginningOfDocument, offset: 10), + "Should return nil when offsetting to an out of bounds position" + ) + + XCTAssertNil( + field.position(from: field.beginningOfDocument, offset: -1), + "Should return nil when offsetting to an out of bounds position" + ) + } + + func test_positionInDirection() { + let field = makeSUT(value: "123456") + + XCTAssertEqual( + field.position(from: field.beginningOfDocument, in: .right, offset: 1), + OneTimeCodeTextField.TextPosition(1) + ) + + XCTAssertEqual( + field.position(from: field.endOfDocument, in: .left, offset: 1), + OneTimeCodeTextField.TextPosition(5) + ) + + // Y axis + XCTAssertEqual( + field.position(from: field.beginningOfDocument, in: .up, offset: 1), + field.beginningOfDocument + ) + + XCTAssertEqual( + field.position(from: field.beginningOfDocument, in: .down, offset: 1), + field.endOfDocument + ) + } + + func test_compare() { + let field = makeSUT(value: "123456") + + XCTAssertEqual( + field.compare(field.beginningOfDocument, to: field.beginningOfDocument), + .orderedSame + ) + + XCTAssertEqual( + field.compare(field.beginningOfDocument, to: field.endOfDocument), + .orderedAscending + ) + + XCTAssertEqual( + field.compare(field.endOfDocument, to: field.beginningOfDocument), + .orderedDescending + ) + } + + func test_offsetToPosition() { + let field = makeSUT(value: "123456") + + XCTAssertEqual(field.offset(from: field.beginningOfDocument, to: field.endOfDocument), 6) + XCTAssertEqual(field.offset(from: field.endOfDocument, to: field.beginningOfDocument), -6) + XCTAssertEqual( + field.offset(from: field.beginningOfDocument, to: OneTimeCodeTextField.TextPosition(3)), + 3 + ) + } + + func test_positionFarthestInDirection() throws { + let field = makeSUT(value: "123456") + + let position = try XCTUnwrap( + OneTimeCodeTextField.TextRange( + start: field.beginningOfDocument, + end: field.endOfDocument + ) + ) + + XCTAssertEqual( + field.position(within: position, farthestIn: .left), + field.beginningOfDocument + ) + + XCTAssertEqual( + field.position(within: position, farthestIn: .right), + field.endOfDocument + ) + + // Y axis + XCTAssertEqual( + field.position(within: position, farthestIn: .up), + field.beginningOfDocument + ) + + XCTAssertEqual( + field.position(within: position, farthestIn: .down), + field.endOfDocument + ) + } + + func test_characterRangeByExtendingInDirection() throws { + let field = makeSUT(value: "123456") + + let position = OneTimeCodeTextField.TextPosition(3) + + XCTAssertEqual( + field.characterRange(byExtending: position, in: .left), + OneTimeCodeTextField.TextRange(start: field.beginningOfDocument, end: position) + ) + + XCTAssertEqual( + field.characterRange(byExtending: position, in: .right), + OneTimeCodeTextField.TextRange(start: position, end: field.endOfDocument) + ) + + // Y axis + XCTAssertNil(field.characterRange(byExtending: position, in: .up)) + XCTAssertNil(field.characterRange(byExtending: position, in: .down)) + } + + func test_firstRectForRange_singleDigit() { + let sut = makeSUT(value: "123456") + + // A [0,1] text range + let range = OneTimeCodeTextField.TextRange( + start: OneTimeCodeTextField.TextPosition(0), + end: OneTimeCodeTextField.TextPosition(1) + ) + let rect = sut.firstRect(for: range) + XCTAssertEqual(rect.minX, 0, accuracy: 0.2) + XCTAssertEqual(rect.minY, 0, accuracy: 0.2) + XCTAssertEqual(rect.width, 46.0, accuracy: 0.2) + XCTAssertEqual(rect.height, 60, accuracy: 0.2) + } + + func test_firstRectForRange_multipleDigits() { + let sut = makeSUT(value: "123456") + + // A [0,3] Text range + let range = OneTimeCodeTextField.TextRange( + start: OneTimeCodeTextField.TextPosition(0), + end: OneTimeCodeTextField.TextPosition(3) + ) + let rect = sut.firstRect(for: range) + XCTAssertEqual(rect.minX, 0, accuracy: 0.2) + XCTAssertEqual(rect.minY, 0, accuracy: 0.2) + XCTAssertEqual(rect.width, 150, accuracy: 0.2) + XCTAssertEqual(rect.height, 60, accuracy: 0.2) + } + + func test_caretRectForPosition() { + let sut = makeSUT() + let frame = sut.caretRect(for: OneTimeCodeTextField.TextPosition(1)) + XCTAssertEqual(frame.minX, 74, accuracy: 0.2) + XCTAssertEqual(frame.minY, 18, accuracy: 0.2) + XCTAssertEqual(frame.width, 2, accuracy: 0.2) + XCTAssertEqual(frame.height, 24, accuracy: 0.2) + } + +} + +// MARK: - Factory methods + +extension OneTimeCodeTextFieldTests { + + fileprivate func makeSUT(numberOfDigits: Int = 6) -> OneTimeCodeTextField { + let sut = OneTimeCodeTextField(numberOfDigits: numberOfDigits, theme: LinkUI.appearance.asElementsTheme) + sut.frame = CGRect(x: 0, y: 0, width: 320, height: 60) + sut.layoutIfNeeded() + return sut + } + + fileprivate func makeSUT(value: String) -> OneTimeCodeTextField { + let sut = makeSUT() + sut.value = value + return sut + } + +} diff --git a/Stripe/StripeiOSTests/OperationDebouncerTests.swift b/Stripe/StripeiOSTests/OperationDebouncerTests.swift new file mode 100644 index 00000000..7a6f3ae0 --- /dev/null +++ b/Stripe/StripeiOSTests/OperationDebouncerTests.swift @@ -0,0 +1,45 @@ +// +// OperationDebouncerTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 1/23/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class OperationDebouncerTests: XCTestCase { + + func testEnqueueShouldDebounce() { + let sut = makeSUT() + + let expectation = self.expectation(description: "Should execute the block just once") + expectation.assertForOverFulfill = true + + // Call `enqueue(block:)` 3 times + for _ in 0..<3 { + sut.enqueue { + expectation.fulfill() + } + } + + Thread.sleep(forTimeInterval: 1) + + wait(for: [expectation], timeout: 1) + } + +} + +extension OperationDebouncerTests { + + func makeSUT() -> OperationDebouncer { + return OperationDebouncer(debounceTime: .milliseconds(500)) + } + +} diff --git a/Stripe/StripeiOSTests/PKPayment+StripeTest.swift b/Stripe/StripeiOSTests/PKPayment+StripeTest.swift new file mode 100644 index 00000000..c71eee52 --- /dev/null +++ b/Stripe/StripeiOSTests/PKPayment+StripeTest.swift @@ -0,0 +1,36 @@ +// +// PKPayment+StripeTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 7/6/15. +// Copyright © 2015 Stripe, Inc. All rights reserved. +// + +import PassKit +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class PKPayment_StripeTest: XCTestCase { + func testIsSimulated() { + let payment = PKPayment() + let paymentToken = PKPaymentToken() + + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wundeclared-selector" + paymentToken.perform(Selector(("setTransactionIdentifier:")), with: "Simulated Identifier") + payment.perform(#selector(setter: STPPaymentMethodCardParams.token), with: paymentToken) + // #pragma clang diagnostic pop + + XCTAssertTrue(payment.stp_isSimulated()) + } + + func testTransactionIdentifier() { + let identifier = PKPayment.stp_testTransactionIdentifier() + XCTAssertTrue(identifier.contains("ApplePayStubs~4242424242424242~0~USD~")) + } +} diff --git a/Stripe/StripeiOSTests/PayWithLinkButtonSnapshotTests.swift b/Stripe/StripeiOSTests/PayWithLinkButtonSnapshotTests.swift new file mode 100644 index 00000000..8567bf41 --- /dev/null +++ b/Stripe/StripeiOSTests/PayWithLinkButtonSnapshotTests.swift @@ -0,0 +1,111 @@ +// +// PayWithLinkButtonSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 11/17/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class PayWithLinkButtonSnapshotTests: FBSnapshotTestCase { + + private let emailAddress = "customer@example.com" + private let longEmailAddress = "long.customer.name@example.com" + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testDefault() { + let sut = makeSUT() + sut.linkAccount = makeAccountStub(email: emailAddress, isRegistered: false) + verify(sut) + + sut.isHighlighted = true + verify(sut, identifier: "Highlighted") + } + + func testDefault_rounded() { + let sut = makeSUT() + sut.cornerRadius = 16 + sut.linkAccount = makeAccountStub(email: emailAddress, isRegistered: false) + verify(sut) + } + + func testDisabled() { + let sut = makeSUT() + sut.isEnabled = false + verify(sut) + } + + func testRegistered() { + let sut = makeSUT() + sut.linkAccount = makeAccountStub(email: emailAddress, isRegistered: true) + verify(sut) + } + + func testRegistered_rounded() { + let sut = makeSUT() + sut.cornerRadius = 16 + sut.linkAccount = makeAccountStub(email: emailAddress, isRegistered: true) + verify(sut) + } + + func testRegistered_square() { + let sut = makeSUT() + sut.cornerRadius = 0 + sut.linkAccount = makeAccountStub(email: emailAddress, isRegistered: true) + verify(sut) + } + + func testRegistered_withLongEmailAddress() { + let sut = PayWithLinkButton() + sut.linkAccount = makeAccountStub(email: longEmailAddress, isRegistered: true) + verify(sut) + } + + func verify( + _ sut: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + sut.autosizeHeight(width: 300) + STPSnapshotVerifyView(sut, identifier: identifier, file: file, line: line) + } + +} + +extension PayWithLinkButtonSnapshotTests { + + fileprivate struct LinkAccountStub: PaymentSheetLinkAccountInfoProtocol { + let email: String + let redactedPhoneNumber: String? + let isRegistered: Bool + let isLoggedIn: Bool + } + + fileprivate func makeAccountStub(email: String, isRegistered: Bool) -> LinkAccountStub { + return LinkAccountStub( + email: email, + redactedPhoneNumber: "+1********55", + isRegistered: isRegistered, + isLoggedIn: false + ) + } + + fileprivate func makeSUT() -> PayWithLinkButton { + return PayWithLinkButton() + } + +} diff --git a/Stripe/StripeiOSTests/PayWithLinkViewController-WalletViewModelTests.swift b/Stripe/StripeiOSTests/PayWithLinkViewController-WalletViewModelTests.swift new file mode 100644 index 00000000..27bf3bf4 --- /dev/null +++ b/Stripe/StripeiOSTests/PayWithLinkViewController-WalletViewModelTests.swift @@ -0,0 +1,144 @@ +// +// PayWithLinkViewController-WalletViewModelTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 3/31/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class PayWithLinkViewController_WalletViewModelTests: XCTestCase { + + func test_shouldRecollectCardCVC() throws { + let sut = try makeSUT() + + // Card with passing CVC checks + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.card + XCTAssertFalse(sut.shouldRecollectCardCVC) + + // Card with failing CVC checks + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.cardWithFailingChecks + XCTAssertTrue( + sut.shouldRecollectCardCVC, + "Should recollect CVC when CVC checks are failing" + ) + + // Expired card + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.expiredCard + XCTAssertTrue(sut.shouldRecollectCardCVC, "Should recollect CVC when card has expired") + + // Bank account (CVC not supported) + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.bankAccount + XCTAssertFalse(sut.shouldRecollectCardCVC) + } + + func test_shouldRecollectCardExpiry() throws { + let sut = try makeSUT() + + // Non-expired card + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.card + XCTAssertFalse(sut.shouldRecollectCardExpiryDate) + + // Expired card + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.expiredCard + XCTAssertTrue( + sut.shouldRecollectCardExpiryDate, + "Should recollect new expiry date when card has expired" + ) + + // Bank account (CVC not supported) + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.bankAccount + XCTAssertFalse(sut.shouldRecollectCardCVC) + } + + func test_shouldShowInstantDebitMandate() throws { + let sut = try makeSUT() + + // Card + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.card + XCTAssertFalse(sut.shouldShowInstantDebitMandate) + + // Bank account + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.bankAccount + XCTAssertTrue(sut.shouldShowInstantDebitMandate) + } + + func test_confirmButtonStatus_shouldHandleNoSelection() throws { + let sut = try makeSUT() + + // No selection + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.notExisting + XCTAssertEqual( + sut.confirmButtonStatus, + .disabled, + "Button should be disabled when no payment method is selected" + ) + + // Selection + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.card + XCTAssertEqual(sut.confirmButtonStatus, .enabled) + } + + func test_confirmButtonStatus_shouldHandleCVCRecollectionRequirements() throws { + let sut = try makeSUT() + + sut.selectedPaymentMethodIndex = LinkStubs.PaymentMethodIndices.cardWithFailingChecks + XCTAssertEqual( + sut.confirmButtonStatus, + .disabled, + "Button should be disabled when no CVC is provided and a card with failing CVC checks is selected" + ) + + // Provide a CVC + sut.cvc = "123" + XCTAssertEqual(sut.confirmButtonStatus, .enabled) + } +} + +extension PayWithLinkViewController_WalletViewModelTests { + + func makeSUT() throws -> PayWithLinkViewController.WalletViewModel { + // Link settings don't live in the PaymentIntent object itself, but in the /elements/sessions API response + // So we construct a minimal response (see STPPaymentIntentTest.testDecodedObjectFromAPIResponseMapping) to parse them + let paymentIntentJson = try XCTUnwrap(STPTestUtils.jsonNamed(STPTestJSONPaymentIntent)) + let orderedPaymentJson = ["card", "link"] + let paymentIntentResponse = [ + "payment_intent": paymentIntentJson, + "ordered_payment_method_types": orderedPaymentJson, + ] as [String: Any] + let linkSettingsJson = ["link_funding_sources": ["CARD"]] + let response = [ + "payment_method_preference": paymentIntentResponse, + "link_settings": linkSettingsJson, + ] + let paymentIntent = try XCTUnwrap( + STPPaymentIntent.decodedObject(fromAPIResponse: response) + ) + + return PayWithLinkViewController.WalletViewModel( + // TODO(ramont): Fully mock `PaymentSheetLinkAccount and remove this. + linkAccount: .init( + email: "user@example.com", + session: LinkStubs.consumerSession(), + publishableKey: nil + ), + context: .init( + intent: .paymentIntent(paymentIntent), + configuration: .init(), + shouldOfferApplePay: false, + shouldFinishOnClose: false, + callToAction: nil + ), + paymentMethods: LinkStubs.paymentMethods() + ) + } + +} diff --git a/Stripe/StripeiOSTests/PaymentAnalyticTest.swift b/Stripe/StripeiOSTests/PaymentAnalyticTest.swift new file mode 100644 index 00000000..3b25a0ce --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentAnalyticTest.swift @@ -0,0 +1,33 @@ +// +// PaymentAnalyticTest.swift +// StripeiOS Tests +// +// Created by Mel Ludowise on 5/26/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +final class PaymentAnalyticTest: XCTestCase { + + func testParams() { + let analytic = GenericPaymentAnalytic( + event: .cardScanCancelled, + paymentConfiguration: STPPaymentConfiguration(), + productUsage: [ + STPPaymentContext.stp_analyticsIdentifier + ], + additionalParams: [:] + ) + + XCTAssertNotNil(analytic.params["apple_pay_enabled"] as? NSNumber) + XCTAssertNotNil(analytic.params["ocr_type"] as? String) + } +} diff --git a/Stripe/StripeiOSTests/PaymentMethodMessagingViewFunctionalTest.swift b/Stripe/StripeiOSTests/PaymentMethodMessagingViewFunctionalTest.swift new file mode 100644 index 00000000..b8b926b6 --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentMethodMessagingViewFunctionalTest.swift @@ -0,0 +1,81 @@ +// +// PaymentMethodMessagingViewFunctionalTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 9/28/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +@_spi(STP)@testable import Stripe +@_spi(STP)@testable import StripeCore +@_spi(STP)@testable import StripeCoreTestUtils +@_spi(STP)@testable import StripePaymentsUI +import XCTest + +class PaymentMethodMessagingViewFunctionalTest: STPNetworkStubbingTestCase { + let mockAnalyticsClient = MockAnalyticsClient() + + override func setUp() { +// recordingMode = true + super.setUp() + mockAnalyticsClient.reset() + PaymentMethodMessagingView.analyticsClient = mockAnalyticsClient + URLSession.shared.configuration.urlCache?.removeAllCachedResponses() + } + + func testCreatesViewFromServerResponse() { + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + let config = PaymentMethodMessagingView.Configuration( + apiClient: apiClient, + paymentMethods: PaymentMethodMessagingView.Configuration.PaymentMethod.allCases, + currency: "USD", + amount: 1099 + ) + let createViewExpectation = expectation(description: "") + PaymentMethodMessagingView.create(configuration: config) { result in + switch result { + case .failure(let error): + XCTFail(error.localizedDescription) + case .success(let view): + // We can't snapshot test the real view, since its appearance can change + XCTAssertTrue(view.label.attributedText?.length ?? 0 > 0) + XCTAssertTrue( + self.mockAnalyticsClient.productUsage.contains( + PaymentMethodMessagingView.stp_analyticsIdentifier + ) + ) + XCTAssertTrue( + self.mockAnalyticsClient.loggedAnalytics.contains { analytic in + analytic.event == .paymentMethodMessagingViewLoadSucceeded + } + ) + } + createViewExpectation.fulfill() + } + waitForExpectations(timeout: 10) + } + + func testInitializingWithBadConfigurationReturnsError() { + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + let config = PaymentMethodMessagingView.Configuration( + apiClient: apiClient, + paymentMethods: [.klarna], + currency: "FOO", + amount: -100 + ) + let createViewExpectation = expectation(description: "") + PaymentMethodMessagingView.create(configuration: config) { result in + guard case .failure = result else { + XCTFail() + return + } + XCTAssertTrue( + self.mockAnalyticsClient.loggedAnalytics.contains { analytic in + analytic.event == .paymentMethodMessagingViewLoadFailed + } + ) + createViewExpectation.fulfill() + } + waitForExpectations(timeout: 10) + } +} diff --git a/Stripe/StripeiOSTests/PaymentMethodMessagingViewSnapshotTests.swift b/Stripe/StripeiOSTests/PaymentMethodMessagingViewSnapshotTests.swift new file mode 100644 index 00000000..034cf1df --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentMethodMessagingViewSnapshotTests.swift @@ -0,0 +1,106 @@ +// +// PaymentMethodMessagingViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 9/26/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +@_spi(STP)@testable import StripePaymentsUI +import UIKit + +@MainActor +class PaymentMethodMessagingViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() +// self.recordMode = true + } + + /// - Note: This mock HTML should include all HTML tags the server can send down + let mockHTML = + """ + +
+ As low as 4 interest-free payments of $24.75 🎉 + """ + var configuration: PaymentMethodMessagingView.Configuration { + return .init(paymentMethods: [.afterpayClearpay, .klarna], currency: "USD", amount: 100) + } + + func testDefaults() async { + guard + let mockAttributedString = try? await PaymentMethodMessagingView.makeAttributedString( + from: mockHTML, + configuration: configuration + ) + else { + XCTFail() + return + } + let view = PaymentMethodMessagingView( + attributedString: mockAttributedString, + modalURL: "https://stripe.com", + configuration: configuration + ) + verify(view) + } + + func testCustomFontAndCustomTextColor() async { + var configuration = configuration + configuration.font = UIFont(name: "AmericanTypewriter", size: 10)! + configuration.textColor = .darkGray + guard + let mockAttributedString = try? await PaymentMethodMessagingView.makeAttributedString( + from: self.mockHTML, + configuration: configuration + ) + else { + XCTFail() + return + } + let view = PaymentMethodMessagingView( + attributedString: mockAttributedString, + modalURL: "https://stripe.com", + configuration: configuration + ) + verify(view) + } + + // Remove _ to snapshot a view using the production endpoint + // We can't test this, even if we stub the network response, because the response contains image URLs + func _testReal() { + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + let config = PaymentMethodMessagingView.Configuration( + apiClient: apiClient, + paymentMethods: PaymentMethodMessagingView.Configuration.PaymentMethod.allCases, + currency: "USD", + amount: 1099 + ) + let createViewExpectation = expectation(description: "") + PaymentMethodMessagingView.create(configuration: config) { [weak self] result in + switch result { + case .failure(let error): + XCTFail(error.localizedDescription) + case .success(let view): + self?.verify(view) + } + createViewExpectation.fulfill() + } + waitForExpectations(timeout: 10) + } + + // MARK: - Helpers + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } +} diff --git a/Stripe/StripeiOSTests/PaymentSheet+APITest.swift b/Stripe/StripeiOSTests/PaymentSheet+APITest.swift new file mode 100644 index 00000000..5787173e --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentSheet+APITest.swift @@ -0,0 +1,671 @@ +// +// PaymentSheet+APITest.swift +// StripeiOS Tests +// +// Created by Jaime Park on 6/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) @_spi(ExperimentalPaymentSheetDecouplingAPI) import StripePaymentSheet + +class PaymentSheetAPITest: XCTestCase { + + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + lazy var paymentHandler: STPPaymentHandler = { + return STPPaymentHandler( + apiClient: apiClient, + formSpecPaymentHandler: PaymentSheetFormSpecPaymentHandler() + ) + }() + lazy var configuration: PaymentSheet.Configuration = { + var config = PaymentSheet.Configuration() + config.apiClient = apiClient + config.allowsDelayedPaymentMethods = true + config.shippingDetails = { + return .init( + address: .init( + country: "US", + line1: "Line 1" + ), + name: "Jane Doe", + phone: "5551234567" + ) + } + return config + }() + + lazy var newCardPaymentOption: PaymentSheet.PaymentOption = { + let cardParams = STPPaymentMethodCardParams() + cardParams.number = "4242424242424242" + cardParams.cvc = "123" + cardParams.expYear = 32 + cardParams.expMonth = 12 + let newCardPaymentOption: PaymentSheet.PaymentOption = .new( + confirmParams: .init( + params: .init( + card: cardParams, + billingDetails: .init(), + metadata: nil + ), + type: .card + ) + ) + + return newCardPaymentOption + }() + + override class func setUp() { + super.setUp() + // `PaymentSheet.load()` uses the `LinkAccountService` to lookup the Link user account. + // Override the default cookie store since Keychain is not available in this test case. + LinkAccountService.defaultCookieStore = LinkInMemoryCookieStore() + } + + // MARK: - load and confirm tests + + func testPaymentSheetLoadAndConfirmWithPaymentIntent() { + let expectation = XCTestExpectation(description: "Retrieve Payment Intent With Preferences") + let types = ["ideal", "card", "bancontact", "sofort"] + let expected = [.card, .iDEAL, .bancontact, .sofort] + .filter { PaymentSheet.supportedPaymentMethods.contains($0) } + + // 0. Create a PI on our test backend + fetchPaymentIntent(types: types) { result in + switch result { + case .success(let clientSecret): + // 1. Load the PI + PaymentSheet.load( + mode: .paymentIntentClientSecret(clientSecret), + configuration: self.configuration + ) { result in + switch result { + case .success(let paymentIntent, let paymentMethods, _): + XCTAssertEqual( + Set(paymentIntent.recommendedPaymentMethodTypes), + Set(expected) + ) + XCTAssertEqual(paymentMethods, []) + // 2. Confirm the intent with a new card + + PaymentSheet.confirm( + configuration: self.configuration, + authenticationContext: self, + intent: paymentIntent, + paymentOption: self.newCardPaymentOption, + paymentHandler: self.paymentHandler + ) { result in + switch result { + case .completed: + // 3. Fetch the PI + self.apiClient.retrievePaymentIntent(withClientSecret: clientSecret) + { paymentIntent, _ in + // Make sure the PI is succeeded and contains shipping + XCTAssertNotNil(paymentIntent?.shipping) + XCTAssertEqual( + paymentIntent?.shipping?.name, + self.configuration.shippingDetails()?.name + ) + XCTAssertEqual( + paymentIntent?.shipping?.phone, + self.configuration.shippingDetails()?.phone + ) + XCTAssertEqual( + paymentIntent?.shipping?.address?.line1, + self.configuration.shippingDetails()?.address.line1 + ) + XCTAssertEqual(paymentIntent?.status, .succeeded) + } + case .canceled: + XCTFail("Confirm canceled") + case .failed(let error): + XCTFail("Failed to confirm: \(error)") + } + expectation.fulfill() + } + case .failure(let error): + print(error) + } + } + + case .failure(let error): + print(error) + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testPaymentSheetLoadWithSetupIntent() { + let expectation = XCTestExpectation(description: "Retrieve Setup Intent With Preferences") + let types = ["ideal", "card", "bancontact", "sofort"] + let expected: [STPPaymentMethodType] = [.card, .iDEAL, .bancontact, .sofort] + fetchSetupIntent(types: types) { result in + switch result { + case .success(let clientSecret): + PaymentSheet.load( + mode: .setupIntentClientSecret(clientSecret), + configuration: self.configuration + ) { result in + switch result { + case .success(let setupIntent, let paymentMethods, _): + XCTAssertEqual( + Set(setupIntent.recommendedPaymentMethodTypes), + Set(expected) + ) + XCTAssertEqual(paymentMethods, []) + expectation.fulfill() + case .failure(let error): + print(error) + } + } + + case .failure(let error): + print(error) + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testPaymentSheetLoadAndConfirmWithDeferredIntent() { + let loadExpectation = XCTestExpectation(description: "Load PaymentSheet") + let confirmExpectation = XCTestExpectation(description: "Confirm deferred intent") + let callbackExpectation = XCTestExpectation(description: "Confirm callback invoked") + + let types = ["card", "cashapp"] + let expected: [STPPaymentMethodType] = [.card, .cashApp] + let confirmHandler: PaymentSheet.IntentConfiguration.ConfirmHandler = {_, intentCreationCallback in + self.fetchPaymentIntent(types: types, currency: "USD") { result in + switch result { + case .success(let clientSecret): + intentCreationCallback(.success(clientSecret)) + callbackExpectation.fulfill() + case .failure(let error): + print(error) + } + } + } + let intentConfig = PaymentSheet.IntentConfiguration(mode: .payment(amount: 1000, + currency: "USD", + setupFutureUsage: .onSession), + paymentMethodTypes: types, + confirmHandler: confirmHandler) + PaymentSheet.load( + mode: .deferredIntent(intentConfig), + configuration: self.configuration + ) { result in + switch result { + case .success(let intent, let paymentMethods, _): + XCTAssertEqual( + Set(intent.recommendedPaymentMethodTypes), + Set(expected) + ) + XCTAssertEqual(paymentMethods, []) + loadExpectation.fulfill() + guard case .deferredIntent(elementsSession: let elementsSession, intentConfig: _) = intent else { + XCTFail() + return + } + + PaymentSheet.confirm(configuration: self.configuration, + authenticationContext: self, + intent: .deferredIntent(elementsSession: elementsSession, + intentConfig: intentConfig), + paymentOption: self.newCardPaymentOption, + paymentHandler: self.paymentHandler) { result in + switch result { + case .completed: + confirmExpectation.fulfill() + case .canceled: + XCTFail("Confirm canceled") + case .failed(let error): + XCTFail("Failed to confirm: \(error)") + } + } + + case .failure(let error): + print(error) + } + } + + wait(for: [loadExpectation, confirmExpectation, callbackExpectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testPaymentSheetLoadAndConfirmWithDeferredIntent_serverSideConfirmation() { + let loadExpectation = XCTestExpectation(description: "Load PaymentSheet") + let confirmExpectation = XCTestExpectation(description: "Confirm deferred intent") + let callbackExpectation = XCTestExpectation(description: "Confirm callback invoked") + + let types = ["card", "cashapp"] + let expected: [STPPaymentMethodType] = [.card, .cashApp] + let serverSideConfirmHandler: PaymentSheet.IntentConfiguration.ConfirmHandlerForServerSideConfirmation = {paymentMethodID, _, intentCreationCallback in + self.fetchPaymentIntent(types: types, + currency: "USD", + paymentMethodID: paymentMethodID, + confirm: true) { result in + switch result { + case .success(let clientSecret): + intentCreationCallback(.success(clientSecret)) + callbackExpectation.fulfill() + case .failure(let error): + print(error) + } + } + } + let intentConfig = PaymentSheet.IntentConfiguration(mode: .payment(amount: 1000, + currency: "USD", + setupFutureUsage: .onSession), + paymentMethodTypes: types, + confirmHandlerForServerSideConfirmation: serverSideConfirmHandler) + PaymentSheet.load( + mode: .deferredIntent(intentConfig), + configuration: self.configuration + ) { result in + switch result { + case .success(let intent, let paymentMethods, _): + XCTAssertEqual( + Set(intent.recommendedPaymentMethodTypes), + Set(expected) + ) + XCTAssertEqual(paymentMethods, []) + loadExpectation.fulfill() + guard case .deferredIntent(elementsSession: let elementsSession, intentConfig: _) = intent else { + XCTFail() + return + } + + PaymentSheet.confirm(configuration: self.configuration, + authenticationContext: self, + intent: .deferredIntent(elementsSession: elementsSession, + intentConfig: intentConfig), + paymentOption: self.newCardPaymentOption, + paymentHandler: self.paymentHandler) { result in + switch result { + case .completed: + confirmExpectation.fulfill() + case .canceled: + XCTFail("Confirm canceled") + case .failed(let error): + XCTFail("Failed to confirm: \(error)") + } + } + + case .failure(let error): + print(error) + } + } + + wait(for: [loadExpectation, confirmExpectation, callbackExpectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testPaymentSheetLoadAndConfirmWithPaymentIntentAttachedPaymentMethod() { + let expectation = XCTestExpectation( + description: "Load PaymentIntent with an attached payment method" + ) + // 0. Create a PI on our test backend with an already attached pm + STPTestingAPIClient.shared().createPaymentIntent(withParams: [ + "amount": 1050, + "payment_method": "pm_card_visa", + ]) { clientSecret, error in + guard let clientSecret = clientSecret, error == nil else { + XCTFail() + return + } + + // 1. Load the PI + PaymentSheet.load( + mode: .paymentIntentClientSecret(clientSecret), + configuration: self.configuration + ) { result in + guard case .success(let paymentIntent, _, _) = result else { + XCTFail() + return + } + // 2. Confirm with saved card + PaymentSheet.confirm( + configuration: self.configuration, + authenticationContext: self, + intent: paymentIntent, + paymentOption: .saved(paymentMethod: .init(stripeId: "pm_card_visa")), + paymentHandler: self.paymentHandler + ) { result in + switch result { + case .completed: + // 3. Fetch the PI + self.apiClient.retrievePaymentIntent(withClientSecret: clientSecret) { + paymentIntent, + _ in + // Make sure the PI is succeeded and contains shipping + XCTAssertNotNil(paymentIntent?.shipping) + XCTAssertEqual( + paymentIntent?.shipping?.name, + self.configuration.shippingDetails()?.name + ) + XCTAssertEqual( + paymentIntent?.shipping?.phone, + self.configuration.shippingDetails()?.phone + ) + XCTAssertEqual( + paymentIntent?.shipping?.address?.line1, + self.configuration.shippingDetails()?.address.line1 + ) + XCTAssertEqual(paymentIntent?.status, .succeeded) + expectation.fulfill() + } + case .canceled: + XCTFail("Confirm canceled") + case .failed(let error): + XCTFail("Failed to confirm: \(error)") + } + } + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testPaymentSheetLoadWithSetupIntentAttachedPaymentMethod() { + let expectation = XCTestExpectation( + description: "Load SetupIntent with an attached payment method" + ) + STPTestingAPIClient.shared().createSetupIntent(withParams: [ + "payment_method": "pm_card_visa", + ]) { clientSecret, error in + guard let clientSecret = clientSecret, error == nil else { + XCTFail() + expectation.fulfill() + return + } + + PaymentSheet.load( + mode: .setupIntentClientSecret(clientSecret), + configuration: self.configuration + ) { result in + defer { expectation.fulfill() } + guard case .success = result else { + XCTFail() + return + } + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + // MARK: - update tests + + func testUpdate() { + var intentConfig = PaymentSheet.IntentConfiguration(mode: .payment(amount: 1000, currency: "USD")) { _, _ in + // These tests don't confirm, so this is unused + } + let firstUpdateExpectation = expectation(description: "First update completes") + let secondUpdateExpectation = expectation(description: "Second update completes") + // Given a PaymentSheet.FlowController instance... + PaymentSheet.FlowController.create(intentConfiguration: intentConfig, configuration: configuration) { result in + switch result { + case .success(let sut): + // ...the vc's intent should match the initial intent config... + XCTAssertFalse(sut.viewController.intent.isSettingUp) + XCTAssertTrue(sut.viewController.intent.isPaymentIntent) + // ...and updating the intent config should succeed... + intentConfig.mode = .setup(currency: nil, setupFutureUsage: .offSession) + sut.update(intentConfiguration: intentConfig) { error in + XCTAssertNil(error) + XCTAssertNil(sut.paymentOption) + XCTAssertTrue(sut.viewController.intent.isSettingUp) + XCTAssertFalse(sut.viewController.intent.isPaymentIntent) + firstUpdateExpectation.fulfill() + + // ...updating the intent config multiple times should succeed... + intentConfig.mode = .payment(amount: 100, currency: "USD", setupFutureUsage: nil) + sut.update(intentConfiguration: intentConfig) { error in + XCTAssertNil(error) + XCTAssertNil(sut.paymentOption) + XCTAssertFalse(sut.viewController.intent.isSettingUp) + XCTAssertTrue(sut.viewController.intent.isPaymentIntent) + secondUpdateExpectation.fulfill() + } + } + case .failure(let error): + XCTFail(error.nonGenericDescription) + } + } + waitForExpectations(timeout: 10) + } + + func testUpdateFails() { + var intentConfig = PaymentSheet.IntentConfiguration(mode: .payment(amount: 1000, currency: "USD")) { _, _ in + // These tests don't confirm, so this is unused + } + + let failedUpdateExpectation = expectation(description: "First update fails") + let secondUpdateExpectation = expectation(description: "Second update succeeds") + PaymentSheet.FlowController.create(intentConfiguration: intentConfig, configuration: configuration) { result in + switch result { + case .success(let sut): + // ...updating w/ an invalid intent config should fail... + intentConfig.mode = .setup(currency: "Invalid currency", setupFutureUsage: .offSession) + sut.update(intentConfiguration: intentConfig) { updateError in + XCTAssertNotNil(updateError) + // ...the paymentOption should be nil... + XCTAssertNil(sut.paymentOption) + failedUpdateExpectation.fulfill() + // Note: `confirm` has an assertionFailure if paymentOption is nil, so we don't check it here. + + // ...updating should succeed after failing to update + intentConfig.mode = .setup(currency: "USD", setupFutureUsage: .offSession) + sut.update(intentConfiguration: intentConfig) { error in + XCTAssertNil(error) + // TODO(Update:) Change this to validate it preserves the paymentOption + XCTAssertNil(sut.paymentOption) + secondUpdateExpectation.fulfill() + } + } + case .failure(let error): + XCTFail(error.localizedDescription) + } + } + waitForExpectations(timeout: 10) + } + + func testUpdateIgnoresInFlightUpdate() { + var intentConfig = PaymentSheet.IntentConfiguration(mode: .payment(amount: 1000, currency: "USD")) { _, _ in + // These tests don't confirm, so this is unused + } + + let firstUpdateExpectation = expectation(description: "First update should not invoke callback") + firstUpdateExpectation.isInverted = true + let secondUpdateExpectation = expectation(description: "Second update succeeds") + var flowController: PaymentSheet.FlowController! + PaymentSheet.FlowController.create(intentConfiguration: intentConfig, configuration: configuration) { result in + switch result { + case .success(let sut): + flowController = sut + flowController.update(intentConfiguration: intentConfig) { _ in + firstUpdateExpectation.fulfill() + } + + intentConfig.mode = .setup(currency: "USD", setupFutureUsage: .offSession) + flowController.update(intentConfiguration: intentConfig) { error in + XCTAssertNil(error) + // TODO(Update:) Change this to validate it preserves the paymentOption + XCTAssertNil(flowController.paymentOption) + secondUpdateExpectation.fulfill() + } + case .failure(let error): + XCTFail(error.localizedDescription) + } + } + waitForExpectations(timeout: 10) + } + + // MARK: - other tests + + func testMakeShippingParamsReturnsNilIfPaymentIntentHasDifferentShipping() { + // Given a PI with shipping... + let pi = STPFixtures.paymentIntent() + guard let shipping = pi.shipping else { + XCTFail("PI should contain shipping") + return + } + // ...and a configuration with *a different* shipping + var config = configuration + // ...PaymentSheet should set shipping params on /confirm + XCTAssertNotNil(PaymentSheet.makeShippingParams(for: pi, configuration: config)) + + // However, if the PI and config have the same shipping... + config.shippingDetails = { + return .init( + address: AddressViewController.AddressDetails.Address( + city: shipping.address?.city, + country: shipping.address?.country ?? "pi.shipping is missing country", + line1: shipping.address?.line1 ?? "pi.shipping is missing line1", + line2: shipping.address?.line2, + postalCode: shipping.address?.postalCode, + state: shipping.address?.state + ), + name: pi.shipping?.name, + phone: pi.shipping?.phone + ) + } + // ...PaymentSheet should not set shipping params on /confirm + XCTAssertNil(PaymentSheet.makeShippingParams(for: pi, configuration: config)) + } + + /// Setting SFU to `true` when a customer is set should set the parameter to `off_session`. + func testPaymentIntentParamsWithSFUTrueAndCustomer() { + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "") + paymentIntentParams.paymentMethodOptions = STPConfirmPaymentMethodOptions() + paymentIntentParams.paymentMethodOptions?.setSetupFutureUsageIfNecessary( + true, + paymentMethodType: PaymentSheet.PaymentMethodType.card, + customer: .init(id: "", ephemeralKeySecret: "") + ) + + let params = STPFormEncoder.dictionary(forObject: paymentIntentParams) + guard + let paymentMethodOptions = params["payment_method_options"] as? [String: Any], + let card = paymentMethodOptions["card"] as? [String: Any], + let setupFutureUsage = card["setup_future_usage"] as? String + else { + XCTFail("Incorrect params") + return + } + + XCTAssertEqual(setupFutureUsage, "off_session") + } + + /// Setting SFU to `false` when a customer is set should set the parameter to an empty string. + func testPaymentIntentParamsWithSFUFalseAndCustomer() { + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "") + paymentIntentParams.paymentMethodOptions = STPConfirmPaymentMethodOptions() + paymentIntentParams.paymentMethodOptions?.setSetupFutureUsageIfNecessary( + false, + paymentMethodType: PaymentSheet.PaymentMethodType.card, + customer: .init(id: "", ephemeralKeySecret: "") + ) + + let params = STPFormEncoder.dictionary(forObject: paymentIntentParams) + guard + let paymentMethodOptions = params["payment_method_options"] as? [String: Any], + let card = paymentMethodOptions["card"] as? [String: Any], + let setupFutureUsage = card["setup_future_usage"] as? String + else { + XCTFail("Incorrect params") + return + } + + XCTAssertEqual(setupFutureUsage, "") + } + + /// Setting SFU to `true` when no customer is set shouldn't set the parameter. + func testPaymentIntentParamsWithSFUTrueAndNoCustomer() { + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "") + paymentIntentParams.paymentMethodOptions = STPConfirmPaymentMethodOptions() + paymentIntentParams.paymentMethodOptions?.setSetupFutureUsageIfNecessary( + false, + paymentMethodType: PaymentSheet.PaymentMethodType.card, + customer: nil + ) + + let params = STPFormEncoder.dictionary(forObject: paymentIntentParams) + XCTAssertEqual((params["payment_method_options"] as! [String: Any]).count, 0) + } + + /// Setting SFU to `false` when no customer is set shouldn't set the parameter. + func testPaymentIntentParamsWithSFUFalseAndNoCustomer() { + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "") + paymentIntentParams.paymentMethodOptions = STPConfirmPaymentMethodOptions() + paymentIntentParams.paymentMethodOptions?.setSetupFutureUsageIfNecessary( + false, + paymentMethodType: PaymentSheet.PaymentMethodType.card, + customer: nil + ) + + let params = STPFormEncoder.dictionary(forObject: paymentIntentParams) + XCTAssertEqual((params["payment_method_options"] as! [String: Any]).count, 0) + } + + // MARK: - helper methods + + func fetchPaymentIntent( + types: [String], + currency: String = "eur", + paymentMethodID: String? = nil, + confirm: Bool = false, + completion: @escaping (Result<(String), Error>) -> Void + ) { + var params = [String: Any]() + params["amount"] = 1050 + params["currency"] = currency + params["payment_method_types"] = types + params["confirm"] = confirm + if let paymentMethodID = paymentMethodID { + params["payment_method"] = paymentMethodID + } + + STPTestingAPIClient + .shared() + .createPaymentIntent( + withParams: params + ) { clientSecret, error in + guard let clientSecret = clientSecret, + error == nil + else { + completion(.failure(error!)) + return + } + + completion(.success(clientSecret)) + } + } + + func fetchSetupIntent(types: [String], completion: @escaping (Result<(String), Error>) -> Void) + { + STPTestingAPIClient + .shared() + .createSetupIntent( + withParams: [ + "payment_method_types": types, + ] + ) { clientSecret, error in + guard let clientSecret = clientSecret, + error == nil + else { + completion(.failure(error!)) + return + } + + completion(.success(clientSecret)) + } + } + +} + +extension PaymentSheetAPITest: STPAuthenticationContext { + func authenticationPresentingViewController() -> UIViewController { + return UIViewController() + } +} diff --git a/Stripe/StripeiOSTests/PaymentSheetAddressTests.swift b/Stripe/StripeiOSTests/PaymentSheetAddressTests.swift new file mode 100644 index 00000000..e14a5d8a --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentSheetAddressTests.swift @@ -0,0 +1,312 @@ +// +// PaymentSheetAddressTests.swift +// StripeiOS Tests +// +// Created by Nick Porter on 7/25/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable import Stripe +@testable@_spi(STP) import StripePaymentSheet + +class PaymentSheetAddressTests: XCTestCase { + + func testEditDistanceEqualAddress() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: address), 0) + } + + func testEditDistanceOneCharDiff() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "Sa Francisco", // One char diff here + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 1) + } + + func testEditDistanceDifferentCity() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "Freemont", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 11) + } + + func testEditDistanceMissingCityOriginal() { + let address = PaymentSheet.Address( + city: nil, + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 13) + } + + func testEditDistanceMissingCityOther() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: nil, + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 13) + } + + func testEditDistanceMissingCountryOriginal() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: nil, + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 2) + } + + func testEditDistanceMissingCountryOther() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: nil, + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 2) + } + + func testEditDistanceMissingLineOneOriginal() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: nil, + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 16) + } + + func testEditDistanceMissingLineOneOther() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: nil, + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 16) + } + + func testEditDistanceMissingLineTwoOriginal() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + line2: nil, + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + line2: "Apt. 112", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 8) + } + + func testEditDistanceMissingLineTwoOther() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + line2: "Apt. 112", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + line2: nil, + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 8) + } + + func testEditDistanceMissingPostalCodeOriginal() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: nil, + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 5) + } + + func testEditDistanceMissingPostalCodeOther() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: nil, + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 5) + } + + func testEditDistanceMissingStateOriginal() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: nil + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 10) + } + + func testEditDistanceMissingStateOther() { + let address = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: "California" + ) + + let otherAddress = PaymentSheet.Address( + city: "San Francisco", + country: "AT", + line1: "510 Townsend St.", + postalCode: "94102", + state: nil + ) + + XCTAssertEqual(address.editDistance(from: otherAddress), 10) + } + +} diff --git a/Stripe/StripeiOSTests/PaymentSheetFormFactorySnapshotTest.swift b/Stripe/StripeiOSTests/PaymentSheetFormFactorySnapshotTest.swift new file mode 100644 index 00000000..2ff0cf82 --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentSheetFormFactorySnapshotTest.swift @@ -0,0 +1,437 @@ +// +// PaymentSheetFormFactorySnapshotTest.swift +// StripeiOSTests +// +// Created by Eduardo Urias on 2/23/23. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +@_spi(STP) @testable import StripePaymentSheet +@_spi(STP) @testable import StripeUICore +import XCTest + +final class PaymentSheetFormFactorySnapshotTest: FBSnapshotTestCase { + override func setUp() { + super.setUp() +// recordMode = true + } + + func testCard_AutomaticFields_NoDefaults() { + let configuration = PaymentSheet.Configuration() + let factory = factory(for: .card, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testCard_AutomaticFields_DefaultAddress() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.address = defaultAddress + let factory = factory(for: .card, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testCard_AllFields_NoDefaults() { + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory(for: .card, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testCard_AllFields_WithDefaults() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory(for: .card, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testCard_CardInfoOnly() { + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .never + configuration.billingDetailsCollectionConfiguration.email = .never + configuration.billingDetailsCollectionConfiguration.phone = .never + configuration.billingDetailsCollectionConfiguration.address = .never + let factory = factory(for: .card, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testCard_CardInfoWithName() { + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .never + configuration.billingDetailsCollectionConfiguration.phone = .never + configuration.billingDetailsCollectionConfiguration.address = .never + let factory = factory(for: .card, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUSBankAccount_AutomaticFields_NoDefaults() { + let configuration = PaymentSheet.Configuration() + let factory = factory(for: .USBankAccount, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUSBankAccount_AutomaticFields_WithDefaults() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.billingDetailsCollectionConfiguration.attachDefaultsToPaymentMethod = true + let factory = factory(for: .USBankAccount, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUSBankAccount_AllFields_NoDefaults() { + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory(for: .USBankAccount, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUSBankAccount_AllFields_WithDefaults() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory(for: .USBankAccount, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUSBankAccount_NoFields() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.billingDetailsCollectionConfiguration.attachDefaultsToPaymentMethod = true + configuration.billingDetailsCollectionConfiguration.name = .never + configuration.billingDetailsCollectionConfiguration.email = .never + configuration.billingDetailsCollectionConfiguration.phone = .never + configuration.billingDetailsCollectionConfiguration.address = .never + let factory = factory(for: .USBankAccount, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUpi_AutomaticFields() { + let configuration = PaymentSheet.Configuration() + let factory = factory(for: .UPI, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUpi_AllFields_NoDefaults() { + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory(for: .UPI, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUpi_AllFields_WithDefaults() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory(for: .UPI, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testUpi_SomeFields_NoDefaults() { + // Same result as automatic fields. + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .never + configuration.billingDetailsCollectionConfiguration.address = .never + let factory = factory(for: .UPI, configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Afterpay_AutomaticFields_NoDefaults() { + loadSpecs() + + let configuration = PaymentSheet.Configuration() + let factory = factory( + for: .dynamic("afterpay_clearpay"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Afterpay_AllFields_NoDefaults() { + loadSpecs() + + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory( + for: .dynamic("afterpay_clearpay"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Afterpay_AllFields_WithDefaults() { + loadSpecs() + + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory( + for: .dynamic("afterpay_clearpay"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Afterpay_MinimalFields() { + loadSpecs() + + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .never + configuration.billingDetailsCollectionConfiguration.email = .never + configuration.billingDetailsCollectionConfiguration.phone = .never + configuration.billingDetailsCollectionConfiguration.address = .never + let factory = factory( + for: .dynamic("afterpay_clearpay"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Klarna_AutomaticFields_NoDefaults() { + loadSpecs() + + let configuration = PaymentSheet.Configuration() + let factory = factory( + for: .dynamic("klarna"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Klarna_AllFields_NoDefaults() { + loadSpecs() + + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory( + for: .dynamic("klarna"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Klarna_AllFields_WithDefaults() { + loadSpecs() + + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + let factory = factory( + for: .dynamic("klarna"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + + func testLpm_Klarna_MinimalFields() { + loadSpecs() + + var configuration = PaymentSheet.Configuration() + configuration.billingDetailsCollectionConfiguration.name = .never + configuration.billingDetailsCollectionConfiguration.email = .never + configuration.billingDetailsCollectionConfiguration.phone = .never + configuration.billingDetailsCollectionConfiguration.address = .never + let factory = factory( + for: .dynamic("klarna"), + configuration: configuration) + let formElement = factory.make() + let view = formElement.view + view.autosizeHeight(width: 375) + STPSnapshotVerifyView(view) + } + +} + +extension PaymentSheetFormFactorySnapshotTest { + private func usAddressSpecProvider() -> AddressSpecProvider { + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + return specProvider + } + + private func factory( + for paymentMethodType: PaymentSheet.PaymentMethodType, + configuration: PaymentSheet.Configuration + ) -> PaymentSheetFormFactory { + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [paymentMethodType.stpPaymentMethodType!]) + return PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: paymentMethodType, + addressSpecProvider: usAddressSpecProvider() + ) + } + + private func loadSpecs() { + let expectation = expectation(description: "FormSpecs loaded") + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + wait(for: [expectation], timeout: 5.0) + } +} diff --git a/Stripe/StripeiOSTests/PaymentSheetFormFactoryTest.swift b/Stripe/StripeiOSTests/PaymentSheetFormFactoryTest.swift new file mode 100644 index 00000000..05ed693e --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentSheetFormFactoryTest.swift @@ -0,0 +1,1907 @@ +// +// PaymentSheetFormFactoryTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/19/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore +@testable@_spi(STP) import StripeUICore + +class MockElement: Element { + var paramsUpdater: (IntentConfirmParams) -> IntentConfirmParams? + + init( + paramsUpdater: @escaping (IntentConfirmParams) -> IntentConfirmParams? + ) { + self.paramsUpdater = paramsUpdater + } + + func updateParams(params: IntentConfirmParams) -> IntentConfirmParams? { + return paramsUpdater(params) + } + + weak var delegate: ElementDelegate? + lazy var view: UIView = { UIView() }() +} + +class PaymentSheetFormFactoryTest: XCTestCase { + func testUpdatesParams() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "Name" + configuration.defaultBillingDetails.email = "email@stripe.com" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit") + ) + let name = factory.makeName() + let email = factory.makeEmail() + let checkbox = factory.makeSaveCheckbox { _ in } + + let form = FormElement(elements: [name, email, checkbox]) + let params = form.updateParams(params: IntentConfirmParams(type: .dynamic("sepa_debit"))) + + XCTAssertEqual(params?.paymentMethodParams.billingDetails?.name, "Name") + XCTAssertEqual(params?.paymentMethodParams.billingDetails?.email, "email@stripe.com") + XCTAssertEqual(params?.paymentMethodParams.type, .SEPADebit) + XCTAssertEqual(params?.paymentMethodType, .dynamic("sepa_debit")) + } + + func testSpecFromJSONProvider() { + let e = expectation(description: "Loads form specs file") + let provider = FormSpecProvider() + provider.load { loaded in + XCTAssertTrue(loaded) + e.fulfill() + } + waitForExpectations(timeout: 1, handler: nil) + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("eps") + ) + + guard let spec = factory.specFromJSONProvider(provider: provider) else { + XCTFail("Unable to load EPS Spec") + return + } + + XCTAssertEqual(spec.fields.count, 5) + XCTAssertEqual( + spec.fields.first, + .name(FormSpec.NameFieldSpec(apiPath: ["v1": "billing_details[name]"], translationId: nil)) + ) + XCTAssertEqual(spec.type, "eps") + } + + func testNameOverrideApiPathBySpec() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "someName" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let name = factory.makeName(apiPath: "custom_location[name]") + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = name.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.billingDetails?.name) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[name]"] + as! String, + "someName" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + + // Using the params as previous customer input... + let name_with_previous_customer_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method"), + previousCustomerInput: updatedParams + ).makeName(apiPath: "custom_location[name]") + // ...should result in a valid element filled out with the previous customer input + XCTAssertEqual(name_with_previous_customer_input.element.text, "someName") + XCTAssertEqual(name_with_previous_customer_input.validationState, .valid) + } + + func testNameValueWrittenToDefaultLocation() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "someName" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let name = factory.makeName() + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = name.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.name, "someName") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[name]"] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + + // Using the params as previous customer input... + let name_with_previous_customer_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method"), + previousCustomerInput: updatedParams + ).makeName() + // ...should result in a valid element filled out with the previous customer input + XCTAssertEqual(name_with_previous_customer_input.element.text, "someName") + XCTAssertEqual(name_with_previous_customer_input.validationState, .valid) + } + + func testNameValueWrittenToLocationDefinedAPIPath() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "someName" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let nameSpec = FormSpec.NameFieldSpec( + apiPath: ["v1": "custom_location[name]"], + translationId: nil + ) + let spec = FormSpec( + type: "mock_pm", + async: false, + fields: [.name(nameSpec)], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.billingDetails?.name) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[name]"] + as! String, + "someName" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + } + + func testNameValueWrittenToLocationUndefinedAPIPath() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.name = "someName" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let nameSpec = FormSpec.NameFieldSpec(apiPath: nil, translationId: nil) + let spec = FormSpec( + type: "mock_pm", + async: false, + fields: [.name(nameSpec)], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[name]"] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.name, "someName") + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + } + + func testEmailOverrideApiPathBySpec() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.email = "email@stripe.com" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let email = factory.makeEmail(apiPath: "custom_location[email]") + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = email.updateParams(params: params) + + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[email]"] + as! String, + "email@stripe.com" + ) + XCTAssertNil(updatedParams?.paymentMethodParams.billingDetails?.email) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + + // Using the params as previous customer input... + let email_with_previous_customer_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method"), + previousCustomerInput: updatedParams + ).makeName(apiPath: "custom_location[email]") + // ...should result in a valid element filled out with the previous customer input + XCTAssertEqual(email_with_previous_customer_input.element.text, "email@stripe.com") + XCTAssertEqual(email_with_previous_customer_input.validationState, .valid) + } + + func testEmailValueWrittenToDefaultLocation() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.email = "email@stripe.com" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let email = factory.makeEmail() + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = email.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.email, "email@stripe.com") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[email]"] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + + // Using the params as previous customer input... + let email_with_previous_customer_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method"), + previousCustomerInput: updatedParams + ).makeEmail() + // ...should result in a valid element filled out with the previous customer input + XCTAssertEqual(email_with_previous_customer_input.element.text, "email@stripe.com") + XCTAssertEqual(email_with_previous_customer_input.validationState, .valid) + } + + func testEmailValueWrittenToLocationDefinedAPIPath() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.email = "email@stripe.com" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let emailSpec = FormSpec.BaseFieldSpec(apiPath: ["v1": "custom_location[email]"]) + let spec = FormSpec( + type: "mock_pm", + async: false, + fields: [.email(emailSpec)], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.billingDetails?.email) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[email]"] + as! String, + "email@stripe.com" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + } + + func testPhoneValueWrittenToDefaultLocation() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.phone = "+15555555555" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + let phoneElement = factory.makePhone() + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = phoneElement.updateParams(params: params) + + XCTAssertEqual( + updatedParams?.paymentMethodParams.billingDetails?.phone, + "+15555555555" + ) + + // Using the params as previous customer input... + let phone_with_previous_customer_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method"), + previousCustomerInput: updatedParams + ).makePhone() + // ...should result in a valid element filled out with the previous customer input + XCTAssertEqual(phone_with_previous_customer_input.element.selectedCountryCode, "US") + XCTAssertEqual(phone_with_previous_customer_input.validationState, .valid) + } + + func testEmailValueWrittenToLocationUndefinedAPIPath() { + var configuration = PaymentSheet.Configuration() + configuration.defaultBillingDetails.email = "email@stripe.com" + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mock_payment_method") + ) + + let emailSpec = FormSpec.BaseFieldSpec(apiPath: nil) + let spec = FormSpec( + type: "mock_pm", + async: false, + fields: [.email(emailSpec)], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("mock_payment_method")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.email, "email@stripe.com") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[email]"] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "mock_payment_method") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .unknown) + } + + func testMakeFormElement_dropdown() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit") + ) + let selectorSpec = FormSpec.SelectorSpec( + translationId: .eps_bank, + items: [ + .init(displayText: "d1", apiValue: "123"), + .init(displayText: "d2", apiValue: "456"), + ], + apiPath: ["v1": "custom_location[selector]"] + ) + let spec = FormSpec( + type: "sepa_debit", + async: false, + fields: [.selector(selectorSpec)], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("sepa_debit")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_location[selector]"] + as! String, + "123" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sepa_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .SEPADebit) + + // Given a dropdown... + let dropdown = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit") + ).makeDropdown(for: selectorSpec) + // ...with a selection *different* from the default of 0 + dropdown.element.select(index: 1) + // ...using the params as previous customer input to create a new dropdown... + let previousCustomerInput = dropdown.updateParams(params: IntentConfirmParams(type: .dynamic("sepa_debit"))) + let dropdown_with_previous_customer_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit"), + previousCustomerInput: previousCustomerInput + ).makeDropdown(for: selectorSpec) + + // ...should result in a valid element filled out with the previous customer input + XCTAssertEqual(dropdown_with_previous_customer_input.element.selectedIndex, 1) + XCTAssertEqual(dropdown_with_previous_customer_input.validationState, .valid) + } + + func testMakeFormElement_KlarnaCountry_UndefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("klarna") + ) + let spec = FormSpec( + type: "klarna", + async: false, + fields: [.klarna_country(.init(apiPath: nil))], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("klarna")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.address?.country, "US") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters[ + "billing_details[address][country]" + ] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "klarna") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .klarna) + } + + func testMakeFormElement_KlarnaCountry_DefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("klarna") + ) + let spec = FormSpec( + type: "klarna", + async: false, + fields: [.klarna_country(.init(apiPath: ["v1": "billing_details[address][country]"]))], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("klarna")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.billingDetails?.address?.country) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters[ + "billing_details[address][country]" + ] as! String, + "US" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "klarna") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .klarna) + } + + func testMakeFormElement_BSBNumber() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let bsb = factory.makeBSB(apiPath: nil) + bsb.element.setText("000-000") + + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + let updatedParams = bsb.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.auBECSDebit?.bsbNumber, "000000") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters["au_becs_debit[bsb_number]"] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + // Using the params as previous customer input... + let bsb_with_previous_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit"), + previousCustomerInput: updatedParams + ).makeBSB() + // ...should result in a valid, filled out element + XCTAssert(bsb_with_previous_input.validationState == .valid) + let updatedParams_with_previous_input = bsb_with_previous_input.updateParams(params: .init(type: .dynamic("au_becs_debit"))) + XCTAssertEqual(updatedParams_with_previous_input?.paymentMethodParams.auBECSDebit?.bsbNumber, "000000") + } + + func testMakeFormElement_BSBNumber_withAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let bsb = factory.makeBSB(apiPath: "custom_path[bsb_number]") + bsb.element.setText("000-000") + + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + let updatedParams = bsb.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.auBECSDebit?.bsbNumber) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["custom_path[bsb_number]"] + as! String, + "000000" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + // Using the params as previous customer input... + let bsb_with_previous_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit"), + previousCustomerInput: updatedParams + ).makeBSB(apiPath: "custom_path[bsb_number]") + // ...should result in a valid, filled out element + XCTAssert(bsb_with_previous_input.validationState == .valid) + let updatedParams_with_previous_input = bsb_with_previous_input.updateParams(params: .init(type: .dynamic("au_becs_debit"))) + XCTAssertEqual( + updatedParams_with_previous_input?.paymentMethodParams.additionalAPIParameters["custom_path[bsb_number]"] + as! String, + "000000" + ) + } + + func testMakeFormElement_BSBNumber_UndefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let spec = FormSpec( + type: "au_becs_debit", + async: false, + fields: [.au_becs_bsb_number(.init(apiPath: nil))], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + guard let wrappedElement = firstWrappedTextFieldElement(formElement: formElement.element) else { + XCTFail("Unable to get firstElement") + return + } + + wrappedElement.element.setText("000-000") + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.auBECSDebit?.bsbNumber, "000000") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters["au_becs_debit[bsb_number]"] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + + // Using the params as previous customer input... + let bsb_with_previous_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit"), + previousCustomerInput: updatedParams + ).makeBSB() + // ...should result in a valid, filled out element + XCTAssert(bsb_with_previous_input.validationState == .valid) + let updatedParams_with_previous_input = bsb_with_previous_input.updateParams(params: .init(type: .dynamic("au_becs_debit"))) + XCTAssertEqual(updatedParams_with_previous_input?.paymentMethodParams.auBECSDebit?.bsbNumber, "000000") + } + + func testMakeFormElement_BSBNumber_DefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let spec = FormSpec( + type: "au_becs_debit", + async: false, + fields: [.au_becs_bsb_number(.init(apiPath: ["v1": "au_becs_debit[bsb_number]"]))], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + guard let wrappedElement = firstWrappedTextFieldElement(formElement: formElement.element) else { + XCTFail("Unable to get firstElement") + return + } + + wrappedElement.element.setText("000-000") + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.auBECSDebit?.bsbNumber) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["au_becs_debit[bsb_number]"] + as! String, + "000000" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + } + + func testMakeFormElement_AUBECSAccountNumber_UndefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let spec = FormSpec( + type: "au_becs_debit", + async: false, + fields: [.au_becs_account_number(.init(apiPath: nil))], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + guard let wrappedElement = firstWrappedTextFieldElement(formElement: formElement.element) else { + XCTFail("Unable to get firstElement") + return + } + + wrappedElement.element.setText("000123456") + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.auBECSDebit?.accountNumber, "000123456") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters[ + "au_becs_debit[account_number]" + ] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + + // Using the params as previous customer input... + let form_with_previous_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit"), + previousCustomerInput: updatedParams + ).makeFormElementFromSpec(spec: spec) + // ...should result in a valid, filled out element + XCTAssert(form_with_previous_input.validationState == .valid) + let updatedParams_with_previous_input = form_with_previous_input.updateParams(params: .init(type: .dynamic("au_becs_debit"))) + XCTAssertEqual(updatedParams_with_previous_input?.paymentMethodParams.auBECSDebit?.accountNumber, "000123456") + } + + func testMakeFormElement_AUBECSAccountNumber_DefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let spec = FormSpec( + type: "au_becs_debit", + async: false, + fields: [ + .au_becs_account_number(.init(apiPath: ["v1": "au_becs_debit[account_number]"])), + ], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + guard let wrappedElement = firstWrappedTextFieldElement(formElement: formElement.element) else { + XCTFail("Unable to get firstElement") + return + } + + wrappedElement.element.setText("000123456") + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.auBECSDebit?.accountNumber) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters[ + "au_becs_debit[account_number]" + ] as! String, + "000123456" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + + // Using the params as previous customer input... + let form_with_previous_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit"), + previousCustomerInput: updatedParams + ).makeFormElementFromSpec(spec: spec) + // ...should result in a valid, filled out element + XCTAssert(form_with_previous_input.validationState == .valid) + let updatedParams_with_previous_input = form_with_previous_input.updateParams(params: .init(type: .dynamic("au_becs_debit"))) + XCTAssertEqual( + updatedParams_with_previous_input?.paymentMethodParams.additionalAPIParameters[ + "au_becs_debit[account_number]" + ] as! String, + "000123456" + ) + } + + func testMakeFormElement_AUBECSAccountNumber() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let accountNum = factory.makeAUBECSAccountNumber(apiPath: nil) + accountNum.element.setText("000123456") + + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + let updatedParams = accountNum.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.auBECSDebit?.accountNumber, "000123456") + XCTAssertNil( + updatedParams?.paymentMethodParams.additionalAPIParameters[ + "au_becs_debit[account_number]" + ] + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + } + + func testMakeFormElement_AUBECSAccountNumber_withAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit") + ) + let accountNum = factory.makeAUBECSAccountNumber(apiPath: "custom_path[account_number]") + accountNum.element.setText("000123456") + + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + let updatedParams = accountNum.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.auBECSDebit?.accountNumber) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters[ + "custom_path[account_number]" + ] as! String, + "000123456" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + } + + func testMakeFormElement_BillingAddress_UndefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sofort") + ) + let spec = FormSpec( + type: "sofort", + async: false, + fields: [.country(.init(apiPath: nil, allowedCountryCodes: ["AT", "BE"]))], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("sofort")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.address?.country, "AT") + XCTAssert(updatedParams?.paymentMethodParams.additionalAPIParameters.isEmpty ?? false) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sofort") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .sofort) + } + + func testMakeFormElement_Country_DefinedAPIPath_forSofort() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sofort") + ) + let spec = FormSpec( + type: "sofort", + async: false, + fields: [ + .country( + .init(apiPath: ["v1": "sofort[country]"], allowedCountryCodes: ["AT", "BE"]) + ), + ], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("sofort")) + + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.sofort?.country) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["sofort[country]"] + as! String, + "AT" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sofort") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .sofort) + } + + func testMakeFormElement_Country() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sofort") + ) + let country = factory.makeCountry(countryCodes: ["AT", "BE"], apiPath: nil) + (country as! PaymentMethodElementWrapper).element.select(index: 1) // select a different index than the default of 0 + + let params = IntentConfirmParams(type: .dynamic("sofort")) + let updatedParams = country.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.address?.country, "BE") + XCTAssert(updatedParams?.paymentMethodParams.additionalAPIParameters.isEmpty ?? false) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sofort") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .sofort) + + // Using the params as previous customer input... + let country_with_previous_input = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sofort"), + previousCustomerInput: updatedParams + ).makeCountry(countryCodes: ["AT", "BE"], apiPath: nil) + // ...should result in a valid, filled out element + XCTAssert(country_with_previous_input.validationState == .valid) + let updatedParams_with_previous_input = country_with_previous_input.updateParams(params: .init(type: .dynamic("sofort"))) + XCTAssertEqual(updatedParams_with_previous_input?.paymentMethodParams.billingDetails?.address?.country, "BE") + } + + func testMakeFormElement_Country_withAPIPath() { + let configuration = PaymentSheet.Configuration() + func makeCountry(previousCustomerInput: IntentConfirmParams?) -> PaymentMethodElement { + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sofort"), + previousCustomerInput: previousCustomerInput + ) + let country = factory.makeCountry(countryCodes: ["AT", "BE"], apiPath: "sofort[country]") + return country + } + let country = makeCountry(previousCustomerInput: nil) + (country as! PaymentMethodElementWrapper).element.select(index: 1) // select a different index than the default of 0 + + let params = IntentConfirmParams(type: .dynamic("sofort")) + let updatedParams = country.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.sofort?.country) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["sofort[country]"] + as! String, + "BE" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sofort") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .sofort) + + // Using the params as previous customer input... + let country_with_previous_input = makeCountry(previousCustomerInput: updatedParams) + // ...should result in a valid, filled out element + XCTAssert(country_with_previous_input.validationState == .valid) + let updatedParams_with_previous_input = country_with_previous_input.updateParams(params: .init(type: .dynamic("sofort"))) + XCTAssertEqual( + updatedParams_with_previous_input?.paymentMethodParams.additionalAPIParameters["sofort[country]"] as! String, + "BE" + ) + } + + func testMakeFormElement_Iban_UndefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + func makeForm(previousCustomerInput: IntentConfirmParams?) -> PaymentMethodElementWrapper { + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit"), + previousCustomerInput: previousCustomerInput + ) + let spec = FormSpec( + type: "sepa_debit", + async: false, + fields: [.iban(.init(apiPath: nil))], + selectorIcon: nil, + nextActionSpec: nil + ) + return factory.makeFormElementFromSpec(spec: spec) + } + let formElement = makeForm(previousCustomerInput: nil) + let params = IntentConfirmParams(type: .dynamic("sepa_debit")) + guard let wrappedElement = firstWrappedTextFieldElement(formElement: formElement.element) else { + XCTFail("Unable to get firstElement") + return + } + + wrappedElement.element.setText("GB33BUKB20201555555555") + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.sepaDebit?.iban, "GB33BUKB20201555555555") + XCTAssert(updatedParams?.paymentMethodParams.additionalAPIParameters.isEmpty ?? false) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sepa_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .SEPADebit) + + // Using the params as previous customer input... + let form_with_previous_input = makeForm(previousCustomerInput: updatedParams) + // ...should result in a valid, filled out element + let updatedParams_with_previous_input = form_with_previous_input.updateParams(params: .init(type: .dynamic("sepa_debit"))) + XCTAssertEqual(updatedParams_with_previous_input?.paymentMethodParams.sepaDebit?.iban, "GB33BUKB20201555555555") + } + + func testMakeFormElement_Iban_DefinedAPIPath() { + let configuration = PaymentSheet.Configuration() + func makeForm(previousCustomerInput: IntentConfirmParams?) -> PaymentMethodElementWrapper { + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit"), + previousCustomerInput: previousCustomerInput + ) + let spec = FormSpec( + type: "sepa_debit", + async: false, + fields: [.iban(.init(apiPath: ["v1": "sepa_debit[iban]"]))], + selectorIcon: nil, + nextActionSpec: nil + ) + return factory.makeFormElementFromSpec(spec: spec) + } + + let formElement = makeForm(previousCustomerInput: nil) + let params = IntentConfirmParams(type: .dynamic("sepa_debit")) + guard let wrappedElement = firstWrappedTextFieldElement(formElement: formElement.element) else { + XCTFail("Unable to get firstElement") + return + } + + wrappedElement.element.setText("GB33BUKB20201555555555") + let updatedParams = formElement.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.sepaDebit?.iban) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["sepa_debit[iban]"] + as! String, + "GB33BUKB20201555555555" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sepa_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .SEPADebit) + + // Using the params as previous customer input... + let form_with_previous_input = makeForm(previousCustomerInput: updatedParams) + // ...should result in a valid, filled out element + let updatedParams_with_previous_input = form_with_previous_input.updateParams(params: .init(type: .dynamic("sepa_debit"))) + XCTAssertEqual( + updatedParams_with_previous_input?.paymentMethodParams.additionalAPIParameters["sepa_debit[iban]"] + as! String, + "GB33BUKB20201555555555" + ) + } + + func testMakeFormElement_Iban() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit") + ) + let iban = factory.makeIban(apiPath: nil) + iban.element.setText("GB33BUKB20201555555555") + + let params = IntentConfirmParams(type: .dynamic("sepa_debit")) + let updatedParams = iban.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.sepaDebit?.iban, "GB33BUKB20201555555555") + XCTAssert(updatedParams?.paymentMethodParams.additionalAPIParameters.isEmpty ?? false) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sepa_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .SEPADebit) + } + + func testMakeFormElement_Iban_withAPIPath() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("sepa_debit") + ) + let iban = factory.makeIban(apiPath: "sepa_debit[iban]") + iban.element.setText("GB33BUKB20201555555555") + + let params = IntentConfirmParams(type: .dynamic("sepa_debit")) + let updatedParams = iban.updateParams(params: params) + + XCTAssertNil(updatedParams?.paymentMethodParams.sepaDebit?.iban) + XCTAssertEqual( + updatedParams?.paymentMethodParams.additionalAPIParameters["sepa_debit[iban]"] + as! String, + "GB33BUKB20201555555555" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "sepa_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .SEPADebit) + } + + func testMakeFormElement_email_with_unknownField() { + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("luxe_bucks") + ) + let spec = FormSpec( + type: "luxe_bucks", + async: false, + fields: [ + .unknown("some_unknownField1"), + .email(.init(apiPath: nil)), + .unknown("some_unknownField2"), + ], + selectorIcon: nil, + nextActionSpec: nil + ) + let formElement = factory.makeFormElementFromSpec(spec: spec) + let params = IntentConfirmParams(type: .dynamic("luxe_bucks")) + guard let wrappedElement = firstWrappedTextFieldElement(formElement: formElement.element) else { + XCTFail("Unable to get firstElement") + return + } + + wrappedElement.element.setText("email@stripe.com") + let updatedParams = formElement.updateParams(params: params) + + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.email, "email@stripe.com") + XCTAssert(updatedParams?.paymentMethodParams.additionalAPIParameters.isEmpty ?? false) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "luxe_bucks") + } + + func testMakeFormElement_BillingAddress() { + let addressSpecProvider = AddressSpecProvider() + addressSpecProvider.addressSpecs = [ + "US": AddressSpec( + format: "%N%n%O%n%A%n%C, %S %Z", + require: "ACSZ", + cityNameType: nil, + stateNameType: .state, + zip: "\\d{5}", + zipNameType: .zip + ), + ] + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("au_becs_debit"), + addressSpecProvider: addressSpecProvider + ) + let accountNum = factory.makeBillingAddressSection(countries: nil) + accountNum.element.line1?.setText("123 main") + accountNum.element.line2?.setText("#501") + accountNum.element.city?.setText("AnywhereTown") + accountNum.element.state?.setRawData("California") + accountNum.element.postalCode?.setText("55555") + + let params = IntentConfirmParams(type: .dynamic("au_becs_debit")) + let updatedParams = accountNum.updateParams(params: params) + + XCTAssertEqual( + updatedParams?.paymentMethodParams.billingDetails?.address?.line1, + "123 main" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.address?.line2, "#501") + XCTAssertEqual(updatedParams?.paymentMethodParams.billingDetails?.address?.country, "US") + XCTAssertEqual( + updatedParams?.paymentMethodParams.billingDetails?.address?.city, + "AnywhereTown" + ) + XCTAssertEqual( + updatedParams?.paymentMethodParams.billingDetails?.address?.state, + "California" + ) + XCTAssertEqual( + updatedParams?.paymentMethodParams.billingDetails?.address?.postalCode, + "55555" + ) + XCTAssertEqual(updatedParams?.paymentMethodParams.rawTypeString, "au_becs_debit") + XCTAssertEqual(updatedParams?.paymentMethodParams.type, .AUBECSDebit) + } + + func testMakeFormElement_AddressElementUsesDefaultCountries() { + let addressSpecProvider = addressSpecProvider(countries: ["US", "FR"]) + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mockPM"), + addressSpecProvider: addressSpecProvider + ) + let billingAddressSpec = FormSpec.BillingAddressSpec(allowedCountryCodes: nil) + let spec = FormSpec( + type: "mockPM", + async: false, + fields: [.billing_address(billingAddressSpec)], + selectorIcon: nil, + nextActionSpec: nil + ) + + let formElement = factory.makeFormElementFromSpec(spec: spec) + guard let addressSectionElement = firstAddressSectionElement(formElement: formElement.element) + else { + XCTFail("failed to get address section element") + return + } + + XCTAssertEqual(addressSectionElement.countryCodes.count, 2) + XCTAssertTrue(addressSectionElement.countryCodes.contains("US")) + XCTAssertTrue(addressSectionElement.countryCodes.contains("FR")) + } + + func testMakeFormElement_AddressElementUsesAllowedCountryCodes_FR() { + let addressSpecProvider = addressSpecProvider(countries: ["US", "FR"]) + let configuration = PaymentSheet.Configuration() + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .dynamic("mockPM"), + addressSpecProvider: addressSpecProvider + ) + let billingAddressSpec = FormSpec.BillingAddressSpec(allowedCountryCodes: ["FR"]) + let spec = FormSpec( + type: "mockPM", + async: false, + fields: [.billing_address(billingAddressSpec)], + selectorIcon: nil, + nextActionSpec: nil + ) + + let formElement = factory.makeFormElementFromSpec(spec: spec) + guard let addressSectionElement = firstAddressSectionElement(formElement: formElement.element) + else { + XCTFail("failed to get address section element") + return + } + + XCTAssertEqual(addressSectionElement.countryCodes.count, 1) + XCTAssertTrue(addressSectionElement.countryCodes.contains("FR")) + } + + func testNonCardsAndUSBankAccountsDontHaveSaveForFutureUseCheckbox() { + let configuration = PaymentSheet.Configuration() + let intent = Intent.paymentIntent(STPFixtures.paymentIntent()) + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "ACSZP", + require: "AZ", + cityNameType: .post_town, + stateNameType: .state, + zip: "", + zipNameType: .pin + ), + ] + let loadFormSpecs = expectation(description: "Load form specs") + FormSpecProvider.shared.load { _ in + loadFormSpecs.fulfill() + } + waitForExpectations(timeout: 10, handler: nil) + // No payment method should have a checkbox except for cards and US Bank Accounts + for type in PaymentSheet.supportedPaymentMethods.filter({ + $0 != .card && $0 != .USBankAccount + }) { + let factory = PaymentSheetFormFactory( + intent: intent, + configuration: configuration, + paymentMethod: PaymentSheet.PaymentMethodType( + from: STPPaymentMethod.string(from: type)! + ), + addressSpecProvider: specProvider + ) + + var form = factory.make() + if let wrapper = form as? PaymentMethodElementWrapper { + form = wrapper.element + } + + guard let form = form as? FormElement else { + XCTFail() + return + } + if form.getAllUnwrappedSubElements() + .compactMap({ $0 as? CheckboxElement }) + .contains(where: { $0.label.hasPrefix("Save") }) { // Hacky way to differentiate the save checkbox from other checkboxes + XCTFail("\(type) contains a checkbox") + } + } + } + + func testShowsCardCheckbox() { + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.card]) + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .card + ) + XCTAssertEqual(factory.saveMode, .userSelectable) + } + + func testEPSDoesntHideCardCheckbox() { + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.card, .EPS]) + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .card + ) + XCTAssertEqual(factory.saveMode, .userSelectable) + } + + func testBillingAddressSection() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + configuration.defaultBillingDetails.address = defaultAddress + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.card]) + // An address section with defaults... + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .card, + addressSpecProvider: specProvider + ) + let addressSection = factory.makeBillingAddressSection(countries: nil) + + // ...should update params + let intentConfirmParams = addressSection.updateParams( + params: IntentConfirmParams(type: .card) + ) + guard let billingDetails = intentConfirmParams?.paymentMethodParams.billingDetails?.address + else { + XCTFail() + return + } + + XCTAssertEqual(billingDetails.line1, defaultAddress.line1) + XCTAssertEqual(billingDetails.line2, defaultAddress.line2) + XCTAssertEqual(billingDetails.city, defaultAddress.city) + XCTAssertEqual(billingDetails.postalCode, defaultAddress.postalCode) + XCTAssertEqual(billingDetails.state, defaultAddress.state) + XCTAssertEqual(billingDetails.country, defaultAddress.country) + } + + func testPreferDefaultBillingDetailsOverShippingDetails() { + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + configuration.defaultBillingDetails.address = .init(line1: "Billing line 1") + configuration.shippingDetails = { + return .init(address: .init(country: "US", line1: "Shipping line 1"), name: "Name") + } + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.card]) + // An address section with both default billing and default shipping... + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .card, + addressSpecProvider: specProvider + ) + let addressSection = factory.makeBillingAddressSection(countries: nil) + // ...sets the defaults to use billing and not shipping + XCTAssertEqual(addressSection.element.line1?.text, "Billing line 1") + // ...and doesn't show the shipping checkbox + XCTAssertTrue(addressSection.element.sameAsCheckbox.view.isHidden) + } + + func testApplyDefaults_Card_Applied() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + configuration.billingDetailsCollectionConfiguration.attachDefaultsToPaymentMethod = true + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.card]) + // An address section with defaults... + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .card, + addressSpecProvider: specProvider + ) + let cardForm = factory.makeCard() + let params = cardForm.applyDefaults(params: IntentConfirmParams(type: .card)) + + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.name, "Jane Doe") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.email, "foo@bar.com") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.phone, "+15555555555") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.line1, "510 Townsend St.") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.line2, "Line 2") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.city, "San Francisco") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.state, "CA") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.country, "US") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.postalCode, "94102") + } + + func testApplyDefaults_Card_NotApplied() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.card]) + // An address section with defaults... + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .card, + addressSpecProvider: specProvider + ) + let formElement = factory.make() + let params = formElement.applyDefaults(params: IntentConfirmParams(type: .card)) + + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.name) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.email) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.phone) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.line1) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.line2) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.city) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.state) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.country) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.postalCode) + } + + func testApplyDefaults_LPM_Applied() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + configuration.billingDetailsCollectionConfiguration.attachDefaultsToPaymentMethod = true + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.afterpayClearpay]) + // An address section with defaults... + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + + let expectation = expectation(description: "FormSpecs loaded") + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + wait(for: [expectation], timeout: 5.0) + + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .dynamic("afterpay_clearpay"), + addressSpecProvider: specProvider + ) + let form = factory.make() + let params = form.applyDefaults(params: IntentConfirmParams(type: .dynamic("afterpay_clearpay"))) + + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.name, "Jane Doe") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.email, "foo@bar.com") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.phone, "+15555555555") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.line1, "510 Townsend St.") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.line2, "Line 2") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.city, "San Francisco") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.state, "CA") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.country, "US") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.postalCode, "94102") + } + + func testApplyDefaults_LPM_NotApplied() { + let defaultAddress = PaymentSheet.Address( + city: "San Francisco", + country: "US", + line1: "510 Townsend St.", + line2: "Line 2", + postalCode: "94102", + state: "CA" + ) + var configuration = PaymentSheet.Configuration() + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + configuration.defaultBillingDetails.name = "Jane Doe" + configuration.defaultBillingDetails.email = "foo@bar.com" + configuration.defaultBillingDetails.phone = "+15555555555" + configuration.defaultBillingDetails.address = defaultAddress + let paymentIntent = STPFixtures.makePaymentIntent(paymentMethodTypes: [.afterpayClearpay]) + // An address section with defaults... + let specProvider = AddressSpecProvider() + specProvider.addressSpecs = [ + "US": AddressSpec( + format: "NOACSZ", + require: "ACSZ", + cityNameType: .city, + stateNameType: .state, + zip: "", + zipNameType: .zip + ), + ] + + let expectation = expectation(description: "FormSpecs loaded") + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + wait(for: [expectation], timeout: 5.0) + + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(paymentIntent), + configuration: configuration, + paymentMethod: .dynamic("afterpay_clearpay"), + addressSpecProvider: specProvider + ) + let form = factory.make() + let params = form.applyDefaults(params: IntentConfirmParams(type: .dynamic("afterpay_clearpay"))) + + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.name) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.email) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.phone) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.line1) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.line2) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.city) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.state) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.country) + XCTAssertNil(params.paymentMethodParams.nonnil_billingDetails.address?.postalCode) + } + + // MARK: - Previous Customer Input tests + + // Covers: + // - Email + // - Name + // - Phone + // - Billing address + // - Card form + // - Save checkbox + func testAppliesPreviousCustomerInput_billing_details_and_card() { + // Given default billing details... + let defaultAddress = PaymentSheet.Address( + city: "should not be used", + country: "should not be used", + line1: "should not be used", + line2: "should not be used", + postalCode: "should not be used", + state: "should not be used" + ) + var configuration = PaymentSheet.Configuration() + // ...and a configuration that requires collection of all billing details... + configuration.billingDetailsCollectionConfiguration.email = .always + configuration.billingDetailsCollectionConfiguration.name = .always + configuration.billingDetailsCollectionConfiguration.phone = .always + configuration.billingDetailsCollectionConfiguration.address = .full + configuration.customer = .init(id: "id", ephemeralKeySecret: "sec") + configuration.defaultBillingDetails.name = "should not be used" + configuration.defaultBillingDetails.email = "should not be usedm" + configuration.defaultBillingDetails.phone = "should not be used" + configuration.defaultBillingDetails.address = defaultAddress + + let expectation = expectation(description: "Load specs") + AddressSpecProvider.shared.loadAddressSpecs { + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + } + waitForExpectations(timeout: 1) + + // ...and previous customer input billing details... + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "Jane Doe" + billingDetails.email = "foo@bar.com" + billingDetails.phone = "5555555555" + billingDetails.address = STPPaymentMethodAddress() + billingDetails.address?.line1 = "510 Townsend St." + billingDetails.address?.line2 = "Line 2" + billingDetails.address?.city = "San Francisco" + billingDetails.address?.state = "CA" + billingDetails.address?.country = "US" + billingDetails.address?.postalCode = "94102" + + // ...and full card details... + let cardValues = STPFixtures.paymentMethodCardParams() + cardValues.expMonth = 3 // Choose a single digit month to exercise the code for padding with leading zeros + let previousCustomerInput = IntentConfirmParams.init( + params: .paramsWith( + card: cardValues, + billingDetails: billingDetails, + metadata: nil), + type: .card + ) + + // ...the card form... + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent(paymentMethodTypes: ["card"])), + configuration: configuration, + paymentMethod: .card, + previousCustomerInput: previousCustomerInput + ) + let cardForm = factory.make() + + // ...should be valid... + XCTAssert(cardForm.validationState == .valid) + // ...and its params should match the defaults above + let params = cardForm.updateParams(params: IntentConfirmParams(type: .card))! + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.name, "Jane Doe") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.email, "foo@bar.com") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.phone, "+15555555555") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.line1, "510 Townsend St.") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.line2, "Line 2") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.city, "San Francisco") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.state, "CA") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.country, "US") + XCTAssertEqual(params.paymentMethodParams.nonnil_billingDetails.address?.postalCode, "94102") + + XCTAssertEqual(params.paymentMethodParams.card?.number, cardValues.number) + XCTAssertEqual(params.paymentMethodParams.card?.expMonth, cardValues.expMonth) + XCTAssertEqual(params.paymentMethodParams.card?.expYear, cardValues.expYear) + XCTAssertEqual(params.paymentMethodParams.card?.cvc, cardValues.cvc) + // ...and the checkbox state should be enabled (the default) + XCTAssertEqual(params.saveForFutureUseCheckboxState, .selected) + } + + func testAppliesPreviousCustomerInput_checkbox() { + let expectation = expectation(description: "Load specs") + AddressSpecProvider.shared.loadAddressSpecs { + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + } + waitForExpectations(timeout: 1) + + func makeCardForm(isSettingUp: Bool, previousCustomerInput: IntentConfirmParams?) -> PaymentMethodElement { + var configuration = PaymentSheet.Configuration._testValue_MostPermissive() + configuration.customer = .init(id: "id", ephemeralKeySecret: "ek") + return PaymentSheetFormFactory( + intent: isSettingUp ? .setupIntent(STPFixtures.setupIntent()) : .paymentIntent(STPFixtures.paymentIntent()), + configuration: configuration, + paymentMethod: .card, + previousCustomerInput: previousCustomerInput + ).make() + } + // A filled out card form in setup mode... + let previousCustomerInput = IntentConfirmParams.init( + params: .paramsWith( + card: STPFixtures.paymentMethodCardParams(), + billingDetails: STPFixtures.paymentMethodBillingDetails(), + metadata: nil), + type: .card + ) + let cardForm_setup = makeCardForm(isSettingUp: true, previousCustomerInput: previousCustomerInput) + // ...should have the checkbox hidden + let cardForm_setup_params = cardForm_setup.updateParams(params: .init(type: .card)) + XCTAssertEqual(cardForm_setup_params?.saveForFutureUseCheckboxState, .hidden) + + // Making another card form for payment using the previous card form's input... + let cardForm_payment = makeCardForm(isSettingUp: false, previousCustomerInput: cardForm_setup_params) + // ...should have the checkbox selected (the default) + let cardForm_payment_params = cardForm_payment.updateParams(params: .init(type: .card)) + XCTAssertEqual(cardForm_payment_params?.saveForFutureUseCheckboxState, .selected) + + // Deselecting the checkbox... + let saveCheckbox = cardForm_payment.getAllUnwrappedSubElements().compactMap({ $0 as? CheckboxElement }).first(where: { $0.label.hasPrefix("Save") }) + saveCheckbox?.isSelected = false + let cardForm_payment_params_checkbox_deselected = cardForm_payment.updateParams(params: .init(type: .card)) + XCTAssertEqual(cardForm_payment_params_checkbox_deselected?.saveForFutureUseCheckboxState, .deselected) + // ...and making another card form... + let cardForm_payment_2 = makeCardForm(isSettingUp: false, previousCustomerInput: cardForm_payment_params_checkbox_deselected) + // ...should have the checkbox deselected, preserving the previous customer input + let cardForm_payment_2_params = cardForm_payment_2.updateParams(params: .init(type: .card)) + XCTAssertEqual(cardForm_payment_2_params?.saveForFutureUseCheckboxState, .deselected) + + } + + func testAppliesPreviousCustomerInput_for_different_payment_method_type() { + let expectation = expectation(description: "Load specs") + AddressSpecProvider.shared.loadAddressSpecs { + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + } + waitForExpectations(timeout: 1) + + // ...Given previous customer input billing details... + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "Jane Doe" + billingDetails.email = "foo@bar.com" + billingDetails.phone = "5555555555" + billingDetails.address = STPPaymentMethodAddress() + billingDetails.address?.line1 = "510 Townsend St." + billingDetails.address?.line2 = "Line 2" + billingDetails.address?.city = "San Francisco" + billingDetails.address?.state = "CA" + billingDetails.address?.country = "US" + billingDetails.address?.postalCode = "94102" + + // ...for Afterpay... + let previousAfterpayCustomerInput = IntentConfirmParams.init( + params: .paramsWith(afterpayClearpay: .init(), billingDetails: billingDetails, metadata: nil), + type: .dynamic("afterpay_clearpay") + ) + + // ...the Afterpay form should be valid + let afterpayFactory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent(paymentMethodTypes: ["afterpay_clearpay"])), + configuration: ._testValue_MostPermissive(), + paymentMethod: .dynamic("afterpay_clearpay"), + previousCustomerInput: previousAfterpayCustomerInput + ) + let afterpayForm = afterpayFactory.make() + XCTAssert(afterpayForm.validationState == .valid) + + // ...but if the customer previous input was for a card... + let previousCardCustomerInput = IntentConfirmParams.init( + params: .paramsWith( + card: STPFixtures.paymentMethodCardParams(), + billingDetails: billingDetails, + metadata: nil), + type: .card + ) + // ...the Afterpay form should be blank and invalid, even though the previous input had full billing details + let afterpayFormWithPreviousCardInput = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent(paymentMethodTypes: ["afterpay_clearpay"])), + configuration: ._testValue_MostPermissive(), + paymentMethod: .dynamic("afterpay_clearpay"), + previousCustomerInput: previousCardCustomerInput + ).make() + XCTAssert(afterpayFormWithPreviousCardInput.validationState != .valid) + // ...and the address section shouldn't be populated with any defaults + guard + let afterpayForm = afterpayFormWithPreviousCardInput as? PaymentMethodElementWrapper, + let addressSectionElement = afterpayForm.element.getAllUnwrappedSubElements().compactMap({ $0 as? AddressSectionElement }).first + else { + XCTFail("expected address section") + return + } + let emptyAddressSectionElement = AddressSectionElement() + XCTAssertEqual(addressSectionElement.addressDetails, emptyAddressSectionElement.addressDetails) + } + + func testAppliesPreviousCustomerInput_klarna_country() { + func makeKlarnaCountry(apiPath: String?, previousCustomerInput: IntentConfirmParams?) -> PaymentMethodElementWrapper { + let factory = PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent(paymentMethodTypes: ["klarna"], currency: "eur")), + configuration: ._testValue_MostPermissive(), + paymentMethod: .dynamic("klarna"), + previousCustomerInput: previousCustomerInput + ) + return factory.makeKlarnaCountry(apiPath: apiPath) as! PaymentMethodElementWrapper + } + let apiPathValues: [String?] = [nil, "billing_details[address][country]"] // Test the same thing with and without an api path + apiPathValues.forEach { apiPath in + // Given a klarna country... + let klarnaCountry = makeKlarnaCountry(apiPath: apiPath, previousCustomerInput: nil) + // ...with a selection *different* from the default of 0 + klarnaCountry.element.select(index: 1) + // ...using its params as previous customer input to create a new klarna country... + let previousCustomerInput = klarnaCountry.updateParams(params: IntentConfirmParams(type: .dynamic("klarna"))) + let klarnaCountry_with_previous_customer_input = makeKlarnaCountry(apiPath: apiPath, previousCustomerInput: previousCustomerInput) + // ...should result in a valid element filled out with the previous customer input + XCTAssertEqual(klarnaCountry_with_previous_customer_input.element.selectedIndex, 1) + XCTAssertEqual(klarnaCountry_with_previous_customer_input.validationState, .valid) + } + } + + func testAppliesPreviousCustomerInput_for_mandate() { + let expectation = expectation(description: "Load specs") + AddressSpecProvider.shared.loadAddressSpecs { + FormSpecProvider.shared.load { _ in + expectation.fulfill() + } + } + waitForExpectations(timeout: 1) + + // Use PayPal as an example PM, since it is an empty form w/ a mandate iff PI+SFU or SI + func makePaypalForm(isSettingUp: Bool, previousCustomerInput: IntentConfirmParams?) -> PaymentMethodElement { + return PaymentSheetFormFactory( + intent: .paymentIntent(STPFixtures.paymentIntent(paymentMethodTypes: ["paypal"], setupFutureUsage: isSettingUp ? .offSession : .none)), + configuration: ._testValue_MostPermissive(), + paymentMethod: .dynamic("paypal"), + previousCustomerInput: previousCustomerInput + ).make() + } + + // 1. nil -> valid Payment form + // A paypal form for *payment* without previous customer input... + let paypalForm_payment = makePaypalForm(isSettingUp: false, previousCustomerInput: nil) + // ...should be valid - it requires no customer input. + guard let paypalForm_payment_paymentOption = paypalForm_payment.updateParams(params: IntentConfirmParams(type: .dynamic("paypal"))) else { + XCTFail("payment option should be non-nil") + return + } + XCTAssertFalse(paypalForm_payment_paymentOption.didDisplayMandate) + + // 2. valid Payment form -> invalid Setup form + // Creating a paypal form for *setup* using the old form as previous customer input... + var paypalForm_setup = makePaypalForm(isSettingUp: true, previousCustomerInput: paypalForm_payment_paymentOption) + // ...should not be valid... + XCTAssertNil(paypalForm_setup.updateParams(params: IntentConfirmParams(type: .dynamic("paypal")))) + // ...until the customer has seen the mandate... + sendEventToSubviews(.viewDidAppear, from: paypalForm_setup.view) + guard let paypalForm_setup_paymentOption = paypalForm_setup.updateParams(params: IntentConfirmParams(type: .dynamic("paypal"))) else { + XCTFail("payment option should be non-nil") + return + } + XCTAssertTrue(paypalForm_setup_paymentOption.didDisplayMandate) + + // 3. valid Setup form -> valid Setup form + // Using the form's previous customer input to create another *setup* paypal form... + paypalForm_setup = makePaypalForm(isSettingUp: true, previousCustomerInput: paypalForm_setup_paymentOption) + // ...should be valid... + guard let paypalForm_setup_paymentOption = paypalForm_setup.updateParams(params: IntentConfirmParams(type: .dynamic("paypal"))) else { + XCTFail("payment option should be non-nil") + return + } + XCTAssertTrue(paypalForm_setup_paymentOption.didDisplayMandate) + } + + // MARK: - Helpers + + func addressSpecProvider(countries: [String]) -> AddressSpecProvider { + let addressSpecProvider = AddressSpecProvider() + let specs = [ + "US": AddressSpec( + format: "%N%n%O%n%A%n%C, %S %Z", + require: "ACSZ", + cityNameType: nil, + stateNameType: .state, + zip: "\\d{5}", + zipNameType: .zip + ), + "FR": AddressSpec( + format: "%O%n%N%n%A%n%Z %C", + require: "ACZ", + cityNameType: nil, + stateNameType: nil, + zip: "\\d{2} ?\\d{3}", + zipNameType: nil + ), + ] + let filteredSpecs = specs.filter { countries.contains($0.key) } + addressSpecProvider.addressSpecs = filteredSpecs + return addressSpecProvider + } + + private func firstWrappedTextFieldElement( + formElement: FormElement + ) -> PaymentMethodElementWrapper? { + guard let sectionElement = formElement.elements.first as? SectionElement, + let wrappedElement = sectionElement.elements.first + as? PaymentMethodElementWrapper + else { + return nil + } + return wrappedElement + } + private func firstAddressSectionElement(formElement: FormElement) -> AddressSectionElement? { + guard + let wrapper = formElement.elements.first + as? PaymentMethodElementWrapper + else { + return nil + } + return wrapper.element + } +} + +extension Element { + /// A convenience method that overwrites the one defined in Element.swift that unwraps any Elements wrapped in `PaymentMethodElementWrapper` + /// and returns all Elements underneath this Element, including this Element. + public func getAllUnwrappedSubElements() -> [Element] { + switch self { + case let container as ContainerElement: + return [container] + container.elements.flatMap { $0.getAllUnwrappedSubElements() } + case let wrappedElement as PaymentMethodElementWrapper: + return wrappedElement.element.getAllUnwrappedSubElements() + case let wrappedElement as PaymentMethodElementWrapper: + return wrappedElement.element.getAllUnwrappedSubElements() + case let wrappedElement as PaymentMethodElementWrapper: + return wrappedElement.element.getAllUnwrappedSubElements() + case let wrappedElement as PaymentMethodElementWrapper: + return wrappedElement.element.getAllUnwrappedSubElements() + default: + return [self] + } + } +} diff --git a/Stripe/StripeiOSTests/PaymentSheetLinkAccountTests.swift b/Stripe/StripeiOSTests/PaymentSheetLinkAccountTests.swift new file mode 100644 index 00000000..fa7d7881 --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentSheetLinkAccountTests.swift @@ -0,0 +1,82 @@ +// +// PaymentSheetLinkAccountTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 3/11/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +final class PaymentSheetLinkAccountTests: XCTestCase { + + func testMakePaymentMethodParams() { + let sut = makeSUT() + + let paymentDetails = makePaymentDetailsStub() + let result = sut.makePaymentMethodParams(from: paymentDetails) + + XCTAssertEqual(result?.type, .link) + XCTAssertEqual(result?.link?.paymentDetailsID, "1") + XCTAssertEqual( + result?.link?.credentials as? [String: String], + [ + "consumer_session_client_secret": "client_secret" + ] + ) + XCTAssertNil(result?.link?.additionalAPIParameters["card"]) + } + + func testMakePaymentMethodParams_withCVC() { + let sut = makeSUT() + + let paymentDetails = makePaymentDetailsStub(withCVC: "12345") + let result = sut.makePaymentMethodParams(from: paymentDetails) + + XCTAssertEqual( + result?.link?.additionalAPIParameters["card"] as? [String: String], + [ + "cvc": "12345" + ] + ) + } + +} + +extension PaymentSheetLinkAccountTests { + + func makePaymentDetailsStub(withCVC cvc: String? = nil) -> ConsumerPaymentDetails { + let card = ConsumerPaymentDetails.Details.Card( + expiryYear: 2030, + expiryMonth: 1, + brand: "visa", + last4: "4242", + checks: nil + ) + + card.cvc = cvc + + return ConsumerPaymentDetails( + stripeID: "1", + details: .card(card: card), + isDefault: true + ) + } + + func makeSUT() -> PaymentSheetLinkAccount { + return PaymentSheetLinkAccount( + email: "user@example.com", + session: LinkStubs.consumerSession(), + publishableKey: nil, + apiClient: STPAPIClient(publishableKey: STPTestingDefaultPublishableKey), + cookieStore: LinkInMemoryCookieStore() + ) + } + +} diff --git a/Stripe/StripeiOSTests/PaymentSheetPaymentMethodTypeTest.swift b/Stripe/StripeiOSTests/PaymentSheetPaymentMethodTypeTest.swift new file mode 100644 index 00000000..dee88fe9 --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentSheetPaymentMethodTypeTest.swift @@ -0,0 +1,731 @@ +// +// PaymentSheetPaymentMethodTypeTest.swift +// StripeiOS Tests +// +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class PaymentSheetPaymentMethodTypeTest: XCTestCase { + + func makeConfiguration( + hasReturnURL: Bool = false + ) -> PaymentSheet.Configuration { + var configuration = PaymentSheet.Configuration() + configuration.returnURL = hasReturnURL ? "foo://bar" : nil + return configuration + } + + // MARK: - Cards + + /// Returns false, card not in `supportedPaymentMethods` + func testSupportsAdding_notInSupportedList_noRequirementsNeeded() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: .card, + configuration: PaymentSheet.Configuration(), + intent: .paymentIntent(STPFixtures.paymentIntent()), + supportedPaymentMethods: [] + ) + , .notSupported + ) + } + + /// Returns true, card in `supportedPaymentMethods` and has no additional requirements + func testSupportsAdding_inSupportedList_noRequirementsNeeded() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: .card, + configuration: PaymentSheet.Configuration(), + intent: .paymentIntent( + STPFixtures.makePaymentIntent(setupFutureUsage: .offSession) + ), + supportedPaymentMethods: [.card] + ), + .supported + ) + } + + /// Returns true, card in `supportedPaymentMethods` and has no additional requirements + func testSupportsAdding_inSupportedList_noRequirementsNeededButProvided() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: .card, + configuration: makeConfiguration(hasReturnURL: true), + intent: .paymentIntent(STPFixtures.makePaymentIntent()), + supportedPaymentMethods: [.card] + ), + .supported + ) + } + + // MARK: - iDEAL + + /// Returns true, iDEAL in `supportedPaymentMethods` and URL requirement and not setting up requirement are met + func testSupportsAdding_inSupportedList_urlConfiguredRequired() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: PaymentSheet.PaymentMethodType.dynamic("ideal"), + configuration: makeConfiguration(hasReturnURL: true), + intent: .paymentIntent(STPFixtures.makePaymentIntent()), + supportedPaymentMethods: [.iDEAL] + ), + .supported + ) + } + + /// Returns true, iDEAL in `supportedPaymentMethods` but URL requirement not is met + func testSupportsAdding_inSupportedList_urlConfiguredRequiredButNotProvided() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: PaymentSheet.PaymentMethodType.dynamic("ideal"), + configuration: makeConfiguration(), + intent: .paymentIntent(STPFixtures.makePaymentIntent()), + supportedPaymentMethods: [.iDEAL] + ), + .missingRequirements([.returnURL]) + ) + } + + // MARK: - Afterpay + + /// Returns false, Afterpay in `supportedPaymentMethods` but shipping requirement not is met + func testSupportsAdding_inSupportedList_urlConfiguredAndShippingRequired_missingShipping() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: PaymentSheet.PaymentMethodType.dynamic("afterpay_clearpay"), + configuration: makeConfiguration(hasReturnURL: true), + intent: .paymentIntent(STPFixtures.makePaymentIntent(shippingProvided: false)), + supportedPaymentMethods: [.afterpayClearpay] + ), + .missingRequirements([.shippingAddress]) + ) + } + + /// Returns false, Afterpay in `supportedPaymentMethods` but URL and shipping requirement not is met + func testSupportsAdding_inSupportedList_urlConfiguredAndShippingRequired_missingURL() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: PaymentSheet.PaymentMethodType.dynamic("afterpay_clearpay"), + configuration: makeConfiguration(hasReturnURL: false), + intent: .paymentIntent(STPFixtures.makePaymentIntent(shippingProvided: false)), + supportedPaymentMethods: [.afterpayClearpay] + ), + .missingRequirements([.shippingAddress, .returnURL]) + ) + } + + /// Returns true, Afterpay in `supportedPaymentMethods` and both URL and shipping requirements are met + func testSupportsAdding_inSupportedList_urlConfiguredAndShippingRequired_bothMet() { + // Afterpay should be supported if PI has shipping... + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: PaymentSheet.PaymentMethodType.dynamic("afterpay_clearpay"), + configuration: makeConfiguration(hasReturnURL: true), + intent: .paymentIntent(STPFixtures.makePaymentIntent(shippingProvided: true)), + supportedPaymentMethods: [.afterpayClearpay] + ), + .supported + ) + // ...and also if configuration.allowsPaymentMethodsThatRequireShipping is true + var config = makeConfiguration(hasReturnURL: true) + config.allowsPaymentMethodsRequiringShippingAddress = true + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: PaymentSheet.PaymentMethodType.dynamic("afterpay_clearpay"), + configuration: config, + intent: .paymentIntent(STPFixtures.makePaymentIntent(shippingProvided: false)), + supportedPaymentMethods: [.afterpayClearpay] + ), + .supported + ) + } + + // MARK: - SEPA family + + let sepaFamily = [ + PaymentSheet.PaymentMethodType.dynamic("ideal"), + PaymentSheet.PaymentMethodType.dynamic("bancontact"), + PaymentSheet.PaymentMethodType.dynamic("sofort"), + PaymentSheet.PaymentMethodType.dynamic("sepa_debit"), + ] + func testCantSetupSEPAFamily() { + // All SEPA family pms... + for pm in sepaFamily { + // ...can't be used for PIs... + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: pm, + configuration: makeConfiguration(hasReturnURL: true), + // ...if setup future usage is provided. + intent: .paymentIntent( + STPFixtures.makePaymentIntent(setupFutureUsage: .offSession) + ), + supportedPaymentMethods: sepaFamily.map { $0.stpPaymentMethodType! } + ), + .missingRequirements([.unavailable, .userSupportsDelayedPaymentMethods]) + ) + + // ...and can't be set up + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: pm, + configuration: makeConfiguration(hasReturnURL: true), + intent: .setupIntent(STPFixtures.setupIntent()), + supportedPaymentMethods: sepaFamily.map { $0.stpPaymentMethodType! } + ), + .missingRequirements([.unavailable, .userSupportsDelayedPaymentMethods]) + ) + } + } + + func testCanAddSEPAFamily() { + // iDEAL and bancontact can be added if returnURL provided + let sepaFamilySynchronous = [ + PaymentSheet.PaymentMethodType.dynamic("ideal"), + PaymentSheet.PaymentMethodType.dynamic("bancontact"), + ] + for pm in sepaFamilySynchronous { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: pm, + configuration: makeConfiguration(hasReturnURL: true), + intent: .paymentIntent(STPFixtures.makePaymentIntent()), + supportedPaymentMethods: sepaFamily.map { $0.stpPaymentMethodType! } + ), + .supported + ) + } + + let sepaFamilyAsynchronous = [ + PaymentSheet.PaymentMethodType.dynamic("sofort"), + PaymentSheet.PaymentMethodType.dynamic("sepa_debit"), + ] + // ...SEPA and sofort also need allowsDelayedPaymentMethod: + for pm in sepaFamilyAsynchronous { + var config = makeConfiguration(hasReturnURL: true) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: pm, + configuration: config, + intent: .paymentIntent(STPFixtures.makePaymentIntent()), + supportedPaymentMethods: sepaFamily.map { $0.stpPaymentMethodType! } + ), + .missingRequirements([.userSupportsDelayedPaymentMethods]) + ) + config.allowsDelayedPaymentMethods = true + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: pm, + configuration: config, + intent: .paymentIntent(STPFixtures.makePaymentIntent()), + supportedPaymentMethods: sepaFamily.map { $0.stpPaymentMethodType! } + ), + .supported + ) + } + } + + // US Bank Account + func testCanAddUSBankAccountBasedOnVerificationMethod() { + var configuration = PaymentSheet.Configuration() + configuration.allowsDelayedPaymentMethods = true + for verificationMethod in STPPaymentMethodOptions.USBankAccount.VerificationMethod.allCases { + let usBankOptions = STPPaymentMethodOptions.USBankAccount( + setupFutureUsage: nil, + verificationMethod: verificationMethod, + allResponseFields: [:] + ) + let paymentMethodOptions = STPPaymentMethodOptions( + usBankAccount: usBankOptions, + allResponseFields: [:] + ) + let pi = STPFixtures.makePaymentIntent( + paymentMethodTypes: [.USBankAccount], + setupFutureUsage: nil, + paymentMethodOptions: paymentMethodOptions, + shippingProvided: false + ) + switch verificationMethod { + case .automatic, .instantOrSkip, .instant: + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: .USBankAccount, + configuration: configuration, + intent: .paymentIntent(pi), + supportedPaymentMethods: [.USBankAccount] + ), + .supported + ) + + case .skip, .microdeposits, .unknown: + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: .USBankAccount, + configuration: configuration, + intent: .paymentIntent(pi), + supportedPaymentMethods: [.USBankAccount] + ), + .missingRequirements([.validUSBankVerificationMethod]) + ) + } + } + } + + func testInit() { + XCTAssertEqual(PaymentSheet.PaymentMethodType(from: "card"), .card) + XCTAssertEqual(PaymentSheet.PaymentMethodType(from: "us_bank_account"), .USBankAccount) + XCTAssertEqual(PaymentSheet.PaymentMethodType(from: "link"), .link) + XCTAssertEqual( + PaymentSheet.PaymentMethodType(from: "mock_payment_method"), + .dynamic("mock_payment_method") + ) + } + + func testString() { + XCTAssertEqual(PaymentSheet.PaymentMethodType.string(from: .card), "card") + XCTAssertEqual( + PaymentSheet.PaymentMethodType.string(from: .USBankAccount), + "us_bank_account" + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.string(from: .link), "link") + XCTAssertNil(PaymentSheet.PaymentMethodType.string(from: .linkInstantDebit)) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.string(from: .dynamic("mock_payment_method")), + "mock_payment_method" + ) + } + + func testDisplayName() { + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("card").displayName, "Card") + XCTAssertEqual(PaymentSheet.PaymentMethodType.card.displayName, "Card") + + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("us_bank_account").displayName, + "US Bank Account" + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.USBankAccount.displayName, "US Bank Account") + + XCTAssertEqual(PaymentSheet.PaymentMethodType.link.displayName, "Link") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("link").displayName, "Link") + + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("alipay").displayName, "Alipay") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("ideal").displayName, "iDEAL") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("fpx").displayName, "FPX") + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("sepa_debit").displayName, + "SEPA Debit" + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("au_becs_debit").displayName, + "AU BECS Direct Debit" + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("grabpay").displayName, "GrabPay") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("giropay").displayName, "giropay") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("eps").displayName, "EPS") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("p24").displayName, "Przelewy24") + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("bancontact").displayName, + "Bancontact" + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("netbanking").displayName, + "NetBanking" + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("oxxo").displayName, "OXXO") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("sofort").displayName, "Sofort") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("upi").displayName, "UPI") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("paypal").displayName, "PayPal") + if Locale.current.regionCode == "GB" { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("afterpay_clearpay").displayName, + "Clearpay" + ) + } else { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("afterpay_clearpay").displayName, + "Afterpay" + ) + } + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("blik").displayName, "BLIK") + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("wechat_pay").displayName, + "WeChat Pay" + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("boleto").displayName, "Boleto") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("link").displayName, "Link") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("klarna").displayName, "Klarna") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("affirm").displayName, "Affirm") + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("").displayName, "") + } + + func testSTPPaymentMethodType() { + XCTAssertEqual(PaymentSheet.PaymentMethodType.card.stpPaymentMethodType, .card) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("card").stpPaymentMethodType, .card) + + XCTAssertEqual( + PaymentSheet.PaymentMethodType.USBankAccount.stpPaymentMethodType, + .USBankAccount + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("us_bank_account").stpPaymentMethodType, + .USBankAccount + ) + + XCTAssertEqual( + PaymentSheet.PaymentMethodType.linkInstantDebit.stpPaymentMethodType, + .linkInstantDebit + ) + + XCTAssertEqual(PaymentSheet.PaymentMethodType.link.stpPaymentMethodType, .link) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("link").stpPaymentMethodType, .link) + + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("alipay").stpPaymentMethodType, + .alipay + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("ideal").stpPaymentMethodType, .iDEAL) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("fpx").stpPaymentMethodType, .FPX) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("sepa_debit").stpPaymentMethodType, + .SEPADebit + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("au_becs_debit").stpPaymentMethodType, + .AUBECSDebit + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("grabpay").stpPaymentMethodType, + .grabPay + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("giropay").stpPaymentMethodType, + .giropay + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("eps").stpPaymentMethodType, .EPS) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("p24").stpPaymentMethodType, + .przelewy24 + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("bancontact").stpPaymentMethodType, + .bancontact + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("netbanking").stpPaymentMethodType, + .netBanking + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("oxxo").stpPaymentMethodType, .OXXO) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("sofort").stpPaymentMethodType, + .sofort + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("upi").stpPaymentMethodType, .UPI) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("paypal").stpPaymentMethodType, + .payPal + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("afterpay_clearpay").stpPaymentMethodType, + .afterpayClearpay + ) + XCTAssertEqual(PaymentSheet.PaymentMethodType.dynamic("blik").stpPaymentMethodType, .blik) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("wechat_pay").stpPaymentMethodType, + .weChatPay + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("boleto").stpPaymentMethodType, + .boleto + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("klarna").stpPaymentMethodType, + .klarna + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.dynamic("affirm").stpPaymentMethodType, + .affirm + ) + XCTAssertNil(PaymentSheet.PaymentMethodType.dynamic("doesNotExist").stpPaymentMethodType) + } + + func testConvertingNonDynamicTypes() { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.card.stpPaymentMethodType, + PaymentSheet.PaymentMethodType.dynamic("card").stpPaymentMethodType + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.USBankAccount.stpPaymentMethodType, + PaymentSheet.PaymentMethodType.dynamic("us_bank_account").stpPaymentMethodType + ) + XCTAssertEqual( + PaymentSheet.PaymentMethodType.link.stpPaymentMethodType, + PaymentSheet.PaymentMethodType.dynamic("link").stpPaymentMethodType + ) + } + + func testPaymentIntentRecommendedPaymentMethodTypes() { + let paymentIntent = constructPI( + paymentMethodTypes: ["card", "us_bank_account", "klarna", "futurePaymentMethod"], + orderedPaymentMethodTypes: ["card", "klarna", "us_bank_account", "futurePaymentMethod"] + )! + let intent = Intent.paymentIntent(paymentIntent) + let types = PaymentSheet.PaymentMethodType.recommendedPaymentMethodTypes(from: intent) + + XCTAssertEqual(types[0], .card) + XCTAssertEqual(types[1], .dynamic("klarna")) + XCTAssertEqual(types[2], .USBankAccount) + XCTAssertEqual(types[3], .dynamic("futurePaymentMethod")) + + } + + func testPaymentIntentRecommendedPaymentMethodTypes_withoutOrderedPaymentMethodTypes() { + let paymentIntent = constructPI(paymentMethodTypes: [ + "card", "us_bank_account", "klarna", "futurePaymentMethod", + ])! + let intent = Intent.paymentIntent(paymentIntent) + let types = PaymentSheet.PaymentMethodType.recommendedPaymentMethodTypes(from: intent) + + XCTAssertEqual(types[0], .card) + XCTAssertEqual(types[1], .USBankAccount) + XCTAssertEqual(types[2], .dynamic("klarna")) + XCTAssertEqual(types[3], .dynamic("futurePaymentMethod")) + } + + func testSetupIntentRecommendedPaymentMethodTypes() { + let setupIntent = constructSI(paymentMethodTypes: [ + "card", "us_bank_account", "klarna", "futurePaymentMethod", + ])! + let intent = Intent.setupIntent(setupIntent) + let types = PaymentSheet.PaymentMethodType.recommendedPaymentMethodTypes(from: intent) + + XCTAssertEqual(types[0], .card) + XCTAssertEqual(types[1], .USBankAccount) + XCTAssertEqual(types[2], .dynamic("klarna")) + XCTAssertEqual(types[3], .dynamic("futurePaymentMethod")) + } + + func testSetupIntentRecommendedPaymentMethodTypes_withoutOrderedPaymentMethodTypes() { + let setupIntent = constructSI( + paymentMethodTypes: ["card", "us_bank_account", "klarna", "futurePaymentMethod"], + orderedPaymentMethodTypes: ["card", "klarna", "us_bank_account", "futurePaymentMethod"] + )! + let intent = Intent.setupIntent(setupIntent) + let types = PaymentSheet.PaymentMethodType.recommendedPaymentMethodTypes(from: intent) + + XCTAssertEqual(types[0], .card) + XCTAssertEqual(types[1], .dynamic("klarna")) + XCTAssertEqual(types[2], .USBankAccount) + XCTAssertEqual(types[3], .dynamic("futurePaymentMethod")) + } + + func testPaymentIntentFilteredPaymentMethodTypes() { + let paymentIntent = constructPI( + paymentMethodTypes: ["card", "klarna", "p24"], + orderedPaymentMethodTypes: ["card", "klarna", "p24"] + )! + let intent = Intent.paymentIntent(paymentIntent) + var configuration = PaymentSheet.Configuration() + configuration.returnURL = "http://return-to-url" + configuration.allowsDelayedPaymentMethods = true + let types = PaymentSheet.PaymentMethodType.filteredPaymentMethodTypes( + from: intent, + configuration: configuration + ) + + XCTAssertEqual(types.count, 3) + XCTAssertEqual(types[0], .card) + XCTAssertEqual(types[1], .dynamic("klarna")) + XCTAssertEqual(types[2], .dynamic("p24")) + } + + func testPaymentIntentFilteredPaymentMethodTypes_withUnfulfilledRequirements() { + let paymentIntent = constructPI( + paymentMethodTypes: ["card", "klarna", "p24"], + orderedPaymentMethodTypes: ["card", "klarna", "p24"] + )! + let intent = Intent.paymentIntent(paymentIntent) + let configuration = PaymentSheet.Configuration() + let types = PaymentSheet.PaymentMethodType.filteredPaymentMethodTypes( + from: intent, + configuration: configuration + ) + + XCTAssertEqual(types.count, 1) + XCTAssertEqual(types[0], .card) + } + + func testPaymentIntentFilteredPaymentMethodTypes_withSetupFutureUsage() { + let paymentIntent = constructPI( + paymentMethodTypes: ["card", "cashapp"], + orderedPaymentMethodTypes: ["card", "cashapp", "mobilepay"], + setupFutureUsage: .onSession + )! + let intent = Intent.paymentIntent(paymentIntent) + var configuration = PaymentSheet.Configuration() + configuration.returnURL = "http://return-to-url" + configuration.allowsDelayedPaymentMethods = true + let types = PaymentSheet.PaymentMethodType.filteredPaymentMethodTypes( + from: intent, + configuration: configuration + ) + + XCTAssertEqual(types.count, 1) + XCTAssertEqual(types[0], .card) + // Cash App is not enabled for saving or reuse so it should be filtered out + } + + func testSetupIntentFilteredPaymentMethodTypes() { + let setupIntent = constructSI(paymentMethodTypes: ["card", "cashapp"])! + let intent = Intent.setupIntent(setupIntent) + var configuration = PaymentSheet.Configuration() + configuration.returnURL = "http://return-to-url" + let types = PaymentSheet.PaymentMethodType.filteredPaymentMethodTypes( + from: intent, + configuration: configuration + ) + + XCTAssertEqual(types.count, 1) + XCTAssertEqual(types[0], .card) + // Cash App is not enabled for saving or reuse so it should be filtered out + } + + func testSetupIntentFilteredPaymentMethodTypes_withoutOrderedPaymentMethodTypes() { + let setupIntent = constructSI(paymentMethodTypes: ["card", "klarna", "p24"])! + let intent = Intent.setupIntent(setupIntent) + let configuration = PaymentSheet.Configuration() + let types = PaymentSheet.PaymentMethodType.filteredPaymentMethodTypes( + from: intent, + configuration: configuration + ) + + XCTAssertEqual(types.count, 1) + XCTAssertEqual(types[0], .card) + } + + func testUnknownPMTypeIsUnsupported() { + let paymentIntent = constructPI(paymentMethodTypes: ["luxe_bucks"])! + let setupIntent = constructSI(paymentMethodTypes: ["luxe_bucks"])! + let paymentMethod = PaymentSheet.PaymentMethodType.dynamic("luxe_bucks") + var configuration = PaymentSheet.Configuration() + configuration.returnURL = "http://return-to-url" + + for intent in [Intent.setupIntent(setupIntent), Intent.paymentIntent(paymentIntent)] { + XCTAssertEqual( + PaymentSheet.PaymentMethodType.supportsAdding( + paymentMethod: paymentMethod, + configuration: configuration, + intent: intent + ), + .missingRequirements([.unavailable]) + ) + } + } + + func testSupport() { + let paymentIntent = constructPI(paymentMethodTypes: ["luxe_bucks"])! + let intent = Intent.paymentIntent(paymentIntent) + var configuration = PaymentSheet.Configuration() + configuration.returnURL = "http://return-to-url" + + XCTAssertEqual( + PaymentSheet.PaymentMethodType.configurationSatisfiesRequirements( + requirements: [.returnURL], + configuration: configuration, + intent: intent + ), + .supported + ) + } + + private func constructPI( + paymentMethodTypes: [String], + orderedPaymentMethodTypes: [String]? = nil, + setupFutureUsage: STPPaymentIntentSetupFutureUsage = .none + ) -> STPPaymentIntent? { + var apiResponse: [AnyHashable: Any?] = [ + "id": "123", + "client_secret": "sec", + "amount": 10, + "currency": "usd", + "status": "requires_payment_method", + "livemode": false, + "created": 1652736692.0, + "payment_method_types": paymentMethodTypes, + "setup_future_usage": setupFutureUsage.stringValue, + ] + if let orderedPaymentMethodTypes = orderedPaymentMethodTypes { + apiResponse["ordered_payment_method_types"] = orderedPaymentMethodTypes + } + guard + let stpPaymentIntent = STPPaymentIntent.decodeSTPPaymentIntentObject( + fromAPIResponse: apiResponse as [AnyHashable: Any] + ) + else { + XCTFail("Failed to decode") + return nil + } + return stpPaymentIntent + } + private func constructSI( + paymentMethodTypes: [String], + orderedPaymentMethodTypes: [String]? = nil + ) -> STPSetupIntent? { + var apiResponse: [AnyHashable: Any] = [ + "id": "123", + "client_secret": "sec", + "status": "requires_payment_method", + "created": 1652736692.0, + "payment_method_types": paymentMethodTypes, + "livemode": false, + ] + if let orderedPaymentMethodTypes = orderedPaymentMethodTypes { + apiResponse["ordered_payment_method_types"] = orderedPaymentMethodTypes + } + guard + let stpSetupIntent = STPSetupIntent.decodeSTPSetupIntentObject( + fromAPIResponse: apiResponse + ) + else { + XCTFail("Failed to decode") + return nil + } + return stpSetupIntent + } + +} + +extension STPFixtures { + static func makePaymentIntent( + paymentMethodTypes: [STPPaymentMethodType]? = nil, + setupFutureUsage: STPPaymentIntentSetupFutureUsage? = nil, + paymentMethodOptions: STPPaymentMethodOptions? = nil, + shippingProvided: Bool = false + ) -> STPPaymentIntent { + var json = STPTestUtils.jsonNamed(STPTestJSONPaymentIntent)! + if let setupFutureUsage = setupFutureUsage { + json["setup_future_usage"] = setupFutureUsage.stringValue + } + if let paymentMethodTypes = paymentMethodTypes { + json["payment_method_types"] = paymentMethodTypes.map { + STPPaymentMethod.string(from: $0) + } + } + if !shippingProvided { + // The payment intent json already has shipping on it, so just remove it if needed + json["shipping"] = nil + } + if let paymentMethodOptions = paymentMethodOptions { + json["payment_method_options"] = paymentMethodOptions.dictionaryValue + } + let pi = STPPaymentIntent.decodedObject(fromAPIResponse: json) + return pi! + } +} diff --git a/Stripe/StripeiOSTests/PaymentSheetTestUtils.swift b/Stripe/StripeiOSTests/PaymentSheetTestUtils.swift new file mode 100644 index 00000000..361e832c --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentSheetTestUtils.swift @@ -0,0 +1,53 @@ +// +// PaymentSheetTestUtils.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/16/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class PaymentSheetTestUtils { + // Copy and pasted from PaymentSheetSnapshotTests + static var snapshotTestTheme: PaymentSheet.Appearance { + var appearance = PaymentSheet.Appearance() + + // Customize the font + var font = PaymentSheet.Appearance.Font() + font.sizeScaleFactor = 0.85 + font.base = UIFont(name: "AvenirNext-Regular", size: 12)! + + appearance.cornerRadius = 0.0 + appearance.borderWidth = 2.0 + appearance.shadow = PaymentSheet.Appearance.Shadow( + color: .orange, + opacity: 0.5, + offset: CGSize(width: 0, height: 2), + radius: 4 + ) + + // Customize the colors + var colors = PaymentSheet.Appearance.Colors() + colors.primary = .systemOrange + colors.background = .cyan + colors.componentBackground = .yellow + colors.componentBorder = .systemRed + colors.componentDivider = .black + colors.text = .red + colors.textSecondary = .orange + colors.componentText = .red + colors.componentPlaceholderText = .systemBlue + colors.icon = .green + colors.danger = .purple + + appearance.font = font + appearance.colors = colors + + return appearance + } +} diff --git a/Stripe/StripeiOSTests/PaymentTypeCellSnapshotTests.swift b/Stripe/StripeiOSTests/PaymentTypeCellSnapshotTests.swift new file mode 100644 index 00000000..ba83bcae --- /dev/null +++ b/Stripe/StripeiOSTests/PaymentTypeCellSnapshotTests.swift @@ -0,0 +1,81 @@ +// +// PaymentTypeCellSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 12/17/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class PaymentTypeCellSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testCardUnselected() { + let cell = PaymentMethodTypeCollectionView.PaymentTypeCell() + cell.paymentMethodType = .card + cell.frame = CGRect( + origin: .zero, + size: CGSize( + width: PaymentMethodTypeCollectionView.cellHeight, + height: PaymentMethodTypeCollectionView.cellHeight + ) + ) + STPSnapshotVerifyView(cell) + } + + func testCardSelected() { + let cell = PaymentMethodTypeCollectionView.PaymentTypeCell() + cell.paymentMethodType = .card + cell.frame = CGRect( + origin: .zero, + size: CGSize( + width: PaymentMethodTypeCollectionView.cellHeight, + height: PaymentMethodTypeCollectionView.cellHeight + ) + ) + cell.isSelected = true + STPSnapshotVerifyView(cell) + } + + @available(iOS 13.0, *) + func testCardUnselected_forceDarkMode() { + let cell = PaymentMethodTypeCollectionView.PaymentTypeCell() + cell.overrideUserInterfaceStyle = .dark + cell.paymentMethodType = .card + cell.frame = CGRect( + origin: .zero, + size: CGSize( + width: PaymentMethodTypeCollectionView.cellHeight, + height: PaymentMethodTypeCollectionView.cellHeight + ) + ) + STPSnapshotVerifyView(cell) + } + + @available(iOS 13.0, *) + func testCardSelected_forceDarkMode() { + let cell = PaymentMethodTypeCollectionView.PaymentTypeCell() + cell.overrideUserInterfaceStyle = .dark + cell.paymentMethodType = .card + cell.frame = CGRect( + origin: .zero, + size: CGSize( + width: PaymentMethodTypeCollectionView.cellHeight, + height: PaymentMethodTypeCollectionView.cellHeight + ) + ) + cell.isSelected = true + STPSnapshotVerifyView(cell) + } +} diff --git a/Stripe/StripeiOSTests/Resources/3DSSource.json b/Stripe/StripeiOSTests/Resources/3DSSource.json new file mode 100644 index 00000000..bff15783 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/3DSSource.json @@ -0,0 +1,57 @@ +// Source: https://stripe.com/docs/sources/three-d-secure +{ + "id": "src_456", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_456", + "created": 1483663790, + "currency": "eur", + "flow": "redirect", + "livemode": false, + "metadata": {}, + "owner": { + "address": { + "city": "Pittsburgh", + "country": "US", + "line1": "123 Fake St", + "line2": "Apt 1", + "postal_code": "19219", + "state": "PA" + }, + "email": "jenny.rosen@example.com", + "name": "Jenny Rosen", + "phone": "555-867-5309", + "verified_address": { + "city": "Pittsburgh", + "country": "US", + "line1": "123 Fake St", + "line2": "Apt 1", + "postal_code": "19219", + "state": "PA" + }, + "verified_email": "jenny.rosen@example.com", + "verified_name": "Jenny Rosen", + "verified_phone": "555-867-5309" + }, + "receiver": { + "address": "test_1MBhWS3uv4ynCfQXF3xQjJkzFPukr4K56N", + "amount_charged": 300, + "amount_received": 200, + "amount_returned": 100, + "refund_attributes_method": "email", + "refund_attributes_status": "missing" + }, + "redirect": { + "return_url": "exampleappschema://stripe_callback", + "status": "pending", + "url": "https://hooks.stripe.com/redirect/authenticate/src_19YlvWAHEMiOZZp1QQlOD79v?client_secret=src_client_secret_kBwCSm6Xz5MQETiJ43hUH8qv" + }, + "status": "pending", + "type": "three_d_secure", + "usage": "single_use", + "three_d_secure": { + "card": "src_123", + "customer": null, + "authenticated": false + } +} diff --git a/Stripe/StripeiOSTests/Resources/AlipaySource.json b/Stripe/StripeiOSTests/Resources/AlipaySource.json new file mode 100644 index 00000000..a38ae906 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/AlipaySource.json @@ -0,0 +1,34 @@ +// Source: https://stripe.com/docs/sources/alipay +{ + "id": "src_123", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_123", + "created": 1445277809, + "currency": "usd", + "flow": "redirect", + "livemode": true, + "owner": { + "address": null, + "email": null, + "name": null, + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": null, + "verified_phone": null, + }, + "redirect": { + "return_url": "https://shop.foo.com/crtABC", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_123?client_secret=src_client_secret_123" + }, + "statement_descriptor": null, + "status": "pending", + "type": "alipay", + "usage": "single_use", + "alipay": { + "statement_descriptor": null, + "native_url": null + } +} diff --git a/Stripe/StripeiOSTests/Resources/ApplePayPaymentMethod.json b/Stripe/StripeiOSTests/Resources/ApplePayPaymentMethod.json new file mode 100644 index 00000000..8831c871 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/ApplePayPaymentMethod.json @@ -0,0 +1,30 @@ +{ + "card": { + "brand": "visa", + "checks": { + }, + "country": "US", + "exp_month": 12, + "exp_year": 2020, + "funding": "credit", + "generated_from": null, + "last4": 4242, + "three_d_secure_usage": { + "supported": 1 + }, + "wallet": { + "apple_pay": { + }, + "dynamic_last4": 4242, + "type": "apple_pay" + } + }, + "created": 1558545782, + "id": "pm_123456789", + "livemode": false, + "metadata": { + }, + "object": "payment_method", + "type": "card" +} + diff --git a/Stripe/StripeiOSTests/Resources/BacsDebitPaymentMethod.json b/Stripe/StripeiOSTests/Resources/BacsDebitPaymentMethod.json new file mode 100644 index 00000000..f42e15cb --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/BacsDebitPaymentMethod.json @@ -0,0 +1,28 @@ +{ + "id": "pm_1G4EmqKlwPmebFhpaMbye30L", + "object": "payment_method", + "bacs_debit": { + "fingerprint": "9eMbmctOrd8i7DYa", + "last4": "2345", + "sort_code": "108800" + }, + "billing_details": { + "address": { + "city": "f", + "country": "US", + "line1": "f", + "line2": "f", + "postal_code": "f", + "state": "FL" + }, + "email": "f@f.1", + "name": "f", + "phone": null + }, + "created": 1579820912, + "customer": "cus_GWUuPERgJF44Dm", + "livemode": false, + "metadata": { + }, + "type": "bacs_debit" +} diff --git a/Stripe/StripeiOSTests/Resources/BancontactSource.json b/Stripe/StripeiOSTests/Resources/BancontactSource.json new file mode 100644 index 00000000..4a9cf384 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/BancontactSource.json @@ -0,0 +1,38 @@ +// Source: https://stripe.com/docs/sources/bancontact +{ + "id": "src_16xhynE8WzK49JbAs9M21jaR", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU", + "created": 1445277809, + "currency": "eur", + "statement_descriptor": null, + "flow": "redirect", + "livemode": true, + "owner": { + "address": null, + "email": null, + "name": "Jenny Rosen", + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": "Jenny Rosen", + "verified_phone": null + }, + "redirect": { + "return_url": "https://shop.example.com/crtA6B28E1", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_16xhynE8WzK49JbAs9M21jaR?client_secret=src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU" + }, + "status": "pending", + "type": "bancontact", + "usage": "single_use", + "bancontact": { + "bank_code": null, + "bic": null, + "bank_name": null, + "iban_last4": null, + "statement_descriptor": null, + "preferred_language": null + } +} diff --git a/Stripe/StripeiOSTests/Resources/BankAccount.json b/Stripe/StripeiOSTests/Resources/BankAccount.json new file mode 100644 index 00000000..0f7c2227 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/BankAccount.json @@ -0,0 +1,18 @@ +{ + "id": "ba_1AZmya2eZvKYlo2CQzt7Fwnz", + "object": "bank_account", + "account": "acct_1032D82eZvKYlo2C", + "account_holder_name": "Jane Austen", + "account_holder_type": "individual", + "bank_name": "STRIPE TEST BANK", + "country": "US", + "currency": "usd", + "default_for_currency": false, + "fingerprint": "1JWtPxqbdX5Gamtc", + "last4": "6789", + "metadata": { + "order_id": "6735" + }, + "routing_number": "110000000", + "status": "new" +} diff --git a/Stripe/StripeiOSTests/Resources/Card.json b/Stripe/StripeiOSTests/Resources/Card.json new file mode 100644 index 00000000..14e3c133 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/Card.json @@ -0,0 +1,26 @@ +// Source: https://stripe.com/docs/api#card_object +{ + "id": "card_103kbR2eZvKYlo2CDczLmw4K", + "object": "card", + "address_city": "Pittsburgh", + "address_country": "US", + "address_line1": "123 Fake St", + "address_line1_check": "pass", + "address_line2": "Apt 1", + "address_state": "PA", + "address_zip": "19219", + "address_zip_check": "pass", + "brand": "Visa", + "country": "US", + "customer": "cus_3kbRl1kVJmQ4Ur", + "currency": "usd", + "cvc_check": "pass", + "dynamic_last4": "5678", + "exp_month": 5, + "exp_year": 2017, + "fingerprint": "Xt5EWLLDS7FJjR1c", + "funding": "credit", + "last4": "4242", + "name": "Jane Austen", + "tokenization_method": null +} diff --git a/Stripe/StripeiOSTests/Resources/CardPaymentMethod.json b/Stripe/StripeiOSTests/Resources/CardPaymentMethod.json new file mode 100644 index 00000000..4541a2fc --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/CardPaymentMethod.json @@ -0,0 +1,61 @@ +// Source: https://stripe.com/docs/api/payment_methods/object +{ + "id": "pm_123456789", + "object": "payment_method", + "billing_details": { + "address": { + "city": "München", + "country": "DE", + "postal_code": "80337", + "line1": "Marienplatz", + "line2": "8", + "state": "Bayern", + }, + "email": "jenny@example.com", + "phone": "+15555555555", + "name": "jenny", + }, + "card": { + "brand": "visa", + "checks": { + }, + "country": "US", + "description": "Visa Classic", + "exp_month": 8, + "exp_year": 2020, + "fingerprint": "6gVyxfIhqc8Z0g0X", + "funding": "credit", + "iin": "424242", + "issuer": "Stripe Payments UK Limited", + "last4": "4242", + "three_d_secure_usage": { + "supported": true + }, + "wallet": { + "type": "visa_checkout", + "visa_checkout": { + "name": "Jenny", + "email": "jenny@example.com", + "billing_address": { + "city": "München", + "country": "DE", + "postal_code": "80337", + "line1": "Marienplatz", + "line2": "8", + "state": "Bayern", + }, + "shipping_address": { + "city": "München", + "country": "DE", + "postal_code": "80337", + "line1": "Marienplatz", + "line2": "8", + "state": "Bayern", + }, + } + } + }, + "created": 123456789, + "livemode": false, + "type": "card" +} diff --git a/Stripe/StripeiOSTests/Resources/CardSource.json b/Stripe/StripeiOSTests/Resources/CardSource.json new file mode 100644 index 00000000..415e4aaa --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/CardSource.json @@ -0,0 +1,33 @@ +// Source: https://stripe.com/docs/sources/cards +{ + "id": "src_123", + "object": "source", + "amount": null, + "client_secret": "src_client_secret_123", + "created": 1483575790, + "currency": null, + "flow": "none", + "livemode": false, + "owner": { + "address": null, + "email": null, + "name": null, + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": null, + "verified_phone": null + }, + "status": "chargeable", + "type": "card", + "usage": "reusable", + "card": { + "brand": "Visa", + "country": "US", + "exp_month": 12, + "exp_year": 2034, + "funding": "debit", + "last4": "5556", + "three_d_secure": "not_supported" + } +} diff --git a/Stripe/StripeiOSTests/Resources/Customer.json b/Stripe/StripeiOSTests/Resources/Customer.json new file mode 100644 index 00000000..79dcda8c --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/Customer.json @@ -0,0 +1,26 @@ +{ + "id": "cus_123", + "object": "customer", + "created": 1463413795, + "default_source": "card_123", + "livemode": false, + "shipping": { + "address": { + "city": "Baltimore", + "country": "AF", + "line1": "67 Ave C", + "line2": "3A", + "postal_code": "10002", + "state": "MD" + }, + "name": "Ben Guo", + "phone": "(555) 555-5555" + }, + "sources": { + "object": "list", + "data": [], + "has_more": false, + "total_count": 0, + "url": "/v1/customers/cus_8SjRZeRUyWjiBz/sources" + }, +} diff --git a/Stripe/StripeiOSTests/Resources/EPSSource.json b/Stripe/StripeiOSTests/Resources/EPSSource.json new file mode 100644 index 00000000..36e2da04 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/EPSSource.json @@ -0,0 +1,34 @@ +// Source: https://stripe.com/docs/sources/eps +{ + "id": "src_16xhynE8WzK49JbAs9M21jaR", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU", + "created": 1445277809, + "currency": "eur", + "flow": "redirect", + "livemode": true, + "owner": { + "address": null, + "email": null, + "name": "Jenny Rosen", + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": "Jenny Rosen", + "verified_phone": null + }, + "redirect": { + "return_url": "https://shop.example.com/crtA6B28E1", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_16xhynE8WzK49JbAs9M21jaR?client_secret=src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU" + }, + "statement_descriptor": null, + "status": "pending", + "type": "eps", + "usage": "single_use", + "eps": { + "reference": null, + "statement_descriptor": null, + } +} diff --git a/Stripe/StripeiOSTests/Resources/ElementsSession.json b/Stripe/StripeiOSTests/Resources/ElementsSession.json new file mode 100644 index 00000000..a9a4fe5d --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/ElementsSession.json @@ -0,0 +1,199 @@ +{ + "account_id": null, + "apple_pay_preference": "enabled", + "business_name": "Mobile Example Account", + "experiments": { + "element_link_autofill_in_link_authentication_element": "control", + "element_link_autofill_in_payment_element": "control_test", + "elements_link_aa": "control", + "elements_link_in_payment_element_only": "treatment", + "elements_link_in_payment_element_only_holdback": "control", + "elements_link_longterm_holdback": "control", + "lpm_discoverability_upe_experiment_1": "control" + }, + "experiments_data": { + "arb_id": "35386406-1724-456b-ab1d-a4c4d8659098", + "experiment_assignments": { + "element_link_autofill_in_link_authentication_element": "control", + "element_link_autofill_in_payment_element": "control_test", + "elements_link_aa": "control", + "elements_link_in_payment_element_only": "treatment", + "elements_link_in_payment_element_only_holdback": "control", + "elements_link_longterm_holdback": "control", + "lpm_discoverability_upe_experiment_1": "control" + } + }, + "flags": { + "elements_disable_paypal_express": true, + "elements_enable_blik": true, + "elements_enable_br_card_installments": false, + "elements_enable_deferred_intent": false, + "elements_enable_demo_pay": false, + "elements_enable_express_checkout": false, + "elements_enable_external_payment_method_paypal": false, + "elements_enable_external_payment_method_venmo": false, + "elements_enable_mobilepay": false, + "elements_enable_mx_card_installments": false, + "elements_enable_payment_element_vertical_layout": false, + "elements_enable_revolut_pay": false, + "elements_link_enable_email_domain_correction": false, + "elements_lpm_discoverability_downward_arrow": false, + "elements_lpm_discoverability_rotating_cycle": false, + "elements_web_lpm_server_driven_ui": true, + "financial_connections_enable_deferred_intent_flow": false, + "merchant_success_log_element_is_visible": true + }, + "google_pay_preference": "enabled", + "link_consumer_info": null, + "link_settings": { + "link_authenticated_change_event_enabled": false, + "link_bank_incentives_enabled": false, + "link_bank_onboarding_enabled": false, + "link_crypto_onramp_bank_upsell": false, + "link_crypto_onramp_elements_logout_disabled": false, + "link_crypto_onramp_force_cvc_reverification": false, + "link_elements_is_crypto_onramp": false, + "link_elements_pageload_sign_up_disabled": false, + "link_email_verification_login_enabled": false, + "link_financial_incentives_experiment_enabled": false, + "link_funding_sources": [ + "CARD" + ], + "link_instant_debits_create_link_account_session_on_instantiation": false, + "link_local_storage_login_enabled": false, + "link_m2_default_integration_enabled": true, + "link_only_for_payment_method_types_enabled": false, + "link_passthrough_mode_enabled": false, + "link_pay_button_element_enabled": true, + "link_session_storage_login_enabled": true + }, + "merchant_country": "US", + "merchant_currency": "usd", + "merchant_id": "acct_1HvTI7Lu5o3P18Zp", + "meta_pay_signed_container_context": null, + "order": null, + "ordered_payment_method_types_and_wallets": [ + "card", + "link", + "apple_pay", + "google_pay", + "us_bank_account", + "afterpay_clearpay", + "klarna", + "cashapp", + "alipay", + "wechat_pay" + ], + "payment_method_preference": { + "object": "payment_method_preference", + "country_code": "US", + "ordered_payment_method_types": [ + "card", + "link", + "us_bank_account", + "afterpay_clearpay", + "klarna", + "cashapp", + "alipay", + "wechat_pay" + ], + "type": "deferred_intent" + }, + "payment_method_specs": [ + { + "async": false, + "fields": [ + { + "type": "afterpay_header" + }, + { + "api_path": { + "v1": "billing_details[name]" + }, + "type": "name" + }, + { + "api_path": { + "v1": "billing_details[email]" + }, + "type": "email" + }, + { + "allowed_country_codes": null, + "type": "billing_address" + } + ], + "selector_icon": { + "light_theme_png": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-afterpay@3x-6776ded2b20306c85d02639aea1e7dc5.png", + "light_theme_svg": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-afterpay-abedc6b87e4e9f917e22bbe6648ba809.svg" + }, + "type": "afterpay_clearpay" + }, + { + "async": false, + "fields": [], + "selector_icon": { + "light_theme_png": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-alipay.png", + "light_theme_svg": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/alipay-22c167d415e209c71b2ac68b7fbc9f43.svg" + }, + "type": "alipay" + }, + { + "async": false, + "fields": [], + "type": "card" + }, + { + "async": false, + "fields": [], + "selector_icon": { + "light_theme_png": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-cashapp@3x-a89c5d8d0651cae2a511bb49a6be1cfc.png", + "light_theme_svg": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-cashapp-981164a833e417d28a8ac2684fda2324.svg" + }, + "type": "cashapp" + }, + { + "async": false, + "fields": [ + { + "type": "klarna_header" + }, + { + "api_path": { + "v1": "billing_details[email]" + }, + "type": "email" + }, + { + "api_path": { + "v1": "billing_details[address][country]" + }, + "type": "klarna_country" + } + ], + "selector_icon": { + "light_theme_png": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-klarna@3x-d8624aa9a5662d719a44d16b9fcca0be.png", + "light_theme_svg": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-klarna-bb91aa8f173a3c72931696b0f752ec73.svg" + }, + "type": "klarna" + }, + { + "async": false, + "fields": [], + "selector_icon": { + "light_theme_png": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/wechat_pay.png", + "light_theme_svg": "https://js.stripe.com/v3/fingerprinted/img/payment-methods/icon-pm-wechat-pay-f62a5a27f646cb5f596c610475d14444.svg" + }, + "type": "wechat_pay" + } + ], + "paypal_express_config": { + "client_id": null, + "paypal_merchant_id": null + }, + "session_id": "elements_session_1MFNQu2rLVE", + "shipping_address_settings": { + "autocomplete_allowed": true + }, + "unactivated_payment_method_types": ["cashapp"] +} diff --git a/Stripe/StripeiOSTests/Resources/EphemeralKey.json b/Stripe/StripeiOSTests/Resources/EphemeralKey.json new file mode 100644 index 00000000..1b3272ad --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/EphemeralKey.json @@ -0,0 +1,14 @@ +{ + "id": "ephkey_123", + "object": "ephemeral_key", + "secret": "ek_test_123", + "created": 1483575790, + "livemode": false, + "expires": 1483579790, + "associated_objects": [ + { + "type": "customer", + "id": "cus_123" + } + ] +} diff --git a/Stripe/StripeiOSTests/Resources/FileUpload.json b/Stripe/StripeiOSTests/Resources/FileUpload.json new file mode 100644 index 00000000..e03a733f --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/FileUpload.json @@ -0,0 +1,10 @@ +// Source: https://stripe.com/docs/api#file_upload_object +{ + "id": "file_1AZl0o2eZvKYlo2CoIkwLzfd", + "object": "file_upload", + "created": 1498674938, + "purpose": "dispute_evidence", + "size": 34478, + "type": "jpg", + "url": "https://stripe-upload-api.s3.amazonaws.com/uploads/file_1AXyapEOD54MuFwSnhlqqvsX?AWSAccessKeyId=KEY_ID&Expires=TIMESTAMP&Signature=SIGNATURE" +} diff --git a/Stripe/StripeiOSTests/Resources/GiropaySource.json b/Stripe/StripeiOSTests/Resources/GiropaySource.json new file mode 100644 index 00000000..a95b7df1 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/GiropaySource.json @@ -0,0 +1,36 @@ +// Source: https://stripe.com/docs/sources/giropay +{ + "id": "src_16xhynE8WzK49JbAs9M21jaR", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU", + "created": 1445277809, + "currency": "eur", + "flow": "redirect", + "livemode": true, + "owner": { + "address": null, + "email": null, + "name": null, + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": "Jenny Rosen", + "verified_phone": null + }, + "redirect": { + "return_url": "https://shop.example.com/crtA6B28E1", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_16xhynE8WzK49JbAs9M21jaR?client_secret=src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU" + }, + "statement_descriptor": null, + "status": "pending", + "type": "giropay", + "usage": "single_use", + "giropay": { + "bank_code": null, + "bic": null, + "bank_name": null, + "statement_descriptor": null + } +} diff --git a/Stripe/StripeiOSTests/Resources/Images.xcassets/Contents.json b/Stripe/StripeiOSTests/Resources/Images.xcassets/Contents.json new file mode 100644 index 00000000..da4a164c --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/Images.xcassets/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "version" : 1, + "author" : "xcode" + } +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/MockFiles/paymentIntentResponse.json b/Stripe/StripeiOSTests/Resources/MockFiles/paymentIntentResponse.json new file mode 100644 index 00000000..53200cdf --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/MockFiles/paymentIntentResponse.json @@ -0,0 +1,53 @@ +{ + "id": "pi_3LN3c2L123456789", + "object": "payment_intent", + "amount": 5099, + "amount_details": { + "tip": {} + }, + "automatic_payment_methods": null, + "canceled_at": null, + "cancellation_reason": null, + "capture_method": "automatic", + "client_secret": "pi_123456_secret_654321", + "confirmation_method": "automatic", + "created": 1658187870, + "currency": "usd", + "description": null, + "last_payment_error": null, + "livemode": false, + "next_action": , + "payment_method": , + "payment_method_options": { + "us_bank_account": { + "verification_method": "automatic" + } + }, + "payment_method_types": [ + "card", + "afterpay_clearpay", + "klarna", + "us_bank_account", + "affirm", + "blik" + ], + "processing": null, + "receipt_email": null, + "setup_future_usage": null, + "shipping": { + "address": { + "city": "San Francisco", + "country": "US", + "line1": "510 Townsend St", + "line2": null, + "postal_code": "94102", + "state": "California" + }, + "carrier": null, + "name": "John Doe", + "phone": null, + "tracking_number": null + }, + "source": null, + "status": +} diff --git a/Stripe/StripeiOSTests/Resources/MultibancoSource.json b/Stripe/StripeiOSTests/Resources/MultibancoSource.json new file mode 100644 index 00000000..05250c28 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/MultibancoSource.json @@ -0,0 +1,42 @@ +// Source: https://stripe.com/docs/sources/multibanco +{ + "id": "src_16xhynE8WzK49JbAs9M21jaR", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU", + "created": 1445277809, + "currency": "eur", + "flow": "receiver", + "livemode": true, + "owner": { + "address": null, + "email": null, + "name": "Jenny Rosen", + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": "Jenny Rosen", + "verified_phone": null + }, + "redirect": { + "return_url": "https://shop.example.com/crtA6B28E1", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_16xhynE8WzK49JbAs9M21jaR?client_secret=src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU" + }, + "receiver": { + "address": "12345-123456789", + "amount_charged": 0, + "amount_received": 0, + "amount_returned": 0, + "refund_attributes_method": "email", + "refund_attributes_status": "missing" + }, + "statement_descriptor": null, + "status": "pending", + "type": "multibanco", + "usage": "single_use", + "multibanco": { + "reference": "12345", + "entity": "123456789", + } +} diff --git a/Stripe/StripeiOSTests/Resources/P24Source.json b/Stripe/StripeiOSTests/Resources/P24Source.json new file mode 100644 index 00000000..5948340c --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/P24Source.json @@ -0,0 +1,33 @@ +// Source: https://stripe.com/docs/sources/p24 +{ + "id": "src_16xhynE8WzK49JbAs9M21jaR", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU", + "created": 1445277809, + "currency": "eur", + "flow": "redirect", + "livemode": true, + "owner": { + "address": null, + "email": "jenny.rosen@example.com", + "name": "Jenny Rosen", + "phone": null, + "verified_address": null, + "verified_email": "jenny.rosen@example.com", + "verified_name": "Jenny Rosen", + "verified_phone": null + }, + "redirect": { + "return_url": "https://shop.example.com/crtA6B28E1", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_16xhynE8WzK49JbAs9M21jaR?client_secret=src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU" + }, + "statement_descriptor": null, + "status": "pending", + "type": "p24", + "usage": "single_use", + "p24": { + "reference": "P24-000-111-222" + } +} diff --git a/Stripe/StripeiOSTests/Resources/PaymentIntent.json b/Stripe/StripeiOSTests/Resources/PaymentIntent.json new file mode 100644 index 00000000..c0e7bfad --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/PaymentIntent.json @@ -0,0 +1,82 @@ +{ + "id": "pi_1Cl15wIl4IdHmuTbCWrpJXN6", + "object": "payment_intent", + "payment_method_types": [ + "card" + ], + "amount": 2345, + "canceled_at": 1530911045, + "capture_method": "manual", + "client_secret": "pi_1Cl15wIl4IdHmuTbCWrpJXN6_secret_EkKtQ7Sg75hLDFKqFG8DtWcaK", + "confirmation_method": "automatic", + "created": 1530911040, + "currency": "usd", + "description": "My Sample PaymentIntent", + "last_payment_error": { + "code": "payment_intent_authentication_failure", + "doc_url": "https://stripe.com/docs/error-codes#payment-intent-authentication-failure", + "message": "The provided PaymentMethod has failed authentication. You can provide payment_method_data or a new PaymentMethod to attempt to fulfill this PaymentIntent again.", + "payment_method": { + "id": "pm_1F5KZ4KlwPmebFhph828lKsZ", + "object": "payment_method", + "billing_details": { + "address": { + }, + }, + "card": { + "brand": "visa", + "checks": { + }, + "country": "US", + "exp_month": 2, + "exp_year": 2042, + "funding": "credit", + "last4": "3063", + "three_d_secure_usage": { + "supported": true + }, + }, + "created": 1565305114, + "livemode": false, + "metadata": { + }, + "type": "card" + }, + "type": "invalid_request_error" + }, + "livemode": false, + "next_source_action": { + "type": "authorize_with_url", + "authorize_with_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/redirect/authenticate/src_1Cl1AeIl4IdHmuTb1L7x083A?client_secret=src_client_secret_DBNwUe9qHteqJ8qQBwNWiigk" + } + }, + "next_action": { + "type": "redirect_to_url", + "redirect_to_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/redirect/authenticate/src_1Cl1AeIl4IdHmuTb1L7x083A?client_secret=src_client_secret_DBNwUe9qHteqJ8qQBwNWiigk" + } + }, + "receipt_email": "danj@example.com", + "shipping": { + "address": { + "city": "San Francisco", + "country": "USA", + "line1": "123 Main St", + "line2": "Apt 456", + "postal_code": "94107", + "state": "CA" + }, + "carrier": "USPS", + "name": "Dan", + "phone": "1-415-555-1234", + "tracking_number": "xyz123abc" + }, + "source": "src_1Cl1AdIl4IdHmuTbseiDWq6m", + "status": "requires_action", + "payment_method_types" : [ + "card" + ], +} diff --git a/Stripe/StripeiOSTests/Resources/SEPADebitSource.json b/Stripe/StripeiOSTests/Resources/SEPADebitSource.json new file mode 100644 index 00000000..18abb191 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/SEPADebitSource.json @@ -0,0 +1,38 @@ +// Source: https://stripe.com/docs/sources/sepa-debit +{ + "id": "src_18HgGjHNCLa1Vra6Y9TIP6tU", + "object": "source", + "amount": null, + "client_secret": "src_client_secret_XcBmS94nTg5o0xc9MSliSlDW", + "created": 1464803577, + "currency": "eur", + "flow": "none", + "livemode": false, + "owner": { + "address": null, + "email": null, + "name": "Jenny Rosen", + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": null, + "verified_phone": null + }, + "status": "chargeable", + "type": "sepa_debit", + "usage": "reusable", + "sepa_debit": { + "bank_code": "37040044", + "branch_code": "a_branch", + "country": "DE", + "fingerprint": "NxdSyRegc9PsMkWy", + "last4": "3001", + "mandate": "NXDSYREGC9PSMKWY", + "mandate_reference": "NXDSYREGC9PSMKWY", + "mandate_url": "https://hooks.stripe.com/adapter/sepa_debit/file/src_18HgGjHNCLa1Vra6Y9TIP6tU/src_client_secret_XcBmS94nTg5o0xc9MSliSlDW" + }, + "verification": { + "attempts_remaining": 5, + "status": "pending" + } +} diff --git a/Stripe/StripeiOSTests/Resources/SetupIntent.json b/Stripe/StripeiOSTests/Resources/SetupIntent.json new file mode 100644 index 00000000..1b293c3e --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/SetupIntent.json @@ -0,0 +1,71 @@ +{ + "id": "seti_123456789", + "object": "setup_intent", + "application": "ca_123456789", + "client_secret": "seti_123456789_secret_123456789", + "created": 123456789, + "customer": "cus_123456", + "description": "My Sample SetupIntent", + "last_setup_error": { + "code": "setup_intent_authentication_failure", + "doc_url": "https://stripe.com/docs/error-codes#setup-intent-authentication-failure", + "message": "The latest attempt to set up the payment method has failed because authentication failed.", + "payment_method": { + "id": "pm_1F5fdSKlwPmebFhpPTD7Tg4l", + "object": "payment_method", + "billing_details": { + "address": { + "city": null, + "country": null, + "line1": null, + "line2": null, + "postal_code": null, + "state": null + }, + "email": null, + "name": null, + "phone": null + }, + "card": { + "brand": "visa", + "checks": { + "address_line1_check": null, + "address_postal_code_check": null, + "cvc_check": null + }, + "country": null, + "exp_month": 2, + "exp_year": 2042, + "funding": "credit", + "generated_from": null, + "last4": "3246", + "three_d_secure_usage": { + "supported": true + }, + "wallet": null + }, + "created": 1565386110, + "customer": null, + "livemode": false, + "metadata": { + }, + "type": "card" + }, + "type": "invalid_request_error" + }, + "livemode": false, + "next_action": { + "type": "redirect_to_url", + "redirect_to_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/redirect/authenticate/src_1Cl1AeIl4IdHmuTb1L7x083A?client_secret=src_client_secret_DBNwUe9qHteqJ8qQBwNWiigk" + } + }, + "on_behalf_of": null, + "payment_method": "pm_123456", + "payment_method_types": [ + "card" + ], + "status": "requires_action", + "usage": "off_session" +} diff --git a/Stripe/StripeiOSTests/Resources/SofortSource.json b/Stripe/StripeiOSTests/Resources/SofortSource.json new file mode 100644 index 00000000..83349507 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/SofortSource.json @@ -0,0 +1,39 @@ +// Source: https://stripe.com/docs/sources/sofort +{ + "id": "src_16xhynE8WzK49JbAs9M21jaR", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU", + "created": 1445277809, + "currency": "eur", + "flow": "redirect", + "livemode": true, + "owner": { + "address": null, + "email": null, + "name": "Jenny Rosen", + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": "Jenny Rosen", + "verified_phone": null + }, + "redirect": { + "return_url": "https://shop.example.com/crtA6B28E1", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_16xhynE8WzK49JbAs9M21jaR?client_secret=src_client_secret_UfwvW2WHpZ0s3QEn9g5x7waU" + }, + "statement_descriptor": null, + "status": "pending", + "type": "sofort", + "usage": "single_use", + "sofort": { + "country": "DE", + "bank_code": null, + "bic": null, + "bank_name": null, + "iban_last4": null, + "preferred_language": null, + "statement_descriptor": null + } +} diff --git a/Stripe/StripeiOSTests/Resources/WeChatPaySource.json b/Stripe/StripeiOSTests/Resources/WeChatPaySource.json new file mode 100644 index 00000000..97dc5738 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/WeChatPaySource.json @@ -0,0 +1,40 @@ +// Source: https://stripe.com/docs/sources/wechat-pay +{ + "id" : "src_1FCC54BNJ02ErVOjfA2jH45g", + "livemode" : true, + "amount" : 1010, + "metadata" : { + + }, + "owner" : { + "address" : null, + "phone" : null, + "verified_address" : null, + "verified_phone" : null, + "verified_email" : null, + "verified_name" : null, + "email" : null, + "name" : null + }, + "statement_descriptor" : null, + "usage" : "single_use", + "type" : "wechat", + "wechat" : { + "android_sign" : "2709CDC079DC1BB820FDB69941BE85C8EB9363D0057A7A3A3F521E64F7F08D52", + "android_package" : "Sign=WXPay", + "android_appId" : "wxa0df51ec63e578ce", + "ios_native_url" : "weixin:\/\/app\/wxa0df51ec63e578ce\/pay\/?appId=wxa0df51ec63e578ce&nonceStr=NLc6rkkjVIswJ43S&package=Sign%3DWXPay&partnerId=268716457&prepayId=wx280519598839065fb52b27b81487775200&timeStamp=1566940800&sign=2709CDC079DC1BB820FDB69941BE85C8EB9363D0057A7A3A3F521E64F7F08D52", + "android_timeStamp" : "1566940800", + "qr_code_url" : null, + "android_partnerId" : "268716457", + "android_nonceStr" : "NLc6rkkjVIswJ43S", + "android_prepayId" : "wx280519598839065fb52b27b81487775200" + }, + "object" : "source", + "created" : 1566940798, + "client_secret" : "src_client_secret_FhbHtMh4dGYSAE7UBnhDhsNb", + "flow" : "none", + "currency" : "usd", + "status" : "pending" +} + diff --git a/Stripe/StripeiOSTests/Resources/iDEALSource.json b/Stripe/StripeiOSTests/Resources/iDEALSource.json new file mode 100644 index 00000000..2ea86eea --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/iDEALSource.json @@ -0,0 +1,32 @@ +// Source: https://stripe.com/docs/sources/ideal +{ + "id": "src_123", + "object": "source", + "amount": 1099, + "client_secret": "src_client_secret_123", + "created": 1445277809, + "currency": "eur", + "flow": "redirect", + "livemode": true, + "owner": { + "address": null, + "email": null, + "name": "Jenny Rosen", + "phone": null, + "verified_address": null, + "verified_email": null, + "verified_name": "Jenny Rosen", + "verified_phone": null, + }, + "redirect": { + "return_url": "https://shop.foo.com/crtABC", + "status": "pending", + "url": "https://pay.stripe.com/redirect/src_123?client_secret=src_client_secret_123" + }, + "status": "pending", + "type": "ideal", + "usage": "single_use", + "ideal": { + "bank": "ing" + } +} diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_content_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_content_0.tail new file mode 100644 index 00000000..beb39d8a --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_content_0.tail @@ -0,0 +1,21 @@ +GET +/content\?amount=.*&client=.*&country=.*¤cy=.*&locale=.*&logo_color=.*&payment_methods%5B0%5D=.*&payment_methods%5B1%5D=.*$ +200 +application/json +Content-Type: application/json +Pragma: no-cache +content-security-policy: report-uri /csp-report?p=%2Fcontent;block-all-mixed-content;default-src 'none' 'report-sample';base-uri 'none';form-action 'none';style-src 'unsafe-inline';frame-ancestors 'self';connect-src 'self';img-src 'self' https://b.stripecdn.com +Server: nginx +Expires: 0 +referrer-policy: strict-origin-when-cross-origin +Cache-Control: max-age=0, no-cache, no-store, must-revalidate +Date: Thu, 05 Jan 2023 00:12:44 GMT +x-robots-tag: none +Content-Length: 730 +Strict-Transport-Security: max-age=63072000; includeSubDomains; preload +x-content-type-options: nosniff + +{ + "display_l_html" : "4 interest-free payments of $2.75.", + "learn_more_modal_url" : "js.stripe.com\/v3\/unified-message-redirect.html#componentName=unifiedMessage&controllerId=__privateStripeController12345&locale=en-US&publicOptions%5Bamount%5D=1099&publicOptions%5Bclient%5D=ios&publicOptions%5BcountryCode%5D=US&publicOptions%5Bcurrency%5D=USD&publicOptions%5BpaymentMethods%5D%5B0%5D=klarna&publicOptions%5BpaymentMethods%5D%5B1%5D=afterpay_clearpay" +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_afterpay_logo_black.png_1.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_afterpay_logo_black.png_1.tail new file mode 100644 index 00000000..ad415082 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_afterpay_logo_black.png_1.tail @@ -0,0 +1,22 @@ +GET +/payment-method-messaging-statics-srv/assets/afterpay_logo_black.png$ +200 +image/png;base64 +Content-Type: image/png +x-amz-cf-id: CL_PrqlF-NosFX-UP-voNKv9MioRVkOrDGElIW27VLEvbys68JeTTg== +Etag: "93e30f2ddc5e7cb0f466f1a6ae40e261" +Last-Modified: Tue, 25 Oct 2022 03:57:14 GMT +Via: 1.1 22d43bf299ac98b08849f5a01a8af246.cloudfront.net (CloudFront) +Server: Cloudfront +x-amz-cf-pop: SFO5-P2 +timing-allow-origin: * +Cache-Control: max-age=31536000, public +Date: Thu, 05 Jan 2023 00:12:45 GMT +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +Content-Length: 2755 +x-content-type-options: nosniff +Accept-Ranges: bytes +Vary: Accept-Encoding,Origin +x-cache: RefreshHit from cloudfront + +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 \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_klarna_logo_black.png_2.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_klarna_logo_black.png_2.tail new file mode 100644 index 00000000..5aa861f2 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testCreatesViewFromServerResponse/get_payment-method-messaging-statics-srv_assets_klarna_logo_black.png_2.tail @@ -0,0 +1,23 @@ +GET +/payment-method-messaging-statics-srv/assets/klarna_logo_black.png$ +200 +image/png;base64 +Content-Type: image/png +x-cache: HIT +x-cache-hits: 1 +Age: 87267 +Via: 1.1 varnish +Server: Fastly +timing-allow-origin: * +Cache-Control: max-age=31536000, public +Date: Thu, 05 Jan 2023 00:12:44 GMT +x-request-id: cea29c5c-f2f8-4416-90a0-8a0af1410447 +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +x-timer: S1672877565.720133,VS0,VE1 +Content-Length: 1344 +x-content-type-options: nosniff +Accept-Ranges: bytes +Vary: Accept-Encoding, Origin +x-served-by: cache-pao17469-PAO + +iVBORw0KGgoAAAANSUhEUgAAAKgAAABICAYAAABiD742AAAACXBIWXMAACE4AAAhOAFFljFgAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAATVSURBVHgB7Zw/VBNBEMZHn12sYw09temxBWtolVZTS620aKu2pAZr+vSxhhpr6M33svsIMTs7uzeXdzm/33v3gNxe9m52duffHiKEEEIIIYQQQgghhBBCCCGEEEIIIYQQQgghhBDSEs/mxyvl/J3YKf2eVPv7+fEghMhCQX8p59+LXUl/zo9h4tyn+TFb+nsY2q9jFtoTIs/Fj6Fy7o8QUoGXgg4y5++FkAq8FHRXOfcg9ClJJV4KOlLO3QghlWxCQX8LIZV4KCiUUwuQZkJIJR4KeqScQ/ROBSXVvJBm7IseIF1LuyB7sCeLpH/MJCAguwlHV4Mz3Cssz3KxIt530wm9rTJZSxMFhVk/zrRpS0ExAEfhZ67/iSwmUmqlP1j6fRzariMWLTDoh6FvTM6r+XER2hwr/eB6pNtOZKGcqdRctDoTKavklcoEbVPPeiAdoYmCnorue0IQJQK2MAj95gYhsh8Or0IBBjU3KVPsyEI5h5l2OI97xjOeS35FRfuxlMtkK1bSWh8UAtFMOxRiIr5gIL6JfSBWr20KVsBa5QTjwvtA2y+iPy/anEmdTHLFlU5QqqB4KAhtP9MOJs9z9YwD4aFotYykGbUKcZq4tgsyaZ0SBcWKaVnBrsXf9+z9QChAOd+u+fy/kInVB0VQcGJoB9P+XXzBaq0NBHwpBCrLPu+ePPpaXSNO4OgXR59Tu1ecu1j5u08ySWJRUJgYi3mDwLFNztv51vw+pE3gcqy6E7OlYyz+RCUrTducy7/WBfeO+4T8UhkAKONgqa+cTNaNQ5THrSz86a3AYuKnhjYQBoTiHbXDrUitFBjQdcq5DJThh/iBPjG4MbouUc6c63MhesT+MvzESpiTiXZfl+Irk1axKCiEepVpg9n9WvzR/F1rGgsD4rGqRwtROwktfvmtoY2WPcE4bVImrWMNkuBX5vJx8FF3xRct8i2puFisQI6mmQlLLtay82vY8PqIh0xapySKh1nLCTmVEqlF+65N79K3rG6boEsyaZ0SBcXDf860wew+FT80M7QjdjzSMV0xiZoSblomrVOaqIcJyTnY8Bu9okTNZB2KDQxETaWlq2huRu9kUlPqhIOdC5qQWG5aeQFapBzzehowh2fSLzTfsS2Z4BqkwGAdLZVEN2pr8QgYcg458o+vpBlQzmmmj1TuEIMFYfat2gKZaAFiTiaoBpbuCcAr4si9jsJ3jMNnrdfza3czQUjwR/GwqZuMdfsP0sx/w2TQtqdBcG/kMdmNdvDF+mTWV4Gb9VU57ymTM0nvBUA/3pXDJzTZUW8Nmiwl0lw/uZ1RsVyIlSPu1ewzlljAQyZaoURkA6a+6SsfcWOtBh7C6rynuBT/7XvbDmTS9hsLOVo38R7vJOVKdACraNNVDf1QSZ+C3HSbMsntNWj9fTOv145h6nNJYo+gCUr6TsoqJhDiVlRNKoFMPkpZkj5uHLFwnvgc/Wl+sAsIkrQZaP2XNTFoyqWW4KjfhfapfnPlRAgGAxJTKvjO3ZV7wXfgfXwo5iy0tVSCppIeaIsstEG3XI97nFRcjwmLiVsik5HY/mfBNHx3fIsiZlasdX8SGMiWvMawQSgTQgghhBBCCCGEEEIIIYQQQgghhBBCCCGEEEIIIYQQQgghhBBCyLbzF/UEBgKxu0bnAAAAAElFTkSuQmCC \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testInitializingWithBadConfigurationReturnsError/get_content_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testInitializingWithBadConfigurationReturnsError/get_content_0.tail new file mode 100644 index 00000000..5d7808fb --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/PaymentMethodMessagingViewFunctionalTest/testInitializingWithBadConfigurationReturnsError/get_content_0.tail @@ -0,0 +1,18 @@ +GET +/content\?amount=.*&client=.*&country=.*¤cy=.*&locale=.*&logo_color=.*&payment_methods%5B0%5D=.*$ +400 +application/json +Content-Type: application/json +Pragma: no-cache +content-security-policy: report-uri /csp-report?p=%2Fcontent;block-all-mixed-content;default-src 'none' 'report-sample';base-uri 'none';form-action 'none';style-src 'unsafe-inline';frame-ancestors 'self';connect-src 'self';img-src 'self' https://b.stripecdn.com +Server: nginx +Expires: 0 +referrer-policy: strict-origin-when-cross-origin +Cache-Control: max-age=0, no-cache, no-store, must-revalidate +Date: Tue, 20 Dec 2022 19:39:09 GMT +x-robots-tag: none +Content-Length: 25 +Strict-Transport-Security: max-age=63072000; includeSubDomains; preload +x-content-type-options: nosniff + +unsupported_currency: FOO \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateSourceWithPayment/post_v1_tokens_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateSourceWithPayment/post_v1_tokens_0.tail new file mode 100644 index 00000000..d2b65339 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateSourceWithPayment/post_v1_tokens_0.tail @@ -0,0 +1,25 @@ +POST +/v1/tokens$ +400 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Wed, 24 Jul 2019 23:15:47 GMT +stripe-version: 2019-05-16 +access-control-allow-credentials: true +Content-Length: 213 +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +Connection: keep-alive +request-id: req_l0slDo629t27qU + +{ + "error" : { + "message" : "It's been too long since the user approved this payment in your app. You have to create a token within 24 hours of their in-app approval.", + "type" : "invalid_request_error" + } +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPayment/post_v1_tokens_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPayment/post_v1_tokens_0.tail new file mode 100644 index 00000000..6a049041 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPayment/post_v1_tokens_0.tail @@ -0,0 +1,25 @@ +POST +/v1/tokens$ +400 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Wed, 24 Jul 2019 23:15:48 GMT +stripe-version: 2019-05-16 +access-control-allow-credentials: true +Content-Length: 213 +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +Connection: keep-alive +request-id: req_zWJiyZuMbx4GGM + +{ + "error" : { + "message" : "It's been too long since the user approved this payment in your app. You have to create a token within 24 hours of their in-app approval.", + "type" : "invalid_request_error" + } +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPaymentClassic/post_v1_tokens_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPaymentClassic/post_v1_tokens_0.tail new file mode 100644 index 00000000..6a049041 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPApplePayFunctionalTest/testCreateTokenWithPaymentClassic/post_v1_tokens_0.tail @@ -0,0 +1,25 @@ +POST +/v1/tokens$ +400 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Wed, 24 Jul 2019 23:15:48 GMT +stripe-version: 2019-05-16 +access-control-allow-credentials: true +Content-Length: 213 +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +Connection: keep-alive +request-id: req_zWJiyZuMbx4GGM + +{ + "error" : { + "message" : "It's been too long since the user approved this payment in your app. You have to create a token within 24 hours of their in-app approval.", + "type" : "invalid_request_error" + } +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_create_payment_intent_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_create_payment_intent_0.tail new file mode 100644 index 00000000..9a98d8c2 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_create_payment_intent_0.tail @@ -0,0 +1,18 @@ +POST +/create_payment_intent$ +200 +text/html +Content-Type: text/html;charset=utf-8 +Alt-Svc: clear +Set-Cookie: rack.session=7mmwWRPQtNDztyFI7vcbHHpfb8Kg%2FGwJsi3yjFhDBPc3p%2BUXtXSrShd9xGHLKAcTVyYSsKSiFqPPBxi6xIFSMMSL4%2B42nLw3XOMtvFKncxuLuqvbGD1n3sZNjMSqqHT3r%2B%2FrsWQN6KU%2Bk%2B51dxHhZ9WmczQyqhXOZdMrC2fbjl8FdeaBohIIr59FPd3GemuC0c8D%2Ba3pnw7sxGzl277UKBMaxA7c68OlhQYdLRMhAVA%3D; path=/ +Server: Google Frontend +X-Cloud-Trace-Context: 1b98cc26fedf5037b09e133679db4bfc;o=1 +Via: 1.1 google +x-xss-protection: 1; mode=block +Date: Tue, 22 Mar 2022 22:17:55 GMT +X-Robots-Tag: noindex, nofollow +Content-Length: 147 +x-content-type-options: nosniff +x-frame-options: SAMEORIGIN + +{"intent":"pi_3KgG1XFY0qyl6XeW1TLgmwD3","secret":"pi_3KgG1XFY0qyl6XeW1TLgmwD3_secret_VEKa074h9bDuCe2ArEs0Mpcr7","status":"requires_payment_method"} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_3ds2_authenticate_2.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_3ds2_authenticate_2.tail new file mode 100644 index 00000000..cef59e00 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_3ds2_authenticate_2.tail @@ -0,0 +1,47 @@ +POST +/v1/3ds2/authenticate$ +200 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +idempotency-key: 8f01ab53-7fbb-4305-8298-ca08ac71acaa +original-request: req_DINGjWXMtxQmA8 +stripe-should-retry: false +Server: nginx +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Tue, 22 Mar 2022 22:17:58 GMT +stripe-version: 2020-08-27 +access-control-allow-credentials: true +Content-Length: 1699 +Connection: keep-alive +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +request-id: req_DINGjWXMtxQmA8 + +{ + "object" : "three_d_secure_2", + "fallback_redirect_url" : null, + "source" : "src_1KgG1ZFY0qyl6XeWK2TSv8yQ", + "id" : "threeds2_1KgG1aFY0qyl6XeWupDexdTO", + "livemode" : false, + "creq" : "eyJ0aHJlZURTU2VydmVyVHJhbnNJRCI6ImQzZDdjYTQxLTRhMzUtNDcyMS04MGZkLTg5NGEwNTM3ZmNhMCIsImFjc1RyYW5zSUQiOiI1NjY0OWE0Mi1iODY5LTQ0YzgtYTA0Yi01ZmRlYmY2ZWVhMzgiLCJjaGFsbGVuZ2VXaW5kb3dTaXplIjoiMDUiLCJtZXNzYWdlVHlwZSI6IkNSZXEiLCJtZXNzYWdlVmVyc2lvbiI6IjIuMS4wIn0=", + "created" : 1647987478, + "ares" : { + "threeDSServerTransID" : "d3d7ca41-4a35-4721-80fd-894a0537fca0", + "acsSignedContent" : "eyJhbGciOiJFUzI1NiJ9.eyJhY3NFcGhlbVB1YktleSI6eyJjcnYiOiJQLTI1NiIsImt0eSI6IkVDIiwieCI6IjV3dV9yWHhVX19SQTVqQzF2dWpPOWRLZWtnVW41V1dPSHRnSF9nek1kZkkiLCJ5IjoieG4xcjlmaHVrRlpTMjRkZDN4OFFaalJJTzRCYUFTelJuQ2N5ZU1QNTRpUSJ9LCJzZGtFcGhlbVB1YktleSI6eyJ5IjoiTlcwUVcyOE5acGJUYjMwWnZxS2pfQmRDT0tfVlBocWNxYjZIeEFWUE54dyIsIngiOiJ1UlhZVGFTMU1FcV94R1JYd29xc3NsV24xZnVFMzRDVnBYSndqbWNVRVFFIiwia3R5IjoiRUMiLCJjcnYiOiJQLTI1NiJ9LCJhY3NVUkwiOiJodHRwczovL3Rlc3Rtb2RlLWFjcy5zdHJpcGUuY29tLzNkX3NlY3VyZV8yX3Rlc3QvYWNjdF8xRzZtMXBGWTBxeWw2WGVXL3RocmVlZHMyXzFLZ0cxYUZZMHF5bDZYZVd1cERleGRUTy9hcHBfY2hhbGxlbmdlL0RCcWNtaGNNVUotWHgySW5Ld0UxbmpsaVJVdk1uN2xQZkJWbkdCVnFOWEE9In0.tfD-Qynyhh6QZtBQbpffb-TE1ziFyacELVwnNAO1xbqpfCIryg_ZIvS8VOv3hzQKTXXjbTAHLTbFngzqR0wuCA", + "acsTransID" : "56649a42-b869-44c8-a04b-5fdebf6eea38", + "transStatus" : "C", + "messageType" : "ARes", + "messageVersion" : "2.1.0", + "acsURL" : null, + "messageExtension" : null, + "cardholderInfo" : null, + "authenticationType" : "02", + "acsChallengeMandated" : "Y", + "sdkTransID" : "999c2ad5-6cd7-48fd-932e-8275649f477e" + }, + "error" : null, + "state" : "challenge_required" +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_payment_intents_pi_3KgG1XFY0qyl6XeW1TLgmwD3_confirm_1.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_payment_intents_pi_3KgG1XFY0qyl6XeW1TLgmwD3_confirm_1.tail new file mode 100644 index 00000000..40d86f3a --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentHandlerStubbedTests/testCanPresentErrorsAreReported/post_v1_payment_intents_pi_3KgG1XFY0qyl6XeW1TLgmwD3_confirm_1.tail @@ -0,0 +1,119 @@ +POST +/v1/payment_intents/pi_3KgG1XFY0qyl6XeW1TLgmwD3/confirm$ +200 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +idempotency-key: 6af5227c-d9f3-47b1-b337-bbb219f822cc +original-request: req_twrhuRdZas5Bvh +stripe-should-retry: false +Server: nginx +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Tue, 22 Mar 2022 22:17:57 GMT +stripe-version: 2020-08-27 +access-control-allow-credentials: true +Content-Length: 7975 +Connection: keep-alive +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +request-id: req_twrhuRdZas5Bvh + +{ + "source" : null, + "canceled_at" : null, + "status" : "requires_action", + "amount" : 100, + "capture_method" : "automatic", + "livemode" : false, + "shipping" : null, + "total_details" : { + "amount_discount" : 0, + "amount_tax" : 0 + }, + "object" : "payment_intent", + "currency" : "usd", + "last_payment_error" : null, + "amount_subtotal" : 100, + "automatic_payment_methods" : null, + "cancellation_reason" : null, + "next_action" : { + "type" : "use_stripe_sdk", + "use_stripe_sdk" : { + "directory_server_name" : "visa", + "three_ds_optimizations" : "k", + "directory_server_encryption" : { + "root_certificate_authorities" : [ + "-----BEGIN CERTIFICATE-----\nMIIDojCCAoqgAwIBAgIQE4Y1TR0\/BvLB+WUF1ZAcYjANBgkqhkiG9w0BAQUFADBr\nMQswCQYDVQQGEwJVUzENMAsGA1UEChMEVklTQTEvMC0GA1UECxMmVmlzYSBJbnRl\ncm5hdGlvbmFsIFNlcnZpY2UgQXNzb2NpYXRpb24xHDAaBgNVBAMTE1Zpc2EgZUNv\nbW1lcmNlIFJvb3QwHhcNMDIwNjI2MDIxODM2WhcNMjIwNjI0MDAxNjEyWjBrMQsw\nCQYDVQQGEwJVUzENMAsGA1UEChMEVklTQTEvMC0GA1UECxMmVmlzYSBJbnRlcm5h\ndGlvbmFsIFNlcnZpY2UgQXNzb2NpYXRpb24xHDAaBgNVBAMTE1Zpc2EgZUNvbW1l\ncmNlIFJvb3QwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCvV95WHm6h\n2mCxlCfLF9sHP4CFT8icttD0b0\/Pmdjh28JIXDqsOTPHH2qLJj0rNfVIsZHBAk4E\nlpF7sDPwsRROEW+1QK8bRaVK7362rPKgH1g\/EkZgPI2h4H3PVz4zHvtH8aoVlwdV\nZqW1LS7YgFmypw23RuwhY\/81q6UCzyr0TP579ZRdhE2o8mCP2w4lPJ9zcc+U30rq\n299yOIzzlr3xF7zSujtFWsan9sYXiwGd\/BmoKoMWuDpI\/k4+oKsGGelT84ATB+0t\nvz8KPFUgOSwsAGl0lUq8ILKpeeUYiZGo3BxN77t+Nwtd\/jmliFKMAGzsGHxBvfaL\ndXe6YJ2E5\/4tAgMBAAGjQjBAMA8GA1UdEwEB\/wQFMAMBAf8wDgYDVR0PAQH\/BAQD\nAgEGMB0GA1UdDgQWBBQVOIMPPyw\/cDMezUb+B4wg4NfDtzANBgkqhkiG9w0BAQUF\nAAOCAQEAX\/FBfXxcCLkr4NWSR\/pnXKUTwwMhmytMiUbPWU3J\/qVAtmPN3XEolWcR\nzCSs00Rsca4BIGsDoo8Ytyk6feUWYFN4PMCvFYP3j1IzJL1kk5fui\/fbGKhtcbP3\nLBfQdCVp9\/5rPJS+TUtBjE7ic9DjkCJzQ83z7+pzzkWKsKZJ\/0x9nXGIxHYdkFsd\n7v3M9+79YKWxehZx0RbQfBI8bGmX265fOZpwLwU8GUYEmSA20GBuYQa7FkKMcPcw\n++DbZqMAAb3mLNqRX6BGi01qnD093QVG\/na\/oAo85ADmJ7f\/hC3euiInlhBx6yLt\n398znM\/jra6O1I7mT1GvFpLgXPYHDw==\n-----END CERTIFICATE-----\n", + "-----BEGIN CERTIFICATE-----\nMIIFqTCCA5GgAwIBAgIPUT6WAAAA20Qn7qzgvuFIMA0GCSqGSIb3DQEBCwUAMG8x\nCzAJBgNVBAYTAlVTMQ0wCwYDVQQKDARWSVNBMS8wLQYDVQQLDCZWaXNhIEludGVy\nbmF0aW9uYWwgU2VydmljZSBBc3NvY2lhdGlvbjEgMB4GA1UEAwwXVmlzYSBQdWJs\naWMgUlNBIFJvb3QgQ0EwHhcNMjEwMzE2MDAwMDAwWhcNNDEwMzE1MDAwMDAwWjBv\nMQswCQYDVQQGEwJVUzENMAsGA1UECgwEVklTQTEvMC0GA1UECwwmVmlzYSBJbnRl\ncm5hdGlvbmFsIFNlcnZpY2UgQXNzb2NpYXRpb24xIDAeBgNVBAMMF1Zpc2EgUHVi\nbGljIFJTQSBSb290IENBMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA\n2WEbXLS3gI6LOY93bP7Kz6EO9L1QXlr8l+fTkJWZldJ6QuwZ1cv4369tfjeJ8O5w\nSJiDcVw7eNdOP73LfAtwHlTnUnb0e9ILTTipc5bkNnAevocrJACsrpiQ8jBI9ttp\ncqKUeJgzW4Ie25ypirKroVD42b4E0iICK2cZ5QfD4BSzUnftp4Bqh8AfpGvG1lre\nCaD53qrsy5SUadY\/NaeUGOkqdPvDSNoDIdrbExwnZaSFUmjQT1svKwMqGo2GFrgJ\n4cULEp4NNj5rga8YTTZ7Xo5MblHrLpSPOmJev30KWi\/BcbvtCNYNWBTg7UMzP3cK\nMQ1pGLvG2PgvFTZSRvH3QzngJRgrDYYOJ6kj9ave+6yOOFqj80ZCuH0Nugt2mMS3\nc3+Nksaw+6H3cQPsE\/Gv5zjfsKleRhEFtE1gyrdUg1DMgu8o\/YhKM7FAqkXUn74z\nwoRFgx3Mi5OaGTQbg+NlwJgR4sVHXCV4s9b8PjneLhzWMn353SFARF9dnO7LDBqq\ntT6WltJu1z9x2Ze0UVNZvxKGcyCkLody29O8j9\/MGZ8SOSUu4U6NHrebKuuf9Fht\nn6PqQ4ppkhy6sReXeV5NVGfVpDYY5ZAKEWqTYgMULWpQ2Py4BGpFzBe07jXkyulR\npoKvz14iXeA0oq16c94DrFYX0jmrWLeU4a\/TCZQLFIsCAwEAAaNCMEAwHQYDVR0O\nBBYEFEtNpg77oBHorQvi8PMKAC+sixb7MA8GA1UdEwEB\/wQFMAMBAf8wDgYDVR0P\nAQH\/BAQDAgEGMA0GCSqGSIb3DQEBCwUAA4ICAQC5BU9qQSZYPcgCp2x0Juq59kMm\nXuBly094DaEnPqvtCgwwAirkv8x8\/QSOxiWWiu+nveyuR+j6Gz\/fJaV4u+J5QEDy\ncfk605Mw3HIcJOeZvDgk1eyOmQwUP6Z\/BdQTNJmZ92Z8dcG5yWCxLBrqPH7ro3Ss\njhYq9duIJU7jfizCJCN4W8tp0D2pWBe1\/CYNswP4GMs5jQ5+ZQKN\/L5JFdwVTu7X\nPt8b5zfgbmmQpVmUn0oFwm3OI++Z6gEpNmW5bd\/2oUIZoG96Qff2fauVMAYiWQvN\nnL3y1gkRguTOSMVUCCiGfdvwu5ygowillvV2nHb7+YibQ9N5Z2spP0o9Zlfzoat2\n7WFpyK47TiUdu\/4toarLKGZP+hbA\/F4xlnM\/8EfZkE1DeTTI0lhN3O8yEsHrtRl1\nOuQZ\/IexHO8UGU6jvn4TWo10HYeXzrGckL7oIXfGTrjPzfY62T5HDW\/BAEZS+9Tk\nijz25YM0fPPz7IdlEG+k4q4YwZ82j73Y9kDEM5423mrWorq\/Bq7I5Y8v0LTY9GWH\nYrpElYf0WdOXAbsfwQiT6qnRio+p82VyqlY8Jt6VVA6CDy\/iHKwcj1ELEnDQfVv9\nhedoxmnQ6xe\/nK8czclu9hQJRv5Lh9gk9Q8DKK2nmgzZ8SSQ+lr3mSSeY8JOMRlE\n+RKdOQIChWthTJKh7w==\n-----END CERTIFICATE-----\n" + ], + "certificate" : "-----BEGIN CERTIFICATE-----\nMIIGAzCCA+ugAwIBAgIQDaAlB1IbPwgx5esGu9tLIjANBgkqhkiG9w0BAQsFADB2\nMQswCQYDVQQGEwJVUzENMAsGA1UECgwEVklTQTEvMC0GA1UECwwmVmlzYSBJbnRl\ncm5hdGlvbmFsIFNlcnZpY2UgQXNzb2NpYXRpb24xJzAlBgNVBAMMHlZpc2EgZUNv\nbW1lcmNlIElzc3VpbmcgQ0EgLSBHMjAeFw0yMTA4MjMxNTMyMzNaFw0yNDA4MjIx\nNTMyMzNaMIGhMRgwFgYDVQQHDA9IaWdobGFuZHMgUmFuY2gxETAPBgNVBAgMCENv\nbG9yYWRvMQswCQYDVQQGEwJVUzENMAsGA1UECgwEVklTQTEvMC0GA1UECwwmVmlz\nYSBJbnRlcm5hdGlvbmFsIFNlcnZpY2UgQXNzb2NpYXRpb24xJTAjBgNVBAMMHDNk\nczIucnNhLmVuY3J5cHRpb24udmlzYS5jb20wggEiMA0GCSqGSIb3DQEBAQUAA4IB\nDwAwggEKAoIBAQCy34cZ88+xfenoccRD1jOi6uVCPXo2xyabXcKntxl7h1kHahac\nmpnuiH+kSgSg4DEHDXHg0WBcpMp0cB67dUE1XDxLAxN0gL5fXpVX7dUjI9tS8lcW\nndChHxZTA8HcXUtv1IwU1L3luhgNkog509bRw\/V1GLukW6CwFRkMI\/8fecV8EUcw\nIGiBr4\/cAcaPnLxFWm\/SFL2NoixiNf6LnwHrU4YIHsPQCIAM1km4XPDb7Gk2S3o0\nkkXroU87yoiHzFHbEZUN\/tO0Juyz8K6AtGBKoppv1hEHz9MFNzLlvGPo7wcPpovb\nMYtwxj10KhtfEKh0sS0yMl1Uvw36JmuwjaC3AgMBAAGjggFfMIIBWzAMBgNVHRMB\nAf8EAjAAMB8GA1UdIwQYMBaAFL0nYyikrlS3yCO3wTVCF+nGeF+FMGcGCCsGAQUF\nBwEBBFswWTAwBggrBgEFBQcwAoYkaHR0cDovL2Vucm9sbC52aXNhY2EuY29tL2VD\nb21tRzIuY3J0MCUGCCsGAQUFBzABhhlodHRwOi8vb2NzcC52aXNhLmNvbS9vY3Nw\nMEYGA1UdIAQ\/MD0wMQYIKwYBBQUHAgEwJTAjBggrBgEFBQcCARYXaHR0cDovL3d3\ndy52aXNhLmNvbS9wa2kwCAYGZ4EDAQEBMBMGA1UdJQQMMAoGCCsGAQUFBwMCMDUG\nA1UdHwQuMCwwKqAooCaGJGh0dHA6Ly9lbnJvbGwudmlzYWNhLmNvbS9lQ29tbUcy\nLmNybDAdBgNVHQ4EFgQU\/JtqQ7VLWNd3\/9zQjpnsR2rz+cwwDgYDVR0PAQH\/BAQD\nAgSwMA0GCSqGSIb3DQEBCwUAA4ICAQBYOGCI\/bYG2gmLgh7UXg5qrt4xeDYe4RXe\n5xSjFkTelNvdf+KykB+oQzw8ZobIY+pKsPihM6IrtoJQuzOLXPV5L9U4j1qa\/NZB\nGZTXFMwKGN\/v0\/tAj3h8wefcLPWb15RsXEpZmA87ollezpXeEHXPhFIit7cHoG5P\nfem9yMuDISI97qbnIKNtFENJr+fMkWIykQ0QnkM1rt99Yv2ZE4GWZN7VJ0zXFqOF\nNF2IVwnTIZ21eDiCOjQr6ohq7bChDMelB5XvEuhfe400DqDP+e5pPHo81ecXkjJK\ngS5grYYZIbeDBdQL1Cgs1mGu6On8ecr0rcpRlQh++BySg9MKkzJdLt1vsYmxfrfb\nkUaLglTdYAU2nYaOEDR4NvkRxfzegXyXkOqfPTmfkrg+OB0LeuICITJGJ0cuZD5W\nGUNaT9WruEANBRJNVjSX1UeJUnCpz4nitT1ml069ONjEowyWUcKvTr4\/nrargv2R\npOD4RPJMti6kG+bm9OeATiSgVNmO5lkAS4AkOop2IcbRFcVKJUTOhx2Q37L4nuAH\nTCXQ9vwT4yWz6fVaCfL\/FTvCGMilLPzXC\/00OPA2ZtWvClvFh\/uHJBjRUnj6WXp3\nO9p9uHfdV9eKJH37k94GUSMjBKQ6aIru1VUvSOmUPrDz5JbQB7bP+IzUaFHeweZX\nOWumZmyGDw==\n-----END CERTIFICATE-----\n", + "directory_server_id" : "A000000003", + "algorithm" : "RSA" + }, + "merchant" : "acct_1G6m1pFY0qyl6XeW", + "one_click_authn" : null, + "type" : "stripe_3ds2_fingerprint", + "three_d_secure_2_source" : "src_1KgG1ZFY0qyl6XeWK2TSv8yQ", + "three_ds_method_url" : "", + "server_transaction_id" : "d3d7ca41-4a35-4721-80fd-894a0537fca0" + } + }, + "payment_method" : { + "object" : "payment_method", + "id" : "pm_1KgG1YFY0qyl6XeWB4kI9vPK", + "billing_details" : { + "email" : null, + "phone" : null, + "name" : null, + "address" : { + "state" : null, + "country" : null, + "line2" : null, + "city" : null, + "line1" : null, + "postal_code" : "12345" + } + }, + "card" : { + "checks" : { + "address_postal_code_check" : null, + "cvc_check" : null, + "address_line1_check" : null + }, + "exp_month" : 1, + "country" : "IE", + "funding" : "credit", + "generated_from" : null, + "wallet" : null, + "brand" : "visa", + "last4" : "3220", + "networks" : { + "available" : [ + "visa" + ], + "preferred" : null + }, + "three_d_secure_usage" : { + "supported" : true + }, + "exp_year" : 2024 + }, + "livemode" : false, + "created" : 1647987476, + "type" : "card", + "customer" : null + }, + "client_secret" : "pi_3KgG1XFY0qyl6XeW1TLgmwD3_secret_VEKa074h9bDuCe2ArEs0Mpcr7", + "id" : "pi_3KgG1XFY0qyl6XeW1TLgmwD3", + "confirmation_method" : "automatic", + "processing" : null, + "receipt_email" : null, + "payment_method_types" : [ + "card" + ], + "setup_future_usage" : null, + "created" : 1647987475, + "description" : null +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentMethodFunctionalTest/testCreateBacsPaymentMethod/post_v1_payment_methods_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentMethodFunctionalTest/testCreateBacsPaymentMethod/post_v1_payment_methods_0.tail new file mode 100644 index 00000000..8dc396c9 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPaymentMethodFunctionalTest/testCreateBacsPaymentMethod/post_v1_payment_methods_0.tail @@ -0,0 +1,48 @@ +POST +/v1/payment_methods$ +200 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Thu, 30 Jan 2020 22:42:49 GMT +stripe-version: 2019-05-16 +access-control-allow-credentials: true +Content-Length: 609 +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +Connection: keep-alive +request-id: req_8ee0FF29PSVnan + +{ + "object" : "payment_method", + "id" : "pm_1G6linL6pqDH2fDJbWxS74Bc", + "billing_details" : { + "email" : "email@email.com", + "phone" : "555-555-5555", + "name" : "Isaac Asimov", + "address" : { + "state" : null, + "country" : "GB", + "line2" : null, + "city" : "London", + "line1" : "Stripe, 7th Floor The Bower Warehouse", + "postal_code" : "EC1V 9NR" + } + }, + "livemode" : false, + "bacs_debit" : { + "fingerprint" : "UkSG0HfCGxxrja1H", + "last4" : "2345", + "sort_code" : "108800" + }, + "created" : 1580424169, + "type" : "bacs_debit", + "customer" : null, + "metadata" : { + + } +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePin/get_v1_issuing_cards_ic_token_pin_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePin/get_v1_issuing_cards_ic_token_pin_0.tail new file mode 100644 index 00000000..3f742eb6 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePin/get_v1_issuing_cards_ic_token_pin_0.tail @@ -0,0 +1,86 @@ +GET +/v1/issuing/cards/ic_token/pin\?verification%5Bid%5D=.*&verification%5Bone_time_code%5D=.*$ +200 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Tue, 30 Apr 2019 21:48:31 GMT +stripe-version: 2015-10-12 +access-control-allow-credentials: true +Content-Length: 1470 +Connection: keep-alive +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +request-id: req_VzRTj5hCU2Banp + +{ + "pin" : "2345", + "object" : "issuing.card_pin", + "card" : { + "id" : "ic_token", + "last4" : "1234", + "livemode" : true, + "shipping" : null, + "metadata" : { + + }, + "brand" : "Visa", + "authorization_controls" : { + "max_approvals" : null, + "currency" : null, + "allowed_categories" : null, + "spending_limits" : null, + "blocked_categories" : null, + "max_amount" : null + }, + "type" : "virtual", + "cardholder" : { + "id" : "ich_token", + "livemode" : true, + "phone_number" : "+1415", + "metadata" : { + + }, + "authorization_controls" : { + "blocked_categories" : [ + + ], + "spending_limits" : [ + + ], + "allowed_categories" : [ + + ] + }, + "type" : "individual", + "object" : "issuing.cardholder", + "billing" : { + "address" : { + "state" : "CA", + "country" : "US", + "line2" : "123", + "city" : "San Francisco", + "line1" : "510 Townsend St", + "postal_code" : "94103" + }, + "name" : "Arnaud Cavailhez" + }, + "created" : 1536780742, + "is_default" : false, + "email" : "acavailhez@stripe.com", + "name" : "Arnaud Cavailhez", + "status" : "active" + }, + "object" : "issuing.card", + "exp_month" : 9, + "exp_year" : 2021, + "created" : 1536781947, + "currency" : "usd", + "name" : "Arnaud Cavailhez", + "status" : "active" + } +} diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePinWithError/get_v1_issuing_cards_ic_token_pin_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePinWithError/get_v1_issuing_cards_ic_token_pin_0.tail new file mode 100644 index 00000000..a7ab44e3 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testRetrievePinWithError/get_v1_issuing_cards_ic_token_pin_0.tail @@ -0,0 +1,26 @@ +GET +/v1/issuing/cards/ic_token/pin\?verification%5Bid%5D=.*&verification%5Bone_time_code%5D=.*$ +400 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Tue, 30 Apr 2019 21:45:51 GMT +stripe-version: 2015-10-12 +access-control-allow-credentials: true +Content-Length: 168 +Connection: keep-alive +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +request-id: req_2Y8kP3hSuVSeFk + +{ + "error" : { + "message" : "Verification challenge does not exist or is already redeemed", + "type" : "invalid_request_error", + "code" : "already_redeemed" + } +} diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testUpdatePin/post_v1_issuing_cards_ic_token_pin_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testUpdatePin/post_v1_issuing_cards_ic_token_pin_0.tail new file mode 100644 index 00000000..ea9ec62b --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPinManagementServiceFunctionalTest/testUpdatePin/post_v1_issuing_cards_ic_token_pin_0.tail @@ -0,0 +1,88 @@ +POST +/v1/issuing/cards/ic_token/pin$ +200 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +access-control-allow-methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +access-control-expose-headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +access-control-max-age: 300 +Cache-Control: no-cache, no-store +Date: Thu, 02 May 2019 18:08:22 GMT +stripe-version: 2015-10-12 +access-control-allow-credentials: true +Content-Length: 1531 +Connection: keep-alive +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +request-id: req_rbkiAN37e2DSZk + +{ + "pin" : "3456", + "object" : "issuing.card_pin", + "card" : { + "id" : "ic_token", + "last4" : "1234", + "livemode" : true, + "replacement_for" : null, + "metadata" : { + + }, + "brand" : "Visa", + "shipping" : null, + "authorization_controls" : { + "max_approvals" : null, + "currency" : null, + "allowed_categories" : null, + "spending_limits" : null, + "blocked_categories" : null, + "max_amount" : null + }, + "replacement_reason" : null, + "type" : "virtual", + "cardholder" : { + "id" : "ich_token", + "livemode" : true, + "phone_number" : "+1415", + "metadata" : { + + }, + "authorization_controls" : { + "blocked_categories" : [ + + ], + "spending_limits" : [ + + ], + "allowed_categories" : [ + + ] + }, + "type" : "individual", + "object" : "issuing.cardholder", + "billing" : { + "address" : { + "state" : "CA", + "country" : "US", + "line2" : "123", + "city" : "San Francisco", + "line1" : "510 Townsend St", + "postal_code" : "94103" + }, + "name" : "Arnaud Cavailhez" + }, + "created" : 1536780742, + "is_default" : false, + "email" : "acavailhez@stripe.com", + "name" : "Arnaud Cavailhez", + "status" : "active" + }, + "object" : "issuing.card", + "exp_month" : 9, + "exp_year" : 2021, + "created" : 1536781947, + "currency" : "usd", + "name" : "Arnaud Cavailhez", + "status" : "active" + } +} diff --git a/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPushProvisioningDetailsFunctionalTest/testRetrievePushProvisioningDetails/get_v1_issuing_cards_ic_1C0Xig4JYtv6MPZK91WoXa9u_push_provisioning_details_0.tail b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPushProvisioningDetailsFunctionalTest/testRetrievePushProvisioningDetails/get_v1_issuing_cards_ic_1C0Xig4JYtv6MPZK91WoXa9u_push_provisioning_details_0.tail new file mode 100644 index 00000000..6095a284 --- /dev/null +++ b/Stripe/StripeiOSTests/Resources/recorded_network_traffic/STPPushProvisioningDetailsFunctionalTest/testRetrievePushProvisioningDetails/get_v1_issuing_cards_ic_1C0Xig4JYtv6MPZK91WoXa9u_push_provisioning_details_0.tail @@ -0,0 +1,27 @@ +GET +/v1/issuing/cards/ic_1C0Xig4JYtv6MPZK91WoXa9u/push_provisioning_details\?ios%5Bcertificates%5D%5B0%5D=.*&ios%5Bcertificates%5D%5B1%5D=.*&ios%5Bnonce%5D=.*&ios%5Bnonce_signature%5D=.*$ +200 +application/json +Content-Type: application/json +Access-Control-Allow-Origin: * +Access-Control-Allow-Methods: GET, POST, HEAD, OPTIONS, DELETE +Server: nginx +Access-Control-Expose-Headers: Request-Id, Stripe-Manage-Version, X-Stripe-External-Auth-Required, X-Stripe-Privileged-Session-Required +Access-Control-Max-Age: 300 +Stripe-Issuing-Push-Provisioning-Platform: ios +Cache-Control: no-cache, no-store +Date: Fri, 30 Nov 2018 23:12:54 GMT +Stripe-Version: 2015-10-12 +Access-Control-Allow-Credentials: true +Content-Length: 1278 +Connection: keep-alive +Strict-Transport-Security: max-age=31556926; includeSubDomains; preload +Request-Id: req_QTDJy3AYaxI1tL + +{ + "activation_data" : "TUJQQUMtMS1GSy00MDU1NTEuMS0tVERFQS1FNUVCNUJCQjhGMENDMjdCQjU5MUNCRTdCMTVDN0U2RjBENTQ5RDM1NkZERTRDQUZBNDBGOUNCMTA1MzI5NUQ4N0RBRTk0MTE0QTQyNENDQTY0NDAxRTFCQTExOTRBRDM=", + "object" : "issuing.push_provisioning_details", + "ephemeral_public_key" : "BHdLb5BNpoPrh9Btay8LwQ5oELoziQwJL7HagE3xB5mrbdgLa5iiwogu34Y32\/xBEviaN31s\/ONRXSetYT745ig=", + "card" : "ic_1C0Xig4JYtv6MPZK91WoXa9u", + "contents" : "UYeMxRqiYrjVzwqKcCGRVbgFXRspbDIKkWWly8e55caWkHYmO2DNnFtqD3y5S6bvGbe+bkNPCIkT1UzQQChCQOkb0P+cbVDBLaFGaDeJW0Mhca8\/6GwbUx9lp4H3czqszG504PkiA89dNvnbtwUmlQOpk+B\/IAMnkaXjD2xUBUtPX9xEr5EvckkDSHHFmpy5rbGfqnWsPbJNPwUiE+6mYbt643DqF9RpmgdFN84DImuMU1W0xshbkN7voq63L\/6UgTW7liTzWVzKUT36TtaTw5TGKVf1Niqu5CHNu2NpDEnzrvcwUCgphxRVgezJyFfq1NjhZVlGA2nUKZuRvc\/XjBwdE0fr4Enw5XfHbRQHorpv\/S2rX4Cmn4VHJE1JDHWK3Wrn4HmMOCH+psVi+T5hPvy6+\/v+0zRRmGGeFKQEx0soItHiQauN2\/zO4QoC2DCQOAKGj1KSzqHhTgdxBcBu4TIOQRsIXu6zk1ItenHIdq4thD8vF8m3wgJ8Y3KcG3TwwgbjomxOjO4rX0AA9q2V6w0TXWQ1eWC8WfX11J30Zt\/SbyYHoU8KrIdM2ANcOvIFHENnUNBcL7AO0+tjv9lVO7M9w7hKMiVJOnDGMeH2OQfnTBOJI8SEHGm2kDRNw\/5+VGJ7pHA1wZ9y2IS6EvY8IeNhqZ7HdBXhn18X4AWThZqmNvGuZoEU\/ZlPmfed\/c0BgqSw5x6K9GuC5G9Nyee3O1E6wETf4Z3goNbFnRYNJ8m+A4DxUKbvhhRSNva4d\/lRkoNuQj2ztPiLSAPVt1H7Dk3ryeyXCrqprHsjn5T35jXFOlm5whuyeGgeXWt3DUxsYXZGTiohZyU3bZELqW3EUSwjwQ==" +} \ No newline at end of file diff --git a/Stripe/StripeiOSTests/Resources/stp_test_upload_image.jpeg b/Stripe/StripeiOSTests/Resources/stp_test_upload_image.jpeg new file mode 100644 index 0000000000000000000000000000000000000000..bb7e0e73e748c3d72c5609783209296b3907eaac GIT binary patch literal 3340 zcmeHKTWl0%82$da?4|n9g@TAN4pSl=znF(%eKMk5F+`oQ?jv}_Zj3GvlEJC`#v-+%r& z=ii5KytCdH(Abyl%>p3+J@^A}5{|5L^+AAK4weEy1Xwa3SY%iNedLxJ?$lPFtSukybYZI0UG9(kyawQ=vlFUi1 zij-2Mb^v;u03D1*y>Y6hOvdRM?oZ)OMt7GS$5KSms0eD&%xglyEQ_vciIR{IVMT{) zsfA(9;quy$Zlw92j~(YZy_n|Lx8)K!E2EY4o-tcnJ9g(lVQjdND)JpGxfQPBmaVep zsGM6KF)E6i<^ypBxqn;axuC=uPV*abtGSG6YiOudNboq?E)FXF-P!4}c$Ma-U$R=Q z3RPJ!?IBT0rBY%dDJGL~tcX`e4M%n3My2_hLbp~a*t+HDros7&YTn%JqXph? zP4*wX`-PU0u`zMe9x%<3H2*J)Fjbi|i~8Vbx9VtV-d~k?LXJ!Fz_nH3MRX=kTk~lx zqm^78^#tk()Dx&DP*0$q!2d6SnJ!B+u(hgU4+Y*&(7w84I%cJ0TAVB-puMX%$M{Vm zBH%3nZ9H@9V-t_V0p|Pz(Oo}Y>AK%L0kH^nj)W#WP%%Qsh<6HZ!)B5t0fJ$8AQa88 zp>QPHa03<`ZUlrX@O!{@-n*jWpt&d}^NYM9^Dl-#m&Po%kP|H(Q1r!Op& z^Y+$g!`z$Z-8{eL_C<@AENyLDw!A&1WV%;n@9636TRU*qy7e3G-dHGVgF_|#o{F>i z-l{t~_TWPgZ`=OJqdT5>@~Nkvd3M+C=k`AT!iz7x{K~8Q4jepm__f!Mym9o+w@$tN z&b#lu|G|eJo%#5aPtQ)AoBZ;tufO^ByYJ8caPgN*mw)~3_bY$=>HES~cghwXr=0!G zR}6ho3_~+v-xr~(zVR5tE|5aAI@g5NO|$vrmPk|A&OQ52Mw{jS3v=@J>4v!rTPIsC z`qlzx*Jo_&e{xncHsk9v$l&*d4+=M$qHw8b1cMe3Y={jInnpN4#7DFS4@Ck6iehk! khS*T;*>w}|bNpf;-Z{9DCirKfW6%LtuP)xZ1H6g905i&OlmGw# literal 0 HcmV?d00001 diff --git a/Stripe/StripeiOSTests/RotatingCardBrandsViewSnapshotTests.swift b/Stripe/StripeiOSTests/RotatingCardBrandsViewSnapshotTests.swift new file mode 100644 index 00000000..62033d27 --- /dev/null +++ b/Stripe/StripeiOSTests/RotatingCardBrandsViewSnapshotTests.swift @@ -0,0 +1,38 @@ +// +// RotatingCardBrandsViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 6/27/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +@_spi(STP) import StripePayments +import XCTest + +@testable import Stripe +@testable @_spi(STP) import StripePaymentSheet + +class RotatingCardBrandsViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() +// recordMode = true + } + + func testAllCardBrands() { + let rotatingCardBrandsView = RotatingCardBrandsView() + rotatingCardBrandsView.cardBrands = RotatingCardBrandsView.orderedCardBrands(from: STPCardBrand.allCases) + rotatingCardBrandsView.autosizeHeight(width: 140) + STPSnapshotVerifyView(rotatingCardBrandsView) + } + + func testSingleCardBrand() { + let rotatingCardBrandsView = RotatingCardBrandsView() + rotatingCardBrandsView.cardBrands = [.visa] + rotatingCardBrandsView.autosizeHeight(width: 140) + STPSnapshotVerifyView(rotatingCardBrandsView) + } + +} diff --git a/Stripe/StripeiOSTests/RotatingCardBrandsViewTests.swift b/Stripe/StripeiOSTests/RotatingCardBrandsViewTests.swift new file mode 100644 index 00000000..b52ff221 --- /dev/null +++ b/Stripe/StripeiOSTests/RotatingCardBrandsViewTests.swift @@ -0,0 +1,42 @@ +// +// RotatingCardBrandsViewTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 6/27/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripePayments +import XCTest + +@testable import Stripe +@testable @_spi(STP) import StripePaymentSheet + +class RotatingCardBrandsViewTests: XCTestCase { + + func testOrdering() { + XCTAssertEqual([.visa, + .mastercard, + .amex, + .discover, + .dinersClub, + .JCB, + .unionPay, + ], RotatingCardBrandsView.orderedCardBrands(from: STPCardBrand.allCases)) + } + + func testRotatesOnMoreThreeOrMoreBrands() { + let rotatingCardBrandsView = RotatingCardBrandsView() + rotatingCardBrandsView.cardBrands = [.visa] + XCTAssertTrue(rotatingCardBrandsView.rotatingCardBrandView.isHidden) + rotatingCardBrandsView.cardBrands = [.visa, .mastercard] + XCTAssertTrue(rotatingCardBrandsView.rotatingCardBrandView.isHidden) + rotatingCardBrandsView.cardBrands = [.visa, .mastercard, .amex] + XCTAssertTrue(rotatingCardBrandsView.rotatingCardBrandView.isHidden) + rotatingCardBrandsView.cardBrands = [.visa, .mastercard, .amex, .dinersClub] + XCTAssertFalse(rotatingCardBrandsView.rotatingCardBrandView.isHidden) + rotatingCardBrandsView.cardBrands = [.visa, .mastercard, .amex, .dinersClub, .JCB] + XCTAssertFalse(rotatingCardBrandsView.rotatingCardBrandView.isHidden) + } + +} diff --git a/Stripe/StripeiOSTests/STPAPIClientNetworkBridgeTest.m b/Stripe/StripeiOSTests/STPAPIClientNetworkBridgeTest.m new file mode 100644 index 00000000..00c9d6f3 --- /dev/null +++ b/Stripe/StripeiOSTests/STPAPIClientNetworkBridgeTest.m @@ -0,0 +1,374 @@ +// +// STPAPIClientNetworkBridgeTest.m +// StripeiOS +// +// Created by David Estes on 9/23/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +@import Stripe; +@import XCTest; +@import PassKit; +@import StripeCoreTestUtils; +#import "STPNetworkStubbingTestCase.h" +#import "STPTestingAPIClient.h" +#import "STPFixtures.h" + +@interface StripeAPIBridgeNetworkTest : XCTestCase + +@property (nonatomic) STPAPIClient *client; + +@end + +// These are a little redundant with the existing +// API tests, but it's a good way to test that the +// bridge works correctly. +@implementation StripeAPIBridgeNetworkTest + +- (void)setUp { + self.client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + [super setUp]; +} + +// MARK: Bank Account +- (void)testCreateTokenWithBankAccount { + XCTestExpectation *exp = [self expectationWithDescription:@"Request complete"]; + STPBankAccountParams *params = [[STPBankAccountParams alloc] init]; + params.accountNumber = @"000123456789"; + params.routingNumber = @"110000000"; + params.country = @"US"; + + [self.client createTokenWithBankAccount:params completion:^(STPToken *token, NSError *error) { + XCTAssertNotNil(token); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: PII + +- (void)testCreateTokenWithPII { + XCTestExpectation *exp = [self expectationWithDescription:@"Create token"]; + + [self.client createTokenWithPersonalIDNumber:@"123456789" completion:^(STPToken *token, NSError *error) { + XCTAssertNotNil(token); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateTokenWithSSNLast4 { + XCTestExpectation *exp = [self expectationWithDescription:@"Create SSN"]; + + [self.client createTokenWithSSNLast4:@"1234" completion:^(STPToken *token, NSError *error) { + XCTAssertNotNil(token); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: Connect Accounts + +- (void)testCreateConnectAccount { + XCTestExpectation *exp = [self expectationWithDescription:@"Create connect account"]; + STPConnectAccountCompanyParams *companyParams = [[STPConnectAccountCompanyParams alloc] init]; + companyParams.name = @"Company"; + STPConnectAccountParams *params = [[STPConnectAccountParams alloc] initWithCompany:companyParams]; + [self.client createTokenWithConnectAccount:params completion:^(STPToken *token, NSError *error) { + XCTAssertNotNil(token); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: Upload + +- (void)testUploadFile { + XCTestExpectation *exp = [self expectationWithDescription:@"Upload file"]; + UIImage *image = [UIImage imageNamed:@"stp_test_upload_image.jpeg" + inBundle:[NSBundle bundleForClass:self.class] + compatibleWithTraitCollection:nil]; + + [self.client uploadImage:image purpose:STPFilePurposeDisputeEvidence completion:^(STPFile *file, NSError *error) { + XCTAssertNotNil(file); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: Credit Cards + +- (void)testCardToken { + XCTestExpectation *exp = [self expectationWithDescription:@"Create card token"]; + STPCardParams *params = [[STPCardParams alloc] init]; + params.number = @"4242424242424242"; + params.expYear = 42; + params.expMonth = 12; + params.cvc = @"123"; + + [self.client createTokenWithCard:params completion:^(STPToken *token, NSError *error) { + XCTAssertNotNil(token); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCVCUpdate { + XCTestExpectation *exp = [self expectationWithDescription:@"CVC Update"]; + + [self.client createTokenForCVCUpdate:@"123" completion:^(STPToken *token, NSError *error) { + XCTAssertNotNil(token); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: Sources + +- (void)testCreateRetrieveAndPollSource { + XCTestExpectation *exp = [self expectationWithDescription:@"Upload file"]; + XCTestExpectation *expR = [self expectationWithDescription:@"Retrieve source"]; + XCTestExpectation *expP = [self expectationWithDescription:@"Poll source"]; + + STPCardParams *card = [[STPCardParams alloc] init]; + card.number = @"4242424242424242"; + card.expYear = 42; + card.expMonth = 12; + card.cvc = @"123"; + + STPSourceParams *params = [STPSourceParams cardParamsWithCard:card]; + + [self.client createSourceWithParams:params completion:^(STPSource *source, NSError *error) { + XCTAssertNotNil(source); + XCTAssertNil(error); + [exp fulfill]; + + [self.client retrieveSourceWithId:source.stripeID clientSecret:source.clientSecret completion:^(STPSource *source2, NSError *error2) { + XCTAssertNotNil(source2); + XCTAssertNil(error2); + [expR fulfill]; + }]; + + [self.client startPollingSourceWithId:source.stripeID clientSecret:source.clientSecret timeout:10 completion:^(STPSource *source2, NSError *error2) { + XCTAssertNotNil(source2); + XCTAssertNil(error2); + [self.client stopPollingSourceWithId:source.stripeID]; + [expP fulfill]; + }]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: Payment Intents + +- (void)testRetrievePaymentIntent { + XCTestExpectation *exp = [self expectationWithDescription:@"Fetch"]; + XCTestExpectation *exp2 = [self expectationWithDescription:@"Fetch with expansion"]; + + STPTestingAPIClient *testClient = [[STPTestingAPIClient alloc] init]; + [testClient createPaymentIntentWithParams:nil completion:^(NSString *clientSecret, NSError *error) { + XCTAssertNil(error); + + [self.client retrievePaymentIntentWithClientSecret:clientSecret completion:^(STPPaymentIntent *pi, NSError *error2) { + XCTAssertNotNil(pi); + XCTAssertNil(error2); + [exp fulfill]; + }]; + + [self.client retrievePaymentIntentWithClientSecret:clientSecret expand:@[@"metadata"] completion:^(STPPaymentIntent *pi, NSError *error2) { + XCTAssertNotNil(pi); + XCTAssertNil(error2); + [exp2 fulfill]; + }]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmPaymentIntent { + XCTestExpectation *exp = [self expectationWithDescription:@"Confirm"]; + XCTestExpectation *exp2 = [self expectationWithDescription:@"Confirm with expansion"]; + STPTestingAPIClient *testClient = [[STPTestingAPIClient alloc] init]; + + STPPaymentMethodCardParams *card = [[STPPaymentMethodCardParams alloc] init]; + card.number = @"4242424242424242"; + card.expYear = @42; + card.expMonth = @12; + card.cvc = @"123"; + + [testClient createPaymentIntentWithParams:nil completion:^(NSString *clientSecret, NSError *error) { + XCTAssertNil(error); + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:card billingDetails:nil metadata:nil]; + + [self.client confirmPaymentIntentWithParams:params completion:^(STPPaymentIntent *pi, NSError *error2) { + XCTAssertNotNil(pi); + XCTAssertNil(error2); + [exp fulfill]; + }]; + }]; + + [testClient createPaymentIntentWithParams:nil completion:^(NSString *clientSecret, NSError *error) { + XCTAssertNil(error); + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:card billingDetails:nil metadata:nil]; + + [self.client confirmPaymentIntentWithParams:params completion:^(STPPaymentIntent *pi, NSError *error2) { + XCTAssertNotNil(pi); + XCTAssertNil(error2); + [exp2 fulfill]; + }]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: Setup Intents + +- (void)testRetrieveSetupIntent { + XCTestExpectation *exp = [self expectationWithDescription:@"Fetch"]; + + STPTestingAPIClient *testClient = [[STPTestingAPIClient alloc] init]; + [testClient createSetupIntentWithParams:nil completion:^(NSString *clientSecret, NSError *error) { + XCTAssertNil(error); + + [self.client retrieveSetupIntentWithClientSecret:clientSecret completion:^(STPSetupIntent *si, NSError *error2) { + XCTAssertNotNil(si); + XCTAssertNil(error2); + [exp fulfill]; + }]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmSetupIntent { + XCTestExpectation *exp = [self expectationWithDescription:@"Confirm"]; + STPTestingAPIClient *testClient = [[STPTestingAPIClient alloc] init]; + + STPPaymentMethodCardParams *card = [[STPPaymentMethodCardParams alloc] init]; + card.number = @"4242424242424242"; + card.expYear = @42; + card.expMonth = @12; + card.cvc = @"123"; + + [testClient createSetupIntentWithParams:nil completion:^(NSString *clientSecret, NSError *error) { + XCTAssertNil(error); + + STPSetupIntentConfirmParams *params = [[STPSetupIntentConfirmParams alloc] initWithClientSecret:clientSecret]; + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:card billingDetails:nil metadata:nil]; + + [self.client confirmSetupIntentWithParams:params completion:^(STPSetupIntent *si, NSError *error2) { + XCTAssertNotNil(si); + XCTAssertNil(error2); + [exp fulfill]; + }]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: Payment Methods + +- (void)testCreatePaymentMethod { + XCTestExpectation *exp = [self expectationWithDescription:@"Create PaymentMethod"]; + + STPPaymentMethodCardParams *card = [[STPPaymentMethodCardParams alloc] init]; + card.number = @"4242424242424242"; + card.expYear = @42; + card.expMonth = @12; + card.cvc = @"123"; + + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + + [self.client createPaymentMethodWithParams:params completion:^(STPPaymentMethod *pm, NSError *error) { + XCTAssertNotNil(pm); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + + +// MARK: Radar + +- (void)testCreateRadarSession { + XCTestExpectation *exp = [self expectationWithDescription:@"Create session"]; + + [self.client createRadarSessionWithCompletion:^(STPRadarSession *session, NSError *error) { + XCTAssertNotNil(session); + XCTAssertNil(error); + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +// MARK: ApplePay + +- (void)testCreateApplePayToken { + XCTestExpectation *exp = [self expectationWithDescription:@"CreateToken"]; + XCTestExpectation *exp2 = [self expectationWithDescription:@"CreateSource"]; + XCTestExpectation *exp3 = [self expectationWithDescription:@"CreatePM"]; + PKPayment *payment = [STPFixtures applePayPayment]; + [self.client createTokenWithPayment:payment completion:^(STPToken *token, NSError *error) { + // The certificate used to sign our fake Apple Pay test payment is invalid, which makes sense. + // Expect an error. + XCTAssertNil(token); + XCTAssertNotNil(error); + [exp fulfill]; + }]; + + [self.client createSourceWithPayment:payment completion:^(STPSource *source, NSError *error) { + XCTAssertNil(source); + XCTAssertNotNil(error); + [exp2 fulfill]; + }]; + + [self.client createPaymentMethodWithPayment:payment completion:^(STPPaymentMethod *pm, NSError *error) { + XCTAssertNil(pm); + XCTAssertNotNil(error); + [exp3 fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testPKPaymentError { + XCTestExpectation *exp = [self expectationWithDescription:@"Upload file"]; + STPCardParams *params = [[STPCardParams alloc] init]; + params.number = @"4242424242424242"; + params.expYear = 20; + params.expMonth = 12; + params.cvc = @"123"; + + [self.client createTokenWithCard:params completion:^(STPToken *token, NSError *error) { + XCTAssertNil(token); + XCTAssertNotNil(error); + XCTAssertNotNil([STPAPIClient pkPaymentErrorForStripeError:error]); + + [exp fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPAPIClientStubbedTest.swift b/Stripe/StripeiOSTests/STPAPIClientStubbedTest.swift new file mode 100644 index 00000000..3bbab4cf --- /dev/null +++ b/Stripe/StripeiOSTests/STPAPIClientStubbedTest.swift @@ -0,0 +1,291 @@ +// +// STPAPIClientStubbedTest.swift +// StripeiOS Tests +// +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import OHHTTPStubs +import OHHTTPStubsSwift +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeApplePay +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet + +class STPAPIClientStubbedTest: APIStubbedTestCase { + + func testSetupIntent_LinkAccountSessionForUSBankAccount() { + let sut = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains( + "/setup_intents/seti_12345/link_account_sessions" + ) ?? false + } response: { urlRequest in + guard let data = urlRequest.httpBodyOrBodyStream, + let body = String(data: data, encoding: .utf8) + else { + return HTTPStubsResponse( + data: "".data(using: .utf8)!, + statusCode: 400, + headers: nil + ) + } + XCTAssert(body.contains("client_secret=si_client_secret_123")) + XCTAssert( + body.contains( + "payment_method_data%5Bbilling_details%5D%5Bemail%5D=test%40example.com" + ) + ) + XCTAssert( + body.contains("payment_method_data%5Bbilling_details%5D%5Bname%5D=Test%20Tester") + ) + XCTAssert(body.contains("payment_method_data%5Btype%5D=us_bank_account")) + + let jsonText = """ + { + "id": "xxxxx", + "object": "link_account_session", + "client_secret": "las_client_secret_123456", + "linked_accounts": { + "object": "list", + "data": [], + "has_more": false, + "total_count": 0, + "url": "/v1/linked_accounts" + }, + "livemode": false + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + let expectCallback = expectation(description: "bindings serialize/deserialize") + sut.createLinkAccountSession( + setupIntentID: "seti_12345", + clientSecret: "si_client_secret_123", + paymentMethodType: .USBankAccount, + customerName: "Test Tester", + customerEmailAddress: "test@example.com" + ) { intent, _ in + guard let intent = intent else { + XCTFail("Intent was null") + return + } + XCTAssertEqual(intent.clientSecret, "las_client_secret_123456") + expectCallback.fulfill() + } + + wait(for: [expectCallback], timeout: 2.0) + } + + func testPaymentIntent_LinkAccountSessionForUSBankAccount() { + let sut = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains( + "/payment_intents/pi_12345/link_account_sessions" + ) ?? false + } response: { urlRequest in + guard let data = urlRequest.httpBodyOrBodyStream, + let body = String(data: data, encoding: .utf8) + else { + return HTTPStubsResponse( + data: "".data(using: .utf8)!, + statusCode: 400, + headers: nil + ) + } + XCTAssert(body.contains("client_secret=si_client_secret_123")) + XCTAssert( + body.contains( + "payment_method_data%5Bbilling_details%5D%5Bemail%5D=test%40example.com" + ) + ) + XCTAssert( + body.contains("payment_method_data%5Bbilling_details%5D%5Bname%5D=Test%20Tester") + ) + XCTAssert(body.contains("payment_method_data%5Btype%5D=us_bank_account")) + + let jsonText = """ + { + "id": "las_12345", + "object": "link_account_session", + "client_secret": "las_client_secret_654321", + "linked_accounts": { + "object": "list", + "data": [ + + ], + "has_more": false, + "total_count": 0, + "url": "/v1/linked_accounts" + }, + "livemode": false + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + let expectCallback = expectation(description: "bindings serialize/deserialize") + sut.createLinkAccountSession( + paymentIntentID: "pi_12345", + clientSecret: "si_client_secret_123", + paymentMethodType: .USBankAccount, + customerName: "Test Tester", + customerEmailAddress: "test@example.com" + ) { intent, _ in + guard let intent = intent else { + XCTFail("Intent was null") + return + } + XCTAssertEqual(intent.clientSecret, "las_client_secret_654321") + expectCallback.fulfill() + } + + wait(for: [expectCallback], timeout: 2.0) + } + + func testSetupIntent_LinkAccountSessionAttach() { + let sut = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains( + "/setup_intents/seti_12345/link_account_sessions/las_123456/attach" + ) ?? false + } response: { urlRequest in + guard let data = urlRequest.httpBodyOrBodyStream, + let body = String(data: data, encoding: .utf8) + else { + return HTTPStubsResponse( + data: "".data(using: .utf8)!, + statusCode: 400, + headers: nil + ) + } + XCTAssert(body.contains("client_secret=si_client_secret_123")) + + let jsonText = """ + { + "id": "seti_12345", + "object": "setup_intent", + "cancellation_reason": null, + "client_secret": "seti_abc_secret_def", + "created": 1647000000, + "description": null, + "last_setup_error": null, + "livemode": false, + "next_action": null, + "payment_method": "pm_abcdefg", + "payment_method_options": { + "us_bank_account": { + "verification_method": "instant" + } + }, + "payment_method_types": [ + "us_bank_account" + ], + "status": "requires_confirmation", + "usage": "off_session" + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + let expectCallback = expectation(description: "bindings serialize/deserialize") + sut.attachLinkAccountSession( + setupIntentID: "seti_12345", + linkAccountSessionID: "las_123456", + clientSecret: "si_client_secret_123" + ) { intent, _ in + guard let intent = intent else { + XCTFail("Intent was null") + return + } + XCTAssertEqual(intent.paymentMethodID, "pm_abcdefg") + expectCallback.fulfill() + } + + wait(for: [expectCallback], timeout: 2.0) + } + + func testPaymentIntent_LinkAccountSessionAttach() { + let sut = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains( + "/payment_intents/pi_12345/link_account_sessions/las_123456/attach" + ) ?? false + } response: { urlRequest in + guard let data = urlRequest.httpBodyOrBodyStream, + let body = String(data: data, encoding: .utf8) + else { + return HTTPStubsResponse( + data: "".data(using: .utf8)!, + statusCode: 400, + headers: nil + ) + } + XCTAssert(body.contains("client_secret=pi_client_secret_123")) + + let jsonText = """ + { + "id": "pi_12345", + "object": "payment_intent", + "amount": 100, + "currency": "usd", + "cancellation_reason": null, + "client_secret": "seti_abc_secret_def", + "created": 1647000000, + "description": null, + "last_setup_error": null, + "livemode": false, + "next_action": null, + "payment_method": "pm_abcdefg", + "payment_method_options": { + "us_bank_account": { + "verification_method": "instant" + } + }, + "payment_method_types": [ + "us_bank_account" + ], + "status": "requires_payment_method" + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + let expectCallback = expectation(description: "bindings serialize/deserialize") + sut.attachLinkAccountSession( + paymentIntentID: "pi_12345", + linkAccountSessionID: "las_123456", + clientSecret: "pi_client_secret_123" + ) { intent, _ in + guard let intent = intent else { + XCTFail("Intent was null") + return + } + XCTAssertEqual(intent.paymentMethodId, "pm_abcdefg") + expectCallback.fulfill() + } + + wait(for: [expectCallback], timeout: 2.0) + } +} diff --git a/Stripe/StripeiOSTests/STPAPIClientTest.swift b/Stripe/StripeiOSTests/STPAPIClientTest.swift new file mode 100644 index 00000000..6ed06c1b --- /dev/null +++ b/Stripe/StripeiOSTests/STPAPIClientTest.swift @@ -0,0 +1,148 @@ +// +// STPAPIClientTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 12/19/14. +// Copyright (c) 2014 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeApplePay +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPAPIClientTest: XCTestCase { + func testSharedClient() { + XCTAssert(STPAPIClient.shared === STPAPIClient.shared) + } + + func testSetDefaultPublishableKey() { + let clientInitializedBefore = STPAPIClient() + StripeAPI.defaultPublishableKey = "test" + let clientInitializedAfter = STPAPIClient() + let sharedClient = STPAPIClient.shared + XCTAssertEqual(clientInitializedBefore.publishableKey, "test") + XCTAssertEqual(clientInitializedAfter.publishableKey, "test") + + // Setting the STPAPIClient instance overrides Stripe.defaultPublishableKey... + sharedClient.publishableKey = "test2" + XCTAssertEqual(sharedClient.publishableKey, "test2") + + // ...while Stripe.defaultPublishableKey remains the same + XCTAssertEqual(StripeAPI.defaultPublishableKey, "test") + } + + func testInitWithPublishableKey() { + let sut = STPAPIClient(publishableKey: "pk_foo") + let authHeader = sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: [:] + ).allHTTPHeaderFields?["Authorization"] + XCTAssertEqual(authHeader, "Bearer pk_foo") + } + + func testSetPublishableKey() { + let sut = STPAPIClient(publishableKey: "pk_foo") + var authHeader = sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: [:] + ).allHTTPHeaderFields?["Authorization"] + XCTAssertEqual(authHeader, "Bearer pk_foo") + sut.publishableKey = "pk_bar" + authHeader = + sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: [:] + ) + .allHTTPHeaderFields?["Authorization"] + XCTAssertEqual(authHeader, "Bearer pk_bar") + } + + func testEphemeralKeyOverwritesHeader() { + let sut = STPAPIClient(publishableKey: "pk_foo") + let ephemeralKey = STPFixtures.ephemeralKey() + let additionalHeaders = sut.authorizationHeader(using: ephemeralKey) + let authHeader = sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: additionalHeaders + ).allHTTPHeaderFields?["Authorization"] + XCTAssertEqual(authHeader, "Bearer " + (ephemeralKey.secret)) + } + + func testSetStripeAccount() { + let sut = STPAPIClient(publishableKey: "pk_foo") + var accountHeader = sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: [:] + ).allHTTPHeaderFields?["Stripe-Account"] + XCTAssertNil(accountHeader) + sut.stripeAccount = "acct_123" + accountHeader = + sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: [:] + ) + .allHTTPHeaderFields?["Stripe-Account"] + XCTAssertEqual(accountHeader, "acct_123") + } + + func testInitWithConfiguration() { + let config = STPFixtures.paymentConfiguration() + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wdeprecated" + config.publishableKey = "pk_123" + config.stripeAccount = "acct_123" + + let sut = STPAPIClient(configuration: config) + XCTAssertEqual(sut.publishableKey, config.publishableKey) + XCTAssertEqual(sut.stripeAccount, config.stripeAccount) + // #pragma clang diagnostic pop + + let accountHeader = sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: [:] + ).allHTTPHeaderFields?["Stripe-Account"] + XCTAssertEqual(accountHeader, "acct_123") + } + + private struct MockUAUsageClass: STPAnalyticsProtocol { + static let stp_analyticsIdentifier = "MockUAUsageClass" + } + + func testPaymentUserAgent() { + STPAnalyticsClient.sharedClient.addClass(toProductUsageIfNecessary: MockUAUsageClass.self) + var params: [String: Any] = [:] + params = STPAPIClient.paramsAddingPaymentUserAgent(params) + XCTAssert((params["payment_user_agent"] as! String).contains("MockUAUsageClass")) + XCTAssert((params["payment_user_agent"] as! String).starts(with: "stripe-ios/")) + } + + func testSetAppInfo() { + let sut = STPAPIClient(publishableKey: "pk_foo") + sut.appInfo = STPAppInfo( + name: "MyAwesomeLibrary", + partnerId: "pp_partner_1234", + version: "1.2.34", + url: "https://myawesomelibrary.info" + ) + let userAgentHeader = sut.configuredRequest( + for: URL(string: "https://www.stripe.com")!, + additionalHeaders: [:] + ).allHTTPHeaderFields?["X-Stripe-User-Agent"] + var userAgentHeaderDict: [AnyHashable: Any]? + do { + if let data = userAgentHeader?.data(using: .utf8) { + userAgentHeaderDict = + try JSONSerialization.jsonObject(with: data, options: []) as? [AnyHashable: Any] + } + } catch { + } + XCTAssertEqual(userAgentHeaderDict?["name"] as! String, "MyAwesomeLibrary") + XCTAssertEqual(userAgentHeaderDict?["partner_id"] as! String, "pp_partner_1234") + XCTAssertEqual(userAgentHeaderDict?["version"] as! String, "1.2.34") + XCTAssertEqual(userAgentHeaderDict?["url"] as! String, "https://myawesomelibrary.info") + } +} diff --git a/Stripe/StripeiOSTests/STPAPISettingsBridgeTest.m b/Stripe/StripeiOSTests/STPAPISettingsBridgeTest.m new file mode 100644 index 00000000..58e85b92 --- /dev/null +++ b/Stripe/StripeiOSTests/STPAPISettingsBridgeTest.m @@ -0,0 +1,86 @@ +// +// STPObjcBridgeTest.m +// StripeiOS Tests +// +// Created by David Estes on 9/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +@import Stripe; +@import XCTest; +@import PassKit; +#import "STPNetworkStubbingTestCase.h" +#import "STPTestingAPIClient.h" +#import "STPFixtures.h" + +@interface StripeAPIBridgeTest : XCTestCase + +@end + +@implementation StripeAPIBridgeTest + +- (void)testStripeAPIBridge { + NSString *testKey = @"pk_test_123"; + StripeAPI.defaultPublishableKey = testKey; + XCTAssertEqualObjects(StripeAPI.defaultPublishableKey, testKey); + StripeAPI.defaultPublishableKey = nil; + + StripeAPI.advancedFraudSignalsEnabled = NO; + XCTAssertFalse(StripeAPI.advancedFraudSignalsEnabled); + StripeAPI.advancedFraudSignalsEnabled = YES; + + + StripeAPI.maxRetries = 2; + XCTAssertEqual(StripeAPI.maxRetries, 2); + StripeAPI.maxRetries = 3; + + // Check that this at least doesn't crash + [StripeAPI handleStripeURLCallbackWithURL:[NSURL URLWithString:@"https://example.com"]]; + + StripeAPI.jcbPaymentNetworkSupported = YES; + XCTAssertTrue(StripeAPI.jcbPaymentNetworkSupported); + StripeAPI.jcbPaymentNetworkSupported = NO; + + StripeAPI.additionalEnabledApplePayNetworks = @[PKPaymentNetworkJCB]; + XCTAssertTrue([StripeAPI.additionalEnabledApplePayNetworks containsObject:PKPaymentNetworkJCB]); + StripeAPI.additionalEnabledApplePayNetworks = @[]; + + PKPaymentRequest *request = [StripeAPI paymentRequestWithMerchantIdentifier:@"test" country:@"US" currency:@"USD"]; + request.paymentSummaryItems = @[[PKPaymentSummaryItem summaryItemWithLabel:@"bar" amount:[NSDecimalNumber decimalNumberWithString:@"1.00"]]]; + #pragma clang diagnostic push + #pragma clang diagnostic ignored "-Wdeprecated" + PKPaymentRequest *request2 = [StripeAPI paymentRequestWithMerchantIdentifier:@"test"]; + request2.paymentSummaryItems = @[[PKPaymentSummaryItem summaryItemWithLabel:@"bar" amount:[NSDecimalNumber decimalNumberWithString:@"1.00"]]]; + #pragma clang diagnostic pop + + XCTAssertTrue([StripeAPI canSubmitPaymentRequest:request]); + XCTAssertTrue([StripeAPI canSubmitPaymentRequest:request2]); + + XCTAssertTrue([StripeAPI deviceSupportsApplePay]); +} + +- (void)testSTPAPIClientBridgeKeys { + NSString *testKey = @"pk_test_123"; + StripeAPI.defaultPublishableKey = testKey; + XCTAssertEqualObjects(testKey, StripeAPI.defaultPublishableKey); + StripeAPI.defaultPublishableKey = nil; +} + +- (void)testSTPAPIClientBridgeSettings { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_test_123"]; + STPPaymentConfiguration *config = [[STPPaymentConfiguration alloc] init]; + client.configuration = config; + XCTAssertEqual(config, client.configuration); + + NSString *stripeAccount = @"acct_123"; + client.stripeAccount = stripeAccount; + XCTAssertEqualObjects(stripeAccount, client.stripeAccount); + + STPAppInfo *appInfo = [[STPAppInfo alloc] initWithName:@"test" partnerId:@"abc123" version:@"1.0" url:@"https://example.com"]; + client.appInfo = appInfo; + XCTAssertEqualObjects(appInfo.name, client.appInfo.name); + + XCTAssertNotNil(STPAPIClient.apiVersion); +} + +@end diff --git a/Stripe/StripeiOSTests/STPAUBECSDebitFormViewSnapshotTests.swift b/Stripe/StripeiOSTests/STPAUBECSDebitFormViewSnapshotTests.swift new file mode 100644 index 00000000..202d43db --- /dev/null +++ b/Stripe/StripeiOSTests/STPAUBECSDebitFormViewSnapshotTests.swift @@ -0,0 +1,118 @@ +// +// STPAUBECSDebitFormViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPAUBECSDebitFormViewSnapshotTests: FBSnapshotTestCase { + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func testDefaultAppearance() { + let view = _newFormView() + _size(toFit: view) + STPSnapshotVerifyView(view, identifier: "STPAUBECSDebitFormView.defaultAppearance") + } + + func testNoDataCustomization() { + let view = _newFormView() + + _applyCustomization(view) + + _size(toFit: view) + + STPSnapshotVerifyView(view, identifier: "STPAUBECSDebitFormView.noDataCustomization") + } + + func testWithDataAppearance() { + let view = _newFormView() + view.nameTextField().text = "Jenny Rosen" + view.emailTextField().text = "jrosen@example.com" + view.bsbNumberTextField().text = "111111" + view.accountNumberTextField().text = "123456" + _size(toFit: view) + + STPSnapshotVerifyView(view, identifier: "STPAUBECSDebitFormView.withDataAppearance") + } + + func testWithDataCustomization() { + let view = _newFormView() + view.nameTextField().text = "Jenny Rosen" + view.emailTextField().text = "jrosen@example.com" + view.bsbNumberTextField().text = "111111" + view.accountNumberTextField().text = "123456" + _applyCustomization(view) + _size(toFit: view) + + STPSnapshotVerifyView(view, identifier: "STPAUBECSDebitFormView.withDataAppearance") + } + + func testInvalidBSBAndEmailAppearance() { + let view = _newFormView() + view.nameTextField().text = "Jenny Rosen" + view.emailTextField().text = "jrosen" + view.bsbNumberTextField().text = "666666" + view.accountNumberTextField().text = "123456" + _size(toFit: view) + + STPSnapshotVerifyView( + view, + identifier: "STPAUBECSDebitFormView.invalidBSBAndEmailAppearance" + ) + } + + func testInvalidBSBAndEmailCustomization() { + let view = _newFormView() + view.nameTextField().text = "Jenny Rosen" + view.emailTextField().text = "jrosen" + view.bsbNumberTextField().text = "666666" + view.accountNumberTextField().text = "123456" + _applyCustomization(view) + _size(toFit: view) + + STPSnapshotVerifyView( + view, + identifier: "STPAUBECSDebitFormView.invalidBSBAndEmailCustomization" + ) + } + + // MARK: - Helpers + func _newFormView() -> STPAUBECSDebitFormView { + let formView = STPAUBECSDebitFormView(companyName: "Snapshotter") + formView.frame = CGRect(x: 0.0, y: 0.0, width: 320.0, height: 600.0) + return formView + } + + func _applyCustomization(_ view: STPAUBECSDebitFormView?) { + view?.formFont = UIFont.boldSystemFont(ofSize: 12.0) + view?.formTextColor = UIColor.blue + view?.formTextErrorColor = UIColor.orange + view?.formPlaceholderColor = UIColor.black + view?.formCursorColor = UIColor.red + view?.formBackgroundColor = UIColor( + red: 255.0 / 255.0, + green: 45.0 / 255.0, + blue: 85.0 / 255.0, + alpha: 1.0 + ) + } + + func _size(toFit view: STPAUBECSDebitFormView?) { + var adjustedFrame = view?.frame + adjustedFrame?.size.height = + view?.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize).height ?? 0.0 + view?.frame = adjustedFrame! + } +} diff --git a/Stripe/StripeiOSTests/STPAUBECSFormViewModelTests.swift b/Stripe/StripeiOSTests/STPAUBECSFormViewModelTests.swift new file mode 100644 index 00000000..829f50c6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPAUBECSFormViewModelTests.swift @@ -0,0 +1,584 @@ +// +// STPAUBECSFormViewModelTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPAUBECSFormViewModelTests: XCTestCase { + func testBECSDebitParams() { + do { + // Test empty data + let model = STPAUBECSFormViewModel() + XCTAssertNil(model.becsDebitParams, "params with no data should be nil") + } + + do { + // Test complete/valid data + let model = STPAUBECSFormViewModel() + model.accountNumber = "123456" + model.bsbNumber = "111-111" + + let params = model.becsDebitParams + XCTAssertNotNil(params, "Failed to create BECS Debit params") + XCTAssertEqual(params?.accountNumber, "123456") + XCTAssertEqual(params?.bsbNumber, "111111") + } + + do { + // Test complete/valid data w/o formatting + let model = STPAUBECSFormViewModel() + model.accountNumber = "123456" + model.bsbNumber = "111111" + + let params = model.becsDebitParams + XCTAssertNotNil(params, "Failed to create BECS Debit params") + XCTAssertEqual(params?.accountNumber, "123456") + XCTAssertEqual(params?.bsbNumber, "111111") + } + + do { + // Test complete/valid accountNumber, incomplete bsb number + let model = STPAUBECSFormViewModel() + model.accountNumber = "123456" + model.bsbNumber = "111-" + + let params = model.becsDebitParams + XCTAssertNil(params, "Should not create params with incomplete bsb number") + } + + do { + // Test incomplete accountNumber, complete/valid bsb number + let model = STPAUBECSFormViewModel() + model.accountNumber = "1234" + model.bsbNumber = "111-111" + + let params = model.becsDebitParams + XCTAssertNil(params, "Should not create params with incomplete account number") + } + + do { + // Test invalid accountNumber, complete/valid bsb number + let model = STPAUBECSFormViewModel() + model.accountNumber = "12345678910" + model.bsbNumber = "111-111" + + let params = model.becsDebitParams + XCTAssertNil(params, "Should not create params with invalid account number") + } + + do { + // Test complete/valid accountNumber, invalid bsb number + let model = STPAUBECSFormViewModel() + model.accountNumber = "123456" + model.bsbNumber = "666-666" + + let params = model.becsDebitParams + XCTAssertNil(params, "Should not create params with incomplete bsb number") + } + } + + func testPaymentMethodParams() { + do { + /// Test empty + let model = STPAUBECSFormViewModel() + XCTAssertNil(model.paymentMethodParams, "params with no data should be nil") + } + + do { + /// name: + + /// email: + + /// bsb: + (formatting) + /// account: + + let model = STPAUBECSFormViewModel() + model.name = "Jenny Rosen" + model.email = "jrosen@example.com" + model.accountNumber = "123456" + model.bsbNumber = "111-111" + + let params = model.paymentMethodParams + XCTAssertNotNil(params, "Failed to create BECS Debit params") + XCTAssertEqual(params?.billingDetails?.name, "Jenny Rosen") + XCTAssertEqual(params?.billingDetails?.email, "jrosen@example.com") + XCTAssertEqual(params?.auBECSDebit?.accountNumber, "123456") + XCTAssertEqual(params?.auBECSDebit?.bsbNumber, "111111") + } + + do { + /// name: + + /// email: + + /// bsb: + + /// account: + + let model = STPAUBECSFormViewModel() + model.name = "Jenny Rosen" + model.email = "jrosen@example.com" + model.accountNumber = "123456" + model.bsbNumber = "111111" + + let params = model.paymentMethodParams + XCTAssertNotNil(params, "Failed to create BECS Debit params") + XCTAssertEqual(params?.billingDetails?.name, "Jenny Rosen") + XCTAssertEqual(params?.billingDetails?.email, "jrosen@example.com") + XCTAssertEqual(params?.auBECSDebit?.accountNumber, "123456") + XCTAssertEqual(params?.auBECSDebit?.bsbNumber, "111111") + } + + do { + /// name: + + /// email: + + /// bsb: x (incomplete) + /// account: + + let model = STPAUBECSFormViewModel() + model.name = "Jenny Rosen" + model.email = "jrosen@example.com" + model.accountNumber = "123456" + model.bsbNumber = "111-" + + let params = model.paymentMethodParams + XCTAssertNil(params, "Should not create params with incomplete bsb number") + } + + do { + /// name: + + /// email: + + /// bsb: + + /// account: x (incomplete) + let model = STPAUBECSFormViewModel() + model.name = "Jenny Rosen" + model.email = "jrosen@example.com" + model.accountNumber = "1234" + model.bsbNumber = "111-111" + + let params = model.paymentMethodParams + XCTAssertNil(params, "Should not create params with incomplete account number") + } + + do { + /// name: + + /// email: + + /// bsb: + + /// account: x + let model = STPAUBECSFormViewModel() + model.name = "Jenny Rosen" + model.email = "jrosen@example.com" + model.accountNumber = "12345678910" + model.bsbNumber = "111-111" + + let params = model.paymentMethodParams + XCTAssertNil(params, "Should not create params with invalid account number") + } + + do { + /// name: + + /// email: + + /// bsb: x + /// account: + + let model = STPAUBECSFormViewModel() + model.name = "Jenny Rosen" + model.email = "jrosen@example.com" + model.accountNumber = "123456" + model.bsbNumber = "666-666" + + let params = model.paymentMethodParams + XCTAssertNil(params, "Should not create params with incomplete bsb number") + } + + do { + /// name: x + /// email: + + /// bsb: + (formatting) + /// account: + + let model = STPAUBECSFormViewModel() + model.name = "" + model.email = "jrosen@example.com" + model.accountNumber = "123456" + model.bsbNumber = "111-111" + + let params = model.paymentMethodParams + XCTAssertNil(params, "Should not create payment method params without name.") + } + + do { + /// name: + + /// email: x + /// bsb: + (formatting) + /// account: + + let model = STPAUBECSFormViewModel() + model.name = "Jenny Rosen" + model.email = "jrose" + model.accountNumber = "123456" + model.bsbNumber = "111-111" + + let params = model.paymentMethodParams + XCTAssertNil(params, "Should not create payment method params with invalid email.") + } + } + + func testBSBLabelForInput() { + do { + // empty test + let model = STPAUBECSFormViewModel() + var isErrorString = true + var bsbLabel = model.bsbLabel( + forInput: "", + editing: false, + isErrorString: &isErrorString + ) + XCTAssertFalse(isErrorString, "Empty input shouldn't be an error.") + XCTAssertNil(bsbLabel, "No bsb label for empty input.") + + isErrorString = true + bsbLabel = model.bsbLabel(forInput: nil, editing: true, isErrorString: &isErrorString) + XCTAssertFalse(isErrorString, "nil input shouldn't be an error.") + XCTAssertNil(bsbLabel, "No bsb label for nil input.") + } + + do { + // invalid test + let model = STPAUBECSFormViewModel() + var isErrorString = false + var bsbLabel = model.bsbLabel( + forInput: "666-666", + editing: false, + isErrorString: &isErrorString + ) + XCTAssertTrue(isErrorString, "Invalid input should be an error.") + XCTAssertEqual(bsbLabel, "The BSB you entered is invalid.") + + isErrorString = false + bsbLabel = model.bsbLabel( + forInput: "666-666", + editing: true, + isErrorString: &isErrorString + ) + XCTAssertTrue(isErrorString, "Invalid input should be an error (editing).") + XCTAssertEqual(bsbLabel, "The BSB you entered is invalid.") + } + + do { + // incomplete test + let model = STPAUBECSFormViewModel() + var isErrorString = false + var bsbLabel = model.bsbLabel( + forInput: "111-11", + editing: false, + isErrorString: &isErrorString + ) + XCTAssertTrue(isErrorString, "Incomplete input should be an error when not editing.") + XCTAssertEqual(bsbLabel, "The BSB you entered is incomplete.") + + isErrorString = true + bsbLabel = model.bsbLabel( + forInput: "111-11", + editing: true, + isErrorString: &isErrorString + ) + XCTAssertFalse(isErrorString, "Incomplete input should not be an error when editing.") + XCTAssertEqual(bsbLabel, "St George Bank (division of Westpac Bank)") + } + + do { + // valid test + let model = STPAUBECSFormViewModel() + var isErrorString = true + var bsbLabel = model.bsbLabel( + forInput: "111-111", + editing: false, + isErrorString: &isErrorString + ) + XCTAssertFalse(isErrorString, "Complete input should be not an error when not editing.") + XCTAssertEqual(bsbLabel, "St George Bank (division of Westpac Bank)") + + isErrorString = true + bsbLabel = model.bsbLabel( + forInput: "111-111", + editing: true, + isErrorString: &isErrorString + ) + XCTAssertFalse(isErrorString, "Complete input should not be an error when editing.") + XCTAssertEqual(bsbLabel, "St George Bank (division of Westpac Bank)") + } + } + + func testIsInputValid() { + do { + // name + let model = STPAUBECSFormViewModel() + XCTAssertTrue( + model.isInputValid("", for: .name, editing: false), + "Name should always be valid." + ) + XCTAssertTrue( + model.isInputValid("Jen", for: .name, editing: true), + "Name should always be valid." + ) + } + + do { + // email + let model = STPAUBECSFormViewModel() + XCTAssertFalse( + model.isInputValid("jrosen", for: .email, editing: false), + "Partial email is invalid when not editing." + ) + XCTAssertTrue( + model.isInputValid("jrosen", for: .email, editing: true), + "Partial email is valid when editing." + ) + + XCTAssertTrue( + model.isInputValid("", for: .email, editing: false), + "Empty email is always valid." + ) + XCTAssertTrue( + model.isInputValid("", for: .email, editing: true), + "Empty email is always valid." + ) + + XCTAssertTrue( + model.isInputValid("jrosen@example.com", for: .email, editing: false), + "Valid email." + ) + XCTAssertTrue( + model.isInputValid("jrosen@example.com", for: .email, editing: true), + "Valid email." + ) + } + + do { + // bsb + let model = STPAUBECSFormViewModel() + XCTAssertFalse( + model.isInputValid("111-1", for: .BSBNumber, editing: false), + "Partial bsb is invalid when not editing." + ) + XCTAssertTrue( + model.isInputValid("111-1", for: .BSBNumber, editing: true), + "Partial bsb is valid when editing." + ) + + XCTAssertTrue( + model.isInputValid("", for: .BSBNumber, editing: false), + "Empty bsb is always valid." + ) + XCTAssertTrue( + model.isInputValid("", for: .BSBNumber, editing: true), + "Empty bsb is always valid." + ) + + XCTAssertTrue( + model.isInputValid("111-111", for: .BSBNumber, editing: false), + "Valid bsb." + ) + XCTAssertTrue( + model.isInputValid("111-111", for: .BSBNumber, editing: true), + "Valid bsb." + ) + + XCTAssertFalse( + model.isInputValid("666-6", for: .BSBNumber, editing: false), + "Invalid partial bsb is always invalid." + ) + XCTAssertFalse( + model.isInputValid("666-6", for: .BSBNumber, editing: true), + "Invalid partial bsb is always invalid." + ) + + XCTAssertFalse( + model.isInputValid("666-666", for: .BSBNumber, editing: false), + "Invalid full bsb is always invalid." + ) + XCTAssertFalse( + model.isInputValid("666-666", for: .BSBNumber, editing: true), + "Invalid full bsb is always invalid." + ) + } + + do { + // account + let model = STPAUBECSFormViewModel() + XCTAssertFalse( + model.isInputValid("1234", for: .accountNumber, editing: false), + "Partial account number is invalid when not editing." + ) + XCTAssertTrue( + model.isInputValid("1234", for: .accountNumber, editing: true), + "Partial account number is valid when editing." + ) + + XCTAssertTrue( + model.isInputValid("", for: .accountNumber, editing: false), + "Empty account number is always valid." + ) + XCTAssertTrue( + model.isInputValid("", for: .accountNumber, editing: true), + "Empty account number is always valid." + ) + + XCTAssertTrue( + model.isInputValid("12345", for: .accountNumber, editing: false), + "Valid account number." + ) + XCTAssertTrue( + model.isInputValid("12345", for: .accountNumber, editing: true), + "Valid account number." + ) + + XCTAssertFalse( + model.isInputValid("12345678910", for: .accountNumber, editing: false), + "Invalid account number is always invalid." + ) + XCTAssertFalse( + model.isInputValid("12345678910", for: .accountNumber, editing: true), + "Invalid account number is always invalid." + ) + } + } + + func testIsFieldComplete() { + do { + // name + let model = STPAUBECSFormViewModel() + XCTAssertFalse( + model.isFieldComplete(withInput: "", in: .name, editing: false), + "Empty name is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "", in: .name, editing: true), + "Empty name is not complete." + ) + + XCTAssertTrue( + model.isFieldComplete(withInput: "Jen", in: .name, editing: false), + "Non-empty name is complete." + ) + XCTAssertTrue( + model.isFieldComplete(withInput: "Jenny Rosen", in: .name, editing: true), + "Non-empty name is complete." + ) + } + + do { + // email + let model = STPAUBECSFormViewModel() + XCTAssertFalse( + model.isFieldComplete(withInput: "jrosen", in: .email, editing: false), + "Partial email is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "jrosen", in: .email, editing: true), + "Partial email is not complete." + ) + + XCTAssertTrue( + model.isFieldComplete(withInput: "jrosen@example.com", in: .email, editing: false), + "Full email is complete." + ) + XCTAssertTrue( + model.isFieldComplete(withInput: "jrosen@example.com", in: .email, editing: true), + "Full email is complete." + ) + } + + do { + // bsb + let model = STPAUBECSFormViewModel() + XCTAssertFalse( + model.isFieldComplete(withInput: "111-1", in: .BSBNumber, editing: false), + "Partial bsb is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "111-1", in: .BSBNumber, editing: true), + "Partial bsb is not complete." + ) + + XCTAssertFalse( + model.isFieldComplete(withInput: "", in: .BSBNumber, editing: false), + "Empty bsb is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "", in: .BSBNumber, editing: true), + "Empty bsb is not complete." + ) + + XCTAssertTrue( + model.isFieldComplete(withInput: "111-111", in: .BSBNumber, editing: false), + "Full bsb is complete." + ) + XCTAssertTrue( + model.isFieldComplete(withInput: "111-111", in: .BSBNumber, editing: true), + "Full bsb is complete." + ) + + XCTAssertFalse( + model.isFieldComplete(withInput: "666-6", in: .BSBNumber, editing: false), + "Invalid partial bsb is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "666-6", in: .BSBNumber, editing: true), + "Invalid partial bsb is not complete." + ) + + XCTAssertFalse( + model.isFieldComplete(withInput: "666-666", in: .BSBNumber, editing: false), + "Invalid full bsb is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "666-666", in: .BSBNumber, editing: true), + "Invalid full bsb is not complete." + ) + } + + do { + // account + let model = STPAUBECSFormViewModel() + XCTAssertFalse( + model.isFieldComplete(withInput: "1234", in: .accountNumber, editing: false), + "Partial account number is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "1234", in: .accountNumber, editing: true), + "Partial account number is not complete." + ) + + XCTAssertFalse( + model.isFieldComplete(withInput: "", in: .accountNumber, editing: false), + "Empty account number is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "", in: .accountNumber, editing: true), + "Empty account number is not complete." + ) + + XCTAssertTrue( + model.isFieldComplete(withInput: "12345", in: .accountNumber, editing: false), + "Min length account number is complete when not editing." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "12345", in: .accountNumber, editing: true), + "Min length account number is not complete when editing." + ) + + XCTAssertTrue( + model.isFieldComplete(withInput: "123456789", in: .accountNumber, editing: true), + "Max length account number is complete when editing." + ) + + XCTAssertFalse( + model.isFieldComplete(withInput: "12345678910", in: .accountNumber, editing: false), + "Invalid account number is not complete." + ) + XCTAssertFalse( + model.isFieldComplete(withInput: "12345678910", in: .accountNumber, editing: true), + "Invalid account number is not complete." + ) + } + } +} diff --git a/Stripe/StripeiOSTests/STPAddCardViewControllerLocalizationTests.swift b/Stripe/StripeiOSTests/STPAddCardViewControllerLocalizationTests.swift new file mode 100644 index 00000000..72458301 --- /dev/null +++ b/Stripe/StripeiOSTests/STPAddCardViewControllerLocalizationTests.swift @@ -0,0 +1,106 @@ +// +// STPAddCardViewControllerLocalizationTests.swift +// StripeiOS Tests +// +// Created by Brian Dorfman on 10/17/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPAddCardViewControllerLocalizationTests: FBSnapshotTestCase { + override func setUp() { + super.setUp() + + // self.recordMode = true + } + + func performSnapshotTest(forLanguage language: String?, delivery: Bool) { + let config = STPFixtures.paymentConfiguration() + config.companyName = "Test Company" + config.requiredBillingAddressFields = .full + config.shippingType = delivery ? .delivery : .shipping + config.cardScanningEnabled = true + STPLocalizationUtils.overrideLanguage(to: language) + + let addCardVC = STPAddCardViewController( + configuration: config, + theme: STPTheme.defaultTheme + ) + addCardVC.shippingAddress = STPAddress() + addCardVC.shippingAddress?.line1 = "1" // trigger "use shipping address" button + + let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: addCardVC)! + + if delivery { + addCardVC.addressViewModel.addressFieldTableViewCountryCode = "INVALID" + STPSnapshotVerifyView(viewToTest, identifier: "delivery") + } else { + /// This method rejects nil or empty country codes to stop strange looking behavior + /// when scrolling to the top "unset" position in the picker, so put in + /// an invalid country code instead to test seeing the "Country" placeholder + addCardVC.addressViewModel.addressFieldTableViewCountryCode = "INVALID" + STPSnapshotVerifyView(viewToTest, identifier: "no_country") + + addCardVC.addressViewModel.addressFieldTableViewCountryCode = "US" + STPSnapshotVerifyView(viewToTest, identifier: "US") + + addCardVC.addressViewModel.addressFieldTableViewCountryCode = "GB" + STPSnapshotVerifyView(viewToTest, identifier: "GB") + + addCardVC.addressViewModel.addressFieldTableViewCountryCode = "CA" + STPSnapshotVerifyView(viewToTest, identifier: "CA") + + addCardVC.addressViewModel.addressFieldTableViewCountryCode = "MX" + STPSnapshotVerifyView(viewToTest, identifier: "MX") + } + + STPLocalizationUtils.overrideLanguage(to: nil) + } + + func testGerman() { + performSnapshotTest(forLanguage: "de", delivery: false) + performSnapshotTest(forLanguage: "de", delivery: true) + } + + func testEnglish() { + performSnapshotTest(forLanguage: "en", delivery: false) + performSnapshotTest(forLanguage: "en", delivery: true) + } + + func testSpanish() { + performSnapshotTest(forLanguage: "es", delivery: false) + performSnapshotTest(forLanguage: "es", delivery: true) + } + + func testFrench() { + performSnapshotTest(forLanguage: "fr", delivery: false) + performSnapshotTest(forLanguage: "fr", delivery: true) + } + + func testItalian() { + performSnapshotTest(forLanguage: "it", delivery: false) + performSnapshotTest(forLanguage: "it", delivery: true) + } + + func testJapanese() { + performSnapshotTest(forLanguage: "ja", delivery: false) + performSnapshotTest(forLanguage: "ja", delivery: true) + } + + func testDutch() { + performSnapshotTest(forLanguage: "nl", delivery: false) + performSnapshotTest(forLanguage: "nl", delivery: true) + } + + func testChinese() { + performSnapshotTest(forLanguage: "zh-Hans", delivery: false) + performSnapshotTest(forLanguage: "zh-Hans", delivery: true) + } +} diff --git a/Stripe/StripeiOSTests/STPAddCardViewControllerTest.swift b/Stripe/StripeiOSTests/STPAddCardViewControllerTest.swift new file mode 100644 index 00000000..39e70eb6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPAddCardViewControllerTest.swift @@ -0,0 +1,292 @@ +// +// STPAddCardViewControllerTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 7/5/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import OHHTTPStubs +import OHHTTPStubsSwift +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class MockDelegate: NSObject, STPAddCardViewControllerDelegate { + func addCardViewControllerDidCancel(_ addCardViewController: STPAddCardViewController) { + + } + + var addCardViewControllerDidCreatePaymentMethodBlock: + (STPAddCardViewController, STPPaymentMethod, STPErrorBlock) -> Void = { _, _, _ in } + func addCardViewController( + _ addCardViewController: STPAddCardViewController, + didCreatePaymentMethod paymentMethod: STPPaymentMethod, + completion: @escaping STPErrorBlock + ) { + addCardViewControllerDidCreatePaymentMethodBlock( + addCardViewController, + paymentMethod, + completion + ) + } +} + +class STPAddCardViewControllerTest: APIStubbedTestCase { + + func paymentMethodAPIFilter( + expectedCardParams: STPPaymentMethodCardParams, + urlRequest: URLRequest + ) -> Bool { + if urlRequest.url?.absoluteString.contains("payment_methods") ?? false { + let cardNumber = urlRequest.queryItems?.first(where: { item in + item.name == "card[number]" + }) + XCTAssertEqual(cardNumber!.value, expectedCardParams.number) + return true + } + return false + } + + func buildAddCardViewController() -> STPAddCardViewController? { + let config = STPFixtures.paymentConfiguration() + let theme = STPTheme.defaultTheme + let vc = STPAddCardViewController( + configuration: config, + theme: theme + ) + XCTAssertNotNil(vc.view) + return vc + } + + func testPrefilledBillingAddress_removeAddress() { + let config = STPFixtures.paymentConfiguration() + config.requiredBillingAddressFields = .postalCode + let sut = STPAddCardViewController( + configuration: config, + theme: STPTheme.defaultTheme + ) + let address = STPAddress() + address.name = "John Smith Doe" + address.phone = "8885551212" + address.email = "foo@example.com" + address.line1 = "55 John St" + address.city = "Harare" + address.postalCode = "10002" + // Zimbabwe does not require zip codes, while the default locale for tests (US) does + address.country = "ZW" + // Sanity checks + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) + + let prefilledInfo = STPUserInformation() + prefilledInfo.billingAddress = address + sut.prefilledInformation = prefilledInfo + + XCTAssertNoThrow(sut.loadView()) + XCTAssertNoThrow(sut.viewDidLoad()) + } + + func testPrefilledBillingAddress_viewDidLoadHappensBeforeSettingAddress() { + let config = STPFixtures.paymentConfiguration() + config.requiredBillingAddressFields = .full + let sut = STPAddCardViewController( + configuration: config, + theme: STPTheme.defaultTheme + ) + XCTAssertNoThrow(sut.loadView()) + XCTAssertNoThrow(sut.viewDidLoad()) + + let address = STPAddress() + address.name = "John Smith Doe" + address.line1 = "55 John St" + address.city = "Harare" + address.postalCode = "10002" + + let prefilledInfo = STPUserInformation() + prefilledInfo.billingAddress = address + sut.prefilledInformation = prefilledInfo + + let nameCell = sut.addressViewModel.addressCells.first { $0.type == .name }! + XCTAssertEqual( nameCell.contents, "John Smith Doe") + + let line1Cell = sut.addressViewModel.addressCells.first { $0.type == .line1 }! + XCTAssertEqual( line1Cell.contents, "55 John St") + + let cityCell = sut.addressViewModel.addressCells.first { $0.type == .city }! + XCTAssertEqual( cityCell.contents, "Harare") + + let zipCell = sut.addressViewModel.addressCells.first { $0.type == .zip }! + XCTAssertEqual( zipCell.contents, "10002") + } + + func testPrefilledBillingAddress_addAddress() { + // Zimbabwe does not require zip codes, while the default locale for tests (US) does + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_ZW") as Locale) { + // Sanity checks + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) + let config = STPFixtures.paymentConfiguration() + config.requiredBillingAddressFields = .postalCode + let sut = STPAddCardViewController( + configuration: config, + theme: STPTheme.defaultTheme + ) + let address = STPAddress() + address.name = "John Smith Doe" + address.phone = "8885551212" + address.email = "foo@example.com" + address.line1 = "55 John St" + address.city = "New York" + address.state = "NY" + address.postalCode = "10002" + address.country = "US" + + let prefilledInfo = STPUserInformation() + prefilledInfo.billingAddress = address + sut.prefilledInformation = prefilledInfo + + XCTAssertNoThrow(sut.loadView()) + XCTAssertNoThrow(sut.viewDidLoad()) + } + } + + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Warc-performSelector-leaks" + func testNextWithCreatePaymentMethodError() { + let sut = buildAddCardViewController()! + let expectedCardParams = STPFixtures.paymentMethodCardParams() + sut.paymentCell?.paymentField!.cardParams = expectedCardParams + + let exp = expectation(description: "createPaymentMethodWithCard network request") + stub { urlRequest in + return self.paymentMethodAPIFilter( + expectedCardParams: expectedCardParams, + urlRequest: urlRequest + ) + } response: { _ in + XCTAssertTrue(sut.loading) + let paymentMethod = ["error": "intentionally_invalid"] + defer { + exp.fulfill() + } + return HTTPStubsResponse(jsonObject: paymentMethod, statusCode: 200, headers: nil) + } + sut.apiClient = stubbedAPIClient() + // tap next button + let nextButton = sut.navigationItem.rightBarButtonItem + _ = nextButton?.target?.perform(nextButton?.action, with: nextButton) + + waitForExpectations(timeout: 2, handler: nil) + + // It takes a few more spins on the runloop before we get a response from + // the HTTP stubs, so we'll wait 0.5 seconds before checking the loading indicator. + let loadExp = expectation(description: "loading has stopped") + DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) { + XCTAssertFalse(sut.loading) + loadExp.fulfill() + } + waitForExpectations(timeout: 1, handler: nil) + } + + func testNextWithCreatePaymentMethodSuccessAndDidCreatePaymentMethodError() { + let sut = buildAddCardViewController()! + let createPaymentMethodExp = expectation(description: "createPaymentMethodWithCard") + + let expectedCardParams = STPFixtures.paymentMethodCardParams() + let expectedPaymentMethod = STPFixtures.paymentMethod() + let expectedPaymentMethodData = STPFixtures.paymentMethodJSON() + + stub { urlRequest in + return self.paymentMethodAPIFilter( + expectedCardParams: expectedCardParams, + urlRequest: urlRequest + ) + } response: { _ in + XCTAssertTrue(sut.loading) + defer { + createPaymentMethodExp.fulfill() + } + return HTTPStubsResponse( + jsonObject: expectedPaymentMethodData, + statusCode: 200, + headers: nil + ) + } + + let mockDelegate = MockDelegate() + sut.apiClient = stubbedAPIClient() + sut.delegate = mockDelegate + sut.paymentCell?.paymentField!.cardParams = expectedCardParams + + let didCreatePaymentMethodExp = expectation(description: "didCreatePaymentMethod") + + mockDelegate.addCardViewControllerDidCreatePaymentMethodBlock = { + (_, paymentMethod, completion) in + XCTAssertTrue(sut.loading) + let error = NSError.stp_genericFailedToParseResponseError() + XCTAssertEqual(paymentMethod.stripeId, expectedPaymentMethod.stripeId) + completion(error) + XCTAssertFalse(sut.loading) + didCreatePaymentMethodExp.fulfill() + } + + // tap next button + let nextButton = sut.navigationItem.rightBarButtonItem + _ = nextButton?.target?.perform(nextButton?.action, with: nextButton) + + waitForExpectations(timeout: 2, handler: nil) + } + + func testNextWithCreateTokenSuccessAndDidCreateTokenSuccess() { + let sut = buildAddCardViewController()! + + let createPaymentMethodExp = expectation(description: "createPaymentMethodWithCard") + + let expectedCardParams = STPFixtures.paymentMethodCardParams() + let expectedPaymentMethod = STPFixtures.paymentMethod() + let expectedPaymentMethodData = STPFixtures.paymentMethodJSON() + + stub { urlRequest in + return self.paymentMethodAPIFilter( + expectedCardParams: expectedCardParams, + urlRequest: urlRequest + ) + } response: { _ in + XCTAssertTrue(sut.loading) + defer { + createPaymentMethodExp.fulfill() + } + return HTTPStubsResponse( + jsonObject: expectedPaymentMethodData, + statusCode: 200, + headers: nil + ) + } + + let mockDelegate = MockDelegate() + sut.apiClient = stubbedAPIClient() + sut.delegate = mockDelegate + sut.paymentCell?.paymentField!.cardParams = expectedCardParams + + let didCreatePaymentMethodExp = expectation(description: "didCreatePaymentMethod") + mockDelegate.addCardViewControllerDidCreatePaymentMethodBlock = { + (_, paymentMethod, completion) in + XCTAssertTrue(sut.loading) + XCTAssertEqual(paymentMethod.stripeId, expectedPaymentMethod.stripeId) + completion(nil) + XCTAssertFalse(sut.loading) + didCreatePaymentMethodExp.fulfill() + } + + // tap next button + let nextButton = sut.navigationItem.rightBarButtonItem + _ = nextButton?.target?.perform(nextButton?.action, with: nextButton) + + waitForExpectations(timeout: 2, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPAddressTests.m b/Stripe/StripeiOSTests/STPAddressTests.m new file mode 100644 index 00000000..8f7b381f --- /dev/null +++ b/Stripe/StripeiOSTests/STPAddressTests.m @@ -0,0 +1,525 @@ +// +// STPAddressTests.m +// Stripe +// +// Created by Ben Guo on 4/13/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import +#import +#import +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPAddressTests : XCTestCase + +@end + +@implementation STPAddressTests + +- (void)testInitWithPKContact_complete { + PKContact *contact = [PKContact new]; + { + NSPersonNameComponents *name = [NSPersonNameComponents new]; + name.givenName = @"John"; + name.familyName = @"Doe"; + contact.name = name; + + contact.emailAddress = @"foo@example.com"; + contact.phoneNumber = [CNPhoneNumber phoneNumberWithStringValue:@"888-555-1212"]; + + CNMutablePostalAddress *address = [CNMutablePostalAddress new]; + address.street = @"55 John St"; + address.city = @"New York"; + address.state = @"NY"; + address.postalCode = @"10002"; + address.ISOCountryCode = @"US"; + address.country = @"United States"; + contact.postalAddress = address.copy; + } + + STPAddress *address = [[STPAddress alloc] initWithPKContact:contact]; + XCTAssertEqualObjects(@"John Doe", address.name); + XCTAssertEqualObjects(@"8885551212", address.phone); + XCTAssertEqualObjects(@"foo@example.com", address.email); + XCTAssertEqualObjects(@"55 John St", address.line1); + XCTAssertEqualObjects(@"New York", address.city); + XCTAssertEqualObjects(@"NY", address.state); + XCTAssertEqualObjects(@"10002", address.postalCode); + XCTAssertEqualObjects(@"US", address.country); +} + +- (void)testInitWithPKContact_partial { + PKContact *contact = [PKContact new]; + { + NSPersonNameComponents *name = [NSPersonNameComponents new]; + name.givenName = @"John"; + contact.name = name; + + CNMutablePostalAddress *address = [CNMutablePostalAddress new]; + address.state = @"VA"; + contact.postalAddress = address.copy; + } + + STPAddress *address = [[STPAddress alloc] initWithPKContact:contact]; + XCTAssertEqualObjects(@"John", address.name); + XCTAssertNil(address.phone); + XCTAssertNil(address.email); + XCTAssertNil(address.line1); + XCTAssertNil(address.city); + XCTAssertEqualObjects(@"VA", address.state); + XCTAssertNil(address.postalCode); + XCTAssertNil(address.country); +} + +- (void)testInitWithCNContact_complete { + if ([CNContact class] == nil) { + // Method not supported by iOS version + return; + } + + CNMutableContact *contact = [CNMutableContact new]; + { + contact.givenName = @"John"; + contact.familyName = @"Doe"; + + contact.emailAddresses = @[ + [CNLabeledValue labeledValueWithLabel:CNLabelHome + value:@"foo@example.com"], + [CNLabeledValue labeledValueWithLabel:CNLabelWork + value:@"bar@example.com"], + + + ]; + + contact.phoneNumbers = @[ + [CNLabeledValue labeledValueWithLabel:CNLabelHome + value:[CNPhoneNumber phoneNumberWithStringValue:@"888-555-1212"]], + [CNLabeledValue labeledValueWithLabel:CNLabelWork + value:[CNPhoneNumber phoneNumberWithStringValue:@"555-555-5555"]], + + + ]; + + CNMutablePostalAddress *address = [CNMutablePostalAddress new]; + address.street = @"55 John St"; + address.city = @"New York"; + address.state = @"NY"; + address.postalCode = @"10002"; + address.ISOCountryCode = @"US"; + address.country = @"United States"; + contact.postalAddresses = @[ + [CNLabeledValue labeledValueWithLabel:CNLabelHome + value:address], + ]; + } + + STPAddress *address = [[STPAddress alloc] initWithCNContact:contact]; + XCTAssertEqualObjects(@"John Doe", address.name); + XCTAssertEqualObjects(@"8885551212", address.phone); + XCTAssertEqualObjects(@"foo@example.com", address.email); + XCTAssertEqualObjects(@"55 John St", address.line1); + XCTAssertEqualObjects(@"New York", address.city); + XCTAssertEqualObjects(@"NY", address.state); + XCTAssertEqualObjects(@"10002", address.postalCode); + XCTAssertEqualObjects(@"US", address.country); +} + +- (void)testInitWithCNContact_partial { + if ([CNContact class] == nil) { + // Method not supported by iOS version + return; + } + + CNMutableContact *contact = [CNMutableContact new]; + { + contact.givenName = @"John"; + + CNMutablePostalAddress *address = [CNMutablePostalAddress new]; + address.state = @"VA"; + contact.postalAddresses = @[ + [CNLabeledValue labeledValueWithLabel:CNLabelHome + value:address], + ]; + } + + STPAddress *address = [[STPAddress alloc] initWithCNContact:contact]; + XCTAssertEqualObjects(@"John", address.name); + XCTAssertNil(address.phone); + XCTAssertNil(address.email); + XCTAssertNil(address.line1); + XCTAssertNil(address.city); + XCTAssertEqualObjects(@"VA", address.state); + XCTAssertNil(address.postalCode); + XCTAssertNil(address.country); +} + +- (void)testPKContactValue { + STPAddress *address = [STPAddress new]; + address.name = @"John Smith Doe"; + address.phone = @"8885551212"; + address.email = @"foo@example.com"; + address.line1 = @"55 John St"; + address.city = @"New York"; + address.state = @"NY"; + address.postalCode = @"10002"; + address.country = @"US"; + + PKContact *contact = [address PKContactValue]; + XCTAssertEqualObjects(contact.name.givenName, @"John"); + XCTAssertEqualObjects(contact.name.familyName, @"Smith Doe"); + XCTAssertEqualObjects(contact.phoneNumber.stringValue, @"8885551212"); + XCTAssertEqualObjects(contact.emailAddress, @"foo@example.com"); + CNPostalAddress *postalAddress = contact.postalAddress; + XCTAssertEqualObjects(postalAddress.street, @"55 John St"); + XCTAssertEqualObjects(postalAddress.city, @"New York"); + XCTAssertEqualObjects(postalAddress.state, @"NY"); + XCTAssertEqualObjects(postalAddress.postalCode, @"10002"); + XCTAssertEqualObjects(postalAddress.country, @"US"); +} + +- (void)testContainsRequiredFieldsNone { + STPAddress *address = [STPAddress new]; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsNone]); + address.line1 = @"55 John St"; + address.city = @"New York"; + address.state = @"NY"; + address.postalCode = @"10002"; + address.country = @"US"; + address.phone = @"8885551212"; + address.email = @"foo@example.com"; + address.name = @"John Doe"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsNone]); + address.country = @"UK"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsNone]); +} + +- (void)testContainsRequiredFieldsZip { + STPAddress *address = [STPAddress new]; + + // nil country is treated as generic postal requirement + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsPostalCode]); + address.country = @"IE"; //should pass for country which doesn't require zip/postal + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsPostalCode]); + address.country = @"US"; + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsPostalCode]); + address.postalCode = @"10002"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsPostalCode]); + address.postalCode = @"ABCDE"; + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsPostalCode]); + address.country = @"UK"; // should pass for alphanumeric countries + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsPostalCode]); + address.country = nil; // nil treated as alphanumeric + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsPostalCode]); +} + +- (void)testContainsRequiredFieldsFull { + STPAddress *address = [STPAddress new]; + + /** + Required fields for full are: + line1, city, country, state (US only) and a valid postal code (based on country) + */ + + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsFull]); + address.country = @"US"; + address.line1 = @"55 John St"; + + // Fail on partial + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsFull]); + + address.city = @"New York"; + + // For US fail if missing state or zip + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsFull]); + address.state = @"NY"; + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsFull]); + address.postalCode = @"ABCDE"; + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsFull]); + //postal must be numeric for US + address.postalCode = @"10002"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); + address.phone = @"8885551212"; + address.email = @"foo@example.com"; + address.name = @"John Doe"; + // Name/phone/email should have no effect + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); + + // Non US countries don't require state + address.country = @"UK"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); + address.state = nil; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); + // alphanumeric postal ok in some countries + address.postalCode = @"ABCDE"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); + // UK requires ZIP + address.postalCode = nil; + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsFull]); + + + address.country = @"IE"; // Doesn't require postal or state, but allows them + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); + address.postalCode = @"ABCDE"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); + address.state = @"Test"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsFull]); +} + +- (void)testContainsRequiredFieldsName { + STPAddress *address = [STPAddress new]; + + XCTAssertFalse([address containsRequiredFields:STPBillingAddressFieldsName]); + address.name = @"Jane Doe"; + XCTAssertTrue([address containsRequiredFields:STPBillingAddressFieldsName]); +} + +- (void)testContainsContentForBillingAddressFields { + STPAddress *address = [STPAddress new]; + + // Empty address should return false for everything + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsNone]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsPostalCode]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsFull]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsName]); + + // 1+ characters in postalCode will return true for .PostalCode && .Full + address.postalCode = @"0"; + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsNone]); + XCTAssertTrue([address containsContentForBillingAddressFields:STPBillingAddressFieldsPostalCode]); + XCTAssertTrue([address containsContentForBillingAddressFields:STPBillingAddressFieldsFull]); + // empty string returns false + address.postalCode = @""; + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsNone]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsPostalCode]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsFull]); + address.postalCode = nil; + + // 1+ characters in name will return true for .Name + address.name = @"Jane Doe"; + XCTAssertTrue([address containsContentForBillingAddressFields:STPBillingAddressFieldsName]); + // empty string returns false + address.name = @""; + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsName]); + address.name = nil; + + // Test every other property that contributes to the full address, ensuring it returns True for .Full only + // This is *not* refactoring-safe, but I think it's better than a bunch of duplicated code + for (NSString *propertyName in @[@"line1", @"line2", @"city", @"state", @"country"]) { + for (NSString *testValue in @[@"a", @"0", @"Foo Bar"]) { + [address setValue:testValue forKey:propertyName]; + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsNone]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsPostalCode]); + XCTAssertTrue([address containsContentForBillingAddressFields:STPBillingAddressFieldsFull]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsName]); + [address setValue:nil forKey:propertyName]; + } + + // Make sure that empty string is treated like nil, and returns false for these properties + [address setValue:@"" forKey:propertyName]; + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsNone]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsPostalCode]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsFull]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsName]); + [address setValue:nil forKey:propertyName]; + } + + // ensure it still returns false for everything since it has been cleared + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsNone]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsPostalCode]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsFull]); + XCTAssertFalse([address containsContentForBillingAddressFields:STPBillingAddressFieldsName]); +} + +- (void)testContainsRequiredShippingAddressFields { + STPAddress *address = [STPAddress new]; + XCTAssertTrue([address containsRequiredShippingAddressFields:nil]); + NSSet *allFields = [NSSet setWithArray:@[STPContactField.postalAddress, + STPContactField.emailAddress, + STPContactField.phoneNumber, + STPContactField.name]]; + XCTAssertFalse([address containsRequiredShippingAddressFields:allFields]); + + address.name = @"John Smith"; + XCTAssertTrue(([address containsRequiredShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertFalse(([address containsRequiredShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + + address.email = @"john@example.com"; + XCTAssertTrue(([address containsRequiredShippingAddressFields:[NSSet setWithArray:@[STPContactField.name, STPContactField.emailAddress]]])); + XCTAssertFalse(([address containsRequiredShippingAddressFields:allFields])); + + address.phone = @"5555555555"; + XCTAssertTrue(([address containsRequiredShippingAddressFields:[NSSet setWithArray:@[STPContactField.name, STPContactField.emailAddress, STPContactField.phoneNumber]]])); + address.phone = @"555"; + XCTAssertFalse(([address containsRequiredShippingAddressFields:[NSSet setWithArray:@[STPContactField.name, STPContactField.emailAddress, STPContactField.phoneNumber]]])); + XCTAssertFalse(([address containsRequiredShippingAddressFields:allFields])); + address.country = @"GB"; + XCTAssertTrue(([address containsRequiredShippingAddressFields:[NSSet setWithArray:@[STPContactField.name, STPContactField.emailAddress]]])); + address.phone = @"5555555555"; + XCTAssertTrue(([address containsRequiredShippingAddressFields:[NSSet setWithArray:@[STPContactField.name, STPContactField.emailAddress, STPContactField.phoneNumber]]])); + + address.country = @"US"; + address.phone = @"5555555555"; + address.line1 = @"55 John St"; + address.city = @"New York"; + address.state = @"NY"; + address.postalCode = @"12345"; + XCTAssertTrue([address containsRequiredShippingAddressFields:allFields]); +} + +- (void)testContainsContentForShippingAddressFields { + STPAddress *address = [STPAddress new]; + + // Empty address should return false for everything + XCTAssertFalse(([address containsContentForShippingAddressFields:nil])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress]]])); + + // Name + address.name = @"Smith"; + XCTAssertFalse(([address containsContentForShippingAddressFields:nil])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress]]])); + address.name = @""; + + // Phone + address.phone = @"1"; + XCTAssertFalse(([address containsContentForShippingAddressFields:nil])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress]]])); + address.phone = @""; + + // Email + address.email = @"f"; + XCTAssertFalse(([address containsContentForShippingAddressFields:nil])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber]]])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress]]])); + address.email = @""; + + // Test every property that contributes to the full address + // This is *not* refactoring-safe, but I think it's better than a bunch more duplicated code + for (NSString *propertyName in @[@"line1", @"line2", @"city", @"state", @"postalCode", @"country"]) { + for (NSString *testValue in @[@"a", @"0", @"Foo Bar"]) { + [address setValue:testValue forKey:propertyName]; + XCTAssertFalse(([address containsContentForShippingAddressFields:nil])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress]]])); + [address setValue:@"" forKey:propertyName]; + } + } + + // ensure it still returns false for everything with empty strings + XCTAssertFalse(([address containsContentForShippingAddressFields:nil])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress]]])); + + // Try a hybrid address, and make sure some bitwise combinations work + address.name = @"a"; + address.phone = @"1"; + address.line1 = @"_"; + XCTAssertFalse(([address containsContentForShippingAddressFields:nil])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name]]])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber]]])); + XCTAssertFalse(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.emailAddress]]])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress]]])); + + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.name, STPContactField.emailAddress]]])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.phoneNumber, STPContactField.emailAddress]]])); + XCTAssertTrue(([address containsContentForShippingAddressFields:[NSSet setWithArray:@[STPContactField.postalAddress, + STPContactField.emailAddress, + STPContactField.phoneNumber, + STPContactField.name]]])); + +} + + +- (void)testShippingInfoForCharge { + STPAddress *address = [STPFixtures address]; + PKShippingMethod *method = [[PKShippingMethod alloc] init]; + method.label = @"UPS Ground"; + NSDictionary *info = [STPAddress shippingInfoForChargeWithAddress:address + shippingMethod:method]; + NSDictionary *expected = @{ + @"address": @{ + @"city": address.city, + @"country": address.country, + @"line1": address.line1, + @"line2": address.line2, + @"postal_code": address.postalCode, + @"state": address.state + }, + @"name": address.name, + @"phone": address.phone, + @"carrier": method.label, + }; + XCTAssertEqualObjects(expected, info); +} + +#pragma mark STPFormEncodable Tests + +- (void)testRootObjectName { + XCTAssertNil([STPAddress rootObjectName]); +} + +- (void)testPropertyNamesToFormFieldNamesMapping { + STPAddress *address = [STPAddress new]; + + NSDictionary *mapping = [STPAddress propertyNamesToFormFieldNamesMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([address respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); +} + +#pragma mark NSCopying Tests + +- (void)testCopyWithZone { + STPAddress *address = [STPFixtures address]; + STPAddress *copiedAddress = [address copy]; + + XCTAssertNotEqual(address, copiedAddress, @"should be different objects"); + + // The property names we expect to *not* be equal objects + NSArray *notEqualProperties = @[ + // these include the object's address, so they won't be the same across copies + @"debugDescription", + @"description", + @"hash", + ]; + // use runtime inspection to find the list of properties. If a new property is + // added to the fixture, but not the `copyWithZone:` implementation, this should catch it + for (NSString *property in [STPTestUtils propertyNamesOf:address]) { + if ([notEqualProperties containsObject:property]) { + XCTAssertNotEqualObjects([address valueForKey:property], + [copiedAddress valueForKey:property], + @"%@", property); + } else { + XCTAssertEqualObjects([address valueForKey:property], + [copiedAddress valueForKey:property], + @"%@", property); + } + } +} + +@end diff --git a/Stripe/StripeiOSTests/STPAddressViewModelTest.swift b/Stripe/StripeiOSTests/STPAddressViewModelTest.swift new file mode 100644 index 00000000..429368e2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPAddressViewModelTest.swift @@ -0,0 +1,205 @@ +// +// STPAddressViewModelTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 10/21/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPAddressViewModelTest: XCTestCase { + func testInitWithRequiredBillingFields() { + var sut = STPAddressViewModel(requiredBillingFields: .none, availableCountries: nil) + XCTAssertTrue(sut.addressCells.count == 0) + XCTAssertTrue(sut.isValid) + + sut = STPAddressViewModel(requiredBillingFields: .postalCode, availableCountries: nil) + XCTAssertTrue(sut.addressCells.count == 0) + + sut = STPAddressViewModel(requiredBillingFields: .full, availableCountries: nil) + XCTAssertTrue(sut.addressCells.count == 7) + let types: [STPAddressFieldType] = [ + .name, + .line1, + .line2, + .country, + .zip, + .city, + .state, + ] + for i in 0..(), + availableCountries: nil + ) + XCTAssertTrue(sut.addressCells.count == 0) + + sut = STPAddressViewModel( + requiredShippingFields: Set([.name]), + availableCountries: nil + ) + XCTAssertTrue(sut.addressCells.count == 1) + let cell1 = sut.addressCells[0] + XCTAssertEqual(cell1.type, .name) + + sut = STPAddressViewModel( + requiredShippingFields: Set([.name, .emailAddress]), + availableCountries: nil + ) + XCTAssertTrue(sut.addressCells.count == 2) + var types: [STPAddressFieldType] = [.name, .email] + for i in 0..([ + .postalAddress, .emailAddress, .phoneNumber, + ]), + availableCountries: nil + ) + XCTAssertTrue(sut.addressCells.count == 9) + types = [ + .email, + .name, + .line1, + .line2, + .country, + .zip, + .city, + .state, + .phone, + ] + for i in 0..([ + .postalAddress, + .emailAddress, + .phoneNumber, + ]), + availableCountries: nil + ) + sut.addressCells[0].contents = "foo@example.com" + sut.addressCells[1].contents = "John Smith" + sut.addressCells[2].contents = "55 John St" + sut.addressCells[3].contents = "#3B" + sut.addressCells[4].contents = "US" + sut.addressCells[5].contents = "10002" + sut.addressCells[6].contents = "New York" + sut.addressCells[7].contents = "NY" + sut.addressCells[8].contents = "555-555-5555" + + XCTAssertEqual(sut.address.email, "foo@example.com") + XCTAssertEqual(sut.address.name, "John Smith") + XCTAssertEqual(sut.address.line1, "55 John St") + XCTAssertEqual(sut.address.line2, "#3B") + XCTAssertEqual(sut.address.city, "New York") + XCTAssertEqual(sut.address.state, "NY") + XCTAssertEqual(sut.address.postalCode, "10002") + XCTAssertEqual(sut.address.country, "US") + XCTAssertEqual(sut.address.phone, "555-555-5555") + } + + func testSetAddress() { + let address = STPAddress() + address.email = "foo@example.com" + address.name = "John Smith" + address.line1 = "55 John St" + address.line2 = "#3B" + address.city = "New York" + address.state = "NY" + address.postalCode = "10002" + address.country = "US" + address.phone = "555-555-5555" + + let sut = STPAddressViewModel( + requiredShippingFields: Set([ + .postalAddress, + .emailAddress, + .phoneNumber, + ]), + availableCountries: nil + ) + sut.address = address + XCTAssertEqual(sut.addressCells[0].contents, "foo@example.com") + XCTAssertEqual(sut.addressCells[1].contents, "John Smith") + XCTAssertEqual(sut.addressCells[2].contents, "55 John St") + XCTAssertEqual(sut.addressCells[3].contents, "#3B") + XCTAssertEqual(sut.addressCells[4].contents, "US") + XCTAssertEqual(sut.addressCells[4].textField.text, "United States") + XCTAssertEqual(sut.addressCells[5].contents, "10002") + XCTAssertEqual(sut.addressCells[6].contents, "New York") + XCTAssertEqual(sut.addressCells[7].contents, "NY") + XCTAssertEqual(sut.addressCells[8].contents, "555-555-5555") + } + + func testIsValid_Zip() { + let sut = STPAddressViewModel(requiredBillingFields: .postalCode, availableCountries: nil) + + let address = STPAddress() + + address.country = "US" + sut.address = address + // The AddressViewModel shouldn't request any information when requesting ZIPs. + XCTAssertEqual(sut.addressCells.count, 0) + + address.postalCode = "94016" + sut.address = address + XCTAssertTrue(sut.isValid) + + address.country = "MO" // in Macao, postalCode is optional + address.postalCode = nil + sut.address = address + XCTAssertEqual(sut.addressCells.count, 0) + XCTAssertTrue(sut.isValid, "in Macao, postalCode is optional, valid without one") + } + + func testIsValid_Full() { + let sut = STPAddressViewModel(requiredBillingFields: .full, availableCountries: nil) + XCTAssertFalse(sut.isValid) + sut.addressCells[0].contents = "John Smith" + sut.addressCells[1].contents = "55 John St" + sut.addressCells[2].contents = "#3B" + XCTAssertFalse(sut.isValid) + sut.addressCells[3].contents = "10002" + sut.addressCells[4].contents = "New York" + sut.addressCells[5].contents = "NY" + sut.addressCells[6].contents = "US" + XCTAssertTrue(sut.isValid) + } + + func testIsValid_Name() { + let sut = STPAddressViewModel(requiredBillingFields: .name, availableCountries: nil) + + let address = STPAddress() + + address.name = "" + sut.address = address + XCTAssertEqual(sut.addressCells.count, 1) + XCTAssertFalse(sut.isValid) + + address.name = "Jane Doe" + sut.address = address + XCTAssertEqual(sut.addressCells.count, 1) + XCTAssertTrue(sut.isValid) + } +} diff --git a/Stripe/StripeiOSTests/STPAnalyticsClientPaymentSheetTest.swift b/Stripe/StripeiOSTests/STPAnalyticsClientPaymentSheetTest.swift new file mode 100644 index 00000000..7bf2252b --- /dev/null +++ b/Stripe/StripeiOSTests/STPAnalyticsClientPaymentSheetTest.swift @@ -0,0 +1,268 @@ +// +// STPAnalyticsClientPaymentSheetTest.swift +// StripeiOS Tests +// +// Created by Mel Ludowise on 5/26/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet + +class STPAnalyticsClientPaymentSheetTest: XCTestCase { + private var client: STPAnalyticsClient! + + override func setUp() { + super.setUp() + client = STPAnalyticsClient() + } + + func testPaymentSheetInit() { + let customerConfig = PaymentSheet.CustomerConfiguration(id: "", ephemeralKeySecret: "") + let applePayConfig = PaymentSheet.ApplePayConfiguration( + merchantId: "", + merchantCountryCode: "" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: false, + configuration: makeConfig(applePay: nil, customer: nil) + ).rawValue, + "mc_complete_init_default" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: true, + configuration: makeConfig(applePay: nil, customer: nil) + ).rawValue, + "mc_custom_init_default" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: false, + configuration: makeConfig(applePay: applePayConfig, customer: nil) + ).rawValue, + "mc_complete_init_applepay" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: true, + configuration: makeConfig(applePay: applePayConfig, customer: nil) + ).rawValue, + "mc_custom_init_applepay" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: false, + configuration: makeConfig(applePay: nil, customer: customerConfig) + ).rawValue, + "mc_complete_init_customer" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: true, + configuration: makeConfig(applePay: nil, customer: customerConfig) + ).rawValue, + "mc_custom_init_customer" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: false, + configuration: makeConfig(applePay: applePayConfig, customer: customerConfig) + ).rawValue, + "mc_complete_init_customer_applepay" + ) + XCTAssertEqual( + client.paymentSheetInitEventValue( + isCustom: true, + configuration: makeConfig(applePay: applePayConfig, customer: customerConfig) + ).rawValue, + "mc_custom_init_customer_applepay" + ) + } + + func testPaymentSheetAddsUsage() { + let client = STPAnalyticsClient.sharedClient + _ = PaymentSheet( + paymentIntentClientSecret: "", + configuration: PaymentSheet.Configuration() + ) + XCTAssertTrue(client.productUsage.contains("PaymentSheet")) + + _ = PaymentSheet.FlowController( + intent: .paymentIntent(STPFixtures.paymentIntent()), + savedPaymentMethods: [], + isLinkEnabled: false, + configuration: PaymentSheet.Configuration() + ) + XCTAssertTrue(client.productUsage.contains("PaymentSheet.FlowController")) + } + + func testVariousPaymentSheetEvents() { + let client = STPTestingAnalyticsClient() + let event1 = XCTestExpectation(description: "mc_custom_sheet_newpm_show") + client.registerExpectation(event1) + client.logPaymentSheetShow( + isCustom: true, + paymentMethod: .newPM, + linkEnabled: false, + activeLinkSession: false, + currency: "USD" + ) + + let event2 = XCTestExpectation(description: "mc_complete_sheet_savedpm_show") + client.registerExpectation(event2) + client.logPaymentSheetShow( + isCustom: false, + paymentMethod: .savedPM, + linkEnabled: false, + activeLinkSession: false, + currency: "USD" + ) + + let event3 = XCTestExpectation(description: "mc_complete_payment_savedpm_success") + client.registerExpectation(event3) + client.logPaymentSheetPayment( + isCustom: false, + paymentMethod: .savedPM, + result: .completed, + linkEnabled: false, + activeLinkSession: false, + currency: "USD" + ) + + let event4 = XCTestExpectation(description: "mc_custom_payment_applepay_failure") + client.registerExpectation(event4) + client.logPaymentSheetPayment( + isCustom: true, + paymentMethod: .applePay, + result: .failed(error: PaymentSheetError.unknown(debugDescription: "Error")), + linkEnabled: false, + activeLinkSession: false, + currency: "USD" + ) + + let event5 = XCTestExpectation(description: "mc_custom_paymentoption_applepay_select") + client.registerExpectation(event5) + client.logPaymentSheetPaymentOptionSelect(isCustom: true, paymentMethod: .applePay) + + let event6 = XCTestExpectation(description: "mc_complete_paymentoption_newpm_select") + client.registerExpectation(event6) + client.logPaymentSheetPaymentOptionSelect(isCustom: false, paymentMethod: .newPM) + + wait( + for: [event1, event2, event3, event4, event5, event6], + timeout: STPTestingNetworkRequestTimeout + ) + } + + func testPaymentSheetAnalyticPayload() throws { + // setup + let analytic = PaymentSheetAnalytic( + event: STPAnalyticEvent.mcInitCompleteApplePay, + productUsage: Set([STPPaymentContext.stp_analyticsIdentifier]), + additionalParams: ["testKey": "testVal"] + ) + + let client = STPAnalyticsClient() + client.addAdditionalInfo("test-additional-info") + client.addClass(toProductUsageIfNecessary: STPPaymentContext.self) + + // test + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + let payload = client.payload(from: analytic, apiClient: apiClient) + + // verify + XCTAssertEqual(15, payload.count) + XCTAssertNotNil(payload["device_type"] as? String) + // In xctest, this is the version of Xcode + XCTAssertNotNil(payload["app_version"] as? String) + XCTAssertEqual("none", payload["ocr_type"] as? String) + XCTAssertEqual( + STPAnalyticEvent.mcInitCompleteApplePay.rawValue, + payload["event"] as? String + ) + XCTAssertEqual(STPTestingDefaultPublishableKey, payload["publishable_key"] as? String) + XCTAssertEqual("analytics.stripeios-1.0", payload["analytics_ua"] as? String) + XCTAssertEqual("xctest", payload["app_name"] as? String) + XCTAssertNotNil(payload["os_version"] as? String) + XCTAssertNil(payload["ui_usage_level"]) + XCTAssertTrue(payload["apple_pay_enabled"] as? Bool ?? false) + XCTAssertEqual("legacy", payload["pay_var"] as? String) + XCTAssertEqual(STPAPIClient.STPSDKVersion, payload["bindings_version"] as? String) + XCTAssertEqual("testVal", payload["testKey"] as? String) + XCTAssertEqual("X", payload["install"] as? String) + + let additionalInfo = try XCTUnwrap(payload["additional_info"] as? [String]) + XCTAssertEqual(1, additionalInfo.count) + XCTAssertEqual("test-additional-info", additionalInfo[0]) + + let productUsage = try XCTUnwrap(payload["product_usage"] as? [String]) + XCTAssertEqual(1, productUsage.count) + XCTAssertEqual(STPPaymentContext.stp_analyticsIdentifier, productUsage[0]) + } + + func testLogPaymentSheetPayment_shouldIncludeDuration() throws { + let client = STPTestingAnalyticsClient() + + client.logPaymentSheetShow( + isCustom: false, + paymentMethod: .newPM, + linkEnabled: false, + activeLinkSession: false, + currency: "USD" + ) + + client.logPaymentSheetPayment( + isCustom: false, + paymentMethod: .savedPM, + result: .completed, + linkEnabled: false, + activeLinkSession: false, + currency: "USD" + ) + + let duration = client.lastPayload?["duration"] as? TimeInterval + XCTAssertNotNil(duration) + } +} + +// MARK: - Helpers + +extension STPAnalyticsClientPaymentSheetTest { + fileprivate func makeConfig( + applePay: PaymentSheet.ApplePayConfiguration?, + customer: PaymentSheet.CustomerConfiguration? + ) -> PaymentSheet.Configuration { + var config = PaymentSheet.Configuration() + config.applePay = applePay + config.customer = customer + return config + } +} + +// MARK: - Mock types + +private class STPTestingAnalyticsClient: STPAnalyticsClient { + var expectedEvents: [String: XCTestExpectation] = [:] + + var lastPayload: [String: Any]? + + func registerExpectation(_ expectation: XCTestExpectation) { + expectedEvents[expectation.description] = expectation + } + + override func logPayload(_ payload: [String: Any]) { + if let event = payload["event"] as? String, + let expectedEvent = expectedEvents[event] + { + expectedEvent.fulfill() + } + + lastPayload = payload + } +} diff --git a/Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift b/Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift new file mode 100644 index 00000000..4267cbc3 --- /dev/null +++ b/Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift @@ -0,0 +1,236 @@ +// +// STPAnalyticsClientPaymentsTest.swift +// StripeiOS Tests +// +// Created by Mel Ludowise on 5/26/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import StripeApplePay +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPAnalyticsClientPaymentsTest: XCTestCase { + private var client: STPAnalyticsClient! + + override func setUp() { + super.setUp() + client = STPAnalyticsClient() + } + + func testAdditionalInfo() { + XCTAssertEqual(client.additionalInfo(), []) + + // Add some info + client.addAdditionalInfo("hello") + client.addAdditionalInfo("i'm additional info") + client.addAdditionalInfo("how are you?") + + XCTAssertEqual(client.additionalInfo(), ["hello", "how are you?", "i'm additional info"]) + + // Clear it + client.clearAdditionalInfo() + XCTAssertEqual(client.additionalInfo(), []) + } + + func testPayloadFromAnalytic() throws { + client.addAdditionalInfo("test_additional_info") + + let mockAnalytic = MockAnalytic() + let payload = client.payload(from: mockAnalytic) + + XCTAssertEqual(payload.count, 13) + + // Verify event name is included + XCTAssertEqual(payload["event"] as? String, mockAnalytic.event.rawValue) + + // Verify additionalInfo is included + XCTAssertEqual(payload["additional_info"] as? [String], ["test_additional_info"]) + + // Verify all the analytic params are in the payload + XCTAssertEqual(payload["test_param1"] as? Int, 1) + XCTAssertEqual(payload["test_param2"] as? String, "two") + + // Verify productUsage is included + XCTAssertNotNil(payload["product_usage"]) + + // Verify install method is Xcode + XCTAssertEqual(payload["install"] as? String, "X") + } + + func testPayloadFromErrorAnalytic() throws { + client.addAdditionalInfo("test_additional_info") + + let mockAnalytic = MockErrorAnalytic() + let payload = client.payload(from: mockAnalytic) + + // Verify event name is included + XCTAssertEqual(payload["event"] as? String, mockAnalytic.event.rawValue) + + // Verify additionalInfo is included + XCTAssertEqual(payload["additional_info"] as? [String], ["test_additional_info"]) + + // Verify all the analytic params are in the payload + XCTAssertEqual(payload["test_param1"] as? Int, 1) + XCTAssertEqual(payload["test_param2"] as? String, "two") + + // Verify productUsage is included + XCTAssertNotNil(payload["product_usage"]) + + // Verify error_dictionary is included + let errorDict = try XCTUnwrap(payload["error_dictionary"] as? [String: Any]) + XCTAssertTrue( + NSDictionary(dictionary: errorDict).isEqual( + to: mockAnalytic.error.serializeForLogging() + ) + ) + } + + func testTokenTypeFromParameters() { + let card = STPFixtures.cardParams() + let cardDict = buildTokenParams(card) + XCTAssertEqual(STPAnalyticsClient.tokenType(fromParameters: cardDict), "card") + + let account = STPFixtures.accountParams() + let accountDict = buildTokenParams(account) + XCTAssertEqual(STPAnalyticsClient.tokenType(fromParameters: accountDict), "account") + + let bank = STPFixtures.bankAccountParams() + let bankDict = buildTokenParams(bank) + XCTAssertEqual(STPAnalyticsClient.tokenType(fromParameters: bankDict), "bank_account") + + let applePay = STPFixtures.applePayPayment() + let applePayDict = addTelemetry(applePay.stp_tokenParameters(apiClient: .shared)) + XCTAssertEqual(STPAnalyticsClient.tokenType(fromParameters: applePayDict), "apple_pay") + } + + // MARK: - Tests various classes report usage + + func testCardTextFieldAddsUsage() { + _ = STPPaymentCardTextField() + XCTAssertTrue( + STPAnalyticsClient.sharedClient.productUsage.contains("STPPaymentCardTextField") + ) + } + + func testPaymentContextAddsUsage() { + let keyManager = STPEphemeralKeyManager( + keyProvider: MockKeyProvider(), + apiVersion: "1", + performsEagerFetching: false + ) + let apiClient = STPAPIClient() + let customerContext = STPCustomerContext.init(keyManager: keyManager, apiClient: apiClient) + _ = STPPaymentContext(customerContext: customerContext) + XCTAssertTrue(STPAnalyticsClient.sharedClient.productUsage.contains("STPCustomerContext")) + } + + func testApplePayContextAddsUsage() { + _ = STPApplePayContext(paymentRequest: STPFixtures.applePayRequest(), delegate: nil) + XCTAssertTrue(STPAnalyticsClient.sharedClient.productUsage.contains("STPApplePayContext")) + } + + func testCustomerContextAddsUsage() { + let keyManager = STPEphemeralKeyManager( + keyProvider: MockKeyProvider(), + apiVersion: "1", + performsEagerFetching: false + ) + let apiClient = STPAPIClient() + _ = STPCustomerContext(keyManager: keyManager, apiClient: apiClient) + XCTAssertTrue(STPAnalyticsClient.sharedClient.productUsage.contains("STPCustomerContext")) + } + + func testAddCardVCAddsUsage() { + _ = STPAddCardViewController() + XCTAssertTrue( + STPAnalyticsClient.sharedClient.productUsage.contains("STPAddCardViewController") + ) + } + + func testBankSelectionVCAddsUsage() { + _ = STPBankSelectionViewController() + XCTAssertTrue( + STPAnalyticsClient.sharedClient.productUsage.contains("STPBankSelectionViewController") + ) + } + + func testShippingVCAddsUsage() { + let config = STPFixtures.paymentConfiguration() + config.requiredShippingAddressFields = [STPContactField.postalAddress] + _ = STPShippingAddressViewController( + configuration: config, + theme: .defaultTheme, + currency: nil, + shippingAddress: nil, + selectedShippingMethod: nil, + prefilledInformation: nil + ) + XCTAssertTrue( + STPAnalyticsClient.sharedClient.productUsage.contains( + "STPShippingAddressViewController" + ) + ) + } +} + +// MARK: - Helpers + +extension STPAnalyticsClientPaymentsTest { + fileprivate func buildTokenParams(_ object: T) -> [String: Any] + { + return addTelemetry(STPFormEncoder.dictionary(forObject: object)) + } + + fileprivate func addTelemetry(_ params: [String: Any]) -> [String: Any] { + // STPAPIClient adds these before determining the token type, + // so do the same in the test + return STPTelemetryClient.shared.paramsByAddingTelemetryFields(toParams: params) + } +} + +// MARK: - Mock types + +private struct MockAnalytic: Analytic { + let event = STPAnalyticEvent.sourceCreation + + let params: [String: Any] = [ + "test_param1": 1, + "test_param2": "two", + ] +} + +private struct MockErrorAnalytic: ErrorAnalytic { + let event = STPAnalyticEvent.sourceCreation + + let params: [String: Any] = [ + "test_param1": 1, + "test_param2": "two", + ] + + let error: Error = NSError(domain: "domain", code: 100, userInfo: nil) +} + +private struct MockAnalyticsClass1: STPAnalyticsProtocol { + static let stp_analyticsIdentifier = "MockAnalyticsClass1" +} + +private struct MockAnalyticsClass2: STPAnalyticsProtocol { + static let stp_analyticsIdentifier = "MockAnalyticsClass2" +} + +private class MockKeyProvider: NSObject, STPCustomerEphemeralKeyProvider { + func createCustomerKey( + withAPIVersion apiVersion: String, + completion: @escaping STPJSONResponseCompletionBlock + ) { + guard apiVersion == "1" else { return } + + completion(nil, NSError.stp_genericConnectionError()) + } +} diff --git a/Stripe/StripeiOSTests/STPApplePayContextFunctionalTest.m b/Stripe/StripeiOSTests/STPApplePayContextFunctionalTest.m new file mode 100644 index 00000000..1fd80409 --- /dev/null +++ b/Stripe/StripeiOSTests/STPApplePayContextFunctionalTest.m @@ -0,0 +1,385 @@ +// +// STPApplePayContextFunctionalTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/5/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +@import StripeApplePay; +#import "STPTestingAPIClient.h" + +#import "STPFixtures.h" +#import "StripeiOS_Tests-Swift.h" +@import OHHTTPStubs; + +@interface STPTestApplePayContextDelegate: NSObject +@property (nonatomic) void (^didCompleteDelegateMethod)(STPPaymentStatus status, NSError *error); +@property (nonatomic) void (^didCreatePaymentMethodDelegateMethod)(STPPaymentMethod *paymentMethod, PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion); + +@end + +@implementation STPTestApplePayContextDelegate + +- (void)applePayContext:(__unused STPApplePayContext *)context didCompleteWithStatus:(STPPaymentStatus)status error:(nullable NSError *)error { + self.didCompleteDelegateMethod(status, error); +} + +- (void)applePayContext:(__unused STPApplePayContext *)context didCreatePaymentMethod:(STPPaymentMethod *)paymentMethod paymentInformation:(PKPayment *)paymentInformation completion:(nonnull STPIntentClientSecretCompletionBlock)completion { + self.didCreatePaymentMethodDelegateMethod(paymentMethod, paymentInformation, completion); +} + +@end + + +@interface STPApplePayContext(Testing) +@property (nonatomic, nullable) PKPaymentAuthorizationController *authorizationController; +@end + +API_AVAILABLE(ios(13.0)) +@interface STPApplePayContextFunctionalTest : XCTestCase +@property (nonatomic) STPApplePayContextFunctionalTestAPIClient *apiClient; +@property (nonatomic) STPTestApplePayContextDelegate *delegate; +@property (nonatomic) STPApplePayContext *context; + +@end + +@interface STPTestPKPaymentAuthorizationController : PKPaymentAuthorizationController +@end + +@implementation STPTestPKPaymentAuthorizationController + +// Stub dismissViewControllerAnimated: to just call its completion block +- (void)dismissWithCompletion:(void (^)(void))completion { + completion(); +} + +@end + +@implementation STPApplePayContextFunctionalTest + +- (void)setUp { + self.delegate = [STPTestApplePayContextDelegate new]; + if (@available(iOS 13.0, *)) { + STPApplePayContextFunctionalTestAPIClient *apiClient = [[STPApplePayContextFunctionalTestAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + [apiClient setupStubs]; + apiClient.applePayContext = self.context; + self.apiClient = apiClient; + } else { + XCTSkip("Unsupported iOS version"); + } + + self.context = [[STPApplePayContext alloc] initWithPaymentRequest:[STPFixtures applePayRequest] delegate:self.delegate]; + self.apiClient.applePayContext = self.context; + self.context.apiClient = self.apiClient; + self.context.authorizationController = [[STPTestPKPaymentAuthorizationController alloc] init]; +} + +- (void)tearDown { + [HTTPStubs removeAllStubs]; +} + +- (void)testCompletesManualConfirmationPaymentIntent { + __block NSString *clientSecret; + // A manual confirmation PI confirmed server-side... + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(STPPaymentMethod *paymentMethod, PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + XCTAssertNotNil(paymentInformation); + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{@"confirmation_method": @"manual", @"payment_method": paymentMethod.stripeId, @"confirm": @YES} completion:^(NSString * _Nullable _clientSecret, NSError * __unused error) { + XCTAssertNotNil(_clientSecret); + clientSecret = _clientSecret; + completion(clientSecret, nil); + }]; + }; + + // ...used with ApplePayContext + STPApplePayContext *context = [[STPApplePayContext alloc] initWithPaymentRequest:[STPFixtures applePayRequest] delegate:self.delegate]; + context.apiClient = self.apiClient; + [self _startApplePayForContextWithExpectedStatus:PKPaymentAuthorizationStatusSuccess]; + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + XCTAssertEqual(status, STPPaymentStatusSuccess); + XCTAssertNil(error); + + // ...and results in a successful PI + [self.apiClient retrievePaymentIntentWithClientSecret:clientSecret completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError *paymentIntentRetrieveError) { + XCTAssertNil(paymentIntentRetrieveError); + XCTAssert(paymentIntent.status == STPPaymentIntentStatusSucceeded); + [didCallCompletion fulfill]; + }]; + }; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCompletesAutomaticConfirmationPaymentIntent { + __block NSString *clientSecret; + // An automatic confirmation PI with the PaymentMethod attached... + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable _clientSecret, NSError * __unused error) { + XCTAssertNotNil(_clientSecret); + clientSecret = _clientSecret; + completion(clientSecret, nil); + }]; + }; + + // ...used with ApplePayContext + [self _startApplePayForContextWithExpectedStatus:PKPaymentAuthorizationStatusSuccess]; + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + XCTAssertEqual(status, STPPaymentStatusSuccess); + XCTAssertNil(error); + + // ...and results in a successful PI + [self.apiClient retrievePaymentIntentWithClientSecret:clientSecret completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError *paymentIntentRetrieveError) { + XCTAssertNil(paymentIntentRetrieveError); + XCTAssert(paymentIntent.status == STPPaymentIntentStatusSucceeded); + XCTAssertEqualObjects(paymentIntent.shipping.name, @"Jane Doe"); + XCTAssertEqualObjects(paymentIntent.shipping.address.line1, @"510 Townsend St"); + [didCallCompletion fulfill]; + }]; + }; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCompletesAutomaticConfirmationPaymentIntentManualCapture { + __block NSString *clientSecret; + // An automatic confirmation PI with the PaymentMethod attached... + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{@"capture_method": @"manual"} completion:^(NSString * _Nullable _clientSecret, NSError * __unused error) { + XCTAssertNotNil(_clientSecret); + clientSecret = _clientSecret; + completion(clientSecret, nil); + }]; + }; + + // ...used with ApplePayContext + [self _startApplePayForContextWithExpectedStatus:PKPaymentAuthorizationStatusSuccess]; + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + XCTAssertEqual(status, STPPaymentStatusSuccess); + XCTAssertNil(error); + + // ...and results in a successful PI + [self.apiClient retrievePaymentIntentWithClientSecret:clientSecret completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * paymentIntentRetrieveError) { + XCTAssertNil(paymentIntentRetrieveError); + XCTAssert(paymentIntent.status == STPPaymentIntentStatusRequiresCapture); + [didCallCompletion fulfill]; + }]; + }; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCompletesSetupIntent { + __block NSString *clientSecret; + // An automatic confirmation SI... + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + [[STPTestingAPIClient sharedClient] createSetupIntentWithParams:nil completion:^(NSString * _Nullable _clientSecret, NSError * __unused error) { + XCTAssertNotNil(_clientSecret); + clientSecret = _clientSecret; + completion(clientSecret, nil); + }]; + }; + + // ...used with ApplePayContext + [self _startApplePayForContextWithExpectedStatus:PKPaymentAuthorizationStatusSuccess]; + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + XCTAssertEqual(status, STPPaymentStatusSuccess); + XCTAssertNil(error); + + // ...and results in a successful PI + [self.apiClient retrieveSetupIntentWithClientSecret:clientSecret completion:^(STPSetupIntent * _Nullable setupIntent, NSError *setupIntentRetrieveError) { + XCTAssertNil(setupIntentRetrieveError); + XCTAssert(setupIntent.status == STPSetupIntentStatusSucceeded); + [didCallCompletion fulfill]; + }]; + }; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - Error tests +- (void)testBadPaymentIntentClientSecretErrors { + __block NSString *clientSecret; + // An invalid PaymentIntent client secret... + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + dispatch_async(dispatch_get_main_queue(), ^{ + clientSecret = @"pi_bad_secret_1234"; + completion(clientSecret, nil); + }); + }; + + // ...used with ApplePayContext + [self _startApplePayForContextWithExpectedStatus:PKPaymentAuthorizationStatusFailure]; + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + // ...and results in an error + XCTAssertEqual(status, STPPaymentStatusError); + XCTAssertNotNil(error); + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + [didCallCompletion fulfill]; + }; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testBadSetupIntentClientSecretErrors { + __block NSString *clientSecret; + // An invalid SetupIntent client secret... + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + dispatch_async(dispatch_get_main_queue(), ^{ + clientSecret = @"seti_bad_secret_1234"; + completion(clientSecret, nil); + }); + }; + + // ...used with ApplePayContext + [self _startApplePayForContextWithExpectedStatus:PKPaymentAuthorizationStatusFailure]; + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + // ...and results in an error + XCTAssertEqual(status, STPPaymentStatusError); + XCTAssertNotNil(error); + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + [didCallCompletion fulfill]; + }; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - Cancel tests +- (void)testCancelBeforeIntentConfirmsCancels { + // Cancelling Apple Pay *before* the context attempts to confirms the PI/SI... + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + [self.context paymentAuthorizationControllerDidFinish:self.context.authorizationController]; // Simulate cancel before passing PI to the context + // ...should never retrieve the PI (b/c it is cancelled before) + completion(@"A 'client secret' that triggers an exception if fetched", nil); + }; + + [self.context paymentAuthorizationController:self.context.authorizationController + didAuthorizePayment:[STPFixtures simulatorApplePayPayment] + handler:^(PKPaymentAuthorizationResult * __unused _Nonnull result) {}]; // Simulate user tapping 'Pay' button in Apple Pay + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + // ...and results in a 'user cancel' status + XCTAssertEqual(status, STPPaymentStatusUserCancellation); + XCTAssertNil(error); + [didCallCompletion fulfill]; + }; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCancelAfterPaymentIntentConfirmsStillSucceeds { + // Cancelling Apple Pay *after* the context attempts to confirm the PI... + self.apiClient.shouldSimulateCancelAfterConfirmBegins = true; + + __block NSString *clientSecret; + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable _clientSecret, NSError * __unused error) { + XCTAssertNotNil(_clientSecret); + clientSecret = _clientSecret; + completion(clientSecret, nil); + }]; + }; + + [self.context paymentAuthorizationController:self.context.authorizationController + didAuthorizePayment:[STPFixtures simulatorApplePayPayment] + handler:^(PKPaymentAuthorizationResult * __unused _Nonnull result) {}]; // Simulate user tapping 'Pay' button in Apple Pay + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + XCTAssertEqual(status, STPPaymentStatusSuccess); + XCTAssertNil(error); + + // ...and results in a successful PI + [self.apiClient retrievePaymentIntentWithClientSecret:clientSecret completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * paymentIntentRetrieveError) { + XCTAssertNil(paymentIntentRetrieveError); + XCTAssert(paymentIntent.status == STPPaymentIntentStatusSucceeded); + [didCallCompletion fulfill]; + }]; + }; + + [self waitForExpectationsWithTimeout:20.0 handler:nil]; // give this a longer timeout, it tends to take a while +} + +- (void)testCancelAfterSetupIntentConfirmsStillSucceeds { + // Cancelling Apple Pay *after* the context attempts to confirm the SI... + self.apiClient.shouldSimulateCancelAfterConfirmBegins = true; + + __block NSString *clientSecret; + STPTestApplePayContextDelegate *delegate = self.delegate; + delegate.didCreatePaymentMethodDelegateMethod = ^(__unused STPPaymentMethod *paymentMethod, __unused PKPayment *paymentInformation, STPIntentClientSecretCompletionBlock completion) { + [[STPTestingAPIClient sharedClient] createSetupIntentWithParams:nil completion:^(NSString * _Nullable _clientSecret, NSError * __unused error) { + XCTAssertNotNil(_clientSecret); + clientSecret = _clientSecret; + completion(clientSecret, nil); + }]; + }; + + [self.context paymentAuthorizationController:self.context.authorizationController + didAuthorizePayment:[STPFixtures simulatorApplePayPayment] + handler:^(PKPaymentAuthorizationResult * __unused _Nonnull result) {}]; // Simulate user tapping 'Pay' button in Apple Pay + + // ...calls applePayContext:didCompleteWithStatus:error: + XCTestExpectation *didCallCompletion = [self expectationWithDescription:@"applePayContext:didCompleteWithStatus: called"]; + delegate.didCompleteDelegateMethod = ^(STPPaymentStatus status, NSError *error) { + XCTAssertEqual(status, STPPaymentStatusSuccess); + XCTAssertNil(error); + + // ...and results in a successful SI + [self.apiClient retrieveSetupIntentWithClientSecret:clientSecret completion:^(STPSetupIntent * _Nullable setupIntent, NSError * setupIntentRetrieveError) { + XCTAssertNil(setupIntentRetrieveError); + XCTAssert(setupIntent.status == STPSetupIntentStatusSucceeded); + [didCallCompletion fulfill]; + }]; + }; + + [self waitForExpectationsWithTimeout:20.0 handler:nil]; // give this a longer timeout, it tends to take a while +} + + +#pragma mark - Helper + +/// Simulates user tapping 'Pay' button in Apple Pay sheet +- (void)_startApplePayForContextWithExpectedStatus:(PKPaymentAuthorizationStatus)expectedStatus { + // When the user taps 'Pay', PKPaymentAuthorizationController calls `didAuthorizePayment:completion:` + // After you call its completion block, it calls `paymentAuthorizationControllerDidFinish:` + XCTestExpectation *didCallAuthorizePaymentCompletion = [self expectationWithDescription:@"ApplePayContext called completion block of paymentAuthorizationController:didAuthorizePayment:completion:"]; + [self.context paymentAuthorizationController:self.context.authorizationController didAuthorizePayment:[STPFixtures simulatorApplePayPayment] handler:^(PKPaymentAuthorizationResult * _Nonnull result) { + XCTAssertEqual(expectedStatus, result.status); + dispatch_async(dispatch_get_main_queue(), ^{ + [self.context paymentAuthorizationControllerDidFinish:self.context.authorizationController]; + [didCallAuthorizePaymentCompletion fulfill]; + }); + }]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPApplePayContextFunctionalTestExtras.swift b/Stripe/StripeiOSTests/STPApplePayContextFunctionalTestExtras.swift new file mode 100644 index 00000000..15a7394b --- /dev/null +++ b/Stripe/StripeiOSTests/STPApplePayContextFunctionalTestExtras.swift @@ -0,0 +1,44 @@ +// +// STPApplePayContextFunctionalTestExtras.swift +// StripeiOS Tests +// +// Created by David Estes on 3/23/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import OHHTTPStubs +import OHHTTPStubsSwift + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeApplePay +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPApplePayContextFunctionalTestAPIClient: STPAPIClient { + @objc var applePayContext: STPApplePayContext? + @objc var shouldSimulateCancelAfterConfirmBegins: Bool = false + + @objc func setupStubs() { + stub { urlRequest in + // Hook SetupIntent or PaymentIntent confirmation + if let urlString = urlRequest.url?.absoluteString, + urlString.contains("_intents/"), + urlString.hasSuffix("/confirm") + { + if self.shouldSimulateCancelAfterConfirmBegins { + self.applePayContext!.paymentAuthorizationControllerDidFinish( + self.applePayContext!.authorizationController! + ) + } + } + // Let everything pass through to the underlying API + return false + } response: { _ in + // This doesn't matter, we're not sending responses for anything. + return HTTPStubsResponse() + } + } +} diff --git a/Stripe/StripeiOSTests/STPApplePayContextTest.swift b/Stripe/StripeiOSTests/STPApplePayContextTest.swift new file mode 100644 index 00000000..21bcbc49 --- /dev/null +++ b/Stripe/StripeiOSTests/STPApplePayContextTest.swift @@ -0,0 +1,163 @@ +// +// STPApplePayContextTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 2/20/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripePayments + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeApplePay +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPApplePayTestDelegateiOS11: NSObject, STPApplePayContextDelegate { + func applePayContext( + _ context: STPApplePayContext, + didSelectShippingContact contact: PKContact, + handler completion: @escaping (PKPaymentRequestShippingContactUpdate) -> Void + ) { + completion(PKPaymentRequestShippingContactUpdate()) + } + + func applePayContext( + _ context: STPApplePayContext, + didSelect shippingMethod: PKShippingMethod, + handler completion: @escaping (PKPaymentRequestShippingMethodUpdate) -> Void + ) { + completion(PKPaymentRequestShippingMethodUpdate()) + } + + func applePayContext( + _ context: STPApplePayContext, + didCompleteWith status: STPPaymentStatus, + error: Error? + ) { + } + + func applePayContext( + _ context: STPApplePayContext, + didCreatePaymentMethod paymentMethod: STPPaymentMethod, + paymentInformation: PKPayment, + completion: STPIntentClientSecretCompletionBlock + ) { + } +} + +// MARK: - STPApplePayTestDelegateiOS11 +class STPApplePayContextTest: XCTestCase { + func testiOS11ApplePayDelegateMethodsForwarded() { + // With a user that only implements iOS 11 delegate methods... + let delegate = STPApplePayTestDelegateiOS11() + let request = StripeAPI.paymentRequest( + withMerchantIdentifier: "foo", + country: "US", + currency: "USD" + ) + request.paymentSummaryItems = [ + PKPaymentSummaryItem(label: "bar", amount: NSDecimalNumber(string: "1.00")) + ] + let context = STPApplePayContext(paymentRequest: request, delegate: delegate)! + + // ...the context should respondToSelector appropriately... + XCTAssertTrue( + context.responds( + to: #selector( + PKPaymentAuthorizationControllerDelegate.paymentAuthorizationController( + _: + didSelectShippingContact: + handler: + )) + ) + ) + XCTAssertFalse( + context.responds( + to: #selector( + PKPaymentAuthorizationControllerDelegate.paymentAuthorizationController( + _: + didSelectShippingContact: + completion: + )) + ) + ) + + // ...and forward the PassKit delegate method to its delegate + let vc: PKPaymentAuthorizationController = PKPaymentAuthorizationController() + let contact = PKContact() + let shippingContactExpectation = expectation( + description: "didSelectShippingContact forwarded" + ) + context.paymentAuthorizationController( + vc, + didSelectShippingContact: contact, + handler: { _ in + shippingContactExpectation.fulfill() + } + ) + + let method = PKShippingMethod() + let shippingMethodExpectation = expectation( + description: "didSelectShippingMethod forwarded" + ) + context.paymentAuthorizationController( + vc, + didSelectShippingMethod: method, + handler: { _ in + shippingMethodExpectation.fulfill() + } + ) + waitForExpectations(timeout: 2, handler: nil) + } + + func testConvertsShippingDetails() { + let delegate = STPApplePayTestDelegateiOS11() + let request = StripeAPI.paymentRequest( + withMerchantIdentifier: "foo", + country: "US", + currency: "USD" + ) + request.paymentSummaryItems = [ + PKPaymentSummaryItem(label: "bar", amount: NSDecimalNumber(string: "1.00")) + ] + let context = STPApplePayContext(paymentRequest: request, delegate: delegate) + + let payment = STPFixtures.simulatorApplePayPayment() + let shipping = PKContact() + shipping.name = PersonNameComponentsFormatter().personNameComponents(from: "Jane Doe") + shipping.phoneNumber = CNPhoneNumber(stringValue: "555-555-5555") + let address = CNMutablePostalAddress() + address.street = "510 Townsend St" + address.city = "San Francisco" + address.state = "CA" + address.isoCountryCode = "US" + address.postalCode = "94105" + shipping.postalAddress = address + payment.perform(#selector(setter: PKPaymentRequest.shippingContact), with: shipping) + + let shippingParams = context!._shippingDetails(from: payment) + XCTAssertNotNil(shippingParams) + XCTAssertEqual(shippingParams?.name, "Jane Doe") + XCTAssertNil(shippingParams?.carrier) + XCTAssertEqual(shippingParams?.phone, "555-555-5555") + XCTAssertNil(shippingParams?.trackingNumber) + + XCTAssertEqual(shippingParams?.address.line1, "510 Townsend St") + XCTAssertNil(shippingParams?.address.line2) + XCTAssertEqual(shippingParams?.address.city, "San Francisco") + XCTAssertEqual(shippingParams?.address.state, "CA") + XCTAssertEqual(shippingParams?.address.country, "US") + XCTAssertEqual(shippingParams?.address.postalCode, "94105") + } + + // Tests stp_tokenParameters in StripeApplePay, not StripePayments + func testStpTokenParameters() { + let applePay = STPFixtures.applePayPayment() + let applePayDict = applePay.stp_tokenParameters(apiClient: .shared) + XCTAssertNotNil(applePayDict["pk_token"]) + XCTAssertEqual((applePayDict["card"] as! NSDictionary)["name"] as! String, "Test Testerson") + XCTAssertEqual(applePayDict["pk_token_instrument_name"] as! String, "Master Charge") + } +} diff --git a/Stripe/StripeiOSTests/STPApplePayFunctionalTest.swift b/Stripe/StripeiOSTests/STPApplePayFunctionalTest.swift new file mode 100644 index 00000000..050a20af --- /dev/null +++ b/Stripe/StripeiOSTests/STPApplePayFunctionalTest.swift @@ -0,0 +1,110 @@ +// +// STPApplePayFunctionalTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 12/21/14. +// Copyright (c) 2014 Stripe, Inc. All rights reserved. +// + +import PassKit +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeApplePay +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPApplePayFunctionalTest: STPNetworkStubbingTestCase { + override func setUp() { + // self.recordingMode = YES; + super.setUp() + } + + // TODO: regenerate these fixtures with a fresh/real PKPayment + func testCreateTokenWithPaymentClassic() { + let payment = STPFixtures.applePayPayment() + let client = STPAPIClient(publishableKey: "pk_test_vOo1umqsYxSrP5UXfOeL3ecm") + + let expectation = self.expectation(description: "Apple pay token creation") + client.createToken( + with: payment + ) { token, error in + expectation.fulfill() + XCTAssertNil(token, "token should be nil") + guard let error = error else { + XCTFail("error should not be nil") + return + } + + // Since we can't actually generate a new cryptogram in a CI environment, we should just post a blob of expired token data and + // make sure we get the "too long since tokenization" error. This at least asserts that our blob has been correctly formatted and + // can be decrypted by the backend. + XCTAssert( + ((error as NSError).userInfo[STPError.errorMessageKey] as? NSString)?.range( + of: "too long" + ).location != NSNotFound, + "Error is unrelated to 24-hour expiry: \(error)" + ) + } + waitForExpectations(timeout: 5.0, handler: nil) + } + + func testCreateTokenWithPayment() { + let payment = STPFixtures.applePayPayment() + let client = STPAPIClient(publishableKey: "pk_test_vOo1umqsYxSrP5UXfOeL3ecm") + + let expectation = self.expectation(description: "Apple pay token creation") + StripeAPI.Token.create( + apiClient: client, + payment: payment + ) { result in + expectation.fulfill() + XCTAssertNil(try? result.get(), "token should be nil") + guard case .failure(let error) = result else { + XCTFail("error should not be nil") + return + } + + // Since we can't actually generate a new cryptogram in a CI environment, we should just post a blob of expired token data and + // make sure we get the "too long since tokenization" error. This at least asserts that our blob has been correctly formatted and + // can be decrypted by the backend. + XCTAssert( + ((error as NSError).userInfo[STPError.errorMessageKey] as? NSString)?.range( + of: "too long" + ).location != NSNotFound, + "Error is unrelated to 24-hour expiry: \(error)" + ) + } + waitForExpectations(timeout: 5.0, handler: nil) + } + + func testCreateSourceWithPayment() { + let payment = STPFixtures.applePayPayment() + let client = STPAPIClient(publishableKey: "pk_test_vOo1umqsYxSrP5UXfOeL3ecm") + + let expectation = self.expectation(description: "Apple pay source creation") + client.createSource( + with: payment + ) { source, error in + expectation.fulfill() + XCTAssertNil(source, "token should be nil") + guard let error = error else { + XCTFail("error should not be nil") + return + } + + // Since we can't actually generate a new cryptogram in a CI environment, we should just post a blob of expired token data and + // make sure we get the "too long since tokenization" error. This at least asserts that our blob has been correctly formatted and + // can be decrypted by the backend. + XCTAssert( + ((error as NSError).userInfo[STPError.errorMessageKey] as? NSString)?.range( + of: "too long" + ).location != NSNotFound, + "Error is unrelated to 24-hour expiry: \(error)" + ) + } + waitForExpectations(timeout: 5.0, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPApplePayPaymentOptionTest.m b/Stripe/StripeiOSTests/STPApplePayPaymentOptionTest.m new file mode 100644 index 00000000..07b3cb51 --- /dev/null +++ b/Stripe/StripeiOSTests/STPApplePayPaymentOptionTest.m @@ -0,0 +1,50 @@ +// +// STPApplePayPaymentOptionTest.m +// Stripe +// +// Created by Joey Dong on 7/28/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import + + + +@interface STPApplePayPaymentOptionTest : XCTestCase + +@end + +@implementation STPApplePayPaymentOptionTest + +#pragma mark - STPPaymentOption Tests + +- (void)testImage { + STPApplePayPaymentOption *applePay = [[STPApplePayPaymentOption alloc] init]; + XCTAssert([applePay image]); +} + +- (void)testTemplateImage { + STPApplePayPaymentOption *applePay = [[STPApplePayPaymentOption alloc] init]; + XCTAssert([applePay templateImage]); +} + +- (void)testLabel { + STPApplePayPaymentOption *applePay = [[STPApplePayPaymentOption alloc] init]; + XCTAssertEqualObjects([applePay label], @"Apple Pay"); +} + +#pragma mark - Equality Tests + +- (void)testApplePayEquals { + STPApplePayPaymentOption *applePay1 = [[STPApplePayPaymentOption alloc] init]; + STPApplePayPaymentOption *applePay2 = [[STPApplePayPaymentOption alloc] init]; + + XCTAssertNotEqual(applePay1, applePay2); + + XCTAssertEqualObjects(applePay1, applePay1); + XCTAssertEqualObjects(applePay1, applePay2); + + XCTAssertEqual(applePay1.hash, applePay1.hash); + XCTAssertEqual(applePay1.hash, applePay2.hash); +} +@end diff --git a/Stripe/StripeiOSTests/STPApplePayTest.m b/Stripe/StripeiOSTests/STPApplePayTest.m new file mode 100644 index 00000000..19447a53 --- /dev/null +++ b/Stripe/StripeiOSTests/STPApplePayTest.m @@ -0,0 +1,62 @@ +// +// STPApplePayTest.m +// Stripe +// +// Created by Ben Guo on 6/1/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCore; + +@interface STPApplePayTest : XCTestCase + +@end + +@implementation STPApplePayTest + +- (void)testPaymentRequestWithMerchantIdentifierCountryCurrency { + PKPaymentRequest *paymentRequest = [StripeAPI paymentRequestWithMerchantIdentifier:@"foo" country:@"GB" currency:@"GBP"]; + XCTAssertEqualObjects(paymentRequest.merchantIdentifier, @"foo"); + if (@available(iOS 12, *)) { + NSSet *expectedNetworks = [NSSet setWithArray:@[PKPaymentNetworkAmex, PKPaymentNetworkMasterCard, PKPaymentNetworkVisa, PKPaymentNetworkDiscover, PKPaymentNetworkMaestro]]; + XCTAssertEqualObjects([NSSet setWithArray:paymentRequest.supportedNetworks], expectedNetworks); + } else { + NSSet *expectedNetworks = [NSSet setWithArray:@[PKPaymentNetworkAmex, PKPaymentNetworkMasterCard, PKPaymentNetworkVisa, PKPaymentNetworkDiscover]]; + XCTAssertEqualObjects([NSSet setWithArray:paymentRequest.supportedNetworks], expectedNetworks); + } + XCTAssertEqual(paymentRequest.merchantCapabilities, PKMerchantCapability3DS); + XCTAssertEqualObjects(paymentRequest.countryCode, @"GB"); + XCTAssertEqualObjects(paymentRequest.currencyCode, @"GBP"); + XCTAssertEqualObjects(paymentRequest.requiredBillingContactFields, [NSSet setWithArray:@[PKContactFieldPostalAddress]]); +} + +- (void)testCanSubmitPaymentRequestReturnsYES { + PKPaymentRequest *request = [[PKPaymentRequest alloc] init]; + request.merchantIdentifier = @"foo"; + request.paymentSummaryItems = @[[PKPaymentSummaryItem summaryItemWithLabel:@"bar" amount:[NSDecimalNumber decimalNumberWithString:@"1.00"]]]; + + XCTAssertTrue([StripeAPI canSubmitPaymentRequest:request]); +} + +- (void)testCanSubmitPaymentRequestIfTotalIsZero { + PKPaymentRequest *request = [[PKPaymentRequest alloc] init]; + request.merchantIdentifier = @"foo"; + request.paymentSummaryItems = @[[PKPaymentSummaryItem summaryItemWithLabel:@"bar" amount:[NSDecimalNumber decimalNumberWithString:@"0.00"]]]; + + // "In versions of iOS prior to version 12.0 and watchOS prior to version 5.0, the amount of the grand total must be greater than zero." + if (@available(iOS 12, *)) { + XCTAssertTrue([StripeAPI canSubmitPaymentRequest:request]); + } else { + XCTAssertFalse([StripeAPI canSubmitPaymentRequest:request]); + } +} + +- (void)testCanSubmitPaymentRequestReturnsNOIfMerchantIdentifierIsNil { + PKPaymentRequest *request = [[PKPaymentRequest alloc] init]; + request.paymentSummaryItems = @[[PKPaymentSummaryItem summaryItemWithLabel:@"bar" amount:[NSDecimalNumber decimalNumberWithString:@"1.00"]]]; + + XCTAssertFalse([StripeAPI canSubmitPaymentRequest:request]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPApplePayTest.swift b/Stripe/StripeiOSTests/STPApplePayTest.swift new file mode 100644 index 00000000..670d22bf --- /dev/null +++ b/Stripe/StripeiOSTests/STPApplePayTest.swift @@ -0,0 +1,34 @@ +// +// STPApplePayTest.swift +// StripeiOS Tests +// +// Created by David Estes on 9/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPApplePaySwiftTest: XCTestCase { + func testAdditionalPaymentNetwork() { + XCTAssertFalse(StripeAPI.supportedPKPaymentNetworks().contains(.JCB)) + StripeAPI.additionalEnabledApplePayNetworks = [.JCB] + XCTAssertTrue(StripeAPI.supportedPKPaymentNetworks().contains(.JCB)) + StripeAPI.additionalEnabledApplePayNetworks = [] + } + + // Tests stp_tokenParameters in StripePayments, not StripeApplePay + func testStpTokenParameters() { + let applePay = STPFixtures.applePayPayment() + let applePayDict = applePay.stp_tokenParameters(apiClient: .shared) + XCTAssertNotNil(applePayDict["pk_token"]) + XCTAssertEqual((applePayDict["card"] as! NSDictionary)["name"] as! String, "Test Testerson") + XCTAssertEqual(applePayDict["pk_token_instrument_name"] as! String, "Master Charge") + } +} diff --git a/Stripe/StripeiOSTests/STPBECSDebitAccountNumberValidatorTests.swift b/Stripe/StripeiOSTests/STPBECSDebitAccountNumberValidatorTests.swift new file mode 100644 index 00000000..4e06739c --- /dev/null +++ b/Stripe/StripeiOSTests/STPBECSDebitAccountNumberValidatorTests.swift @@ -0,0 +1,240 @@ +// +// STPBECSDebitAccountNumberValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPBECSDebitAccountNumberValidatorTests: XCTestCase { + func testValidationStateForText() { + let tests = [ + // empty input + [ + "input": "", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.empty.rawValue), + ], + [ + "input": "", + "bsb": "0", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.empty.rawValue), + ], + [ + "input": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.empty.rawValue), + ], + [ + "input": "", + "bsb": "00", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.empty.rawValue), + ], + // incomplete input + [ + "input": "1", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "1", + "bsb": "0", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "1", + "bsb": "00", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "1", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "12345", + "bsb": "06", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + // incomplete input (editing) + [ + "input": "1", + "bsb": "", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "1", + "bsb": "0", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "1", + "bsb": "00", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "1", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "12345", + "bsb": "06", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + [ + "input": "12345678", + "bsb": "", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.incomplete.rawValue), + ], + // complete + [ + "input": "12345", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.complete.rawValue), + ], + [ + "input": "123456", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.complete.rawValue), + ], + [ + "input": "1234567", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.complete.rawValue), + ], + [ + "input": "12345678", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.complete.rawValue), + ], + [ + "input": "123456789", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.complete.rawValue), + ], + // complete (editing) + [ + "input": "123456789", + "bsb": "", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.complete.rawValue), + ], + // invalid + [ + "input": "12345678910", + "bsb": "", + "editing": NSNumber(value: false), + "expected": NSNumber(value: STPTextValidationState.invalid.rawValue), + ], + // invalid (editing) + [ + "input": "12345678910", + "bsb": "", + "editing": NSNumber(value: true), + "expected": NSNumber(value: STPTextValidationState.invalid.rawValue), + ], + ] + + for test in tests { + let input = (test["input"] as? String)! + let bsb = test["bsb"] as? String + let editing = (test["editing"] as? NSNumber)!.boolValue + let expected = STPTextValidationState( + rawValue: (test["expected"] as! NSNumber).intValue + )! + + XCTAssertEqual( + STPBECSDebitAccountNumberValidator.validationState( + forText: input, + withBSBNumber: bsb, + completeOnMaxLengthOnly: editing + ), + expected + ) + } + } + + func testformattedSanitizedTextFromString() { + let tests = [ + [ + "input": "", + "bsb": "00", + "expected": "", + ], + [ + "input": "1", + "bsb": "00", + "expected": "1", + ], + [ + "input": "--111111--", + "bsb": "00", + "expected": "111111", + ], + [ + "input": "12345678910", + "bsb": "00", + "expected": "123456789", + ], + [ + "input": "", + "bsb": "06", + "expected": "", + ], + [ + "input": "1", + "bsb": "06", + "expected": "1", + ], + [ + "input": "--111111--", + "bsb": "06", + "expected": "111111", + ], + [ + "input": "12345678910", + "bsb": "06", + "expected": "123456789", + ], + ] + + for test in tests { + let input = (test["input"])! + let bsb = test["bsb"] + let expected = test["expected"] + XCTAssertEqual( + STPBECSDebitAccountNumberValidator.formattedSanitizedText( + from: input, + withBSBNumber: bsb + ), + expected + ) + } + } +} diff --git a/Stripe/StripeiOSTests/STPBSBNumberValidatorTests.swift b/Stripe/StripeiOSTests/STPBSBNumberValidatorTests.swift new file mode 100644 index 00000000..efe7b37a --- /dev/null +++ b/Stripe/StripeiOSTests/STPBSBNumberValidatorTests.swift @@ -0,0 +1,92 @@ +// +// STPBSBNumberValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPBSBNumberValidatorTests: XCTestCase { + func testValidationStateForText() { + // Don't use the special test key behavior of treating 00 as a valid BSB. + STPAPIClient.shared.publishableKey = "pk_live_not_a_real_key" + let tests: [(String, STPTextValidationState)] = [ + ("", .empty), + ("1", .incomplete), + ("11", .incomplete), + ("00", .invalid), + ("111111", .complete), + ("111-111", .complete), + ("--111-111--", .complete), + ("1234567", .invalid), + ] + + for test in tests { + XCTAssertEqual(STPBSBNumberValidator.validationState(forText: test.0), test.1) + } + } + + func testformattedSanitizedTextFromString() { + let tests = [ + ["", ""], + ["1", "1"], + ["11", "11"], + ["111", "111-"], + ["111111", "111-111"], + ["--111111--", "111-111"], + ["1234567", "123-456"], + ] + + for test in tests { + XCTAssertEqual(STPBSBNumberValidator.formattedSanitizedText(from: test[0]), test[1]) + } + } + + func testIdentityForText() { + let tests = [ + ["", NSNull()], + ["9", NSNull()], + ["94", NSNull()], + ["941", "Delphi Bank (division of Bendigo and Adelaide Bank)"], + ["942", "Bank of Sydney"], + ["942942", "Bank of Sydney"], + ["40", "Commonwealth Bank of Australia"], + ["942-942", "Bank of Sydney"], + ["942942111", "Bank of Sydney"], + ] + + for test in tests { + if test[1] as! NSObject == NSNull() { + XCTAssertNil(STPBSBNumberValidator.identity(forText: test[0] as! String)) + } else { + XCTAssertEqual( + STPBSBNumberValidator.identity(forText: test[0] as! String), + test[1] as? String + ) + } + } + } + + func testIconForText() { + // Don't use the special test key behavior of treating 00 as a valid BSB. + STPAPIClient.shared.publishableKey = "pk_live_not_a_real_key" + let defaultIcon = STPBSBNumberValidator.icon(forText: nil) + XCTAssertNotNil(defaultIcon, "Nil default icon") + + XCTAssertEqual(defaultIcon, STPBSBNumberValidator.icon(forText: "00")) + + let bankIcon = STPBSBNumberValidator.icon(forText: "11") + XCTAssertNotNil(bankIcon, "Nil icon for bank `11`") + XCTAssertFalse((defaultIcon == bankIcon), "Icon for `11` is same as default") + + XCTAssertEqual(bankIcon, STPBSBNumberValidator.icon(forText: "111-111")) + } +} diff --git a/Stripe/StripeiOSTests/STPBankAccountFunctionalTest.m b/Stripe/StripeiOSTests/STPBankAccountFunctionalTest.m new file mode 100644 index 00000000..f9593fc9 --- /dev/null +++ b/Stripe/StripeiOSTests/STPBankAccountFunctionalTest.m @@ -0,0 +1,69 @@ +// +// STPBankAccountFunctionalTest.m +// Stripe +// +// Created by Charles Scalesse on 10/2/14. +// +// + +@import XCTest; +@import StripeCoreTestUtils; + + +#import "STPTestingAPIClient.h" + + +@interface STPBankAccountFunctionalTest : XCTestCase +@end + +@implementation STPBankAccountFunctionalTest + +- (void)testCreateAndRetreiveBankAccountToken { + STPBankAccountParams *bankAccount = [[STPBankAccountParams alloc] init]; + bankAccount.accountNumber = @"000123456789"; + bankAccount.routingNumber = @"110000000"; + bankAccount.country = @"US"; + bankAccount.accountHolderName = @"Jimmy bob"; + bankAccount.accountHolderType = STPBankAccountHolderTypeCompany; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Bank account creation"]; + [client createTokenWithBankAccount:bankAccount + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + XCTAssertNil(error, @"error should be nil %@", error.localizedDescription); + XCTAssertNotNil(token, @"token should not be nil"); + + XCTAssertNotNil(token.tokenId); + XCTAssertEqual(token.type, STPTokenTypeBankAccount); + XCTAssertNotNil(token.bankAccount.stripeID); + XCTAssertEqualObjects(@"STRIPE TEST BANK", token.bankAccount.bankName); + XCTAssertEqualObjects(@"6789", token.bankAccount.last4); + XCTAssertEqualObjects(@"Jimmy bob", token.bankAccount.accountHolderName); + XCTAssertEqual(token.bankAccount.accountHolderType, STPBankAccountHolderTypeCompany); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testInvalidKey { + STPBankAccountParams *bankAccount = [[STPBankAccountParams alloc] init]; + bankAccount.accountNumber = @"000123456789"; + bankAccount.routingNumber = @"110000000"; + bankAccount.country = @"US"; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"not_a_valid_key_asdf"]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Bad bank account creation"]; + + [client createTokenWithBankAccount:bankAccount + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + XCTAssertNil(token, @"token should be nil"); + XCTAssertNotNil(error, @"error should not be nil"); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPBankAccountParamsTest.m b/Stripe/StripeiOSTests/STPBankAccountParamsTest.m new file mode 100644 index 00000000..c7b7e6e3 --- /dev/null +++ b/Stripe/StripeiOSTests/STPBankAccountParamsTest.m @@ -0,0 +1,119 @@ +// +// STPBankAccountParamsTest.m +// Stripe +// +// Created by Joey Dong on 6/19/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + + +@interface STPBankAccountParams () + +- (NSString *)accountHolderTypeString; + +@end + +@interface STPBankAccountParamsTest : XCTestCase + +@end + +@implementation STPBankAccountParamsTest + +#pragma mark - + +- (void)testLast4ReturnsAccountNumberLast4 { + STPBankAccountParams *bankAccountParams = [[STPBankAccountParams alloc] init]; + bankAccountParams.accountNumber = @"000123456789"; + XCTAssertEqualObjects(bankAccountParams.last4, @"6789"); +} + +- (void)testLast4ReturnsNilWhenNoAccountNumberSet { + STPBankAccountParams *bankAccountParams = [[STPBankAccountParams alloc] init]; + XCTAssertNil(bankAccountParams.last4); +} + +- (void)testLast4ReturnsNilWhenAccountNumberIsLessThanLength4 { + STPBankAccountParams *bankAccountParams = [[STPBankAccountParams alloc] init]; + bankAccountParams.accountNumber = @"123"; + XCTAssertNil(bankAccountParams.last4); +} + +#pragma mark - STPBankAccountHolderType Tests + +- (void)testAccountHolderTypeFromString { + XCTAssertEqual([STPBankAccountParams accountHolderTypeFromString:@"individual"], STPBankAccountHolderTypeIndividual); + XCTAssertEqual([STPBankAccountParams accountHolderTypeFromString:@"INDIVIDUAL"], STPBankAccountHolderTypeIndividual); + + XCTAssertEqual([STPBankAccountParams accountHolderTypeFromString:@"company"], STPBankAccountHolderTypeCompany); + XCTAssertEqual([STPBankAccountParams accountHolderTypeFromString:@"COMPANY"], STPBankAccountHolderTypeCompany); + + XCTAssertEqual([STPBankAccountParams accountHolderTypeFromString:@"garbage"], STPBankAccountHolderTypeIndividual); + XCTAssertEqual([STPBankAccountParams accountHolderTypeFromString:@"GARBAGE"], STPBankAccountHolderTypeIndividual); +} + +- (void)testStringFromAccountHolderType { + NSArray *values = @[ + @(STPBankAccountHolderTypeIndividual), + @(STPBankAccountHolderTypeCompany), + ]; + + for (NSNumber *accountHolderTypeNumber in values) { + STPBankAccountHolderType accountHolderType = (STPBankAccountHolderType)[accountHolderTypeNumber integerValue]; + NSString *string = [STPBankAccountParams stringFromAccountHolderType:accountHolderType]; + + switch (accountHolderType) { + case STPBankAccountHolderTypeIndividual: + XCTAssertEqualObjects(string, @"individual"); + break; + case STPBankAccountHolderTypeCompany: + XCTAssertEqualObjects(string, @"company"); + break; + } + } +} + +#pragma mark - Description Tests + +- (void)testDescription { + STPBankAccountParams *bankAccountParams = [[STPBankAccountParams alloc] init]; + XCTAssert(bankAccountParams.description); +} + +#pragma mark - STPFormEncodable Tests + +- (void)testRootObjectName { + XCTAssertEqualObjects([STPBankAccountParams rootObjectName], @"bank_account"); +} + +- (void)testPropertyNamesToFormFieldNamesMapping { + STPBankAccountParams *bankAccountParams = [[STPBankAccountParams alloc] init]; + + NSDictionary *mapping = [STPBankAccountParams propertyNamesToFormFieldNamesMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([bankAccountParams respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); +} + +- (void)testAccountHolderTypeString { + STPBankAccountParams *bankAccountParams = [[STPBankAccountParams alloc] init]; + + bankAccountParams.accountHolderType = STPBankAccountHolderTypeIndividual; + XCTAssertEqualObjects([bankAccountParams accountHolderTypeString], @"individual"); + + bankAccountParams.accountHolderType = STPBankAccountHolderTypeCompany; + XCTAssertEqualObjects([bankAccountParams accountHolderTypeString], @"company"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPBankAccountTest.m b/Stripe/StripeiOSTests/STPBankAccountTest.m new file mode 100644 index 00000000..341f1dc2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPBankAccountTest.m @@ -0,0 +1,148 @@ +// +// STPBankAccountTest.m +// Stripe +// +// Created by Charles Scalesse on 10/2/14. +// +// + +@import XCTest; + + + +#import "STPTestUtils.h" + +@interface STPBankAccount () + ++ (STPBankAccountStatus)statusFromString:(NSString *)string; ++ (NSString *)stringFromStatus:(STPBankAccountStatus)status; + +- (void)setLast4:(NSString *)last4; + +@end + +@interface STPBankAccountTest : XCTestCase + +@end + +@implementation STPBankAccountTest + +#pragma mark - STPBankAccountStatus Tests + +- (void)testStatusFromString { + XCTAssertEqual([STPBankAccount statusFromString:@"new"], STPBankAccountStatusNew); + XCTAssertEqual([STPBankAccount statusFromString:@"NEW"], STPBankAccountStatusNew); + + XCTAssertEqual([STPBankAccount statusFromString:@"validated"], STPBankAccountStatusValidated); + XCTAssertEqual([STPBankAccount statusFromString:@"VALIDATED"], STPBankAccountStatusValidated); + + XCTAssertEqual([STPBankAccount statusFromString:@"verified"], STPBankAccountStatusVerified); + XCTAssertEqual([STPBankAccount statusFromString:@"VERIFIED"], STPBankAccountStatusVerified); + + XCTAssertEqual([STPBankAccount statusFromString:@"verification_failed"], STPBankAccountStatusVerificationFailed); + XCTAssertEqual([STPBankAccount statusFromString:@"VERIFICATION_FAILED"], STPBankAccountStatusVerificationFailed); + + XCTAssertEqual([STPBankAccount statusFromString:@"errored"], STPBankAccountStatusErrored); + XCTAssertEqual([STPBankAccount statusFromString:@"ERRORED"], STPBankAccountStatusErrored); + + XCTAssertEqual([STPBankAccount statusFromString:@"garbage"], STPBankAccountStatusNew); + XCTAssertEqual([STPBankAccount statusFromString:@"GARBAGE"], STPBankAccountStatusNew); +} + +- (void)testStringFromStatus { + NSArray *values = @[ + @(STPBankAccountStatusNew), + @(STPBankAccountStatusValidated), + @(STPBankAccountStatusVerified), + @(STPBankAccountStatusVerificationFailed), + @(STPBankAccountStatusErrored) + ]; + + for (NSNumber *statusNumber in values) { + STPBankAccountStatus status = (STPBankAccountStatus)[statusNumber integerValue]; + NSString *string = [STPBankAccount stringFromStatus:status]; + + switch (status) { + case STPBankAccountStatusNew: + XCTAssertEqualObjects(string, @"new"); + break; + case STPBankAccountStatusValidated: + XCTAssertEqualObjects(string, @"validated"); + break; + case STPBankAccountStatusVerified: + XCTAssertEqualObjects(string, @"verified"); + break; + case STPBankAccountStatusVerificationFailed: + XCTAssertEqualObjects(string, @"verification_failed"); + break; + case STPBankAccountStatusErrored: + XCTAssertEqualObjects(string, @"errored"); + break; + } + } +} + +#pragma mark - Equality Tests + +- (void)testBankAccountEquals { + STPBankAccount *bankAccount1 = [STPBankAccount decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"BankAccount"]]; + STPBankAccount *bankAccount2 = [STPBankAccount decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"BankAccount"]]; + + XCTAssertNotEqual(bankAccount1, bankAccount2); + + XCTAssertEqualObjects(bankAccount1, bankAccount1); + XCTAssertEqualObjects(bankAccount1, bankAccount2); + + XCTAssertEqual(bankAccount1.hash, bankAccount1.hash); + XCTAssertEqual(bankAccount1.hash, bankAccount2.hash); +} + +#pragma mark - Description Tests + +- (void)testDescription { + STPBankAccount *bankAccount = [STPBankAccount decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"BankAccount"]]; + XCTAssert(bankAccount.description); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[ + @"id", + @"last4", + @"bank_name", + @"country", + @"currency", + @"status", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"BankAccount"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPBankAccount decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPBankAccount decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"BankAccount"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:@"BankAccount"]; + STPBankAccount *bankAccount = [STPBankAccount decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(bankAccount.stripeID, @"ba_1AZmya2eZvKYlo2CQzt7Fwnz"); + XCTAssertEqualObjects(bankAccount.accountHolderName, @"Jane Austen"); + XCTAssertEqual(bankAccount.accountHolderType, STPBankAccountHolderTypeIndividual); + XCTAssertEqualObjects(bankAccount.bankName, @"STRIPE TEST BANK"); + XCTAssertEqualObjects(bankAccount.country, @"US"); + XCTAssertEqualObjects(bankAccount.currency, @"usd"); + XCTAssertEqualObjects(bankAccount.fingerprint, @"1JWtPxqbdX5Gamtc"); + XCTAssertEqualObjects(bankAccount.last4, @"6789"); + XCTAssertEqualObjects(bankAccount.routingNumber, @"110000000"); + XCTAssertEqual(bankAccount.status, STPBankAccountStatusNew); + + XCTAssertNotEqual(bankAccount.allResponseFields, response); + XCTAssertEqualObjects(bankAccount.allResponseFields, response); +} + +@end diff --git a/Stripe/StripeiOSTests/STPBinRangeTest.swift b/Stripe/StripeiOSTests/STPBinRangeTest.swift new file mode 100644 index 00000000..8bd42f92 --- /dev/null +++ b/Stripe/StripeiOSTests/STPBinRangeTest.swift @@ -0,0 +1,190 @@ +// +// STPBinRangeTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 5/24/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPBinRangeTest: XCTestCase { + func testAllRanges() { + for binRange in STPBINController.shared.allRanges() { + XCTAssertEqual(binRange.accountRangeLow.count, binRange.accountRangeHigh.count) + } + } + + func testMatchesNumber() { + var binRange = STPBINRange( + panLength: 0, + brand: .unknown, + accountRangeLow: "134", + accountRangeHigh: "167", + country: nil + ) + + XCTAssertFalse(binRange.matchesNumber("0")) + XCTAssertTrue(binRange.matchesNumber("1")) + XCTAssertFalse(binRange.matchesNumber("2")) + + XCTAssertFalse(binRange.matchesNumber("00")) + XCTAssertTrue(binRange.matchesNumber("13")) + XCTAssertTrue(binRange.matchesNumber("14")) + XCTAssertTrue(binRange.matchesNumber("16")) + XCTAssertFalse(binRange.matchesNumber("20")) + + XCTAssertFalse(binRange.matchesNumber("133")) + XCTAssertTrue(binRange.matchesNumber("134")) + XCTAssertTrue(binRange.matchesNumber("135")) + XCTAssertTrue(binRange.matchesNumber("167")) + XCTAssertFalse(binRange.matchesNumber("168")) + + XCTAssertFalse(binRange.matchesNumber("1244")) + XCTAssertTrue(binRange.matchesNumber("1340")) + XCTAssertTrue(binRange.matchesNumber("1344")) + XCTAssertTrue(binRange.matchesNumber("1444")) + XCTAssertTrue(binRange.matchesNumber("1670")) + XCTAssertTrue(binRange.matchesNumber("1679")) + XCTAssertFalse(binRange.matchesNumber("1680")) + + binRange = STPBINRange( + panLength: 0, + brand: .unknown, + accountRangeLow: "004", + accountRangeHigh: "017", + country: nil + ) + + XCTAssertTrue(binRange.matchesNumber("0")) + XCTAssertFalse(binRange.matchesNumber("1")) + + XCTAssertTrue(binRange.matchesNumber("00")) + XCTAssertTrue(binRange.matchesNumber("01")) + XCTAssertFalse(binRange.matchesNumber("10")) + XCTAssertFalse(binRange.matchesNumber("20")) + + XCTAssertFalse(binRange.matchesNumber("000")) + XCTAssertFalse(binRange.matchesNumber("002")) + XCTAssertTrue(binRange.matchesNumber("004")) + XCTAssertTrue(binRange.matchesNumber("009")) + XCTAssertTrue(binRange.matchesNumber("014")) + XCTAssertTrue(binRange.matchesNumber("017")) + XCTAssertFalse(binRange.matchesNumber("019")) + XCTAssertFalse(binRange.matchesNumber("020")) + XCTAssertFalse(binRange.matchesNumber("100")) + + XCTAssertFalse(binRange.matchesNumber("0000")) + XCTAssertFalse(binRange.matchesNumber("0021")) + XCTAssertTrue(binRange.matchesNumber("0044")) + XCTAssertTrue(binRange.matchesNumber("0098")) + XCTAssertTrue(binRange.matchesNumber("0143")) + XCTAssertTrue(binRange.matchesNumber("0173")) + XCTAssertFalse(binRange.matchesNumber("0195")) + XCTAssertFalse(binRange.matchesNumber("0202")) + XCTAssertFalse(binRange.matchesNumber("1004")) + + binRange = STPBINRange( + panLength: 0, + brand: .unknown, + accountRangeLow: "", + accountRangeHigh: "", + country: nil + ) + XCTAssertTrue(binRange.matchesNumber("")) + XCTAssertTrue(binRange.matchesNumber("1")) + } + + func testBinRangesForNumber() { + var binRanges: [STPBINRange]? + + binRanges = STPBINController.shared.binRanges(forNumber: "4136000000008") + XCTAssertEqual(binRanges?.count, 3) + + binRanges = STPBINController.shared.binRanges(forNumber: "4242424242424242") + XCTAssertEqual(binRanges?.count, 2) + + binRanges = STPBINController.shared.binRanges(forNumber: "5555555555554444") + XCTAssertEqual(binRanges?.count, 2) + + binRanges = STPBINController.shared.binRanges(forNumber: "") + XCTAssertEqual(binRanges?.count, STPBINController.shared.allRanges().count) + + binRanges = STPBINController.shared.binRanges(forNumber: "123") + XCTAssertEqual(binRanges?.count, 1) + } + + func testBinRangesForBrand() { + let allBrands: [STPCardBrand] = [ + .visa, + .amex, + .mastercard, + .discover, + .JCB, + .dinersClub, + .unionPay, + .unknown, + ] + for brand in allBrands { + let binRanges = STPBINController.shared.binRanges(for: brand) + for binRange in binRanges { + XCTAssertEqual(binRange.brand, brand) + } + } + } + + func testMostSpecificBinRangeForNumber() { + var binRange: STPBINRange? + + binRange = STPBINController.shared.mostSpecificBINRange(forNumber: "") + XCTAssertNotEqual(binRange?.brand, .unknown) + + binRange = STPBINController.shared.mostSpecificBINRange(forNumber: "4242424242422") + XCTAssertEqual(binRange?.brand, .visa) + XCTAssertEqual(binRange?.panLength, 16) + + binRange = STPBINController.shared.mostSpecificBINRange(forNumber: "4136000000008") + XCTAssertEqual(binRange?.brand, .visa) + XCTAssertEqual(binRange?.panLength, 13) + + binRange = STPBINController.shared.mostSpecificBINRange(forNumber: "4242424242424242") + XCTAssertEqual(binRange?.brand, .visa) + XCTAssertEqual(binRange?.panLength, 16) + } + + func testMostSpecificBinRangePrefersKnownBrand() { + // 624478 is a real world case that returns ranges for UnionPay and NYCE, the latter being handled as unknown. + let mockedRanges = [ + STPBINRange( + panLength: 16, + brand: .unionPay, + accountRangeLow: "6244780000000000", + accountRangeHigh: "6244789999999999", + country: "HK" + ), + STPBINRange( + panLength: 16, + brand: .unknown, + accountRangeLow: "6244780000000000", + accountRangeHigh: "6244789999999999", + country: "CN" + ), + ] + + STPBINController.shared.sRetrievedRanges["624478"] = mockedRanges + STPBINController.shared.sAllRanges += mockedRanges + + let binRange = STPBINController.shared.mostSpecificBINRange(forNumber: "624478") + XCTAssertEqual(binRange.accountRangeLow, "6244780000000000") + XCTAssertEqual(binRange.accountRangeHigh, "6244789999999999") + XCTAssertEqual(binRange.brand, .unionPay) + + // Cleanup added values to avoid issues caused by singleton state. + STPBINController.shared.sRetrievedRanges["624478"] = nil + STPBINController.shared.sAllRanges = STPBINController.STPBINRangeInitialRanges + } +} diff --git a/Stripe/StripeiOSTests/STPBlocks.h b/Stripe/StripeiOSTests/STPBlocks.h new file mode 100644 index 00000000..d9be6f72 --- /dev/null +++ b/Stripe/StripeiOSTests/STPBlocks.h @@ -0,0 +1,255 @@ +// +// STPBlocks.h +// Stripe +// +// Created by Jack Flintermann on 3/23/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import +#import + +@class STP3DS2AuthenticateResponse; +@class STPToken; +@class STPFile; +@class STPSource; +@class STPCustomer; +@protocol STPSourceProtocol; +@class STPPaymentIntent; +@class STPSetupIntent; +@class STPPaymentMethod; +@class STPIssuingCardPin; +@class STPFPXBankStatusResponse; + +/** + These values control the labels used in the shipping info collection form. + */ +typedef NS_ENUM(NSUInteger, STPShippingType) { + /** + Shipping the purchase to the provided address using a third-party + shipping company. + */ + STPShippingTypeShipping, + /** + Delivering the purchase by the seller. + */ + STPShippingTypeDelivery, +}; + +/** + An enum representing the status of a shipping address validation. + */ +typedef NS_ENUM(NSUInteger, STPShippingStatus) { + /** + The shipping address is valid. + */ + STPShippingStatusValid, + /** + The shipping address is invalid. + */ + STPShippingStatusInvalid, +}; + +/** + An enum representing the status of a payment requested from the user. + */ +typedef NS_ENUM(NSUInteger, STPPaymentStatus) { + /** + The payment succeeded. + */ + STPPaymentStatusSuccess, + /** + The payment failed due to an unforeseen error, such as the user's Internet connection being offline. + */ + STPPaymentStatusError, + /** + The user cancelled the payment (for example, by hitting "cancel" in the Apple Pay dialog). + */ + STPPaymentStatusUserCancellation, +}; + +/** + An empty block, called with no arguments, returning nothing. + */ +typedef void (^STPVoidBlock)(void); + +/** + A block that may optionally be called with an error. + + @param error The error that occurred, if any. + */ +typedef void (^STPErrorBlock)(NSError * __nullable error); + +/** + A block that contains a boolean success param and may optionally be called with an error. + + @param success Whether the task succeeded. + @param error The error that occurred, if any. + */ +typedef void (^STPBooleanSuccessBlock)(BOOL success, NSError * __nullable error); + +/** + A callback to be run with a JSON response. + + @param jsonResponse The JSON response, or nil if an error occured. + @param error The error that occurred, if any. + */ +typedef void (^STPJSONResponseCompletionBlock)(NSDictionary * __nullable jsonResponse, NSError * __nullable error); + +/** + A callback to be run with a token response from the Stripe API. + + @param token The Stripe token from the response. Will be nil if an error occurs. @see STPToken + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPTokenCompletionBlock)(STPToken * __nullable token, NSError * __nullable error); + +/** + A callback to be run with a source response from the Stripe API. + + @param source The Stripe source from the response. Will be nil if an error occurs. @see STPSource + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPSourceCompletionBlock)(STPSource * __nullable source, NSError * __nullable error); + +/** + A callback to be run with a source or card response from the Stripe API. + + @param source The Stripe source from the response. Will be nil if an error occurs. @see STPSourceProtocol + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPSourceProtocolCompletionBlock)(id __nullable source, NSError * __nullable error); + +/** + A callback to be run with a PaymentIntent response from the Stripe API. + + @param paymentIntent The Stripe PaymentIntent from the response. Will be nil if an error occurs. @see STPPaymentIntent + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPPaymentIntentCompletionBlock)(STPPaymentIntent * __nullable paymentIntent, NSError * __nullable error); + +/** + A callback to be run with a PaymentIntent response from the Stripe API. + + @param setupIntent The Stripe SetupIntent from the response. Will be nil if an error occurs. @see STPSetupIntent + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPSetupIntentCompletionBlock)(STPSetupIntent * __nullable setupIntent, NSError * __nullable error); + +/** + A callback to be run with a PaymentMethod response from the Stripe API. + + @param paymentMethod The Stripe PaymentMethod from the response. Will be nil if an error occurs. @see STPPaymentMethod + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPPaymentMethodCompletionBlock)(STPPaymentMethod * __nullable paymentMethod, NSError * __nullable error); + +/** + A callback to be run with an array of PaymentMethods response from the Stripe API. + + @param paymentMethods An array of PaymentMethod from the response. Will be nil if an error occurs. @see STPPaymentMethod + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPPaymentMethodsCompletionBlock)(NSArray *__nullable paymentMethods, NSError * __nullable error); + +/** + A callback to be run with a validation result and shipping methods for a + shipping address. + + @param status An enum representing whether the shipping address is valid. + @param shippingValidationError If the shipping address is invalid, an error describing the issue with the address. If no error is given and the address is invalid, the default error message will be used. + @param shippingMethods The shipping methods available for the address. + @param selectedShippingMethod The default selected shipping method for the address. + */ +typedef void (^STPShippingMethodsCompletionBlock)(STPShippingStatus status, NSError * __nullable shippingValidationError, NSArray* __nullable shippingMethods, PKShippingMethod * __nullable selectedShippingMethod); + +/** + A callback to be run with a file response from the Stripe API. + + @param file The Stripe file from the response. Will be nil if an error occurs. @see STPFile + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPFileCompletionBlock)(STPFile * __nullable file, NSError * __nullable error); + +/** + A callback to be run with a customer response from the Stripe API. + + @param customer The Stripe customer from the response, or nil if an error occurred. @see STPCustomer + @param error The error returned from the response, or nil if none occurs. + */ +typedef void (^STPCustomerCompletionBlock)(STPCustomer * __nullable customer, NSError * __nullable error); + +/** + An enum representing the success and error states of PIN management + */ +typedef NS_ENUM(NSUInteger, STPPinStatus) { + /** + The verification object was already redeemed + */ + STPPinSuccess, + /** + The verification object was already redeemed + */ + STPPinErrorVerificationAlreadyRedeemed, + /** + The one-time code was incorrect + */ + STPPinErrorVerificationCodeIncorrect, + /** + The verification object was expired + */ + STPPinErrorVerificationExpired, + /** + The verification object has been attempted too many times + */ + STPPinErrorVerificationTooManyAttempts, + /** + An error occured while retrieving the ephemeral key + */ + STPPinEphemeralKeyError, + /** + An unknown error occured + */ + STPPinUnknownError, +}; + +/** + A callback to be run with a card PIN response from the Stripe API. + + @param cardPin The Stripe card PIN from the response. Will be nil if an error occurs. @see STPIssuingCardPin + @param status The status to help you sort between different error state, or STPPinSuccess when succesful. @see STPPinStatus for possible values. + @param error The error returned from the response, or nil if none occurs. @see StripeError.h for possible values. + */ +typedef void (^STPPinCompletionBlock)(STPIssuingCardPin * __nullable cardPin, STPPinStatus status, NSError * __nullable error); + +/** + A callback to be run with a 3DS2 authenticate response from the Stripe API. + + @param authenticateResponse The Stripe AuthenticateResponse. Will be nil if an error occurs. @see STP3DS2AuthenticateResponse + @param error The error returned from the response, or nil if none occurs. + */ +typedef void (^STP3DS2AuthenticateCompletionBlock)(STP3DS2AuthenticateResponse * _Nullable authenticateResponse, NSError * _Nullable error); + +/** + A callback to be run with a response from the Stripe API containing information about the online status of FPX banks. + + @param bankStatusResponse The response from Stripe containing the status of the various banks. Will be nil if an error occurs. @see STPFPXBankStatusResponse + @param error The error returned from the response, or nil if none occurs. + */ +typedef void (^STPFPXBankStatusCompletionBlock)(STPFPXBankStatusResponse * _Nullable bankStatusResponse, NSError * _Nullable error); + +/** + A block called with a payment status and an optional error. + + @param error The error that occurred, if any. + */ +typedef void (^STPPaymentStatusBlock)(STPPaymentStatus status, NSError * __nullable error); + +/** + A block to be run with the client secret of a PaymentIntent or SetupIntent. + + @param clientSecret The client secret of the PaymentIntent or SetupIntent. See https://stripe.com/docs/api/payment_intents/object#payment_intent_object-client_secret + @param error The error that occurred when creating the Intent, or nil if none occurred. + */ +typedef void (^STPIntentClientSecretCompletionBlock)(NSString * __nullable clientSecret, NSError * __nullable error); + diff --git a/Stripe/StripeiOSTests/STPCardBINMetadataTests.swift b/Stripe/StripeiOSTests/STPCardBINMetadataTests.swift new file mode 100644 index 00000000..bb98e92f --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardBINMetadataTests.swift @@ -0,0 +1,55 @@ +// +// STPCardBINMetadataTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 7/20/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardBINMetadataTests: XCTestCase { + func testAPICall() { + STPAPIClient.shared.publishableKey = STPTestingDefaultPublishableKey + + let expectation = self.expectation(description: "Retrieve card metadata") + + // 625035 is a randomly selected UnionPay BIN + STPBINRange.retrieve( + forPrefix: "625035", + completion: { result in + let cardMetadata = try! result.get() + XCTAssertTrue(cardMetadata.data.count > 0) + XCTAssertEqual(cardMetadata.data.first!.brand, .unionPay) + expectation.fulfill() + } + ) + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testLoadingInBINRange() { + STPAPIClient.shared.publishableKey = STPTestingDefaultPublishableKey + + let expectation = self.expectation(description: "Retrieve card metadata") + let hardCodedBinRanges = STPBINController.shared.allRanges() + STPBINController.shared.retrieveBINRanges(forPrefix: "625035") { result in + let ranges = try! result.get() + XCTAssertTrue(ranges.count > 0) + XCTAssertTrue( + STPBINController.shared.allRanges().count == hardCodedBinRanges.count + ranges.count + ) + for range in ranges { + XCTAssertTrue(STPBINController.shared.allRanges().contains(range)) + } + expectation.fulfill() + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + + } +} diff --git a/Stripe/StripeiOSTests/STPCardBrandTest.m b/Stripe/StripeiOSTests/STPCardBrandTest.m new file mode 100644 index 00000000..aea63cd4 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardBrandTest.m @@ -0,0 +1,68 @@ +// +// STPCardBrandTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/3/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + + + +@interface STPCardBrandTest : XCTestCase + +@end + +@implementation STPCardBrandTest + +- (void)testStringFromBrand { + NSArray *brands = @[ + @(STPCardBrandAmex), + @(STPCardBrandDinersClub), + @(STPCardBrandDiscover), + @(STPCardBrandJCB), + @(STPCardBrandMastercard), + @(STPCardBrandUnionPay), + @(STPCardBrandVisa), + @(STPCardBrandCartesBancaires), + @(STPCardBrandUnknown), + ]; + + for (NSNumber *brandNumber in brands) { + STPCardBrand brand = [brandNumber integerValue]; + NSString *string = [STPCardBrandUtilities stringFromCardBrand:brand]; + + switch (brand) { + case STPCardBrandAmex: + XCTAssertEqualObjects(string, @"American Express"); + break; + case STPCardBrandDinersClub: + XCTAssertEqualObjects(string, @"Diners Club"); + break; + case STPCardBrandDiscover: + XCTAssertEqualObjects(string, @"Discover"); + break; + case STPCardBrandJCB: + XCTAssertEqualObjects(string, @"JCB"); + break; + case STPCardBrandMastercard: + XCTAssertEqualObjects(string, @"Mastercard"); + break; + case STPCardBrandUnionPay: + XCTAssertEqualObjects(string, @"UnionPay"); + break; + case STPCardBrandVisa: + XCTAssertEqualObjects(string, @"Visa"); + break; + case STPCardBrandCartesBancaires: + XCTAssertEqualObjects(string, @"Cartes Bancaires"); + break; + case STPCardBrandUnknown: + XCTAssertEqualObjects(string, @"Unknown"); + break; + } + }; +} + +@end diff --git a/Stripe/StripeiOSTests/STPCardCVCInputTextFieldFormatterTests.swift b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldFormatterTests.swift new file mode 100644 index 00000000..1288fcea --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldFormatterTests.swift @@ -0,0 +1,52 @@ +// +// STPCardCVCInputTextFieldFormatterTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardCVCInputTextFieldFormatterTests: XCTestCase { + + func testAllowedInput() { + let formatter = STPCardCVCInputTextFieldFormatter() + + formatter.cardBrand = .unknown + XCTAssertTrue(formatter.isAllowedInput("1", to: "", at: NSRange(location: 0, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("12", to: "", at: NSRange(location: 0, length: 2))) + XCTAssertTrue(formatter.isAllowedInput("2", to: "1", at: NSRange(location: 1, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("3", to: "12", at: NSRange(location: 2, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("123", to: "", at: NSRange(location: 0, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("4", to: "123", at: NSRange(location: 3, length: 1))) + XCTAssertFalse(formatter.isAllowedInput("5", to: "1234", at: NSRange(location: 4, length: 1))) + + formatter.cardBrand = .amex + XCTAssertTrue(formatter.isAllowedInput("1", to: "", at: NSRange(location: 0, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("12", to: "", at: NSRange(location: 0, length: 2))) + XCTAssertTrue(formatter.isAllowedInput("2", to: "1", at: NSRange(location: 1, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("3", to: "12", at: NSRange(location: 2, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("123", to: "", at: NSRange(location: 0, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("4", to: "123", at: NSRange(location: 3, length: 1))) + XCTAssertFalse(formatter.isAllowedInput("5", to: "1234", at: NSRange(location: 4, length: 1))) + + formatter.cardBrand = .visa + XCTAssertTrue(formatter.isAllowedInput("1", to: "", at: NSRange(location: 0, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("12", to: "", at: NSRange(location: 0, length: 2))) + XCTAssertTrue(formatter.isAllowedInput("2", to: "1", at: NSRange(location: 1, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("3", to: "12", at: NSRange(location: 2, length: 1))) + XCTAssertTrue(formatter.isAllowedInput("123", to: "", at: NSRange(location: 0, length: 1))) + XCTAssertFalse(formatter.isAllowedInput("4", to: "123", at: NSRange(location: 3, length: 1))) + XCTAssertFalse(formatter.isAllowedInput("5", to: "1234", at: NSRange(location: 4, length: 1))) + + XCTAssertFalse(formatter.isAllowedInput("a", to: "123", at: NSRange(location: 0, length: 1))) + } + +} diff --git a/Stripe/StripeiOSTests/STPCardCVCInputTextFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldSnapshotTests.swift new file mode 100644 index 00000000..f9f80057 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldSnapshotTests.swift @@ -0,0 +1,61 @@ +// +// STPCardCVCInputTextFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardCVCInputTextFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testEmpty() { + let field = STPCardCVCInputTextField() + field.sizeToFit() + field.frame.size.width = 200 + + STPSnapshotVerifyView(field) + } + + func testIncomplete() { + let field = STPCardCVCInputTextField() + field.sizeToFit() + field.frame.size.width = 200 + field.text = "1" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + func testValid() { + let field = STPCardCVCInputTextField() + field.sizeToFit() + field.frame.size.width = 200 + field.text = "123" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + func testInvalid() { + let field = STPCardCVCInputTextField() + field.sizeToFit() + field.frame.size.width = 200 + field.text = "12345" + field.textDidChange() + + STPSnapshotVerifyView(field) + } +} diff --git a/Stripe/StripeiOSTests/STPCardCVCInputTextFieldTests.swift b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldTests.swift new file mode 100644 index 00000000..dcd15604 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldTests.swift @@ -0,0 +1,34 @@ +// +// STPCardCVCInputTextFieldTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 8/31/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardCVCInputTextFieldTests: XCTestCase { + + func testTruncatingCVCWhenTooLong() { + let cvcField = STPCardCVCInputTextField() + cvcField.cardBrand = .amex + cvcField.text = String( + repeating: "1", + count: Int(STPCardValidator.maxCVCLength(for: .amex)) + ) + XCTAssertEqual(cvcField.text?.count, Int(STPCardValidator.maxCVCLength(for: .amex))) + + // Switching the card brand to `visa` should truncate the field text to + // the max length allowed for the brand + cvcField.cardBrand = .visa + XCTAssertEqual(cvcField.text?.count, Int(STPCardValidator.maxCVCLength(for: .visa))) + } + +} diff --git a/Stripe/StripeiOSTests/STPCardCVCInputTextFieldValidatorTests.swift b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldValidatorTests.swift new file mode 100644 index 00000000..deb018dc --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardCVCInputTextFieldValidatorTests.swift @@ -0,0 +1,54 @@ +// +// STPCardCVCInputTextFieldValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardCVCInputTextFieldValidatorTests: XCTestCase { + + func testValidation() { + let validator = STPCardCVCInputTextFieldValidator() + validator.cardBrand = .visa + + validator.inputValue = "123" + if case .valid = validator.validationState { + XCTAssertTrue(true) + } else { + XCTAssertTrue(false, "123 should be valid for Visa") + } + + validator.inputValue = "1" + if case .incomplete(let description) = validator.validationState { + XCTAssertTrue(true) + XCTAssertEqual(description, "Your card's security code is incomplete.") + } else { + XCTAssertTrue(false, "1 should be incomplete for Visa") + } + + validator.inputValue = "1234" + if case .invalid(let errorMessage) = validator.validationState { + XCTAssertEqual(errorMessage, "Your card's security code is invalid.") + } else { + XCTAssertTrue(false, "1234 should be invalid for Visa") + } + + validator.cardBrand = .amex + // don't update inputValue so we know validationState is recalculated on cardBrand change + if case .valid = validator.validationState { + XCTAssertTrue(true) + } else { + XCTAssertTrue(false, "1234 should be valid for Amex") + } + } + +} diff --git a/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldFormatterTests.swift b/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldFormatterTests.swift new file mode 100644 index 00000000..ee89f7f3 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldFormatterTests.swift @@ -0,0 +1,64 @@ +// +// STPCardExpiryInputTextFieldFormatterTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardExpiryInputTextFieldFormatterTests: XCTestCase { + + func testAllowedInput() { + let formatter = STPCardExpiryInputTextFieldFormatter() + XCTAssertTrue(formatter.isAllowedInput("1226", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("12/26", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("12 / 26", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("122026", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("12/2026", to: "", at: NSRange(location: 0, length: 0))) + + XCTAssertTrue(formatter.isAllowedInput("1", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("2", to: "1", at: NSRange(location: 1, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("2", to: "12", at: NSRange(location: 2, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("2", to: "12/", at: NSRange(location: 2, length: 0))) + + // the formatter does NOT verify that these are sensical dates (that is delegated to the validator) + XCTAssertTrue(formatter.isAllowedInput("16/1901", to: "", at: NSRange(location: 0, length: 0))) + + XCTAssertFalse(formatter.isAllowedInput("12 / 25 / 26", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertFalse(formatter.isAllowedInput("12 / 25 / 26", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertFalse(formatter.isAllowedInput("12.26", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertFalse(formatter.isAllowedInput("2026/12", to: "", at: NSRange(location: 0, length: 0))) + } + + func testFormattedText() { + let formatter = STPCardExpiryInputTextFieldFormatter() + XCTAssertEqual( + formatter.formattedText(from: "1226", with: [:]), + NSAttributedString(string: "12/26") + ) + XCTAssertEqual( + formatter.formattedText(from: "12/26", with: [:]), + NSAttributedString(string: "12/26") + ) + XCTAssertEqual( + formatter.formattedText(from: "12 / 26", with: [:]), + NSAttributedString(string: "12/26") + ) + XCTAssertEqual( + formatter.formattedText(from: "122026", with: [:]), + NSAttributedString(string: "12/26") + ) + XCTAssertEqual( + formatter.formattedText(from: "12 / 2026", with: [:]), + NSAttributedString(string: "12/26") + ) + } +} diff --git a/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldSnapshotTests.swift new file mode 100644 index 00000000..868afbd7 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldSnapshotTests.swift @@ -0,0 +1,56 @@ +// +// STPCardExpiryInputTextFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardExpiryInputTextFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testEmpty() { + let field = STPCardExpiryInputTextField() + field.sizeToFit() + field.frame.size.width = 200 + + STPSnapshotVerifyView(field) + } + + func testIncomplete() { + let field = STPCardExpiryInputTextField() + field.sizeToFit() + field.frame.size.width = 200 + field.text = "1" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + // We can't have a valid test here because the date would have to change as time marches on + // func testValid() { + // } + + func testInvalid() { + let field = STPCardExpiryInputTextField() + field.sizeToFit() + field.frame.size.width = 200 + field.text = "16/22" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + +} diff --git a/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldValidatorTests.swift b/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldValidatorTests.swift new file mode 100644 index 00000000..9b235605 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardExpiryInputTextFieldValidatorTests.swift @@ -0,0 +1,131 @@ +// +// STPCardExpiryInputTextFieldValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardExpiryInputTextFieldValidatorTests: XCTestCase { + + func testValidation() { + let now = Date() + guard + let nowMonth = Calendar(identifier: .gregorian).dateComponents( + Set([Calendar.Component.month]), + from: now + ).month, + let fullYear = Calendar(identifier: .gregorian).dateComponents( + Set([Calendar.Component.year]), + from: now + ).year + else { + XCTFail("Chaos reigns") + return + } + let nowYear = fullYear % 100 + + let validator = STPCardExpiryInputTextFieldValidator() + validator.inputValue = String(format: "01/%2d", (nowYear + 1) % 100) + if case .valid = validator.validationState { + XCTAssertTrue(true) + } else { + XCTFail("January of next year should be valid") + } + + let oneMonthAhead: String = { + if nowMonth == 12 { + return String(format: "01/%2d", (nowYear + 1) % 100) + } else { + return String(format: "%02d/%2d", nowMonth + 1, nowYear) + } + }() + validator.inputValue = oneMonthAhead + if case .valid = validator.validationState { + XCTAssertTrue(true) + } else { + XCTFail("One month ahead should be valid") + } + + let oneMonthAgo: String = { + if nowMonth == 1 { + return String(format: "01/%2d", max(0, nowYear - 1)) + } else { + return String(format: "%02d/%2d", nowMonth - 1, nowYear) + } + }() + validator.inputValue = oneMonthAgo + if case .invalid(let errorMessage) = validator.validationState { + XCTAssertEqual(errorMessage, "Your card's expiration year is invalid.") + } else { + XCTFail("One month ago should be invalid") + } + + let nonsensical = "16/55" + validator.inputValue = nonsensical + if case .invalid(let errorMessage) = validator.validationState { + XCTAssertEqual(errorMessage, "Your card's expiration date is invalid.") + } else { + XCTFail("Invalid month+year should be invalid") + } + + validator.inputValue = "2" + if case .incomplete(let description) = validator.validationState { + XCTAssertEqual(description, "Your card's expiration date is incomplete.") + } else { + XCTFail("One digit should be incomplete") + } + + validator.inputValue = "2/" + if case .incomplete(let description) = validator.validationState { + XCTAssertEqual(description, "Your card's expiration date is incomplete.") + } else { + XCTFail("One digit with separator should be incomplete") + } + + validator.inputValue = String(format: "1/%2d", (nowYear + 1) % 100) + if case .incomplete(let description) = validator.validationState { + XCTAssertEqual(description, "Your card's expiration date is incomplete.") + } else { + XCTFail("Single digit month should be incomplete") + } + + validator.inputValue = "13/" + if case .invalid(let description) = validator.validationState { + XCTAssertEqual(description, "Your card's expiration month is invalid.") + } else { + XCTFail("Invalid month should be invalid") + } + } + + func testExpiryStringFormatsYear() throws { + let validator = STPCardExpiryInputTextFieldValidator() + + validator.inputValue = "02/24" + + let expiryStrings = try XCTUnwrap(validator.expiryStrings) + + XCTAssertEqual(expiryStrings.month, "02") + XCTAssertEqual(expiryStrings.year, "2024") + } + + func testExpiryStringDoesNotFormatYear() throws { + let validator = STPCardExpiryInputTextFieldValidator() + + validator.inputValue = "02/2024" + + let expiryStrings = try XCTUnwrap(validator.expiryStrings) + + XCTAssertEqual(expiryStrings.month, "02") + XCTAssertEqual(expiryStrings.year, "2024") + } + +} diff --git a/Stripe/StripeiOSTests/STPCardFormViewSnapshotTests.swift b/Stripe/StripeiOSTests/STPCardFormViewSnapshotTests.swift new file mode 100644 index 00000000..6c1bab7e --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardFormViewSnapshotTests.swift @@ -0,0 +1,126 @@ +// +// STPCardFormViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardFormViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testEmpty() { + let formView = STPCardFormView(billingAddressCollection: .automatic) + formView.countryCode = "US" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + STPSnapshotVerifyView(formView) + } + + func testIncomplete() { + let formView = STPCardFormView(billingAddressCollection: .automatic) + formView.countryCode = "US" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 265)) + + formView.numberField.text = "4242" + formView.numberField.textDidChange() + formView.cvcField.text = "123" + formView.cvcField.textDidChange() + + STPSnapshotVerifyView(formView) + } + + // valid expiration date will change over time so we just test without it + func testCompleteWithoutExpiry() { + let formView = STPCardFormView(billingAddressCollection: .automatic) + formView.countryCode = "US" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + formView.numberField.text = "4242424242424242" + formView.numberField.textDidChange() + formView.cvcField.text = "123" + formView.cvcField.textDidChange() + formView.postalCodeField.text = "12345" + + STPSnapshotVerifyView(formView) + } + + func testEmptyHiddenPostalCode() { + let formView = STPCardFormView(billingAddressCollection: .automatic) + formView.countryCode = "AE" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + STPSnapshotVerifyView(formView) + } + + func testWithFullBillingDetails() { + let formView = STPCardFormView(billingAddressCollection: .required) + formView.countryCode = "US" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 400)) + + STPSnapshotVerifyView(formView) + } + + // MARK: - Standalone + + func testDefaultStandalone() { + let formView = STPCardFormView() + formView.countryCode = "US" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + STPSnapshotVerifyView(formView) + } + + func testBorderlessStandalone() { + let formView = STPCardFormView(style: .borderless) + formView.countryCode = "US" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + STPSnapshotVerifyView(formView) + } + + func testCustomBackgroundStandalone() { + let formView = STPCardFormView() + formView.countryCode = "US" + formView.backgroundColor = .green + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + STPSnapshotVerifyView(formView) + } + + func testCustomBackgroundDisabledColorStandalone() { + let formView = STPCardFormView() + formView.countryCode = "US" + formView.disabledBackgroundColor = .green + formView.isUserInteractionEnabled = false + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + STPSnapshotVerifyView(formView) + } + + func testBorderlessStandaloneIncomplete() { + let formView = STPCardFormView(style: .borderless) + formView.countryCode = "US" + formView.frame = CGRect(origin: .zero, size: CGSize(width: 300, height: 225)) + + formView.numberField.text = "4242" + formView.numberField.textDidChange() + formView.cvcField.text = "123" + formView.cvcField.textDidChange() + + STPSnapshotVerifyView(formView) + } + +} diff --git a/Stripe/StripeiOSTests/STPCardFormViewTests.swift b/Stripe/StripeiOSTests/STPCardFormViewTests.swift new file mode 100644 index 00000000..078a280f --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardFormViewTests.swift @@ -0,0 +1,243 @@ +// +// STPCardFormViewTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 1/19/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardFormViewTests: XCTestCase { + + func testMarkFormErrorsLogic() { + let cardForm = STPCardFormView() + + let handledErrorsTypes = [ + "incorrect_number", + "invalid_number", + "invalid_expiry_month", + "invalid_expiry_year", + "expired_card", + "invalid_cvc", + "incorrect_cvc", + "incorrect_zip", + ] + + let unhandledErrorTypes = [ + "card_declined", + "processing_error", + "imaginary_error", + "", + nil, + ] + + for shouldHandle in handledErrorsTypes { + let error = NSError( + domain: STPError.stripeDomain, + code: STPErrorCode.apiError.rawValue, + userInfo: [STPError.stripeErrorCodeKey: shouldHandle] + ) + XCTAssertTrue( + cardForm.markFormErrors(for: error), + "Failed to handle error for \(shouldHandle)" + ) + } + + for shouldNotHandle in unhandledErrorTypes { + let error: NSError + if let shouldNotHandle = shouldNotHandle { + error = NSError( + domain: STPError.stripeDomain, + code: STPErrorCode.apiError.rawValue, + userInfo: [STPError.stripeErrorCodeKey: shouldNotHandle] + ) + } else { + error = NSError( + domain: STPError.stripeDomain, + code: STPErrorCode.apiError.rawValue, + userInfo: nil + ) + } + XCTAssertFalse( + cardForm.markFormErrors(for: error), + "Incorrectly handled \(shouldNotHandle ?? "nil")" + ) + } + } + + func testHidingPostalCodeOnInit() { + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "zh_Hans_HK") as Locale) { + let cardForm = STPCardFormView() + XCTAssertTrue(cardForm.postalCodeField.isHidden) + } + } + + func testHidingPostalUPECodeOnInit() { + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "zh_Hans_HK") as Locale) { + let cardForm = STPCardFormView( + billingAddressCollection: .automatic, + includeCardScanning: false, + mergeBillingFields: false, + style: .standard, + postalCodeRequirement: .upe, + prefillDetails: nil + ) + XCTAssertTrue(cardForm.postalCodeField.isHidden) + } + } + + func testNotHidingPostalUPECodeOnInit() { + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_US") as Locale) { + let cardForm = STPCardFormView( + billingAddressCollection: .automatic, + includeCardScanning: false, + mergeBillingFields: false, + style: .standard, + postalCodeRequirement: .upe, + prefillDetails: nil + ) + XCTAssertFalse(cardForm.postalCodeField.isHidden) + } + } + + func testPanLockedOnInit() { + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_US") as Locale) { + let cardForm = STPCardFormView( + billingAddressCollection: .automatic, + includeCardScanning: false, + mergeBillingFields: false, + style: .standard, + postalCodeRequirement: .upe, + prefillDetails: nil, + inputMode: .panLocked + ) + XCTAssertFalse(cardForm.numberField.isUserInteractionEnabled) + } + } + + func testPrefilledOnInit() { + let prefillDeatils = STPCardFormView.PrefillDetails( + last4: "4242", + expiryMonth: 12, + expiryYear: 25, + cardBrand: .amex + ) + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_US") as Locale) { + let cardForm = STPCardFormView( + billingAddressCollection: .automatic, + includeCardScanning: false, + mergeBillingFields: false, + style: .standard, + postalCodeRequirement: .upe, + prefillDetails: prefillDeatils, + inputMode: .panLocked + ) + + XCTAssertEqual(cardForm.numberField.text, prefillDeatils.formattedLast4) + XCTAssertEqual(cardForm.numberField.cardBrand, prefillDeatils.cardBrand) + XCTAssertEqual(cardForm.expiryField.text, prefillDeatils.formattedExpiry) + XCTAssertEqual(cardForm.cvcField.cardBrand, prefillDeatils.cardBrand) + } + } + + // MARK: Functional Tests + // If these fail it's _possibly_ because the returned error formats have changed + + func helperFunctionalTestNumber(_ cardNumber: String, shouldHandle: Bool) { + let createPaymentIntentExpectation = self.expectation( + description: "createPaymentIntentExpectation" + ) + var retrievedClientSecret: String? + STPTestingAPIClient.shared().createPaymentIntent(withParams: nil) { + (createdPIClientSecret, _) in + if let createdPIClientSecret = createdPIClientSecret { + retrievedClientSecret = createdPIClientSecret + createPaymentIntentExpectation.fulfill() + } else { + XCTFail() + } + } + wait(for: [createPaymentIntentExpectation], timeout: 8) // STPTestingNetworkRequestTimeout + guard let clientSecret = retrievedClientSecret, + let currentYear = Calendar.current.dateComponents([.year], from: Date()).year + else { + XCTFail() + return + } + + // STPTestingDefaultPublishableKey + let client = STPAPIClient(publishableKey: "pk_test_ErsyMEOTudSjQR8hh0VrQr5X008sBXGOu6") + + let expiryYear = NSNumber(value: currentYear + 2) + let expiryMonth = NSNumber(1) + + let cardParams = STPPaymentMethodCardParams() + cardParams.number = cardNumber + cardParams.expYear = expiryYear + cardParams.expMonth = expiryMonth + cardParams.cvc = "123" + + let address = STPPaymentMethodAddress() + address.postalCode = "12345" + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.address = address + + let paymentMethodParams = STPPaymentMethodParams.paramsWith( + card: cardParams, + billingDetails: billingDetails, + metadata: nil + ) + + let paymentIntentParams = STPPaymentIntentParams(clientSecret: clientSecret) + paymentIntentParams.paymentMethodParams = paymentMethodParams + + let confirmExpectation = expectation(description: "confirmExpectation") + client.confirmPaymentIntent(with: paymentIntentParams) { (_, error) in + if let error = error { + let cardForm = STPCardFormView() + if shouldHandle { + XCTAssertTrue( + cardForm.markFormErrors(for: error), + "Failed to handle \(error) for \(cardNumber)" + ) + } else { + XCTAssertFalse( + cardForm.markFormErrors(for: error), + "Incorrectly handled \(error) for \(cardNumber)" + ) + } + confirmExpectation.fulfill() + } else { + XCTFail() + } + } + wait(for: [confirmExpectation], timeout: 8) // STPTestingNetworkRequestTimeout + } + + func testExpiredCard() { + helperFunctionalTestNumber("4000000000000069", shouldHandle: true) + } + + func testIncorrectCVC() { + helperFunctionalTestNumber("4000000000000127", shouldHandle: true) + } + + func testIncorrectCardNumber() { + helperFunctionalTestNumber("4242424242424241", shouldHandle: true) + } + + func testCardDeclined() { + helperFunctionalTestNumber("4000000000000002", shouldHandle: false) + } + + func testProcessingError() { + helperFunctionalTestNumber("4000000000000119", shouldHandle: false) + } +} diff --git a/Stripe/StripeiOSTests/STPCardFunctionalTest.m b/Stripe/StripeiOSTests/STPCardFunctionalTest.m new file mode 100644 index 00000000..198e2c73 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardFunctionalTest.m @@ -0,0 +1,156 @@ +// +// STPCardFunctionalTest.m +// Stripe +// +// Created by Ray Morgan on 7/11/14. +// +// + +@import XCTest; +@import StripeCoreTestUtils; + +#import "STPTestingAPIClient.h" + +@interface STPCardFunctionalTest : XCTestCase +@end + +@implementation STPCardFunctionalTest + +- (void)testCreateCardToken { + STPCardParams *card = [[STPCardParams alloc] init]; + + card.number = @"4242 4242 4242 4242"; + card.expMonth = 6; + card.expYear = 2024; + card.currency = @"usd"; + card.address.line1 = @"123 Fake Street"; + card.address.line2 = @"Apartment 4"; + card.address.city = @"New York"; + card.address.state = @"NY"; + card.address.country = @"USA"; + card.address.postalCode = @"10002"; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Card creation"]; + + [client createTokenWithCard:card + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + + XCTAssertNil(error, @"error should be nil %@", error.localizedDescription); + XCTAssertNotNil(token, @"token should not be nil"); + + XCTAssertNotNil(token.tokenId); + XCTAssertEqual(token.type, STPTokenTypeCard); + XCTAssertEqual(6U, token.card.expMonth); + XCTAssertEqual(2024U, token.card.expYear); + XCTAssertEqualObjects(@"4242", token.card.last4); + XCTAssertEqualObjects(@"usd", token.card.currency); + XCTAssertEqualObjects(@"10002", token.card.address.postalCode); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCardTokenCreationWithInvalidParams { + STPCardParams *card = [[STPCardParams alloc] init]; + + card.number = @"4242 4242 4242 4241"; + card.expMonth = 6; + card.expYear = 2024; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Card creation"]; + + [client createTokenWithCard:card + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + + XCTAssertNotNil(error, @"error should not be nil"); + XCTAssertEqual(error.code, 70); + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + XCTAssertEqualObjects(error.userInfo[[STPError errorParameterKey]], @"number"); + XCTAssertNil(token, @"token should be nil: %@", token.description); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCardTokenCreationWithExpiredCard { + STPCardParams *card = [[STPCardParams alloc] init]; + + card.number = @"4242 4242 4242 4242"; + card.expMonth = 6; + card.expYear = 2013; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Card creation"]; + + [client createTokenWithCard:card + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + + XCTAssertNotNil(error, @"error should not be nil"); + XCTAssertEqual(error.code, 70); + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + XCTAssertEqualObjects(error.userInfo[[STPError cardErrorCodeKey]], [STPError invalidExpYear]); + XCTAssertEqualObjects(error.userInfo[[STPError errorParameterKey]], @"expYear"); + XCTAssertNil(token, @"token should be nil: %@", token.description); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testInvalidKey { + STPCardParams *card = [[STPCardParams alloc] init]; + + card.number = @"4242 4242 4242 4242"; + card.expMonth = 6; + card.expYear = 2024; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"not_a_valid_key_asdf"]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Card failure"]; + [client createTokenWithCard:card + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + XCTAssertNil(token, @"token should be nil"); + XCTAssertNotNil(error, @"error should not be nil"); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateCVCUpdateToken { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"CVC Update Token Creation"]; + + [client createTokenForCVCUpdate:@"1234" + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + + XCTAssertNil(error, @"error should be nil %@", error.localizedDescription); + XCTAssertNotNil(token, @"token should not be nil"); + + XCTAssertNotNil(token.tokenId); + XCTAssertEqual(token.type, STPTokenTypeCvcUpdate, @"token should be type CVC Update"); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testInvalidCVC { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Invalid CVC"]; + + [client createTokenForCVCUpdate:@"1" + completion:^(STPToken *token, NSError *error) { + [expectation fulfill]; + + XCTAssertNil(token, @"token should be nil"); + XCTAssertNotNil(error, @"error should not be nil"); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPCardNumberInputTextFieldFormatterTests.swift b/Stripe/StripeiOSTests/STPCardNumberInputTextFieldFormatterTests.swift new file mode 100644 index 00000000..2c6bc060 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardNumberInputTextFieldFormatterTests.swift @@ -0,0 +1,73 @@ +// +// STPCardNumberInputTextFieldFormatterTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardNumberInputTextFieldFormatterTests: XCTestCase { + + func testAllowedInput() { + let formatter = STPCardNumberInputTextFieldFormatter() + XCTAssertTrue(formatter.isAllowedInput("4242424242424242", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("424242424242424", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue( + formatter.isAllowedInput("4242 4242 4242 4242", to: "", at: NSRange(location: 0, length: 0)) + ) + XCTAssertTrue(formatter.isAllowedInput("42424242 42424242", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("4242 ", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("3566002020360505", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput(" ", to: "4242", at: NSRange(location: 4, length: 0))) + + XCTAssertFalse( + formatter.isAllowedInput("4242.4242.4242.4242", to: "", at: NSRange(location: 0, length: 0)) + ) + XCTAssertFalse(formatter.isAllowedInput("4", to: "4242424242424242", at: NSRange(location: 0, length: 0))) + } + + func testFormatting() { + let formatter = STPCardNumberInputTextFieldFormatter() + var expected: NSMutableAttributedString = NSMutableAttributedString() + + expected = NSMutableAttributedString(string: "4242424242424242") + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 0, length: 3)) + expected.addAttribute(.kern, value: NSNumber(5), range: NSRange(location: 3, length: 1)) + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 4, length: 3)) + expected.addAttribute(.kern, value: NSNumber(5), range: NSRange(location: 7, length: 1)) + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 8, length: 3)) + expected.addAttribute(.kern, value: NSNumber(5), range: NSRange(location: 11, length: 1)) + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 12, length: 4)) + XCTAssertEqual(formatter.formattedText(from: "4242424242424242", with: [:]), expected) + XCTAssertEqual(formatter.formattedText(from: "4242 4242 4242 4242", with: [:]), expected) + + expected = NSMutableAttributedString(string: "4242") + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 0, length: 4)) + XCTAssertEqual(formatter.formattedText(from: "4242", with: [:]), expected) + + expected = NSMutableAttributedString(string: "42424") + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 0, length: 3)) + expected.addAttribute(.kern, value: NSNumber(5), range: NSRange(location: 3, length: 1)) + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 4, length: 1)) + XCTAssertEqual(formatter.formattedText(from: "42424", with: [:]), expected) + + expected = NSMutableAttributedString(string: "378282246310005") // 4, 6, 5, + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 0, length: 3)) + expected.addAttribute(.kern, value: NSNumber(5), range: NSRange(location: 3, length: 1)) + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 4, length: 5)) + expected.addAttribute(.kern, value: NSNumber(5), range: NSRange(location: 9, length: 1)) + expected.addAttribute(.kern, value: NSNumber(0), range: NSRange(location: 10, length: 5)) + // expected.addAttribute(.kern, value: NSNumber(5), range: NSMakeRange(11, 1)) + // expected.addAttribute(.kern, value: NSNumber(0), range: NSMakeRange(12, 4)) + XCTAssertEqual(formatter.formattedText(from: "378282246310005", with: [:]), expected) + } + +} diff --git a/Stripe/StripeiOSTests/STPCardNumberInputTextFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPCardNumberInputTextFieldSnapshotTests.swift new file mode 100644 index 00000000..db9bf090 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardNumberInputTextFieldSnapshotTests.swift @@ -0,0 +1,61 @@ +// +// STPCardNumberInputTextFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardNumberInputTextFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testEmpty() { + let field = STPCardNumberInputTextField() + field.sizeToFit() + field.frame.size.width = 300 + + STPSnapshotVerifyView(field) + } + + func testIncomplete() { + let field = STPCardNumberInputTextField() + field.sizeToFit() + field.frame.size.width = 300 + field.text = "42" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + func testValid() { + let field = STPCardNumberInputTextField() + field.sizeToFit() + field.frame.size.width = 300 + field.text = "4242424242424242" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + func testInvalid() { + let field = STPCardNumberInputTextField() + field.sizeToFit() + field.frame.size.width = 300 + field.text = "4242424242424241" + field.textDidChange() + + STPSnapshotVerifyView(field) + } +} diff --git a/Stripe/StripeiOSTests/STPCardNumberInputTextFieldValidatorTests.swift b/Stripe/StripeiOSTests/STPCardNumberInputTextFieldValidatorTests.swift new file mode 100644 index 00000000..fb5ae08e --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardNumberInputTextFieldValidatorTests.swift @@ -0,0 +1,207 @@ +// +// STPCardNumberInputTextFieldValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardNumberInputTextFieldValidatorTests: XCTestCase { + + static let cardData: [(STPCardBrand, String, STPValidatedInputState)] = { + return [ + ( + .visa, + "4242424242424242", + .valid(message: nil) + ), + ( + .visa, + "4242424242422", + .incomplete(description: "Your card number is incomplete.") + ), + ( + .visa, + "4012888888881881", + .valid(message: nil) + ), + ( + .visa, + "4000056655665556", + .valid(message: nil) + ), + ( + .mastercard, + "5555555555554444", + .valid(message: nil) + ), + ( + .mastercard, + "5200828282828210", + .valid(message: nil) + ), + ( + .mastercard, + "5105105105105100", + .valid(message: nil) + ), + ( + .mastercard, + "2223000010089800", + .valid(message: nil) + ), + ( + .amex, + "378282246310005", + .valid(message: nil) + ), + ( + .amex, + "371449635398431", + .valid(message: nil) + ), + ( + .discover, + "6011111111111117", + .valid(message: nil) + ), + ( + .discover, + "6011000990139424", + .valid(message: nil) + ), + ( + .dinersClub, + "36227206271667", + .valid(message: nil) + ), + ( + .dinersClub, + "3056930009020004", + .valid(message: nil) + ), + ( + .JCB, + "3530111333300000", + .valid(message: nil) + ), + ( + .JCB, + "3566002020360505", + .valid(message: nil) + ), + ( + .unknown, + "1234567812345678", + .invalid(errorMessage: "Your card number is invalid.") + ), + ] + }() + + func testValidation() { + // same tests as in STPCardValidatorTest#testNumberValidation + var tests: [(STPValidatedInputState, String, STPCardBrand)] = [] + + for card in STPCardNumberInputTextFieldValidatorTests.cardData { + tests.append((card.2, card.1, card.0)) + } + + tests.append((.valid(message: nil), "4242 4242 4242 4242", .visa)) + tests.append((.valid(message: nil), "4136000000008", .visa)) + + let badCardNumbers: [(String, STPCardBrand)] = [ + ("0000000000000000", .unknown), + ("9999999999999995", .unknown), + ("1", .unknown), + ("1234123412341234", .unknown), + ("xxx", .unknown), + ("9999999999999999999999", .unknown), + ("42424242424242424242", .visa), + ("4242-4242-4242-4242", .visa), + ] + + for card in badCardNumbers { + tests.append((.invalid(errorMessage: "Your card number is invalid."), card.0, card.1)) + } + + let possibleCardNumbers: [(String, STPCardBrand)] = [ + ("4242", .visa), ("5", .mastercard), ("3", .unknown), ("", .unknown), + (" ", .unknown), ("6011", .discover), ("4012888888881", .visa), + ] + + for card in possibleCardNumbers { + tests.append( + ( + .incomplete( + description: card.0.isEmpty ? nil : "Your card number is incomplete." + ), + card.0, card.1 + ) + ) + } + + let validator = STPCardNumberInputTextFieldValidator() + for test in tests { + let card = test.1 + validator.inputValue = card + let validationState = validator.validationState + let expected = test.0 + if !(validationState == expected) { + XCTFail("Expected \(expected), got \(validationState) for number \"\(card)\"") + } + let expectedCardBrand = test.2 + if !(validator.cardBrand == expectedCardBrand) { + XCTFail( + "Expected \(expectedCardBrand), got \(validator.cardBrand) for number \(card)" + ) + } + } + + validator.inputValue = "1" + XCTAssertEqual( + .invalid(errorMessage: "Your card number is invalid."), + validator.validationState + ) + + validator.inputValue = "0000000000000000" + XCTAssertEqual( + .invalid(errorMessage: "Your card number is invalid."), + validator.validationState + ) + + validator.inputValue = "9999999999999995" + XCTAssertEqual( + .invalid(errorMessage: "Your card number is invalid."), + validator.validationState + ) + + validator.inputValue = "0000000000000000000" + XCTAssertEqual( + .invalid(errorMessage: "Your card number is invalid."), + validator.validationState + ) + + validator.inputValue = "9999999999999999998" + XCTAssertEqual( + .invalid(errorMessage: "Your card number is invalid."), + validator.validationState + ) + + validator.inputValue = "4242424242424" + XCTAssertEqual( + .incomplete(description: "Your card number is incomplete."), + validator.validationState + ) + + validator.inputValue = nil + XCTAssertEqual(.incomplete(description: nil), validator.validationState) + } +} diff --git a/Stripe/StripeiOSTests/STPCardParamsTest.m b/Stripe/StripeiOSTests/STPCardParamsTest.m new file mode 100644 index 00000000..67247652 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardParamsTest.m @@ -0,0 +1,185 @@ +// +// STPCardParamsTest.m +// Stripe +// +// Created by Joey Dong on 6/19/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPCardParamsTest : XCTestCase + +@end + +@implementation STPCardParamsTest + +#pragma mark - + +- (void)testLast4ReturnsCardNumberLast4 { + STPCardParams *cardParams = [[STPCardParams alloc] init]; + cardParams.number = @"4242424242424242"; + XCTAssertEqualObjects(cardParams.last4, @"4242"); +} + +- (void)testLast4ReturnsNilWhenNoCardNumberSet { + STPCardParams *cardParams = [[STPCardParams alloc] init]; + XCTAssertNil(cardParams.last4); +} + +- (void)testLast4ReturnsNilWhenCardNumberIsLessThanLength4 { + STPCardParams *cardParams = [[STPCardParams alloc] init]; + cardParams.number = @"123"; + XCTAssertNil(cardParams.last4); +} + +- (void)testNameSharedWithAddress { + STPCardParams *cardParams = [STPCardParams new]; + + cardParams.name = @"James"; + XCTAssertEqualObjects(cardParams.name, @"James"); + XCTAssertEqualObjects(cardParams.address.name, @"James"); + + STPAddress *address = [STPAddress new]; + address.name = @"Jim"; + + cardParams.address = address; + XCTAssertEqualObjects(cardParams.name, @"Jim"); + XCTAssertEqualObjects(cardParams.address.name, @"Jim"); + + // Doesn't update `name`, since mutation invisible to the STPCardParams + cardParams.address.name = @"Smith"; + XCTAssertEqualObjects(cardParams.name, @"Jim"); + XCTAssertEqualObjects(cardParams.address.name, @"Smith"); +} + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + +- (void)testAddress { + STPCardParams *cardParams = [[STPCardParams alloc] init]; + cardParams.name = @"John Smith"; + cardParams.addressLine1 = @"55 John St"; + cardParams.addressLine2 = @"#3B"; + cardParams.addressCity = @"New York"; + cardParams.addressState = @"NY"; + cardParams.addressZip = @"10002"; + cardParams.addressCountry = @"US"; + + STPAddress *address = cardParams.address; + + XCTAssertEqualObjects(address.name, @"John Smith"); + XCTAssertEqualObjects(address.line1, @"55 John St"); + XCTAssertEqualObjects(address.line2, @"#3B"); + XCTAssertEqualObjects(address.city, @"New York"); + XCTAssertEqualObjects(address.state, @"NY"); + XCTAssertEqualObjects(address.postalCode, @"10002"); + XCTAssertEqualObjects(address.country, @"US"); +} + +- (void)testSetAddress { + STPAddress *address = [[STPAddress alloc] init]; + address.name = @"John Smith"; + address.line1 = @"55 John St"; + address.line2 = @"#3B"; + address.city = @"New York"; + address.state = @"NY"; + address.postalCode = @"10002"; + address.country = @"US"; + + STPCardParams *cardParams = [[STPCardParams alloc] init]; + cardParams.address = address; + + XCTAssertEqualObjects(cardParams.name, @"John Smith"); + XCTAssertEqualObjects(cardParams.addressLine1, @"55 John St"); + XCTAssertEqualObjects(cardParams.addressLine2, @"#3B"); + XCTAssertEqualObjects(cardParams.addressCity, @"New York"); + XCTAssertEqualObjects(cardParams.addressState, @"NY"); + XCTAssertEqualObjects(cardParams.addressZip, @"10002"); + XCTAssertEqualObjects(cardParams.addressCountry, @"US"); +} + +#pragma clang diagnostic pop + +#pragma mark - Description Tests + +- (void)testDescription { + STPCardParams *cardParams = [[STPCardParams alloc] init]; + XCTAssert(cardParams.description); +} + +#pragma mark - STPFormEncodable Tests + +- (void)testRootObjectName { + XCTAssertEqualObjects([STPCardParams rootObjectName], @"card"); +} + +- (void)testPropertyNamesToFormFieldNamesMapping { + STPCardParams *cardParams = [[STPCardParams alloc] init]; + + NSDictionary *mapping = [STPCardParams propertyNamesToFormFieldNamesMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([cardParams respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); +} + +#pragma mark - NSCopying Tests + +- (void)testCopyWithZone { + STPCardParams *cardParams = [STPFixtures cardParams]; + cardParams.address = [STPFixtures address]; + STPCardParams *copiedCardParams = [cardParams copy]; + + XCTAssertNotEqual(cardParams, copiedCardParams, @"should be different objects"); + + // The property names we expect to *not* be equal objects + NSArray *notEqualProperties = @[ + // these include the object's address, so they won't be the same across copies + @"debugDescription", + @"description", + @"hash", + // STPAddress does not override isEqual:, so this is pointer comparison + @"address", + ]; + + // use runtime inspection to find the list of properties. If a new property is + // added to the fixture, but not the `copyWithZone:` implementation, this should catch it + for (NSString *property in [STPTestUtils propertyNamesOf:cardParams]) { + if ([notEqualProperties containsObject:property]) { + XCTAssertNotEqualObjects([cardParams valueForKey:property], + [copiedCardParams valueForKey:property], + @"%@", property); + } else { + XCTAssertEqualObjects([cardParams valueForKey:property], + [copiedCardParams valueForKey:property], + @"%@", property); + } + } +} + +- (void)testAddressIsNotCopied { + STPCardParams *cardParams = [STPFixtures cardParams]; + cardParams.address = [STPFixtures address]; + STPCardParams *secondCardParams = [STPCardParams new]; + + secondCardParams.address = cardParams.address; + cardParams.address.line1 = @"123 Main"; + + XCTAssertEqualObjects(cardParams.address.line1, @"123 Main"); + XCTAssertEqualObjects(secondCardParams.address.line1, @"123 Main"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPCardTest.swift b/Stripe/StripeiOSTests/STPCardTest.swift new file mode 100644 index 00000000..999d77d2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardTest.swift @@ -0,0 +1,263 @@ +// +// STPCardTest.swift +// StripeiOS Tests +// +// Created by Saikat Chakrabarti on 11/5/12. +// Copyright © 2012 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardTest: XCTestCase { + // MARK: - STPCardBrand Tests + + // These are only intended to be deprecated publicly. + // When removed from public header, can remove these pragmas + func testBrandFromString() { + XCTAssertEqual(STPCard.brand(from: "visa"), .visa) + XCTAssertEqual(STPCard.brand(from: "VISA"), .visa) + + XCTAssertEqual(STPCard.brand(from: "american express"), .amex) + XCTAssertEqual(STPCard.brand(from: "AMERICAN EXPRESS"), .amex) + + XCTAssertEqual(STPCard.brand(from: "mastercard"), .mastercard) + XCTAssertEqual(STPCard.brand(from: "MASTERCARD"), .mastercard) + + XCTAssertEqual(STPCard.brand(from: "discover"), .discover) + XCTAssertEqual(STPCard.brand(from: "DISCOVER"), .discover) + + XCTAssertEqual(STPCard.brand(from: "jcb"), .JCB) + XCTAssertEqual(STPCard.brand(from: "JCB"), .JCB) + + XCTAssertEqual(STPCard.brand(from: "diners club"), .dinersClub) + XCTAssertEqual(STPCard.brand(from: "DINERS CLUB"), .dinersClub) + + XCTAssertEqual(STPCard.brand(from: "unionpay"), .unionPay) + XCTAssertEqual(STPCard.brand(from: "UNIONPAY"), .unionPay) + + XCTAssertEqual(STPCard.brand(from: "unknown"), .unknown) + XCTAssertEqual(STPCard.brand(from: "UNKNOWN"), .unknown) + + XCTAssertEqual(STPCard.brand(from: "garbage"), .unknown) + XCTAssertEqual(STPCard.brand(from: "GARBAGE"), .unknown) + } + + // MARK: - STPCardFundingType Tests + + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wdeprecated" + // These are only intended to be deprecated publicly. + // When removed from public header, can remove these pragmas + func testFundingFromString() { + XCTAssertEqual(STPCard.funding(from: "credit"), .credit) + XCTAssertEqual(STPCard.funding(from: "CREDIT"), .credit) + + XCTAssertEqual(STPCard.funding(from: "debit"), .debit) + XCTAssertEqual(STPCard.funding(from: "DEBIT"), .debit) + + XCTAssertEqual(STPCard.funding(from: "prepaid"), .prepaid) + XCTAssertEqual(STPCard.funding(from: "PREPAID"), .prepaid) + + XCTAssertEqual(STPCard.funding(from: "other"), .other) + XCTAssertEqual(STPCard.funding(from: "OTHER"), .other) + + XCTAssertEqual(STPCard.funding(from: "unknown"), .other) + XCTAssertEqual(STPCard.funding(from: "UNKNOWN"), .other) + + XCTAssertEqual(STPCard.funding(from: "garbage"), .other) + XCTAssertEqual(STPCard.funding(from: "GARBAGE"), .other) + } + + // #pragma clang diagnostic pop + func testStringFromFunding() { + let values: [STPCardFundingType] = [ + .credit, + .debit, + .prepaid, + .other, + ] + + for funding in values { + let string = STPCard.string(fromFunding: funding) + + switch funding { + case .credit: + XCTAssertEqual(string, "credit") + case .debit: + XCTAssertEqual(string, "debit") + case .prepaid: + XCTAssertEqual(string, "prepaid") + case .other: + XCTAssertNil(string) + default: + break + } + } + } + + // MARK: - + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wdeprecated" + // These tests can ber removed in the future, they should be covered by + // the equivalent response decodeable tests + func testInitWithIDBrandLast4ExpMonthExpYearFunding() { + let card = STPCard( + id: "card_1AVRojEOD54MuFwSxr93QJSx", + brand: .visa, + last4: "5556", + expMonth: 12, + expYear: 2034, + funding: .debit + ) + XCTAssertEqual(card.stripeID, "card_1AVRojEOD54MuFwSxr93QJSx") + XCTAssertEqual(card.brand, .visa) + XCTAssertEqual(card.last4, "5556") + XCTAssertEqual(card.expMonth, Int(12)) + XCTAssertEqual(card.expYear, Int(2034)) + XCTAssertEqual(card.funding, .debit) + } + + // #pragma clang diagnostic pop + func testIsApplePayCard() { + let card = STPFixtures.card() + + card.allResponseFields = [:] + XCTAssertFalse(card.isApplePayCard) + + card.allResponseFields = [ + "tokenization_method": "android_pay" + ] + XCTAssertFalse(card.isApplePayCard) + + card.allResponseFields = [ + "tokenization_method": "apple_pay" + ] + XCTAssertTrue(card.isApplePayCard) + + card.allResponseFields = [ + "tokenization_method": "garbage" + ] + XCTAssertFalse(card.isApplePayCard) + + card.allResponseFields = [ + "tokenization_method": "" + ] + XCTAssertFalse(card.isApplePayCard) + + // See: https://stripe.com/docs/api#card_object-tokenization_method + } + + func testAddressPopulated() { + let card = STPFixtures.card() + XCTAssertEqual(card.address?.name, "Jane Austen") + XCTAssertEqual(card.address?.line1, "123 Fake St") + XCTAssertEqual(card.address?.line2, "Apt 1") + XCTAssertEqual(card.address?.city, "Pittsburgh") + XCTAssertEqual(card.address?.state, "PA") + XCTAssertEqual(card.address?.postalCode, "19219") + XCTAssertEqual(card.address?.country, "US") + } + + // MARK: - Equality Tests + func testCardEquals() { + let card1 = STPFixtures.card() + let card2 = STPFixtures.card() + + XCTAssertEqual(card1, card1) + XCTAssertEqual(card1, card2) + + XCTAssertEqual(card1.hash, card1.hash) + XCTAssertEqual(card1.hash, card2.hash) + } + + // MARK: - STPAPIResponseDecodable Tests + func testDecodedObjectFromAPIResponseRequiredFields() { + let requiredFields = ["id", "last4", "brand", "exp_month", "exp_year"] + + for field in requiredFields { + var response = STPTestUtils.jsonNamed("Card") + response?.removeValue(forKey: field) + + XCTAssertNil(STPCard.decodedObject(fromAPIResponse: response)) + } + + XCTAssert((STPCard.decodedObject(fromAPIResponse: STPTestUtils.jsonNamed("Card")) != nil)) + } + + func testDecodedObjectFromAPIResponseMapping() { + let response = STPTestUtils.jsonNamed("Card")! + let card = STPCard.decodedObject(fromAPIResponse: response)! + + XCTAssertEqual(card.stripeID, "card_103kbR2eZvKYlo2CDczLmw4K") + + XCTAssertEqual(card.address?.city, "Pittsburgh") + XCTAssertEqual(card.address?.country, "US") + XCTAssertEqual(card.address?.line1, "123 Fake St") + XCTAssertEqual(card.address?.line2, "Apt 1") + XCTAssertEqual(card.address?.state, "PA") + XCTAssertEqual(card.address?.postalCode, "19219") + + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wdeprecated" + + XCTAssertEqual(card.cardId, "card_103kbR2eZvKYlo2CDczLmw4K") + + XCTAssertEqual(card.addressCity, "Pittsburgh") + XCTAssertEqual(card.addressCountry, "US") + XCTAssertEqual(card.addressLine1, "123 Fake St") + XCTAssertEqual(card.addressLine2, "Apt 1") + XCTAssertEqual(card.addressState, "PA") + XCTAssertEqual(card.addressZip, "19219") + XCTAssertNil(card.metadata) + + // #pragma clang diagnostic pop + + XCTAssertEqual(card.brand, .visa) + XCTAssertEqual(card.country, "US") + XCTAssertEqual(card.currency, "usd") + XCTAssertEqual(card.dynamicLast4, "5678") + XCTAssertEqual(card.expMonth, Int(5)) + XCTAssertEqual(card.expYear, Int(2017)) + XCTAssertEqual(card.funding, .credit) + XCTAssertEqual(card.last4, "4242") + XCTAssertEqual(card.name, "Jane Austen") + + XCTAssertEqual(card.allResponseFields as NSDictionary, response as NSDictionary) + } + + // MARK: - STPSourceProtocol Tests + func testStripeID() { + let card = STPFixtures.card() + XCTAssertEqual(card.stripeID, "card_103kbR2eZvKYlo2CDczLmw4K") + } + + // MARK: - STPPaymentOption Tests + func testLabel() { + let card = STPCard.decodedObject(fromAPIResponse: STPTestUtils.jsonNamed("Card"))! + XCTAssertEqual(card.label, "Visa 4242") + } + + // MARK: - + func forEachBrand(_ block: @escaping (_ brand: STPCardBrand) -> Void) { + let values: [STPCardBrand] = [ + .amex, + .dinersClub, + .discover, + .JCB, + .mastercard, + .unionPay, + .visa, + .unknown, + ] + + for brand in values { + block(brand) + } + } +} diff --git a/Stripe/StripeiOSTests/STPCardValidatorTest.swift b/Stripe/StripeiOSTests/STPCardValidatorTest.swift new file mode 100644 index 00000000..ccde57e0 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCardValidatorTest.swift @@ -0,0 +1,471 @@ +// +// STPCardValidatorTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 7/24/15. +// Copyright (c) 2015 Stripe, Inc. All rights reserved. +// + +import UIKit +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCardValidatorTest: XCTestCase { + static let cardData: [(STPCardBrand, String, STPCardValidationState)] = { + return [ + ( + .visa, + "4242424242424242", + .valid + ), + ( + .visa, + "4242424242422", + .incomplete + ), + ( + .visa, + "4012888888881881", + .valid + ), + ( + .visa, + "4000056655665556", + .valid + ), + ( + .mastercard, + "5555555555554444", + .valid + ), + ( + .mastercard, + "5200828282828210", + .valid + ), + ( + .mastercard, + "5105105105105100", + .valid + ), + ( + .mastercard, + "2223000010089800", + .valid + ), + ( + .amex, + "378282246310005", + .valid + ), + ( + .amex, + "371449635398431", + .valid + ), + ( + .discover, + "6011111111111117", + .valid + ), + ( + .discover, + "6011000990139424", + .valid + ), + ( + .dinersClub, + "36227206271667", + .valid + ), + ( + .dinersClub, + "3056930009020004", + .valid + ), + ( + .JCB, + "3530111333300000", + .valid + ), + ( + .JCB, + "3566002020360505", + .valid + ), + ( + .unknown, + "1234567812345678", + .invalid + ), + ] + }() + + func testNumberSanitization() { + let tests = [ + ["4242424242424242", "4242424242424242"], + ["XXXXXX", ""], + ["424242424242424X", "424242424242424"], + ["X4242", "4242"], + ["4242 4242 4242 4242", "4242424242424242"], + ] + for test in tests { + XCTAssertEqual(STPCardValidator.sanitizedNumericString(for: test[0]), test[1]) + } + } + + func testNumberValidation() { + var tests: [(STPCardValidationState, String)] = [] + + for card in STPCardValidatorTest.cardData { + tests.append((card.2, card.1)) + } + + tests.append((.valid, "4242 4242 4242 4242")) + tests.append((.valid, "4136000000008")) + + let badCardNumbers = [ + "0000000000000000", + "9999999999999995", + "1", + "1234123412341234", + "xxx", + "9999999999999999999999", + "42424242424242424242", + "4242-4242-4242-4242", + ] + + for card in badCardNumbers { + tests.append((.invalid, card)) + } + + let possibleCardNumbers = ["4242", "5", "3", "", " ", "6011", "4012888888881"] + + for card in possibleCardNumbers { + tests.append((.incomplete, card)) + } + + for test in tests { + let card = test.1 + let validationState = STPCardValidator.validationState( + forNumber: card, + validatingCardBrand: true + ) + let expected = test.0 + if !(validationState == expected) { + XCTFail("Expected \(expected), got \(validationState) for number \(card)") + } + } + + XCTAssertEqual( + .incomplete, + STPCardValidator.validationState(forNumber: "1", validatingCardBrand: false) + ) + XCTAssertEqual( + .incomplete, + STPCardValidator.validationState( + forNumber: "0000000000000000", + validatingCardBrand: false + ) + ) + XCTAssertEqual( + .incomplete, + STPCardValidator.validationState( + forNumber: "9999999999999995", + validatingCardBrand: false + ) + ) + XCTAssertEqual( + .valid, + STPCardValidator.validationState( + forNumber: "0000000000000000000", + validatingCardBrand: false + ) + ) + XCTAssertEqual( + .valid, + STPCardValidator.validationState( + forNumber: "9999999999999999998", + validatingCardBrand: false + ) + ) + XCTAssertEqual( + .incomplete, + STPCardValidator.validationState(forNumber: "4242424242424", validatingCardBrand: true) + ) + XCTAssertEqual( + .incomplete, + STPCardValidator.validationState(forNumber: nil, validatingCardBrand: true) + ) + } + + func testBrand() { + for test in STPCardValidatorTest.cardData { + XCTAssertEqual(STPCardValidator.brand(forNumber: test.1), test.0) + } + } + + func testLengthsForCardBrand() { + let tests: [(STPCardBrand, Set)] = [ + (.visa, Set([13, 16])), + (.mastercard, Set([16])), + (.amex, Set([15])), + (.discover, Set([16])), + (.dinersClub, Set([14, 16])), + (.JCB, Set([16])), + (.unionPay, Set([16, 19])), + (.unknown, Set([19])), + ] + for test in tests { + let lengths = STPCardValidator.lengths(for: test.0) as NSSet + let expected = test.1 as NSSet + if !lengths.isEqual(expected) { + XCTFail("Invalid lengths for brand \(test.0): expected \(expected), got \(lengths)") + } + } + } + + func testFragmentLength() { + let tests: [(STPCardBrand, Int)] = [ + (.visa, 4), + (.mastercard, 4), + (.amex, 5), + (.discover, 4), + (.dinersClub, 4), + (.JCB, 4), + (.unionPay, 4), + (.unknown, 4), + ] + for test in tests { + XCTAssertEqual(STPCardValidator.fragmentLength(for: test.0), test.1) + } + } + + func testMonthValidation() { + let tests: [(String, STPCardValidationState)] = [ + ("", .incomplete), + ("0", .incomplete), + ("1", .incomplete), + ("2", .valid), + ("9", .valid), + ("10", .valid), + ("12", .valid), + ("13", .invalid), + ("11a", .invalid), + ("x", .invalid), + ("100", .invalid), + ("00", .invalid), + ("13", .invalid), + ] + for test in tests { + XCTAssertEqual(STPCardValidator.validationState(forExpirationMonth: test.0), test.1) + } + } + + func testYearValidation() { + let tests: [(String, String, STPCardValidationState)] = [ + ("12", "15", .valid), + ("8", "15", .valid), + ("9", "15", .valid), + ("11", "16", .valid), + ("11", "99", .valid), + ("01", "99", .valid), + ("1", "99", .valid), + ("00", "99", .invalid), + ("12", "14", .invalid), + ("7", "15", .invalid), + ("12", "00", .invalid), + ("13", "16", .invalid), + ("12", "2", .incomplete), + ("12", "1", .incomplete), + ("12", "0", .incomplete), + ] + + for test in tests { + let state = STPCardValidator.validationState( + forExpirationYear: test.1, + inMonth: test.0, + inCurrentYear: 15, + currentMonth: 8 + ) + XCTAssertEqual(state, test.2) + } + } + + func testCVCLength() { + let tests: [(STPCardBrand, UInt)] = [ + (.visa, 3), + (.mastercard, 3), + (.amex, 4), + (.discover, 3), + (.dinersClub, 3), + (.JCB, 3), + (.unionPay, 3), + (.unknown, 4), + ] + for test in tests { + let maxCVCLength = STPCardValidator.maxCVCLength(for: test.0) + XCTAssertEqual(maxCVCLength, test.1) + } + } + + func testCVCValidation() { + let tests: [(String, STPCardBrand, STPCardValidationState)] = [ + ("x", .visa, .invalid), + ("", .visa, .incomplete), + ("1", .visa, .incomplete), + ("12", .visa, .incomplete), + ("1x3", .visa, .invalid), + ("123", .visa, .valid), + ("123", .amex, .valid), + ("123", .unknown, .valid), + ("1234", .visa, .invalid), + ("1234", .amex, .valid), + ("12345", .amex, .invalid), + ] + + for test in tests { + let state = STPCardValidator.validationState(forCVC: test.0, cardBrand: test.1) + XCTAssertEqual(state, test.2) + } + } + + func testCardValidation() { + // swiftlint:disable:next large_tuple + let tests: [(String, UInt, UInt, String, STPCardValidationState)] = [ + ( + "4242424242424242", + 12, + 15, + "123", + .valid + ), + ( + "4242424242424242", + 12, + 15, + "x", + .invalid + ), + ( + "4242424242424242", + 12, + 15, + "1", + .incomplete + ), + ( + "4242424242424242", + 12, + 14, + "123", + .invalid + ), + ( + "4242424242424242", + 21, + 15, + "123", + .invalid + ), + ( + "42424242", + 12, + 15, + "123", + .incomplete + ), + ( + "378282246310005", + 12, + 15, + "1234", + .valid + ), + ( + "378282246310005", + 12, + 15, + "123", + .valid + ), + ( + "378282246310005", + 12, + 15, + "12345", + .invalid + ), + ( + "1234567812345678", + 12, + 15, + "12345", + .invalid + ), + ] + for test in tests { + let card = STPCardParams() + card.number = test.0 + card.expMonth = test.1 + card.expYear = test.2 + card.cvc = test.3 + let state = STPCardValidator.validationState( + forCard: card, + inCurrentYear: 15, + currentMonth: 8 + ) + if state != test.4 { + XCTFail( + "Wrong validation state for \(String(describing: card.number)). Expected \(test.4), got \(state))" + ) + } + } + } + + func testCBCFetch() { + STPAPIClient.shared.publishableKey = STPTestingDefaultPublishableKey + let mcExp = expectation(description: "Mastercard/CBC") + let visaExp = expectation(description: "Visa/CBC") + let justVisaExp = expectation(description: "Visa Only") + let paramsExp = expectation(description: "Params") + let emptyParamsExp = expectation(description: "Empty Params") + STPCardValidator.possibleBrands(forNumber: "513130") { result in + let brands = try! result.get() + XCTAssertEqual(brands, [.cartesBancaires, .mastercard]) + mcExp.fulfill() + } + STPCardValidator.possibleBrands(forNumber: "455673") { result in + let brands = try! result.get() + XCTAssertEqual(brands, [.cartesBancaires, .visa]) + visaExp.fulfill() + } + STPCardValidator.possibleBrands(forNumber: "424242") { result in + let brands = try! result.get() + XCTAssertEqual(brands, [.visa]) + justVisaExp.fulfill() + } + + let params = STPPaymentMethodCardParams() + params.number = "5131301234" + STPCardValidator.possibleBrands(forCard: params) { result in + let brands = try! result.get() + XCTAssertEqual(brands, [.cartesBancaires, .mastercard]) + paramsExp.fulfill() + } + + let paramsEmpty = STPPaymentMethodCardParams() + STPCardValidator.possibleBrands(forCard: paramsEmpty) { result in + let brands = try! result.get() + XCTAssertEqual(brands, Set(STPCardBrand.allCases)) + emptyParamsExp.fulfill() + } + + wait(for: [mcExp, visaExp, justVisaExp, paramsExp, emptyParamsExp], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPCertTest.swift b/Stripe/StripeiOSTests/STPCertTest.swift new file mode 100644 index 00000000..5280f9fe --- /dev/null +++ b/Stripe/StripeiOSTests/STPCertTest.swift @@ -0,0 +1,68 @@ +// +// STPCertTest.swift +// StripeiOS Tests +// +// Created by Phillip Cohen on 4/14/14. +// Copyright © 2014 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +let STPExamplePublishableKey = "bad_key" + +class STPCertTest: XCTestCase { + func testNoError() { + let expectation = self.expectation(description: "Token creation") + let client = STPAPIClient(publishableKey: STPExamplePublishableKey) + client.createToken( + withParameters: [:]) { token, error in + expectation.fulfill() + // Note that this API request *will* fail, but it will return error + // messages from the server and not be blocked by local cert checks + XCTAssertNil(token, "Expected no token") + XCTAssertNotNil(error, "Expected error") + } + waitForExpectations(timeout: 20.0, handler: nil) + } + + func testExpired() { + createToken( + withBaseURL: URL(string: "https://expired.badssl.com/") + ) { token, error in + XCTAssertNil(token, "Token should be nil.") + XCTAssertEqual((error as NSError?)?.domain, "NSURLErrorDomain") + XCTAssertNotNil( + (error as NSError?)?.userInfo["NSURLErrorFailingURLPeerTrustErrorKey"], + "There should be a secTustRef for Foundation HTTPS errors" + ) + } + } + + func testMismatched() { + createToken( + withBaseURL: URL(string: "https://mismatched.stripe.com") + ) { token, error in + XCTAssertNil(token, "Token should be nil.") + XCTAssertEqual((error as NSError?)?.domain, "NSURLErrorDomain") + } + } + + // helper method + func createToken(withBaseURL baseURL: URL?, completion: @escaping STPTokenCompletionBlock) { + let expectation = self.expectation(description: "Token creation") + let client = STPAPIClient(publishableKey: STPExamplePublishableKey) + client.apiURL = baseURL + client.createToken( + withParameters: [:]) { token, error in + expectation.fulfill() + completion(token, error) + } + waitForExpectations(timeout: 20.0, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPConfirmCardOptionsTest.m b/Stripe/StripeiOSTests/STPConfirmCardOptionsTest.m new file mode 100644 index 00000000..202f8053 --- /dev/null +++ b/Stripe/StripeiOSTests/STPConfirmCardOptionsTest.m @@ -0,0 +1,36 @@ +// +// STPConfirmCardOptionsTest.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 1/10/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPConfirmCardOptionsTest : XCTestCase + +@end + +@implementation STPConfirmCardOptionsTest + +- (void)testCVC { + STPConfirmCardOptions *cardOptions = [[STPConfirmCardOptions alloc] init]; + + XCTAssertNil(cardOptions.cvc, @"Initial/default value should be nil."); + XCTAssertNil(cardOptions.network, @"Initial/default value should be nil."); + + cardOptions.cvc = @"123"; + XCTAssertEqualObjects(cardOptions.cvc, @"123", @"cvc should be set to '123'."); + cardOptions.network = @"visa"; + XCTAssertEqualObjects(cardOptions.network, @"visa", @"network should be set to 'visa'."); +} + +- (void)testEncoding { + NSDictionary *propertyMap = [STPConfirmCardOptions propertyNamesToFormFieldNamesMapping]; + NSDictionary *expected = @{@"cvc": @"cvc", @"network": @"network"}; + XCTAssertEqualObjects(propertyMap, expected, @"Unexpected property to field name mapping."); +} + +@end diff --git a/Stripe/StripeiOSTests/STPConfirmPaymentMethodOptionsTest.m b/Stripe/StripeiOSTests/STPConfirmPaymentMethodOptionsTest.m new file mode 100644 index 00000000..03f8cb21 --- /dev/null +++ b/Stripe/StripeiOSTests/STPConfirmPaymentMethodOptionsTest.m @@ -0,0 +1,40 @@ +// +// STPConfirmPaymentMethodOptionsTest.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 1/10/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPConfirmPaymentMethodOptionsTest : XCTestCase + +@end + +@implementation STPConfirmPaymentMethodOptionsTest + +- (void)testCardOptions { + STPConfirmPaymentMethodOptions *paymentMethodOptions = [[STPConfirmPaymentMethodOptions alloc] init]; + + XCTAssertNil(paymentMethodOptions.cardOptions, @"Default card value should be nil."); + + STPConfirmCardOptions *cardOptions = [[STPConfirmCardOptions alloc] init]; + paymentMethodOptions.cardOptions = cardOptions; + XCTAssertEqual(paymentMethodOptions.cardOptions, cardOptions, @"Should hold reference to set cardOptions."); +} + +- (void)testFormEncoding { + NSDictionary *propertyToFieldMap = [STPConfirmPaymentMethodOptions propertyNamesToFormFieldNamesMapping]; + NSDictionary *expected = @{@"cardOptions": @"card", + @"alipayOptions": @"alipay", + @"blikOptions": @"blik", + @"weChatPayOptions": @"wechat_pay", + @"usBankAccountOptions": @"us_bank_account", + }; + + XCTAssertEqualObjects(propertyToFieldMap, expected, @"Unexpected property to field name mapping."); +} + +@end diff --git a/Stripe/StripeiOSTests/STPConnectAccountAddressTest.m b/Stripe/StripeiOSTests/STPConnectAccountAddressTest.m new file mode 100644 index 00000000..cf741582 --- /dev/null +++ b/Stripe/StripeiOSTests/STPConnectAccountAddressTest.m @@ -0,0 +1,42 @@ +// +// STPConnectAccountAddressTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 8/2/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPConnectAccountAddressTest : XCTestCase + +@end + +@implementation STPConnectAccountAddressTest + +#pragma mark STPFormEncodable Tests + +- (void)testRootObjectName { + XCTAssertNil([STPConnectAccountAddress rootObjectName]); +} + +- (void)testPropertyNamesToFormFieldNamesMapping { + STPConnectAccountAddress *address = [STPConnectAccountAddress new]; + + NSDictionary *mapping = [STPConnectAccountAddress propertyNamesToFormFieldNamesMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([address respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPConnectAccountFunctionalTest.m b/Stripe/StripeiOSTests/STPConnectAccountFunctionalTest.m new file mode 100644 index 00000000..0a7e539f --- /dev/null +++ b/Stripe/StripeiOSTests/STPConnectAccountFunctionalTest.m @@ -0,0 +1,82 @@ +// +// STPConnectAccountFunctionalTest.m +// StripeiOS Tests +// +// Created by Daniel Jackson on 1/8/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +@import StripeCore; +#import "STPFixtures.h" +#import "STPTestingAPIClient.h" + +@interface STPConnectAccountFunctionalTest : XCTestCase + +/// Client with test publishable key +@property (nonatomic, strong, nonnull) STPAPIClient *client; +@property (nonatomic, strong, nonnull) STPConnectAccountIndividualParams *individual; +@property (nonatomic, strong, nonnull) STPConnectAccountCompanyParams *company; + +@end + +@implementation STPConnectAccountFunctionalTest + +- (void)setUp { + [super setUp]; + + self.client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + self.individual = [STPConnectAccountIndividualParams new]; + self.individual.firstName = @"Test"; + NSDateComponents *dob = [NSDateComponents new]; + dob.day = 31; + dob.month = 8; + dob.year = 2006; + self.individual.dateOfBirth = dob; + self.company = [STPConnectAccountCompanyParams new]; + self.company.name = @"Test"; +} + +- (void)testTokenCreation_terms_nil { + XCTAssertNil([[STPConnectAccountParams alloc] initWithTosShownAndAccepted:NO + individual:self.individual], + @"Guard to prevent trying to call this with `NO`"); + XCTAssertNil([[STPConnectAccountParams alloc] initWithTosShownAndAccepted:NO + company:self.company], + @"Guard to prevent trying to call this with `NO`"); +} + +- (void)testTokenCreation_customer { + [self createToken:[[STPConnectAccountParams alloc] initWithCompany:self.company] + shouldSucceed:YES]; +} + +- (void)testTokenCreation_company { + [self createToken:[[STPConnectAccountParams alloc] initWithIndividual:self.individual] + shouldSucceed:YES]; +} + +#pragma mark - + +- (void)createToken:(STPConnectAccountParams *)params shouldSucceed:(BOOL)shouldSucceed { + XCTestExpectation *expectation = [self expectationWithDescription:@"Connect Account Token"]; + + [self.client createTokenWithConnectAccount:params completion:^(STPToken * _Nullable token, NSError * _Nullable error) { + [expectation fulfill]; + + if (shouldSucceed) { + XCTAssertNil(error); + XCTAssertNotNil(token); + XCTAssertNotNil(token.tokenId); + XCTAssertEqual(token.type, STPTokenTypeAccount); + } else { + XCTAssertNil(token); + XCTAssertNotNil(error); + } + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPConnectAccountParamsTest.m b/Stripe/StripeiOSTests/STPConnectAccountParamsTest.m new file mode 100644 index 00000000..70cccea6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPConnectAccountParamsTest.m @@ -0,0 +1,62 @@ +// +// STPConnectAccountParamsTest.m +// StripeiOS Tests +// +// Created by Daniel Jackson on 1/10/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPConnectAccountParams (Testing) ++ (NSString *)stringFromBusinessType:(STPConnectAccountBusinessType)businessType; +@end + +@interface STPConnectAccountParamsTest : XCTestCase +@end + +@implementation STPConnectAccountParamsTest + +#pragma mark - STPFormEncodable Tests + +- (void)testRootObjectName { + XCTAssertEqualObjects([STPConnectAccountParams rootObjectName], @"account"); +} + +- (void)testBusinessType { + STPConnectAccountIndividualParams *individual = [STPConnectAccountIndividualParams new]; + STPConnectAccountCompanyParams *company = [STPConnectAccountCompanyParams new]; + + XCTAssertEqual([[STPConnectAccountParams alloc] initWithIndividual:individual].businessType, STPConnectAccountBusinessTypeIndividual); + XCTAssertEqual([[STPConnectAccountParams alloc] initWithTosShownAndAccepted:YES individual:individual].businessType, STPConnectAccountBusinessTypeIndividual); + + XCTAssertEqual([[STPConnectAccountParams alloc] initWithCompany:company].businessType, STPConnectAccountBusinessTypeCompany); + XCTAssertEqual([[STPConnectAccountParams alloc] initWithTosShownAndAccepted:YES company:company].businessType, STPConnectAccountBusinessTypeCompany); +} + +- (void)testBusinessTypeString { + XCTAssertEqualObjects(@"individual", [STPConnectAccountParams stringFromBusinessType:STPConnectAccountBusinessTypeIndividual]); + XCTAssertEqualObjects(@"company", [STPConnectAccountParams stringFromBusinessType:STPConnectAccountBusinessTypeCompany]); +} + +- (void)testPropertyNamesToFormFieldNamesMapping { + STPConnectAccountIndividualParams *individual = [STPConnectAccountIndividualParams new]; + STPConnectAccountParams *accountParams = [[STPConnectAccountParams alloc] initWithIndividual:individual]; + + NSDictionary *mapping = [STPConnectAccountParams propertyNamesToFormFieldNamesMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([accountParams respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPCountryPickerInputFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPCountryPickerInputFieldSnapshotTests.swift new file mode 100644 index 00000000..00d4df24 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCountryPickerInputFieldSnapshotTests.swift @@ -0,0 +1,31 @@ +// +// STPCountryPickerInputFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 12/2/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPCountryPickerInputFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testDefault() { + let field = STPCountryPickerInputField() + field.sizeToFit() + field.frame.size.width = 200 + + STPSnapshotVerifyView(field) + } +} diff --git a/Stripe/StripeiOSTests/STPCustomerContextTest.swift b/Stripe/StripeiOSTests/STPCustomerContextTest.swift new file mode 100644 index 00000000..f4e11197 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCustomerContextTest.swift @@ -0,0 +1,676 @@ +// +// STPCustomerContextTest.swift +// StripeiOS Tests +// +// Created by David Estes on 9/20/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import OHHTTPStubs +import OHHTTPStubsSwift +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class MockEphemeralKeyManager: STPEphemeralKeyManagerProtocol { + var ephemeralKey: STPEphemeralKey? + var error: Error? + + init( + key: STPEphemeralKey?, + error: Error? + ) { + self.ephemeralKey = key + self.error = error + } + + func getOrCreateKey(_ completion: @escaping STPEphemeralKeyCompletionBlock) { + completion(ephemeralKey, error) + } +} + +class STPCustomerContextTests: APIStubbedTestCase { + func stubRetrieveCustomers( + key: STPEphemeralKey, + returningCustomerJSON: [AnyHashable: Any], + expectedCount: Int, + apiClient: STPAPIClient + ) { + let exp = expectation(description: "retrieveCustomer") + exp.expectedFulfillmentCount = expectedCount + + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/customers") ?? false + && urlRequest.httpMethod == "GET" + } response: { _ in + DispatchQueue.main.async { + // Fulfill after response is sent + exp.fulfill() + } + return HTTPStubsResponse( + jsonObject: returningCustomerJSON, + statusCode: 200, + headers: nil + ) + } + } + + func stubListPaymentMethods( + key: STPEphemeralKey, + paymentMethodJSONs: [[AnyHashable: Any]], + expectedCount: Int, + apiClient: STPAPIClient + ) { + let exp = expectation(description: "listPaymentMethod") + exp.expectedFulfillmentCount = expectedCount + stub { urlRequest in + if urlRequest.url?.absoluteString.contains("/payment_methods") ?? false + && urlRequest.httpMethod == "GET" + { + // Check to make sure we pass the ephemeral key correctly + let keyFromHeader = urlRequest.allHTTPHeaderFields!["Authorization"]? + .replacingOccurrences(of: "Bearer ", with: "") + XCTAssertEqual(keyFromHeader, key.secret) + return true + } + return false + } response: { _ in + let paymentMethodsJSON = """ + { + "object": "list", + "url": "/v1/payment_methods", + "has_more": false, + "data": [ + ] + } + """ + var pmList = + try! JSONSerialization.jsonObject( + with: paymentMethodsJSON.data(using: .utf8)!, + options: [] + ) as! [AnyHashable: Any] + pmList["data"] = paymentMethodJSONs + DispatchQueue.main.async { + // Fulfill after response is sent + exp.fulfill() + } + return HTTPStubsResponse(jsonObject: pmList, statusCode: 200, headers: nil) + } + } + + func testGetOrCreateKeyErrorForwardedToRetrieveCustomer() { + let exp = expectation(description: "retrieveCustomer") + let expectedError = NSError(domain: "test", code: 123, userInfo: nil) + let apiClient = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/customers") ?? false + } response: { _ in + XCTFail("Retrieve customer should not be called") + return HTTPStubsResponse(error: NSError(domain: "test", code: 100, userInfo: nil)) + } + let ekm = MockEphemeralKeyManager(key: nil, error: expectedError) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + sut.retrieveCustomer { customer, error in + XCTAssertNil(customer) + XCTAssertEqual((error as NSError?)?.domain, expectedError.domain) + exp.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testInitRetrievesResourceKeyAndCustomerAndPaymentMethods() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let apiClient = stubbedAPIClient() + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: [], + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + XCTAssertNotNil(sut) + waitForExpectations(timeout: 2, handler: nil) + } + + func testRetrieveCustomerUsesCachedCustomerIfNotExpired() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomer = STPFixtures.customerWithSingleCardTokenSource() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let apiClient = stubbedAPIClient() + + // apiClient.retrieveCustomer should be called once, when the context is initialized. + // When sut.retrieveCustomer is called below, the cached customer will be used. + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: [], + expectedCount: 1, + apiClient: apiClient + ) + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then check cache + waitForExpectations(timeout: 2, handler: nil) + let exp2 = expectation(description: "retrieveCustomer again") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.retrieveCustomer { customer, _ in + XCTAssertEqual(customer!.stripeID, expectedCustomer.stripeID) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testRetrieveCustomerDoesNotUseCachedCustomerIfExpired() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomer = STPFixtures.customerWithSingleCardTokenSource() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let apiClient = stubbedAPIClient() + + // apiClient.retrieveCustomer should be called twice: + // - when the context is initialized, + // - when sut.retrieveCustomer is called below, as the cached customer has expired. + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 2, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: [], + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then reset and check cache + let exp2 = expectation(description: "retrieveCustomer again") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.customerRetrievedDate = Date(timeIntervalSinceNow: -70) + sut.retrieveCustomer { customer, _ in + XCTAssertEqual(customer!.stripeID, expectedCustomer.stripeID) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testRetrieveCustomerDoesNotUseCachedCustomerAfterClearingCache() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomer = STPFixtures.customerWithSingleCardTokenSource() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let apiClient = stubbedAPIClient() + + // apiClient.retrieveCustomer should be called twice: + // - when the context is initialized, + // - when sut.retrieveCustomer is called below, as the cached customer has been cleared. + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 2, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: [], + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then reset and check cache + let exp2 = expectation(description: "retrieveCustomer again") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.clearCache() + sut.retrieveCustomer { customer, _ in + XCTAssertEqual(customer!.stripeID, expectedCustomer.stripeID) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testRetrievePaymentMethodsUsesCacheIfNotExpired() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let expectedPaymentMethods = [STPFixtures.paymentMethod()] + let expectedPaymentMethodsJSON = [STPFixtures.paymentMethodJSON()] + let apiClient = stubbedAPIClient() + + // apiClient.listPaymentMethods should be called once, when the context is initialized. + // When sut.listPaymentMethods is called below, the cached list will be used. + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: expectedPaymentMethodsJSON, + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then check cache + let exp2 = expectation(description: "listPaymentMethods") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.listPaymentMethodsForCustomer { paymentMethods, _ in + XCTAssertEqual(paymentMethods!.count, expectedPaymentMethods.count) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testRetrievePaymentMethodsDoesNotUseCacheIfExpired() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let expectedPaymentMethods = [STPFixtures.paymentMethod()] + let expectedPaymentMethodsJSON = [STPFixtures.paymentMethodJSON()] + let apiClient = stubbedAPIClient() + + // apiClient.listPaymentMethods should be called twice: + // - when the context is initialized, + // - when sut.listPaymentMethods is called below, as the cached list has expired. + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: expectedPaymentMethodsJSON, + expectedCount: 2, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then check cache + let exp2 = expectation(description: "listPaymentMethods") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.paymentMethodsRetrievedDate = Date(timeIntervalSinceNow: -70) + sut.listPaymentMethodsForCustomer { paymentMethods, _ in + XCTAssertEqual(paymentMethods!.count, expectedPaymentMethods.count) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testRetrievePaymentMethodsDoesNotUseCacheAfterClearingCache() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let expectedPaymentMethods = [STPFixtures.paymentMethod()] + let expectedPaymentMethodsJSON = [STPFixtures.paymentMethodJSON()] + let apiClient = stubbedAPIClient() + + // apiClient.listPaymentMethods should be called twice: + // - when the context is initialized, + // - when sut.listPaymentMethods is called below, as the cached list has been cleared + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: expectedPaymentMethodsJSON, + expectedCount: 2, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then check cache + let exp2 = expectation(description: "listPaymentMethods") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.clearCache() + sut.listPaymentMethodsForCustomer { paymentMethods, _ in + XCTAssertEqual(paymentMethods!.count, expectedPaymentMethods.count) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testSetCustomerShippingCallsAPIClientCorrectly() { + let address = STPFixtures.address() + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let apiClient = stubbedAPIClient() + + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: [], + expectedCount: 1, + apiClient: apiClient + ) + + let exp = expectation(description: "updateCustomer") + stub { urlRequest in + if urlRequest.url?.absoluteString.contains("/customers") ?? false + && urlRequest.httpMethod == "POST" + { + let state = urlRequest.queryItems?.first(where: { item in + item.name == "shipping[address][state]" + })! + XCTAssertEqual(state?.value, address.state) + return true + } + return false + } response: { _ in + exp.fulfill() + return HTTPStubsResponse( + jsonObject: expectedCustomerJSON, + statusCode: 200, + headers: nil + ) + } + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + let exp2 = expectation(description: "updateCustomerWithShipping") + sut.updateCustomer(withShippingAddress: address) { error in + XCTAssertNil(error) + exp2.fulfill() + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testAttachPaymentMethodCallsAPIClientCorrectly() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let apiClient = stubbedAPIClient() + let expectedPaymentMethod = STPFixtures.paymentMethod() + let expectedPaymentMethodJSON = STPFixtures.paymentMethodJSON() + let expectedPaymentMethods = [STPFixtures.paymentMethod()] + + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: [], + expectedCount: 1, + apiClient: apiClient + ) + + let exp = expectation(description: "payment method attach") + // We're attaching 2 payment methods: + exp.expectedFulfillmentCount = 2 + stub { urlRequest in + if urlRequest.url?.absoluteString.contains("/payment_method") ?? false + && urlRequest.httpMethod == "POST" + { + return true + } + return false + } response: { _ in + exp.fulfill() + return HTTPStubsResponse( + jsonObject: expectedPaymentMethodJSON, + statusCode: 200, + headers: nil + ) + } + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + let exp2 = expectation(description: "CustomerContext attachPaymentMethod") + sut.attachPaymentMethod(toCustomer: expectedPaymentMethods.first!) { error in + XCTAssertNil(error) + exp2.fulfill() + } + + let exp3 = expectation(description: "CustomerContext attachPaymentMethod with ID") + sut.attachPaymentMethodToCustomer(paymentMethodId: expectedPaymentMethod.stripeId) { + error in + XCTAssertNil(error) + exp3.fulfill() + } + + waitForExpectations(timeout: 2, handler: nil) + } + + func testDetachPaymentMethodCallsAPIClientCorrectly() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let apiClient = stubbedAPIClient() + let expectedPaymentMethod = STPFixtures.paymentMethod() + let expectedPaymentMethodJSON = STPFixtures.paymentMethodJSON() + let expectedPaymentMethods = [STPFixtures.paymentMethod()] + + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: [], + expectedCount: 1, + apiClient: apiClient + ) + + let exp = expectation(description: "payment method detach") + // We're detaching 2 payment methods: + exp.expectedFulfillmentCount = 2 + stub { urlRequest in + if urlRequest.url?.absoluteString.contains("/payment_method") ?? false + && urlRequest.httpMethod == "POST" + { + return true + } + return false + } response: { _ in + exp.fulfill() + return HTTPStubsResponse( + jsonObject: expectedPaymentMethodJSON, + statusCode: 200, + headers: nil + ) + } + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + let exp2 = expectation(description: "CustomerContext detachPaymentMethod") + sut.detachPaymentMethod(fromCustomer: expectedPaymentMethods.first!) { error in + XCTAssertNil(error) + exp2.fulfill() + } + + let exp3 = expectation(description: "CustomerContext detachPaymentMethod with ID") + sut.detachPaymentMethodFromCustomer(paymentMethodId: expectedPaymentMethod.stripeId) { + error in + XCTAssertNil(error) + exp3.fulfill() + } + + waitForExpectations(timeout: 2, handler: nil) + } + + func testFiltersApplePayPaymentMethodsByDefault() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let expectedPaymentMethodsJSON = [ + STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), + ] + let apiClient = stubbedAPIClient() + + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: expectedPaymentMethodsJSON, + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then check cache + let exp2 = expectation(description: "listPaymentMethods") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.listPaymentMethodsForCustomer { paymentMethods, _ in + // Apple Pay should be filtered out + XCTAssertEqual(paymentMethods!.count, 1) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testIncludesApplePayPaymentMethods() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() + let expectedPaymentMethodsJSON = [ + STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), + ] + let apiClient = stubbedAPIClient() + + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: expectedPaymentMethodsJSON, + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + sut.includeApplePayPaymentMethods = true + // Give the mocked API request a little time to complete and cache the customer, then check cache + let exp2 = expectation(description: "listPaymentMethods") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.listPaymentMethodsForCustomer { paymentMethods, _ in + // Apple Pay should be included + XCTAssertEqual(paymentMethods!.count, 2) + exp2.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testFiltersApplePaySourcesByDefault() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithCardAndApplePaySourcesJSON() + let expectedPaymentMethods = [ + STPFixtures.paymentMethod(), STPFixtures.applePayPaymentMethod(), + ] + let expectedPaymentMethodsJSON = [ + STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), + ] + let apiClient = stubbedAPIClient() + + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: expectedPaymentMethodsJSON, + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + // Give the mocked API request a little time to complete and cache the customer, then check cache + let exp = expectation(description: "retrieveCustomer") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.retrieveCustomer { customer, _ in + // Apple Pay should be filtered out + XCTAssertEqual(customer!.sources.count, 1) + exp.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + func testIncludeApplePaySources() { + let customerKey = STPFixtures.ephemeralKey() + let expectedCustomerJSON = STPFixtures.customerWithCardAndApplePaySourcesJSON() + let expectedPaymentMethodsJSON = [ + STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), + ] + let apiClient = stubbedAPIClient() + + stubRetrieveCustomers( + key: customerKey, + returningCustomerJSON: expectedCustomerJSON, + expectedCount: 1, + apiClient: apiClient + ) + stubListPaymentMethods( + key: customerKey, + paymentMethodJSONs: expectedPaymentMethodsJSON, + expectedCount: 1, + apiClient: apiClient + ) + + let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) + let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) + sut.includeApplePayPaymentMethods = true + // Give the mocked API request a little time to complete and cache the customer, then check cache + let exp = expectation(description: "retrieveCustomer") + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { + sut.retrieveCustomer { customer, _ in + // Apple Pay should be filtered out + XCTAssertEqual(customer!.sources.count, 2) + exp.fulfill() + } + } + waitForExpectations(timeout: 2, handler: nil) + } + + let ohhttpDelay = 0.1 +} diff --git a/Stripe/StripeiOSTests/STPCustomerTest.m b/Stripe/StripeiOSTests/STPCustomerTest.m new file mode 100644 index 00000000..a55e8b58 --- /dev/null +++ b/Stripe/StripeiOSTests/STPCustomerTest.m @@ -0,0 +1,68 @@ +// +// STPCustomerTest.m +// Stripe +// +// Created by Ben Guo on 7/14/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import + + + +#import "STPTestUtils.h" + +@interface STPCustomerTest : XCTestCase +@end + +@implementation STPCustomerTest + +- (void)testDecoding_invalidJSON { + STPCustomer *sut = [STPCustomer decodedObjectFromAPIResponse:@{}]; + XCTAssertNil(sut); +} + +- (void)testDecoding_validJSON { + NSMutableDictionary *card1 = [[STPTestUtils jsonNamed:@"Card"] mutableCopy]; + card1[@"id"] = @"card_123"; + + NSMutableDictionary *card2 = [[STPTestUtils jsonNamed:@"Card"] mutableCopy]; + card2[@"id"] = @"card_456"; + + NSMutableDictionary *applePayCard1 = [[STPTestUtils jsonNamed:@"Card"] mutableCopy]; + applePayCard1[@"id"] = @"card_apple_pay1"; + applePayCard1[@"tokenization_method"] = @"apple_pay"; + + NSMutableDictionary *applePayCard2 = [applePayCard1 mutableCopy]; + applePayCard2[@"id"] = @"card_apple_pay2"; + + NSDictionary *cardSource = [STPTestUtils jsonNamed:@"CardSource"]; + NSDictionary *threeDSSource = [STPTestUtils jsonNamed:@"3DSSource"]; + + NSMutableDictionary *customer = [[STPTestUtils jsonNamed:@"Customer"] mutableCopy]; + NSMutableDictionary *sources = [customer[@"sources"] mutableCopy]; + sources[@"data"] = @[applePayCard1, card1, applePayCard2, card2, cardSource, threeDSSource]; + customer[@"default_source"] = card1[@"id"]; + customer[@"sources"] = sources; + + STPCustomer *sut = [STPCustomer decodedObjectFromAPIResponse:customer]; + XCTAssertNotNil(sut); + XCTAssertEqualObjects(sut.stripeID, customer[@"id"]); + XCTAssertTrue(sut.sources.count == 4); + XCTAssertEqualObjects(sut.sources[0].stripeID, card1[@"id"]); + XCTAssertEqualObjects(sut.sources[1].stripeID, card2[@"id"]); + XCTAssertEqualObjects(sut.defaultSource.stripeID, card1[@"id"]); + XCTAssertEqualObjects(sut.sources[2].stripeID, cardSource[@"id"]); + XCTAssertEqualObjects(sut.sources[3].stripeID, threeDSSource[@"id"]); + + XCTAssertEqualObjects(sut.shippingAddress.name, customer[@"shipping"][@"name"]); + XCTAssertEqualObjects(sut.shippingAddress.phone, customer[@"shipping"][@"phone"]); + XCTAssertEqualObjects(sut.shippingAddress.city, customer[@"shipping"][@"address"][@"city"]); + XCTAssertEqualObjects(sut.shippingAddress.country, customer[@"shipping"][@"address"][@"country"]); + XCTAssertEqualObjects(sut.shippingAddress.line1, customer[@"shipping"][@"address"][@"line1"]); + XCTAssertEqualObjects(sut.shippingAddress.line2, customer[@"shipping"][@"address"][@"line2"]); + XCTAssertEqualObjects(sut.shippingAddress.postalCode, customer[@"shipping"][@"address"][@"postal_code"]); + XCTAssertEqualObjects(sut.shippingAddress.state, customer[@"shipping"][@"address"][@"state"]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPE2ETest.swift b/Stripe/StripeiOSTests/STPE2ETest.swift new file mode 100644 index 00000000..2a55aef0 --- /dev/null +++ b/Stripe/StripeiOSTests/STPE2ETest.swift @@ -0,0 +1,155 @@ +// +// STPE2ETest.swift +// StripeiOS Tests +// +// Created by David Estes on 2/16/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Stripe +import XCTest + +class STPE2ETest: XCTestCase { + let E2ETestTimeout: TimeInterval = 120 + + struct E2EExpectation { + var amount: Int + var currency: String + var accountID: String + } + + class E2EBackend { + static let backendAPIURL = URL( + string: "https://stp-mobile-ci-test-backend-e1b3.stripedemos.com/e2e" + )! + + func createPaymentIntent( + completion: @escaping (STPPaymentIntentParams, E2EExpectation) -> Void + ) { + requestAPI("create_pi", method: "POST") { (json) in + let paymentIntentClientSecret = json["paymentIntent"] as! String + let expectedAmount = json["expectedAmount"] as! Int + let expectedCurrency = json["expectedCurrency"] as! String + let expectedAccountID = json["expectedAccountID"] as! String + let publishableKey = json["publishableKey"] as! String + STPAPIClient.shared.publishableKey = publishableKey + completion( + STPPaymentIntentParams(clientSecret: paymentIntentClientSecret), + E2EExpectation( + amount: expectedAmount, + currency: expectedCurrency, + accountID: expectedAccountID + ) + ) + } + } + + func fetchPaymentIntent(id: String, completion: @escaping (E2EExpectation) -> Void) { + requestAPI("fetch_pi", queryItems: [URLQueryItem(name: "pi", value: id)]) { (json) in + let resultAmount = json["amount"] as! Int + let resultCurrency = json["currency"] as! String + let resultAccountID = json["on_behalf_of"] as! String + completion( + E2EExpectation( + amount: resultAmount, + currency: resultCurrency, + accountID: resultAccountID + ) + ) + } + } + + private func requestAPI( + _ resource: String, + method: String = "GET", + queryItems: [URLQueryItem] = [], + completion: @escaping ([String: Any]) -> Void + ) { + var url = URLComponents( + url: Self.backendAPIURL.appendingPathComponent(resource), + resolvingAgainstBaseURL: false + )! + url.queryItems = queryItems + var request = URLRequest(url: url.url!) + request.httpMethod = method + let task = URLSession.shared.dataTask( + with: request, + completionHandler: { (data, _, error) in + guard let data = data, + let json = try? JSONSerialization.jsonObject(with: data, options: []) + as? [String: Any] + else { + XCTFail( + "Did not receive valid JSON response from E2E server. \(String(describing: error))" + ) + return + } + DispatchQueue.main.async { + completion(json) + } + } + ) + task.resume() + } + } + + static let TestPM: STPPaymentMethodParams = { + let testCard = STPPaymentMethodCardParams() + testCard.number = "4242424242424242" + testCard.expYear = 2050 + testCard.expMonth = 12 + testCard.cvc = "123" + return STPPaymentMethodParams(card: testCard, billingDetails: nil, metadata: nil) + }() + + // MARK: LOG.04.01c + // In this test, a PaymentIntent object is created from an example merchant backend, + // confirmed by the iOS SDK, and then retrieved to validate that the original amount, + // currency, and merchant are the same as the original inputs. + func testE2E() throws { + continueAfterFailure = false + let backend = E2EBackend() + let createPI = XCTestExpectation(description: "Create PaymentIntent") + let fetchPIBackend = XCTestExpectation( + description: "Fetch and check PaymentIntent via backend" + ) + let fetchPIClient = XCTestExpectation( + description: "Fetch and check PaymentIntent via client" + ) + let confirmPI = XCTestExpectation(description: "Confirm PaymentIntent") + + // Create a PaymentIntent + backend.createPaymentIntent { (pip, expected) in + createPI.fulfill() + + // Confirm the PaymentIntent using a test card + pip.paymentMethodParams = STPE2ETest.TestPM + STPAPIClient.shared.confirmPaymentIntent(with: pip) { (confirmedPI, confirmError) in + confirmPI.fulfill() + XCTAssertNotNil(confirmedPI) + XCTAssertNil(confirmError) + + // Check the PI information using the backend + backend.fetchPaymentIntent(id: pip.stripeId!) { (expectationResult) in + XCTAssertEqual(expectationResult.amount, expected.amount) + XCTAssertEqual(expectationResult.accountID, expected.accountID) + XCTAssertEqual(expectationResult.currency, expected.currency) + fetchPIBackend.fulfill() + } + + // Check the PI information using the client + STPAPIClient.shared.retrievePaymentIntent(withClientSecret: pip.clientSecret) { + (fetchedPI, fetchError) in + XCTAssertNil(fetchError) + let fetchedPI = fetchedPI! + XCTAssertEqual(fetchedPI.status, .succeeded) + XCTAssertEqual(fetchedPI.amount, expected.amount) + XCTAssertEqual(fetchedPI.currency, expected.currency) + // The client can't check the "on_behalf_of" field, so we check it via the merchant test above. + fetchPIClient.fulfill() + } + } + } + wait(for: [createPI, confirmPI, fetchPIBackend, fetchPIClient], timeout: E2ETestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPElementsSessionTest.swift b/Stripe/StripeiOSTests/STPElementsSessionTest.swift new file mode 100644 index 00000000..6a36bc9a --- /dev/null +++ b/Stripe/StripeiOSTests/STPElementsSessionTest.swift @@ -0,0 +1,54 @@ +// +// STPElementsSessionTest.swift +// StripeiOSTests +// +// Created by Nick Porter on 2/16/23. +// + +import Foundation +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet + +class STPElementsSessionTest: XCTestCase { + + // MARK: - Description Tests + func testDescription() { + let elementsSessionJson = STPTestUtils.jsonNamed("ElementsSession")! + let elementsSession = STPElementsSession.decodedObject(fromAPIResponse: elementsSessionJson)! + + XCTAssertNotNil(elementsSession) + let desc = elementsSession.description + XCTAssertTrue(desc.contains(NSStringFromClass(type(of: elementsSession).self))) + XCTAssertGreaterThan((desc.count), 500, "Custom description should be long") + } + + // MARK: - STPAPIResponseDecodable Tests + func testDecodedObjectFromAPIResponseMapping() { + let elementsSessionJson = STPTestUtils.jsonNamed("ElementsSession")! + let elementsSession = STPElementsSession.decodedObject(fromAPIResponse: elementsSessionJson)! + + XCTAssertEqual( + elementsSession.orderedPaymentMethodTypes, + [ + STPPaymentMethodType.card, + STPPaymentMethodType.link, + STPPaymentMethodType.USBankAccount, + STPPaymentMethodType.afterpayClearpay, + STPPaymentMethodType.klarna, + STPPaymentMethodType.cashApp, + STPPaymentMethodType.alipay, + STPPaymentMethodType.weChatPay, + ] + ) + + XCTAssertEqual( + elementsSession.unactivatedPaymentMethodTypes, + [STPPaymentMethodType.cashApp] + ) + + XCTAssertNotNil(elementsSession.linkSettings) + XCTAssertEqual(elementsSession.countryCode, "US") + XCTAssertNotNil(elementsSession.paymentMethodSpecs) + } + +} diff --git a/Stripe/StripeiOSTests/STPEphemeralKeyManagerTest.m b/Stripe/StripeiOSTests/STPEphemeralKeyManagerTest.m new file mode 100644 index 00000000..b64b5d2d --- /dev/null +++ b/Stripe/StripeiOSTests/STPEphemeralKeyManagerTest.m @@ -0,0 +1,185 @@ +// +// STPEphemeralKeyManagerTest.m +// Stripe +// +// Created by Ben Guo on 5/9/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import +@import Stripe; + + +#import "STPFixtures.h" + +@interface STPEphemeralKeyManager (Testing) +@property (nonatomic) STPEphemeralKey *ephemeralKey; +@property (nonatomic) NSDate *lastEagerKeyRefresh; +@end + +@interface STPEphemeralKeyManagerTest : XCTestCase + +@property (nonatomic) NSString *apiVersion; + +@end + +@implementation STPEphemeralKeyManagerTest + +- (void)setUp { + [super setUp]; + self.apiVersion = @"2015-03-03"; +} + +- (id)mockKeyProviderWithKeyResponse:(NSDictionary *)keyResponse { + XCTestExpectation *exp = [self expectationWithDescription:@"createCustomerKey"]; + id mockKeyProvider = OCMProtocolMock(@protocol(STPEphemeralKeyProvider)); + OCMStub([mockKeyProvider createCustomerKeyWithAPIVersion:[OCMArg isEqual:self.apiVersion] + completion:[OCMArg any]]) + .andDo(^(NSInvocation *invocation) { + __unsafe_unretained STPJSONResponseCompletionBlock completion; + [invocation getArgument:&completion atIndex:3]; + completion(keyResponse, nil); + [exp fulfill]; + }); + return mockKeyProvider; +} + +- (void)testgetOrCreateKeyCreatesNewKeyAfterInit { + STPEphemeralKey *expectedKey = [STPFixtures ephemeralKey]; + NSDictionary *keyResponse = [expectedKey allResponseFields]; + id mockKeyProvider = [self mockKeyProviderWithKeyResponse:keyResponse]; + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + XCTestExpectation *exp = [self expectationWithDescription:@"getOrCreateKey"]; + [sut getOrCreateKey:^(STPEphemeralKey *resourceKey, NSError *error) { + XCTAssertEqualObjects(resourceKey, expectedKey); + XCTAssertNil(error); + [exp fulfill]; + }]; + [self waitForExpectationsWithTimeout:2 handler:nil]; + [mockKeyProvider stopMocking]; +} + +- (void)testgetOrCreateKeyUsesStoredKeyIfNotExpiring { + id mockKeyProvider = OCMProtocolMock(@protocol(STPEphemeralKeyProvider)); + OCMReject([mockKeyProvider createCustomerKeyWithAPIVersion:[OCMArg any] completion:[OCMArg any]]); + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + STPEphemeralKey *expectedKey = [STPFixtures ephemeralKey]; + sut.ephemeralKey = expectedKey; + XCTestExpectation *exp = [self expectationWithDescription:@"getOrCreateKey"]; + [sut getOrCreateKey:^(STPEphemeralKey *resourceKey, NSError *error) { + XCTAssertEqualObjects(resourceKey, expectedKey); + XCTAssertNil(error); + [exp fulfill]; + }]; + [self waitForExpectationsWithTimeout:2 handler:nil]; + [mockKeyProvider stopMocking]; +} + +- (void)testgetOrCreateKeyCreatesNewKeyIfExpiring { + STPEphemeralKey *expectedKey = [STPFixtures ephemeralKey]; + NSDictionary *keyResponse = [expectedKey allResponseFields]; + id mockKeyProvider = [self mockKeyProviderWithKeyResponse:keyResponse]; + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + sut.ephemeralKey = [STPFixtures expiringEphemeralKey]; + XCTestExpectation *exp = [self expectationWithDescription:@"retrieve"]; + [sut getOrCreateKey:^(STPEphemeralKey *resourceKey, NSError *error) { + XCTAssertEqualObjects(resourceKey, expectedKey); + XCTAssertNil(error); + [exp fulfill]; + }]; + [self waitForExpectationsWithTimeout:2 handler:nil]; + [mockKeyProvider stopMocking]; +} + +- (void)testgetOrCreateKeyCoalescesRepeatCalls { + STPEphemeralKey *expectedKey = [STPFixtures ephemeralKey]; + NSDictionary *keyResponse = [expectedKey allResponseFields]; + XCTestExpectation *createExp = [self expectationWithDescription:@"createKey"]; + createExp.assertForOverFulfill = YES; + id mockKeyProvider = OCMProtocolMock(@protocol(STPEphemeralKeyProvider)); + OCMStub([mockKeyProvider createCustomerKeyWithAPIVersion:[OCMArg isEqual:self.apiVersion] + completion:[OCMArg any]]) + .andDo(^(NSInvocation *invocation) { + __unsafe_unretained STPJSONResponseCompletionBlock completion; + [invocation getArgument:&completion atIndex:3]; + [createExp fulfill]; + dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1*NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ + completion(keyResponse, nil); + }); + }); + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + XCTestExpectation *getExp1 = [self expectationWithDescription:@"getOrCreateKey"]; + [sut getOrCreateKey:^(STPEphemeralKey *ephemeralKey, NSError *error) { + XCTAssertEqualObjects(ephemeralKey, expectedKey); + XCTAssertNil(error); + [getExp1 fulfill]; + }]; + XCTestExpectation *getExp2 = [self expectationWithDescription:@"getOrCreateKey"]; + [sut getOrCreateKey:^(STPEphemeralKey *ephemeralKey, NSError *error) { + XCTAssertEqualObjects(ephemeralKey, expectedKey); + XCTAssertNil(error); + [getExp2 fulfill]; + }]; + + [self waitForExpectationsWithTimeout:2 handler:nil]; + [mockKeyProvider stopMocking]; +} + +// This test doesn't work becuase assertions in Swift are always fatal +/* +- (void)testgetOrCreateKeyThrowsExceptionWhenDecodingFails { + XCTestExpectation *exp1 = [self expectationWithDescription:@"createCustomerKey"]; + NSDictionary *invalidKeyResponse = @{@"foo": @"bar"}; + id mockKeyProvider = OCMProtocolMock(@protocol(STPEphemeralKeyProvider)); + OCMStub([mockKeyProvider createCustomerKeyWithAPIVersion:[OCMArg isEqual:self.apiVersion] + completion:[OCMArg any]]) + .andDo(^(NSInvocation *invocation) { + __unsafe_unretained STPJSONResponseCompletionBlock completion; + [invocation getArgument:&completion atIndex:3]; + XCTAssertThrows(completion(invalidKeyResponse, nil)); + [exp1 fulfill]; + }); + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + XCTestExpectation *exp2 = [self expectationWithDescription:@"retrieve"]; + [sut getOrCreateKey:^(STPEphemeralKey *resourceKey, NSError *error) { + XCTAssertNil(resourceKey); + XCTAssertEqualObjects(error, [NSError stp_ephemeralKeyDecodingError]); + [exp2 fulfill]; + }]; + [self waitForExpectationsWithTimeout:2 handler:nil]; + [mockKeyProvider stopMocking]; +} + */ + +- (void)testEnterForegroundRefreshesResourceKeyIfExpiring { + STPEphemeralKey *key = [STPFixtures expiringEphemeralKey]; + NSDictionary *keyResponse = [key allResponseFields]; + id mockKeyProvider = [self mockKeyProviderWithKeyResponse:keyResponse]; + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + XCTAssertNotNil(sut); + [[NSNotificationCenter defaultCenter] postNotificationName:UIApplicationWillEnterForegroundNotification object:nil]; + + [self waitForExpectationsWithTimeout:2 handler:nil]; + [mockKeyProvider stopMocking]; +} + +- (void)testEnterForegroundDoesNotRefreshResourceKeyIfNotExpiring { + id mockKeyProvider = OCMProtocolMock(@protocol(STPEphemeralKeyProvider)); + OCMReject([mockKeyProvider createCustomerKeyWithAPIVersion:[OCMArg any] completion:[OCMArg any]]); + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + sut.ephemeralKey = [STPFixtures ephemeralKey]; + [[NSNotificationCenter defaultCenter] postNotificationName:UIApplicationWillEnterForegroundNotification object:nil]; + [mockKeyProvider stopMocking]; +} + +- (void)testThrottlingEnterForegroundRefreshes { + id mockKeyProvider = OCMProtocolMock(@protocol(STPEphemeralKeyProvider)); + OCMReject([mockKeyProvider createCustomerKeyWithAPIVersion:[OCMArg any] completion:[OCMArg any]]); + STPEphemeralKeyManager *sut = [[STPEphemeralKeyManager alloc] initWithKeyProvider:mockKeyProvider apiVersion:self.apiVersion performsEagerFetching:YES]; + sut.ephemeralKey = [STPFixtures expiringEphemeralKey]; + sut.lastEagerKeyRefresh = [NSDate dateWithTimeIntervalSinceNow:-60]; + [[NSNotificationCenter defaultCenter] postNotificationName:UIApplicationWillEnterForegroundNotification object:nil]; + [mockKeyProvider stopMocking]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPEphemeralKeyTest.swift b/Stripe/StripeiOSTests/STPEphemeralKeyTest.swift new file mode 100644 index 00000000..b424c92a --- /dev/null +++ b/Stripe/StripeiOSTests/STPEphemeralKeyTest.swift @@ -0,0 +1,32 @@ +// +// STPEphemeralKeyTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 5/17/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPEphemeralKeyTest: XCTestCase { + func testDecoding() { + let json = STPTestUtils.jsonNamed("EphemeralKey")! + let key = STPEphemeralKey.decodedObject(fromAPIResponse: json)! + XCTAssertEqual(key.stripeID, json["id"] as! String) + XCTAssertEqual(key.secret, json["secret"] as! String) + XCTAssertEqual( + key.created, + Date(timeIntervalSince1970: TimeInterval((json["created"] as! NSNumber).doubleValue)) + ) + XCTAssertEqual( + key.expires, + Date(timeIntervalSince1970: TimeInterval((json["expires"] as! NSNumber).doubleValue)) + ) + XCTAssertEqual(key.livemode, (json["livemode"] as! NSNumber).boolValue) + XCTAssertEqual(key.customerID, "cus_123") + } +} diff --git a/Stripe/StripeiOSTests/STPErrorBridgeTest.m b/Stripe/StripeiOSTests/STPErrorBridgeTest.m new file mode 100644 index 00000000..b2b71038 --- /dev/null +++ b/Stripe/StripeiOSTests/STPErrorBridgeTest.m @@ -0,0 +1,39 @@ +// +// STPErrorBridgeTest.m +// StripeiOS Tests +// +// Created by David Estes on 9/23/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +@import Stripe; +@import XCTest; +@import PassKit; +#import "STPNetworkStubbingTestCase.h" +#import "STPTestingAPIClient.h" +#import "STPFixtures.h" + +@interface STPErrorBridgeTest : XCTestCase + +@end + +@implementation STPErrorBridgeTest + +- (void)testSTPErrorBridge { + // Grab a constant from each class, just to make sure we didn't forget to include the bridge: + XCTAssertEqual(STPInvalidRequestError, 50); + XCTAssertEqualObjects(STPError.errorMessageKey, @"com.stripe.lib:ErrorMessageKey"); + NSDictionary *json = @{ + @"error": @{ + @"type": @"invalid_request_error", + @"message": @"Your card number is incorrect.", + @"code": @"incorrect_number" + } + }; + + // Make sure we can parse a Stripe response + NSError *expectedError = [NSError stp_errorFromStripeResponse:json]; + XCTAssertEqualObjects(expectedError.domain, STPError.stripeDomain); +} + +@end diff --git a/Stripe/StripeiOSTests/STPFPXBankBrandTest.m b/Stripe/StripeiOSTests/STPFPXBankBrandTest.m new file mode 100644 index 00000000..ef0185e9 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFPXBankBrandTest.m @@ -0,0 +1,131 @@ +// +// STPFPXBankBrandTest.m +// StripeiOS Tests +// +// Created by David Estes on 8/26/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + + + +@interface STPFPXBankBrandTest : XCTestCase + +@end + +@implementation STPFPXBankBrandTest + +- (void)testStringFromBrand { + NSArray *brands = @[ + @(STPFPXBankBrandAffinBank), + @(STPFPXBankBrandAllianceBank), + @(STPFPXBankBrandAmbank), + @(STPFPXBankBrandBankIslam), + @(STPFPXBankBrandBankMuamalat), + @(STPFPXBankBrandBankRakyat), + @(STPFPXBankBrandBSN), + @(STPFPXBankBrandCIMB), + @(STPFPXBankBrandHongLeongBank), + @(STPFPXBankBrandHSBC), + @(STPFPXBankBrandKFH), + @(STPFPXBankBrandMaybank2E), + @(STPFPXBankBrandMaybank2U), + @(STPFPXBankBrandOcbc), + @(STPFPXBankBrandPublicBank), + @(STPFPXBankBrandCIMB), + @(STPFPXBankBrandRHB), + @(STPFPXBankBrandStandardChartered), + @(STPFPXBankBrandUOB), + @(STPFPXBankBrandUnknown), + ]; + + for (NSNumber *brandNumber in brands) { + STPFPXBankBrand brand = [brandNumber integerValue]; + NSString *brandName = [STPFPXBank stringFrom:brand]; + NSString *brandID = [STPFPXBank identifierFrom:brand]; + STPFPXBankBrand reverseTransformedBrand = [STPFPXBank brandFrom:brandID]; + XCTAssertEqual(reverseTransformedBrand, brand); + + switch (brand) { + case STPFPXBankBrandAffinBank: + XCTAssertEqualObjects(brandID, @"affin_bank"); + XCTAssertEqualObjects(brandName, @"Affin Bank"); + break; + case STPFPXBankBrandAllianceBank: + XCTAssertEqualObjects(brandID, @"alliance_bank"); + XCTAssertEqualObjects(brandName, @"Alliance Bank"); + break; + case STPFPXBankBrandAmbank: + XCTAssertEqualObjects(brandID, @"ambank"); + XCTAssertEqualObjects(brandName, @"AmBank"); + break; + case STPFPXBankBrandBankIslam: + XCTAssertEqualObjects(brandID, @"bank_islam"); + XCTAssertEqualObjects(brandName, @"Bank Islam"); + break; + case STPFPXBankBrandBankMuamalat: + XCTAssertEqualObjects(brandID, @"bank_muamalat"); + XCTAssertEqualObjects(brandName, @"Bank Muamalat"); + break; + case STPFPXBankBrandBankRakyat: + XCTAssertEqualObjects(brandID, @"bank_rakyat"); + XCTAssertEqualObjects(brandName, @"Bank Rakyat"); + break; + case STPFPXBankBrandBSN: + XCTAssertEqualObjects(brandID, @"bsn"); + XCTAssertEqualObjects(brandName, @"BSN"); + break; + case STPFPXBankBrandCIMB: + XCTAssertEqualObjects(brandID, @"cimb"); + XCTAssertEqualObjects(brandName, @"CIMB Clicks"); + break; + case STPFPXBankBrandHongLeongBank: + XCTAssertEqualObjects(brandID, @"hong_leong_bank"); + XCTAssertEqualObjects(brandName, @"Hong Leong Bank"); + break; + case STPFPXBankBrandHSBC: + XCTAssertEqualObjects(brandID, @"hsbc"); + XCTAssertEqualObjects(brandName, @"HSBC BANK"); + break; + case STPFPXBankBrandKFH: + XCTAssertEqualObjects(brandID, @"kfh"); + XCTAssertEqualObjects(brandName, @"KFH"); + break; + case STPFPXBankBrandMaybank2E: + XCTAssertEqualObjects(brandID, @"maybank2e"); + XCTAssertEqualObjects(brandName, @"Maybank2E"); + break; + case STPFPXBankBrandMaybank2U: + XCTAssertEqualObjects(brandID, @"maybank2u"); + XCTAssertEqualObjects(brandName, @"Maybank2U"); + break; + case STPFPXBankBrandOcbc: + XCTAssertEqualObjects(brandID, @"ocbc"); + XCTAssertEqualObjects(brandName, @"OCBC Bank"); + break; + case STPFPXBankBrandPublicBank: + XCTAssertEqualObjects(brandID, @"public_bank"); + XCTAssertEqualObjects(brandName, @"Public Bank"); + break; + case STPFPXBankBrandRHB: + XCTAssertEqualObjects(brandID, @"rhb"); + XCTAssertEqualObjects(brandName, @"RHB Bank"); + break; + case STPFPXBankBrandStandardChartered: + XCTAssertEqualObjects(brandID, @"standard_chartered"); + XCTAssertEqualObjects(brandName, @"Standard Chartered"); + break; + case STPFPXBankBrandUOB: + XCTAssertEqualObjects(brandID, @"uob"); + XCTAssertEqualObjects(brandName, @"UOB Bank"); + break; + case STPFPXBankBrandUnknown: + XCTAssertEqualObjects(brandID, @"unknown"); + XCTAssertEqualObjects(brandName, @"Unknown"); + break; + } + }; +} + +@end diff --git a/Stripe/StripeiOSTests/STPFileFunctionalTest.m b/Stripe/StripeiOSTests/STPFileFunctionalTest.m new file mode 100644 index 00000000..897af7c0 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFileFunctionalTest.m @@ -0,0 +1,89 @@ +// +// STPFileFunctionalTest.m +// Stripe +// +// Created by Charles Scalesse on 1/8/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPFileFunctionalTest : XCTestCase +@end + +@implementation STPFileFunctionalTest + + +- (UIImage *)testImage { +return [UIImage imageNamed:@"stp_test_upload_image.jpeg" + inBundle:[NSBundle bundleForClass:self.class] +compatibleWithTraitCollection:nil]; +} + +- (void)testCreateFileForIdentityDocument { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"File creation for identity document"]; + + UIImage *image = [self testImage]; + + [client uploadImage:image + purpose:STPFilePurposeIdentityDocument + completion:^(STPFile * _Nullable file, NSError * _Nullable error) { + [expectation fulfill]; + XCTAssertNil(error, @"error should be nil %@", error.localizedDescription); + + XCTAssertNotNil(file.fileId); + XCTAssertNotNil(file.created); + XCTAssertEqual(file.purpose, STPFilePurposeIdentityDocument); + XCTAssertNotNil(file.size); + XCTAssertEqualObjects(@"jpg", file.type); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateFileForDisputeEvidence { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"File creation for dispute evidence"]; + + UIImage *image = [self testImage]; + + [client uploadImage:image + purpose:STPFilePurposeDisputeEvidence + completion:^(STPFile * _Nullable file, NSError * _Nullable error) { + [expectation fulfill]; + XCTAssertNil(error, @"error should be nil %@", error.localizedDescription); + + XCTAssertNotNil(file.fileId); + XCTAssertNotNil(file.created); + XCTAssertEqual(file.purpose, STPFilePurposeDisputeEvidence); + XCTAssertNotNil(file.size); + XCTAssertEqualObjects(@"jpg", file.type); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testInvalidKey { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"not_a_valid_key_asdf"]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Bad file creation"]; + + UIImage *image = [self testImage]; + + [client uploadImage:image + purpose:STPFilePurposeIdentityDocument + completion:^(STPFile * _Nullable file, NSError * _Nullable error) { + [expectation fulfill]; + XCTAssertNil(file, @"file should be nil"); + XCTAssertNotNil(error, @"error should not be nil"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPFileTest.m b/Stripe/StripeiOSTests/STPFileTest.m new file mode 100644 index 00000000..29622754 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFileTest.m @@ -0,0 +1,118 @@ +// +// STPFileTest.m +// Stripe +// +// Created by Charles Scalesse on 1/8/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + + +#import "STPTestUtils.h" + +@interface STPFile () + ++ (STPFilePurpose)purposeFromString:(NSString *)string; + +@end + +@interface STPFileTest : XCTestCase + +@end + +@implementation STPFileTest + +#pragma mark - STPFilePurpose Tests + +- (void)testPurposeFromString { + XCTAssertEqual([STPFile purposeFromString:@"dispute_evidence"], STPFilePurposeDisputeEvidence); + XCTAssertEqual([STPFile purposeFromString:@"DISPUTE_EVIDENCE"], STPFilePurposeDisputeEvidence); + + XCTAssertEqual([STPFile purposeFromString:@"identity_document"], STPFilePurposeIdentityDocument); + XCTAssertEqual([STPFile purposeFromString:@"IDENTITY_DOCUMENT"], STPFilePurposeIdentityDocument); + + XCTAssertEqual([STPFile purposeFromString:@"unknown"], STPFilePurposeUnknown); + XCTAssertEqual([STPFile purposeFromString:@"UNKNOWN"], STPFilePurposeUnknown); + + XCTAssertEqual([STPFile purposeFromString:@"garbage"], STPFilePurposeUnknown); + XCTAssertEqual([STPFile purposeFromString:@"GARBAGE"], STPFilePurposeUnknown); +} + +- (void)testStringFromPurpose { + NSArray *values = @[ + @(STPFilePurposeDisputeEvidence), + @(STPFilePurposeIdentityDocument), + @(STPFilePurposeUnknown), + ]; + + for (NSNumber *purposeNumber in values) { + STPFilePurpose purpose = (STPFilePurpose)[purposeNumber integerValue]; + NSString *string = [STPFile stringFromPurpose:purpose]; + + switch (purpose) { + case STPFilePurposeDisputeEvidence: + XCTAssertEqualObjects(string, @"dispute_evidence"); + break; + case STPFilePurposeIdentityDocument: + XCTAssertEqualObjects(string, @"identity_document"); + break; + case STPFilePurposeUnknown: + XCTAssertNil(string); + break; + } + } +} + +#pragma mark - Equality Tests + +- (void)testFileEquals { + STPFile *file1 = [STPFile decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"FileUpload"]]; + STPFile *file2 = [STPFile decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"FileUpload"]]; + + XCTAssertNotEqual(file1, file2); + + XCTAssertEqualObjects(file1, file1); + XCTAssertEqualObjects(file1, file2); + + XCTAssertEqual(file1.hash, file1.hash); + XCTAssertEqual(file1.hash, file2.hash); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[ + @"id", + @"created", + @"size", + @"purpose", + @"type", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"FileUpload"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPFile decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPFile decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"FileUpload"]]); +} + +- (void)testInitializingFileWithAttributeDictionary { + NSDictionary *response = [STPTestUtils jsonNamed:@"FileUpload"]; + STPFile *file = [STPFile decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(file.fileId, @"file_1AZl0o2eZvKYlo2CoIkwLzfd"); + XCTAssertEqualObjects(file.created, [NSDate dateWithTimeIntervalSince1970:1498674938]); + XCTAssertEqual(file.purpose, STPFilePurposeDisputeEvidence); + XCTAssertEqualObjects(file.size, @34478); + XCTAssertEqualObjects(file.type, @"jpg"); + + XCTAssertNotEqual(file.allResponseFields, response); + XCTAssertEqualObjects(file.allResponseFields, response); +} + +@end diff --git a/Stripe/StripeiOSTests/STPFixtures+Swift.swift b/Stripe/StripeiOSTests/STPFixtures+Swift.swift new file mode 100644 index 00000000..adb45883 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFixtures+Swift.swift @@ -0,0 +1,67 @@ +// +// STPFixtures+Swift.swift +// StripeiOSTests +// +// Created by Yuki Tokuhiro on 3/22/23. +// + +import Foundation +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +extension STPFixtures { + static func paymentMethodBillingDetails() -> STPPaymentMethodBillingDetails { + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "Jane Doe" + billingDetails.email = "foo@bar.com" + billingDetails.phone = "5555555555" + billingDetails.address = STPPaymentMethodAddress() + billingDetails.address?.line1 = "510 Townsend St." + billingDetails.address?.line2 = "Line 2" + billingDetails.address?.city = "San Francisco" + billingDetails.address?.state = "CA" + billingDetails.address?.country = "US" + billingDetails.address?.postalCode = "94102" + return billingDetails + } + + static func paymentIntent( + paymentMethodTypes: [String], + orderedPaymentMethodTypes: [String]? = nil, + setupFutureUsage: STPPaymentIntentSetupFutureUsage = .none, + currency: String = "usd" + ) -> STPPaymentIntent { + var apiResponse: [AnyHashable: Any?] = [ + "id": "123", + "client_secret": "sec", + "amount": 10, + "currency": currency, + "status": "requires_payment_method", + "livemode": false, + "created": 1652736692.0, + "payment_method_types": paymentMethodTypes, + "setup_future_usage": setupFutureUsage.stringValue, + ] + if let orderedPaymentMethodTypes = orderedPaymentMethodTypes { + apiResponse["ordered_payment_method_types"] = orderedPaymentMethodTypes + } + return STPPaymentIntent.decodeSTPPaymentIntentObject( + fromAPIResponse: apiResponse as [AnyHashable: Any] + )! + } +} + +extension PaymentSheet.Configuration { + /// Provides a Configuration that allows all pm types available + static func _testValue_MostPermissive() -> Self { + var configuration = PaymentSheet.Configuration() + configuration.returnURL = "https://foo.com" + configuration.allowsDelayedPaymentMethods = true + configuration.allowsPaymentMethodsRequiringShippingAddress = true + configuration.applePay = .init(merchantId: "merchant id", merchantCountryCode: "US") + return configuration + } +} diff --git a/Stripe/StripeiOSTests/STPFixtures.h b/Stripe/StripeiOSTests/STPFixtures.h new file mode 100644 index 00000000..1923f146 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFixtures.h @@ -0,0 +1,225 @@ +// +// STPFixtures.h +// Stripe +// +// Created by Ben Guo on 3/28/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import +#import +#import +@import Stripe; +@import StripeCore; +@import StripePayments; +@import StripePaymentsUI; + +NS_ASSUME_NONNULL_BEGIN +extern NSString *const STPTestJSONCustomer; + +extern NSString *const STPTestJSONCard; + +extern NSString *const STPTestJSONPaymentIntent; +extern NSString *const STPTestJSONSetupIntent; +extern NSString *const STPTestJSONPaymentMethodCard; +extern NSString *const STPTestJSONPaymentMethodApplePay; +extern NSString *const STPTestJSONPaymentMethodBacsDebit; + +extern NSString *const STPTestJSONSource3DS; +extern NSString *const STPTestJSONSourceAlipay; +extern NSString *const STPTestJSONSourceBancontact; +extern NSString *const STPTestJSONSourceCard; +extern NSString *const STPTestJSONSourceEPS; +extern NSString *const STPTestJSONSourceGiropay; +extern NSString *const STPTestJSONSourceiDEAL; +extern NSString *const STPTestJSONSourceMultibanco; +extern NSString *const STPTestJSONSourceP24; +extern NSString *const STPTestJSONSourceSEPADebit; +extern NSString *const STPTestJSONSourceSofort; + +@interface STPFixtures : NSObject + +/** + An STPConnectAccountParams object with all of the fields filled in, and + ToS accepted. + */ ++ (STPConnectAccountParams *)accountParams; + +/** + An Address object with all fields filled. + */ ++ (STPAddress *)address; + +/** + A PKPaymentObject with test payment data. + */ ++ (PKPayment *)applePayPayment; + +/** + A PKPayment from the simulator that can be tokenized in testmode. + */ ++ (PKPayment *)simulatorApplePayPayment; + +/** + A valid PKPaymentRequest with dummy data. + */ ++ (PKPaymentRequest *)applePayRequest; + +/** + A BankAccountParams object with all fields filled. + */ ++ (STPBankAccountParams *)bankAccountParams; + +/** + A CardParams object with a valid number, expMonth, expYear, and cvc. + */ ++ (STPCardParams *)cardParams; + +/** + A valid card object + */ ++ (STPCard *)card; + +/** + A Source object with type card + */ ++ (STPSource *)cardSource; + +/** + A Token for a card + */ ++ (STPToken *)cardToken; + +/** + A Customer object with an empty sources array. + */ ++ (STPCustomer *)customerWithNoSources; + +/** + A Customer object with a single card token in its sources array, and + default_source set to that card token. + */ ++ (STPCustomer *)customerWithSingleCardTokenSource; + +/** + The JSON data for a Customer with a single card token in its sources array, and + default_source set to that card token. + */ ++ (NSDictionary *)customerWithSingleCardTokenSourceJSON; + +/** + A Customer object with a single card source in its sources array, and + default_source set to that card source. + */ ++ (STPCustomer *)customerWithSingleCardSourceSource; + +/** + A Customer object with two cards in its sources array, + one a token/card type and one a source object type. + default_source is set to the card token. + */ ++ (STPCustomer *)customerWithCardTokenAndSourceSources; + +/** + A Customer object with a card source, and apple pay card source, and + default_source set to the apple pay source. + */ ++ (STPCustomer *)customerWithCardAndApplePaySources; + +/** + A Customer JSON blob with a card source, and apple pay card source, and + default_source set to the apple pay source. + */ ++ (NSDictionary *)customerWithCardAndApplePaySourcesJSON; + +/** + A customer object with a sources array that includes the listed json sources + in the order they are listed in the array. + + Valid keys are any STPTestJSONSource constants and the STPTestJSONCard constant. + + Ids for the sources will be automatically generated and will be equal to a + string that is the index of the array of that source. + */ ++ (STPCustomer *)customerWithSourcesFromJSONKeys:(NSArray *)jsonSourceKeys + defaultSource:(NSString *)jsonKeyForDefaultSource; + +/** + A Source object with type iDEAL + */ ++ (STPSource *)iDEALSource; + +/** + A Source object with type Alipay + */ ++ (STPSource *)alipaySource; + +/** + A Source object with type WeChat Pay + */ ++ (STPSource *)weChatPaySource; + +/** + A Source object with type Alipay and a native redirect url + */ ++ (STPSource *)alipaySourceWithNativeURL; + +/** + A PaymentIntent object + */ ++ (STPPaymentIntent *)paymentIntent; + +/** + A SetupIntent object + */ ++ (STPSetupIntent *)setupIntent; + +/** + A PaymentConfiguration object with a fake publishable key. Use this to avoid + triggering our asserts when publishable key is nil or invalid. All other values + are at their original defaults. + */ ++ (STPPaymentConfiguration *)paymentConfiguration; + +/** + A customer-scoped ephemeral key that expires in 100 seconds. + */ ++ (STPEphemeralKey *)ephemeralKey; + +/** + A customer-scoped ephemeral key that expires in 10 seconds. + */ ++ (STPEphemeralKey *)expiringEphemeralKey; + +/** + A PaymentMethod object + */ ++ (STPPaymentMethod *)paymentMethod; + +/** + A PaymentMethod JSON dictionary + */ ++ (NSDictionary *)paymentMethodJSON; + +/** + A STPPaymentMethodCardParams object with a valid number, expMonth, expYear, and cvc. + */ ++ (STPPaymentMethodCardParams *)paymentMethodCardParams; + +/** + An Apple Pay Payment Method object. + */ ++ (STPPaymentMethod *)applePayPaymentMethod; + +/** + An Apple Pay Payment Method JSON dictionary. + */ ++ (NSDictionary *)applePayPaymentMethodJSON; + +@end + +@interface STPJsonSources : NSObject + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe/StripeiOSTests/STPFixtures.m b/Stripe/StripeiOSTests/STPFixtures.m new file mode 100644 index 00000000..eed3c26d --- /dev/null +++ b/Stripe/StripeiOSTests/STPFixtures.m @@ -0,0 +1,361 @@ +// +// STPFixtures.m +// Stripe +// +// Created by Ben Guo on 3/28/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +NSString *const STPTestJSONCustomer = @"Customer"; + +NSString *const STPTestJSONCard = @"Card"; + +NSString *const STPTestJSONPaymentIntent = @"PaymentIntent"; +NSString *const STPTestJSONSetupIntent = @"SetupIntent"; + +NSString *const STPTestJSONPaymentMethodCard = @"CardPaymentMethod"; +NSString *const STPTestJSONPaymentMethodApplePay = @"ApplePayPaymentMethod"; +NSString *const STPTestJSONPaymentMethodBacsDebit = @"BacsDebitPaymentMethod"; + +NSString *const STPTestJSONSource3DS = @"3DSSource"; +NSString *const STPTestJSONSourceAlipay = @"AlipaySource"; +NSString *const STPTestJSONSourceBancontact = @"BancontactSource"; +NSString *const STPTestJSONSourceCard = @"CardSource"; +NSString *const STPTestJSONSourceEPS = @"EPSSource"; +NSString *const STPTestJSONSourceGiropay = @"GiropaySource"; +NSString *const STPTestJSONSourceiDEAL = @"iDEALSource"; +NSString *const STPTestJSONSourceMultibanco = @"MultibancoSource"; +NSString *const STPTestJSONSourceP24 = @"P24Source"; +NSString *const STPTestJSONSourceSEPADebit = @"SEPADebitSource"; +NSString *const STPTestJSONSourceSofort = @"SofortSource"; +NSString *const STPTestJSONSourceWeChatPay = @"WeChatPaySource"; + +@import StripeCore; + +@implementation STPFixtures + ++ (STPConnectAccountParams *)accountParams { + STPConnectAccountIndividualParams *params = [STPConnectAccountIndividualParams new]; + return [[STPConnectAccountParams alloc] initWithTosShownAndAccepted:YES + individual:params]; +} + ++ (STPAddress *)address { + STPAddress *address = [STPAddress new]; + address.name = @"Jenny Rosen"; + address.phone = @"5555555555"; + address.email = @"jrosen@example.com"; + address.line1 = @"27 Smith St"; + address.line2 = @"Apt 2"; + address.postalCode = @"10001"; + address.city = @"New York"; + address.state = @"NY"; + address.country = @"US"; + return address; +} + ++ (STPBankAccountParams *)bankAccountParams { + STPBankAccountParams *bankParams = [STPBankAccountParams new]; + // https://stripe.com/docs/testing#account-numbers + bankParams.accountNumber = @"000123456789"; + bankParams.routingNumber = @"110000000"; + bankParams.country = @"US"; + bankParams.currency = @"usd"; + bankParams.accountNumber = @"Jenny Rosen"; + return bankParams; +} + ++ (STPCardParams *)cardParams { + STPCardParams *cardParams = [STPCardParams new]; + cardParams.number = @"4242424242424242"; + cardParams.expMonth = 10; + cardParams.expYear = 99; + cardParams.cvc = @"123"; + return cardParams; +} + ++ (STPPaymentMethodCardParams *)paymentMethodCardParams { + STPPaymentMethodCardParams *cardParams = [STPPaymentMethodCardParams new]; + cardParams.number = @"4242424242424242"; + cardParams.expMonth = @(10); + cardParams.expYear = @(99); + cardParams.cvc = @"123"; + return cardParams; +} + ++ (STPCard *)card { + return [STPCard decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONCard]]; +} + ++ (STPSource *)cardSource { + return [STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONSourceCard]]; +} + ++ (STPToken *)cardToken { + NSDictionary *cardDict = [STPTestUtils jsonNamed:STPTestJSONCard]; + NSDictionary *tokenDict = @{ + @"id": @"id_for_token", + @"object": @"token", + @"livemode": @NO, + @"created": @1353025450.0, + @"type": @"card", + @"used": @NO, + @"card": cardDict + }; + return [STPToken decodedObjectFromAPIResponse:tokenDict]; +} + ++ (STPCustomer *)customerWithNoSources { + return [STPCustomer decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONCustomer]]; +} + ++ (STPCustomer *)customerWithSingleCardTokenSource { + return [STPCustomer decodedObjectFromAPIResponse:[self customerWithSingleCardTokenSourceJSON]]; +} + ++ (NSDictionary *)customerWithSingleCardTokenSourceJSON { + NSMutableDictionary *card1 = [[STPTestUtils jsonNamed:STPTestJSONCard] mutableCopy]; + card1[@"id"] = @"card_123"; + + NSMutableDictionary *customer = [[STPTestUtils jsonNamed:STPTestJSONCustomer] mutableCopy]; + NSMutableDictionary *sources = [customer[@"sources"] mutableCopy]; + sources[@"data"] = @[card1]; + customer[@"default_source"] = card1[@"id"]; + customer[@"sources"] = sources; + + return customer; +} + ++ (STPCustomer *)customerWithSingleCardSourceSource { + NSMutableDictionary *card1 = [[STPTestUtils jsonNamed:STPTestJSONSourceCard] mutableCopy]; + card1[@"id"] = @"card_123"; + + NSMutableDictionary *customer = [[STPTestUtils jsonNamed:STPTestJSONCustomer] mutableCopy]; + NSMutableDictionary *sources = [customer[@"sources"] mutableCopy]; + sources[@"data"] = @[card1]; + customer[@"default_source"] = card1[@"id"]; + customer[@"sources"] = sources; + + return [STPCustomer decodedObjectFromAPIResponse:customer]; +} + ++ (STPCustomer *)customerWithCardTokenAndSourceSources { + NSMutableDictionary *card1 = [[STPTestUtils jsonNamed:STPTestJSONCard] mutableCopy]; + card1[@"id"] = @"card_123"; + + NSMutableDictionary *card2 = [[STPTestUtils jsonNamed:STPTestJSONSourceCard] mutableCopy]; + card2[@"id"] = @"src_456"; + + NSMutableDictionary *customer = [[STPTestUtils jsonNamed:STPTestJSONCustomer] mutableCopy]; + NSMutableDictionary *sources = [customer[@"sources"] mutableCopy]; + sources[@"data"] = @[card1, card2]; + customer[@"default_source"] = card1[@"id"]; + customer[@"sources"] = sources; + + return [STPCustomer decodedObjectFromAPIResponse:customer]; + +} + ++ (STPCustomer *)customerWithCardAndApplePaySources { + return [STPCustomer decodedObjectFromAPIResponse:[self customerWithCardAndApplePaySourcesJSON]]; +} + ++ (NSDictionary *)customerWithCardAndApplePaySourcesJSON { + NSMutableDictionary *card1 = [[STPTestUtils jsonNamed:STPTestJSONSourceCard] mutableCopy]; + card1[@"id"] = @"src_apple_pay_123"; + NSMutableDictionary *cardDict = [card1[@"card"] mutableCopy]; + cardDict[@"tokenization_method"] = @"apple_pay"; + card1[@"card"] = cardDict; + + NSMutableDictionary *card2 = [[STPTestUtils jsonNamed:STPTestJSONSourceCard] mutableCopy]; + card2[@"id"] = @"src_card_456"; + + NSMutableDictionary *customer = [[STPTestUtils jsonNamed:STPTestJSONCustomer] mutableCopy]; + NSMutableDictionary *sources = [customer[@"sources"] mutableCopy]; + sources[@"data"] = @[card1, card2]; + customer[@"default_source"] = card1[@"id"]; + customer[@"sources"] = sources; + + return customer; +} + ++ (STPCustomer *)customerWithSourcesFromJSONKeys:(NSArray *)jsonSourceKeys + defaultSource:(NSString *)jsonKeyForDefaultSource { + NSMutableArray *sourceJSONDicts = [NSMutableArray new]; + NSString *defaultSourceID = nil; + NSUInteger sourceCount = 0; + for (NSString *jsonKey in jsonSourceKeys) { + NSMutableDictionary *sourceDict = [[STPTestUtils jsonNamed:jsonKey] mutableCopy]; + sourceDict[@"id"] = [NSString stringWithFormat:@"%@", @(sourceCount)]; + if ([jsonKeyForDefaultSource isEqualToString:jsonKey]) { + defaultSourceID = sourceDict[@"id"]; + } + sourceCount += 1; + [sourceJSONDicts addObject:sourceDict.copy]; + } + + NSMutableDictionary *customer = [[STPTestUtils jsonNamed:STPTestJSONCustomer] mutableCopy]; + NSMutableDictionary *sources = [customer[@"sources"] mutableCopy]; + sources[@"data"] = sourceJSONDicts.copy; + customer[@"default_source"] = defaultSourceID ?: @""; + customer[@"sources"] = sources; + + return [STPCustomer decodedObjectFromAPIResponse:customer]; +} + ++ (STPSource *)iDEALSource { + return [STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONSourceiDEAL]]; +} + ++ (STPSource *)alipaySource { + return [STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONSourceAlipay]]; +} + ++ (STPSource *)alipaySourceWithNativeURL { + NSMutableDictionary *dictionary = [STPTestUtils jsonNamed:STPTestJSONSourceAlipay].mutableCopy; + NSMutableDictionary *detailsDictionary = ((NSDictionary *)dictionary[@"alipay"]).mutableCopy; + detailsDictionary[@"native_url"] = @"alipay://test"; + dictionary[@"alipay"] = detailsDictionary; + return [STPSource decodedObjectFromAPIResponse:dictionary]; +} + ++ (STPSource *)weChatPaySource { + return [STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONSourceWeChatPay]]; +} + ++ (STPPaymentIntent *)paymentIntent { + return [STPPaymentIntent decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"PaymentIntent"]]; +} + ++ (STPSetupIntent *)setupIntent { + return [STPSetupIntent decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"SetupIntent"]]; +} + ++ (STPPaymentConfiguration *)paymentConfiguration { + STPPaymentConfiguration *config = [STPPaymentConfiguration new]; + return config; +} + ++ (STPEphemeralKey *)ephemeralKey { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"EphemeralKey"] mutableCopy]; + NSTimeInterval interval = 100; + response[@"expires"] = @([[NSDate dateWithTimeIntervalSinceNow:interval] timeIntervalSince1970]); + return [STPEphemeralKey decodedObjectFromAPIResponse:response]; +} + ++ (STPEphemeralKey *)expiringEphemeralKey { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"EphemeralKey"] mutableCopy]; + NSTimeInterval interval = 10; + response[@"expires"] = @([[NSDate dateWithTimeIntervalSinceNow:interval] timeIntervalSince1970]); + return [STPEphemeralKey decodedObjectFromAPIResponse:response]; +} + ++ (PKPaymentRequest *)applePayRequest { + PKPaymentRequest *paymentRequest = [StripeAPI paymentRequestWithMerchantIdentifier:@"foo" country:@"US" currency:@"USD"]; + paymentRequest.paymentSummaryItems = @[[PKPaymentSummaryItem summaryItemWithLabel:@"bar" amount:[NSDecimalNumber decimalNumberWithString:@"10.00"]]]; + return paymentRequest; +} + ++ (PKPayment *)simulatorApplePayPayment { + PKPayment *payment = [PKPayment new]; + PKPaymentToken *paymentToken = [PKPaymentToken new]; + PKPaymentMethod *paymentMethod = [PKPaymentMethod new]; +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wundeclared-selector" + [paymentMethod performSelector:@selector(setDisplayName:) withObject:@"Simulated Instrument"]; + [paymentMethod performSelector:@selector(setNetwork:) withObject:@"AmEx"]; + [paymentToken performSelector:@selector(setTransactionIdentifier:) withObject:@"Simulated Identifier"]; + + [paymentToken performSelector:@selector(setPaymentMethod:) withObject:paymentMethod]; + + [payment performSelector:@selector(setToken:) withObject:paymentToken]; + + // Add shipping + PKContact *shipping = [PKContact new]; + shipping.name = [[NSPersonNameComponentsFormatter new] personNameComponentsFromString:@"Jane Doe"]; + CNMutablePostalAddress *address = [CNMutablePostalAddress new]; + address.street = @"510 Townsend St"; + shipping.postalAddress = address; + [payment performSelector:@selector(setShippingContact:) withObject:shipping]; +#pragma clang diagnostic pop + return payment; +} + ++ (PKPayment *)applePayPayment { + PKPayment *payment = [PKPayment new]; + PKPaymentToken *paymentToken = [PKPaymentToken new]; + NSString *tokenDataString = @"{\"version\":\"EC_v1\",\"data\":\"lF8RBjPvhc2GuhjEh7qFNijDJjxD/ApmGdQhgn8tpJcJDOwn2E1BkOfSvnhrR8BUGT6+zeBx8OocvalHZ5ba/WA/" + @"tDxGhcEcOMp8sIJrXMVcJ6WqT5P1ZY+utmdORhxyH4nUw2wuEY4lAE7/GtEU/RNDhaKx/" + @"m93l0oLlk84qD1ynTA5JP3gjkdX+RK23iCAZDScXCcCU0OnYlJV8sDyf3+8hIo0gpN43AxoY6N1xAsVbGsO4ZjSCahaXbgt0egFug3s7Fyt9W4uzu07SKKCA2+" + @"DNZeZeerefpN1d1YbiCNlxFmffZKLCGdFERc7Ci3+yrHWWnYhKdQh8FeKCiiAvY5gbZJgQ91lNumCuP1IkHdHqxYI0qFk9c2R6KStJDtoUbVEYbxwnGdEJJPiMPjuKlgi7E+" + @"LlBdXiREmlz4u1EA=\",\"signature\":" + @"\"MIAGCSqGSIb3DQEHAqCAMIACAQExDzANBglghkgBZQMEAgEFADCABgkqhkiG9w0BBwEAAKCAMIID4jCCA4igAwIBAgIIJEPyqAad9XcwCgYIKoZIzj0EAwIwejEuMCwGA1UEAwwlQXBwbGUgQX" + @"BwbGljYXRpb24gSW50ZWdyYXRpb24gQ0EgLSBHMzEmMCQGA1UECwwdQXBwbGUgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkxEzARBgNVBAoMCkFwcGxlIEluYy4xCzAJBgNVBAYTAlVTMB4XDTE0MD" + @"kyNTIyMDYxMVoXDTE5MDkyNDIyMDYxMVowXzElMCMGA1UEAwwcZWNjLXNtcC1icm9rZXItc2lnbl9VQzQtUFJPRDEUMBIGA1UECwwLaU9TIFN5c3RlbXMxEzARBgNVBAoMCkFwcGxlIEluYy4xCz" + @"AJBgNVBAYTAlVTMFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEwhV37evWx7Ihj2jdcJChIY3HsL1vLCg9hGCV2Ur0pUEbg0IO2BHzQH6DMx8cVMP36zIg1rrV1O/" + @"0komJPnwPE6OCAhEwggINMEUGCCsGAQUFBwEBBDkwNzA1BggrBgEFBQcwAYYpaHR0cDovL29jc3AuYXBwbGUuY29tL29jc3AwNC1hcHBsZWFpY2EzMDEwHQYDVR0OBBYEFJRX22/" + @"VdIGGiYl2L35XhQfnm1gkMAwGA1UdEwEB/wQCMAAwHwYDVR0jBBgwFoAUI/JJxE+T5O8n5sT2KGw/orv9LkswggEdBgNVHSAEggEUMIIBEDCCAQwGCSqGSIb3Y2QFATCB/" + @"jCBwwYIKwYBBQUHAgIwgbYMgbNSZWxpYW5jZSBvbiB0aGlzIGNlcnRpZmljYXRlIGJ5IGFueSBwYXJ0eSBhc3N1bWVzIGFjY2VwdGFuY2Ugb2YgdGhlIHRoZW4gYXBwbGljYWJsZSBzdGFuZGFyZ" + @"CB0ZXJtcyBhbmQgY29uZGl0aW9ucyBvZiB1c2UsIGNlcnRpZmljYXRlIHBvbGljeSBhbmQgY2VydGlmaWNhdGlvbiBwcmFjdGljZSBzdGF0ZW1lbnRzLjA2BggrBgEFBQcCARYqaHR0cDovL3d3d" + @"y5hcHBsZS5jb20vY2VydGlmaWNhdGVhdXRob3JpdHkvMDQGA1UdHwQtMCswKaAnoCWGI2h0dHA6Ly9jcmwuYXBwbGUuY29tL2FwcGxlYWljYTMuY3JsMA4GA1UdDwEB/" + @"wQEAwIHgDAPBgkqhkiG92NkBh0EAgUAMAoGCCqGSM49BAMCA0gAMEUCIHKKnw+Soyq5mXQr1V62c0BXKpaHodYu9TWXEPUWPpbpAiEAkTecfW6+" + @"W5l0r0ADfzTCPq2YtbS39w01XIayqBNy8bEwggLuMIICdaADAgECAghJbS+/" + @"OpjalzAKBggqhkjOPQQDAjBnMRswGQYDVQQDDBJBcHBsZSBSb290IENBIC0gRzMxJjAkBgNVBAsMHUFwcGxlIENlcnRpZmljYXRpb24gQXV0aG9yaXR5MRMwEQYDVQQKDApBcHBsZSBJbmMuMQsw" + @"CQYDVQQGEwJVUzAeFw0xNDA1MDYyMzQ2MzBaFw0yOTA1MDYyMzQ2MzBaMHoxLjAsBgNVBAMMJUFwcGxlIEFwcGxpY2F0aW9uIEludGVncmF0aW9uIENBIC0gRzMxJjAkBgNVBAsMHUFwcGxlIENl" + @"cnRpZmljYXRpb24gQXV0aG9yaXR5MRMwEQYDVQQKDApBcHBsZSBJbmMuMQswCQYDVQQGEwJVUzBZMBMGByqGSM49AgEGCCqGSM49AwEHA0IABPAXEYQZ12SF1RpeJYEHduiAou/" + @"ee65N4I38S5PhM1bVZls1riLQl3YNIk57ugj9dhfOiMt2u2ZwvsjoKYT/" + @"VEWjgfcwgfQwRgYIKwYBBQUHAQEEOjA4MDYGCCsGAQUFBzABhipodHRwOi8vb2NzcC5hcHBsZS5jb20vb2NzcDA0LWFwcGxlcm9vdGNhZzMwHQYDVR0OBBYEFCPyScRPk+TvJ+bE9ihsP6K7/" + @"S5LMA8GA1UdEwEB/" + @"wQFMAMBAf8wHwYDVR0jBBgwFoAUu7DeoVgziJqkipnevr3rr9rLJKswNwYDVR0fBDAwLjAsoCqgKIYmaHR0cDovL2NybC5hcHBsZS5jb20vYXBwbGVyb290Y2FnMy5jcmwwDgYDVR0PAQH/" + @"BAQDAgEGMBAGCiqGSIb3Y2QGAg4EAgUAMAoGCCqGSM49BAMCA2cAMGQCMDrPcoNRFpmxhvs1w1bKYr/0F+3ZD3VNoo6+8ZyBXkK3ifiY95tZn5jVQQ2PnenC/gIwMi3VRCGwowV3bF3zODuQZ/" + @"0XfCwhbZZPxnJpghJvVPh6fRuZy5sJiSFhBpkPCZIdAAAxggFeMIIBWgIBATCBhjB6MS4wLAYDVQQDDCVBcHBsZSBBcHBsaWNhdGlvbiBJbnRlZ3JhdGlvbiBDQSAtIEczMSYwJAYDVQQLDB1BcH" + @"BsZSBDZXJ0aWZpY2F0aW9uIEF1dGhvcml0eTETMBEGA1UECgwKQXBwbGUgSW5jLjELMAkGA1UEBhMCVVMCCCRD8qgGnfV3MA0GCWCGSAFlAwQCAQUAoGkwGAYJKoZIhvcNAQkDMQsGCSqGSIb3DQ" + @"EHATAcBgkqhkiG9w0BCQUxDxcNMTQxMjIyMDIxMzQyWjAvBgkqhkiG9w0BCQQxIgQgUak8LCvAswLOnY2vlZf/" + @"iG3q04omAr3zV8YTtqvORGYwCgYIKoZIzj0EAwIERjBEAiAuPXMqEQqiTjYadOAvNmohP2yquB4owoQNjuAETkFXMAIgcH6zOxnbTTFmlEocqMztWR+L6OVBH6iTPIFMBNPcq6gAAAAAAAA=\"," + @"\"header\":{\"transactionId\":\"a530c7d68b6a69791d8864df2646c8aa3d09d33b56d8f8162ab23e1b26afe5e9\",\"ephemeralPublicKey\":" + @"\"MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEhKpIc6wTNQGy39bHM0a0qziDb20jMBFZT9XKSdjGULpDGRdyil6MLwMyIf3lQxaV/" + @"P7CQztw28IvYozvKvjBPQ==\",\"publicKeyHash\":\"yRcyn7njT6JL3AY9nmg0KD/xm/ch7gW1sGl2OuEucZY=\"}}"; + NSData *data = [tokenDataString dataUsingEncoding:NSUTF8StringEncoding]; + + NSPersonNameComponents *nameComponents = [[NSPersonNameComponents alloc] init]; + [nameComponents setGivenName:@"Test"]; + [nameComponents setFamilyName:@"Testerson"]; + PKContact *contact = [[PKContact alloc] init]; + contact.name = nameComponents; + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wundeclared-selector" + // Add a fake display name + PKPaymentMethod *paymentMethod = [[PKPaymentMethod alloc] init]; + [paymentMethod performSelector:@selector(setDisplayName:) withObject:@"Master Charge"]; + + [paymentToken performSelector:@selector(setPaymentMethod:) withObject:paymentMethod]; + + [paymentToken performSelector:@selector(setPaymentData:) withObject:data]; + [payment performSelector:@selector(setToken:) withObject:paymentToken]; + [payment performSelector:@selector(setBillingContact:) withObject:contact]; +#pragma clang diagnostic pop + return payment; +} + +#pragma mark - Payment Method + ++ (STPPaymentMethod *)paymentMethod { + return [STPPaymentMethod decodedObjectFromAPIResponse:[self paymentMethodJSON]]; +} + ++ (NSDictionary *)paymentMethodJSON { + return [STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard]; +} + ++ (STPPaymentMethod *)applePayPaymentMethod { + return [STPPaymentMethod decodedObjectFromAPIResponse:[self applePayPaymentMethodJSON]]; +} + ++ (NSDictionary *)applePayPaymentMethodJSON { + return [STPTestUtils jsonNamed:STPTestJSONPaymentMethodApplePay]; +} +@end diff --git a/Stripe/StripeiOSTests/STPFloatingPlaceholderTextFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPFloatingPlaceholderTextFieldSnapshotTests.swift new file mode 100644 index 00000000..b6307cf0 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFloatingPlaceholderTextFieldSnapshotTests.swift @@ -0,0 +1,443 @@ +// +// STPFloatingPlaceholderTextFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/9/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPFloatingPlaceholderTextFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + // MARK: Not Floating + + func testNotFloating_noBackground() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_whiteBackground() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_roundedRectBorderStyle() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_bezelBorderStyle() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_lineBorderStyle() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + // MARK: Floating + + func testFloating_noBackground() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_whiteBackground() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_roundedRectBorderStyle() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_bezelBorderStyle() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_lineBorderStyle() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + // MARK: Right/Left Views Not Floating + + func testNotFloating_noBackground_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_whiteBackground_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_roundedRectBorderStyle_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_bezelBorderStyle_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_lineBorderStyle_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_noBackground_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_whiteBackground_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_roundedRectBorderStyle_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_bezelBorderStyle_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_lineBorderStyle_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_noBackground_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_whiteBackground_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_roundedRectBorderStyle_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_bezelBorderStyle_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testNotFloating_lineBorderStyle_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + // MARK: Right/Left Views Floating + + func testFloating_noBackground_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_whiteBackground_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_roundedRectBorderStyle_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_bezelBorderStyle_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_lineBorderStyle_rightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_noBackground_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_whiteBackground_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_roundedRectBorderStyle_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_bezelBorderStyle_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_lineBorderStyle_leftView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_noBackground_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_whiteBackground_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.backgroundColor = .white + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_roundedRectBorderStyle_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .roundedRect + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_bezelBorderStyle_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .bezel + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } + + func testFloating_lineBorderStyle_leftRightView() { + let textField: STPFloatingPlaceholderTextField = STPFloatingPlaceholderTextField() + textField.placeholder = "Test Placeholder" + textField.text = "Input Text" + textField.leftView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.leftViewMode = .always + textField.rightView = UIImageView(image: STPImageLibrary.unknownCardCardImage()) + textField.rightViewMode = .always + textField.borderStyle = .line + textField.sizeToFit() + STPSnapshotVerifyView(textField) + } +} diff --git a/Stripe/StripeiOSTests/STPFormEncoderTest.swift b/Stripe/StripeiOSTests/STPFormEncoderTest.swift new file mode 100644 index 00000000..b66e07e0 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFormEncoderTest.swift @@ -0,0 +1,210 @@ +// +// STPFormEncoderTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 1/8/15. +// Copyright © 2015 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPTestFormEncodableObject: NSObject, STPFormEncodable { + var additionalAPIParameters: [AnyHashable: Any] = [:] + + @objc var testProperty: String? + @objc var testIgnoredProperty: String? + @objc var testArrayProperty: [AnyHashable]? + @objc var testDictionaryProperty: [AnyHashable: Any]? + @objc var testNestedObjectProperty: STPTestFormEncodableObject? + + class func rootObjectName() -> String? { + return "test_object" + } + + class func propertyNamesToFormFieldNamesMapping() -> [String: String] { + return [ + "testProperty": "test_property", + "testArrayProperty": "test_array_property", + "testDictionaryProperty": "test_dictionary_property", + "testNestedObjectProperty": "test_nested_property", + ] + } +} + +class STPTestNilRootObjectFormEncodableObject: STPTestFormEncodableObject { + override class func rootObjectName() -> String? { + return nil + } +} + +class STPFormEncoderTest: XCTestCase { + // helper test method + func encode(_ object: STPTestFormEncodableObject?) -> String? { + let dictionary = STPFormEncoder.dictionary(forObject: object!) + return URLEncoder.queryString(from: dictionary) + } + + func testFormEncoding_emptyObject() { + let testObject = STPTestFormEncodableObject() + XCTAssertEqual(encode(testObject), "") + } + + func testFormEncoding_normalObject() { + let testObject = STPTestFormEncodableObject() + testObject.testProperty = "success" + testObject.testIgnoredProperty = "ignoreme" + XCTAssertEqual(encode(testObject), "test_object[test_property]=success") + } + + func testFormEncoding_additionalAttributes() { + let testObject = STPTestFormEncodableObject() + testObject.testProperty = "success" + testObject.additionalAPIParameters = [ + "foo": "bar", + "nested": [ + "nested_key": "nested_value" + ], + ] + XCTAssertEqual( + encode(testObject), + "test_object[foo]=bar&test_object[nested][nested_key]=nested_value&test_object[test_property]=success" + ) + } + + func testFormEncoding_arrayValue_empty() { + let testObject = STPTestFormEncodableObject() + testObject.testProperty = "success" + testObject.testArrayProperty = [] + XCTAssertEqual(encode(testObject), "test_object[test_property]=success") + } + + func testFormEncoding_arrayValue() { + let testObject = STPTestFormEncodableObject() + testObject.testProperty = "success" + testObject.testArrayProperty = [NSNumber(value: 1), NSNumber(value: 2), NSNumber(value: 3)] + XCTAssertEqual( + encode(testObject), + "test_object[test_array_property][0]=1&test_object[test_array_property][1]=2&test_object[test_array_property][2]=3&test_object[test_property]=success" + ) + } + + func testFormEncoding_BoolAndNumbers() { + let testObject = STPTestFormEncodableObject() + testObject.testArrayProperty = [ + NSNumber(value: 0), + NSNumber(value: 1), + NSNumber(value: false), + NSNumber(value: true), + NSNumber(value: true), + ] + XCTAssertEqual( + encode(testObject), + """ + test_object[test_array_property][0]=0\ + &test_object[test_array_property][1]=1\ + &test_object[test_array_property][2]=false\ + &test_object[test_array_property][3]=true\ + &test_object[test_array_property][4]=true + """ + ) + } + + func testFormEncoding_arrayOfEncodable() { + let testObject = STPTestFormEncodableObject() + + let inner1 = STPTestFormEncodableObject() + inner1.testProperty = "inner1" + let inner2 = STPTestFormEncodableObject() + inner2.testArrayProperty = ["inner2"] + + testObject.testArrayProperty = [inner1, inner2] + + XCTAssertEqual( + encode(testObject), + """ + test_object[test_array_property][0][test_property]=inner1\ + &test_object[test_array_property][1][test_array_property][0]=inner2 + """ + ) + } + + func testFormEncoding_dictionaryValue_empty() { + let testObject = STPTestFormEncodableObject() + testObject.testProperty = "success" + testObject.testDictionaryProperty = [:] + XCTAssertEqual(encode(testObject), "test_object[test_property]=success") + } + + func testFormEncoding_dictionaryValue() { + let testObject = STPTestFormEncodableObject() + testObject.testProperty = "success" + testObject.testDictionaryProperty = [ + "foo": "bar" + ] + XCTAssertEqual( + encode(testObject), + "test_object[test_dictionary_property][foo]=bar&test_object[test_property]=success" + ) + } + + func testFormEncoding_dictionaryOfEncodable() { + let testObject = STPTestFormEncodableObject() + + let inner1 = STPTestFormEncodableObject() + inner1.testProperty = "inner1" + let inner2 = STPTestFormEncodableObject() + inner2.testArrayProperty = ["inner2"] + + testObject.testDictionaryProperty = [ + "one": inner1, + "two": inner2, + ] + + XCTAssertEqual( + encode(testObject), + """ + test_object[test_dictionary_property][one][test_property]=inner1\ + &test_object[test_dictionary_property][two][test_array_property][0]=inner2 + """ + ) + } + + func testFormEncoding_setOfEncodable() { + let testObject = STPTestFormEncodableObject() + + let inner = STPTestFormEncodableObject() + inner.testProperty = "inner" + + testObject.testArrayProperty = [Set([inner])] + + XCTAssertEqual( + encode(testObject), + "test_object[test_array_property][0][test_property]=inner" + ) + } + + func testFormEncoding_nestedValue() { + let testObject1 = STPTestFormEncodableObject() + let testObject2 = STPTestFormEncodableObject() + testObject2.testProperty = "nested_object" + testObject1.testProperty = "success" + testObject1.testNestedObjectProperty = testObject2 + XCTAssertEqual( + encode(testObject1), + "test_object[test_nested_property][test_property]=nested_object&test_object[test_property]=success" + ) + } + + func testFormEncoding_nilRootObject() { + let testObject = STPTestNilRootObjectFormEncodableObject() + testObject.testProperty = "success" + XCTAssertEqual(encode(testObject), "test_property=success") + } +} diff --git a/Stripe/StripeiOSTests/STPFormTextFieldTest.swift b/Stripe/StripeiOSTests/STPFormTextFieldTest.swift new file mode 100644 index 00000000..83100b32 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFormTextFieldTest.swift @@ -0,0 +1,62 @@ +// +// STPFormTextFieldTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 3/22/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPFormTextFieldTest: XCTestCase { + func testAutoFormattingBehavior_None() { + let sut = STPFormTextField() + sut.autoFormattingBehavior = .none + sut.text = "123456789" + XCTAssertEqual(sut.text, "123456789") + } + + func testAutoFormattingBehavior_PhoneNumbers() { + let sut = STPFormTextField() + sut.autoFormattingBehavior = .phoneNumbers + sut.text = "123456789" + XCTAssertEqual(sut.text, "(123) 456-789") + } + + func testAutoFormattingBehavior_CardNumbers() { + let sut = STPFormTextField() + sut.autoFormattingBehavior = .cardNumbers + sut.text = "4242424242424242" + XCTAssertEqual(sut.text, "4242424242424242") + var range = NSRange() + var value = sut.attributedText!.attribute(.kern, at: 0, effectiveRange: &range) as! Int + XCTAssertEqual(value, 0) + XCTAssertEqual(range.length, Int(3)) + value = sut.attributedText!.attribute(.kern, at: 3, effectiveRange: &range) as! Int + XCTAssertEqual(value, 5) + XCTAssertEqual(range.length, Int(1)) + value = sut.attributedText!.attribute(.kern, at: 4, effectiveRange: &range) as! Int + XCTAssertEqual(value, 0) + XCTAssertEqual(range.length, Int(3)) + value = sut.attributedText!.attribute(.kern, at: 7, effectiveRange: &range) as! Int + XCTAssertEqual(value, 5) + XCTAssertEqual(range.length, Int(1)) + value = sut.attributedText!.attribute(.kern, at: 8, effectiveRange: &range) as! Int + XCTAssertEqual(value, 0) + XCTAssertEqual(range.length, Int(3)) + value = sut.attributedText?.attribute(.kern, at: 11, effectiveRange: &range) as! Int + XCTAssertEqual(value, 5) + XCTAssertEqual(range.length, Int(1)) + value = sut.attributedText?.attribute(.kern, at: 12, effectiveRange: &range) as! Int + XCTAssertEqual(value, 0) + XCTAssertEqual(range.length, Int(4)) + XCTAssertEqual(sut.attributedText!.length, Int(16)) + + sut.placeholder = "enteracardnumber" + XCTAssertNil(sut.attributedPlaceholder!.attribute(.kern, at: 3, effectiveRange: &range)) + } +} diff --git a/Stripe/StripeiOSTests/STPFormViewSnapshotTests.swift b/Stripe/StripeiOSTests/STPFormViewSnapshotTests.swift new file mode 100644 index 00000000..69e17d36 --- /dev/null +++ b/Stripe/StripeiOSTests/STPFormViewSnapshotTests.swift @@ -0,0 +1,165 @@ +// +// STPFormViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/23/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPFormViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testSingleInput() { + let input = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + input.placeholder = "Single input" + let section = STPFormView.Section(rows: [[input]], title: nil, accessoryButton: nil) + let formView = STPFormView(sections: [section]) + formView.frame.size = formView.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + STPSnapshotVerifyView(formView) + } + + func testSingleInputPerRow() { + var rows = [[STPInputTextField]]() + for row in 0..<5 { + let input = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + input.placeholder = "Row \(row)" + rows.append([input]) + } + let section = STPFormView.Section(rows: rows, title: nil, accessoryButton: nil) + let formView = STPFormView(sections: [section]) + formView.frame.size = formView.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + STPSnapshotVerifyView(formView) + } + + func testMultiInputPerRow() { + var rows = [[STPInputTextField]]() + for row in 0..<5 { + var rowInputs = [STPInputTextField]() + for c in ["A", "B", "C"] { + let input = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + input.placeholder = "Row \(row) \(c)" + rowInputs.append(input) + } + + rows.append(rowInputs) + } + let section = STPFormView.Section(rows: rows, title: nil, accessoryButton: nil) + let formView = STPFormView(sections: [section]) + formView.frame.size = formView.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + STPSnapshotVerifyView(formView) + } + + func testMixSingleMultiInputPerRow() { + var rows = [[STPInputTextField]]() + for row in 0..<5 { + var rowInputs = [STPInputTextField]() + for c in ["A", "B", "C"] { + let input = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + input.placeholder = "Row \(row) \(c)" + rowInputs.append(input) + if row % 2 == 0 { + break + } + } + + rows.append(rowInputs) + } + let section = STPFormView.Section(rows: rows, title: nil, accessoryButton: nil) + let formView = STPFormView(sections: [section]) + formView.frame.size = formView.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + STPSnapshotVerifyView(formView) + } + + func testSingleSectionWithTitle() { + var rows = [[STPInputTextField]]() + for row in 0..<5 { + var rowInputs = [STPInputTextField]() + for c in ["A", "B", "C"] { + let input = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + input.placeholder = "Row \(row) \(c)" + rowInputs.append(input) + } + + rows.append(rowInputs) + } + let section = STPFormView.Section(rows: rows, title: "Single Section", accessoryButton: nil) + let formView = STPFormView(sections: [section]) + formView.frame.size = formView.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + STPSnapshotVerifyView(formView) + } + + func testMultiSection() { + var rows1 = [[STPInputTextField]]() + for row in 0..<5 { + var rowInputs = [STPInputTextField]() + for c in ["A", "B", "C"] { + let input = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + input.placeholder = "Row \(row) \(c)" + rowInputs.append(input) + } + + rows1.append(rowInputs) + } + let section1 = STPFormView.Section( + rows: rows1, + title: "First Section", + accessoryButton: nil + ) + + var rows2 = [[STPInputTextField]]() + for row in 0..<5 { + var rowInputs = [STPInputTextField]() + for c in ["A", "B", "C"] { + let input = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + input.placeholder = "Row \(row) \(c)" + rowInputs.append(input) + } + + rows2.append(rowInputs) + } + let section2 = STPFormView.Section( + rows: rows2, + title: "Second Section", + accessoryButton: nil + ) + + let formView = STPFormView(sections: [section1, section2]) + formView.frame.size = formView.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + STPSnapshotVerifyView(formView) + } + +} diff --git a/Stripe/StripeiOSTests/STPGenericInputPickerFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPGenericInputPickerFieldSnapshotTests.swift new file mode 100644 index 00000000..ae04ee33 --- /dev/null +++ b/Stripe/StripeiOSTests/STPGenericInputPickerFieldSnapshotTests.swift @@ -0,0 +1,79 @@ +// +// STPGenericInputPickerFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Mel Ludowise on 2/9/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +final class STPGenericInputPickerFieldSnapshotTests: FBSnapshotTestCase { + + private var field: STPGenericInputPickerField! + + override func setUp() { + super.setUp() + // recordMode = true + + field = STPGenericInputPickerField(dataSource: MockDataSource()) + field.placeholder = "Placeholder" + field.sizeToFit() + field.frame.size.width = 200 + } + + func testEmptySelection() { + STPSnapshotVerifyView(field) + } + + func testWithDefaultSelection() { + // The 0th row should be auto-selected when tapping into the field + field.delegate?.textFieldDidBeginEditing?(field) + + STPSnapshotVerifyView(field) + } + + func testWithExplicitSelection() { + let index = 5 + + // Explicitly select a row + field.pickerView.selectRow(index, inComponent: 0, animated: false) + + // Because we're interacting with the picker programatically, we need to explicitly + // call `resignFirstResponder` to commit the changes. + _ = field.resignFirstResponder() + + STPSnapshotVerifyView(field) + } +} + +/// Simple DataSource that displays numbers 0–9 +private final class MockDataSource: STPGenericInputPickerFieldDataSource { + func numberOfRows() -> Int { + return 10 + } + + func inputPickerField( + _ pickerField: STPGenericInputPickerField, + titleForRow row: Int + ) + -> String? + { + return "\(row)" + } + + func inputPickerField( + _ pickerField: STPGenericInputPickerField, + inputValueForRow row: Int + ) + -> String? + { + return "\(row)" + } +} diff --git a/Stripe/StripeiOSTests/STPGenericInputPickerFieldValidatorTest.swift b/Stripe/StripeiOSTests/STPGenericInputPickerFieldValidatorTest.swift new file mode 100644 index 00000000..0a04d8b8 --- /dev/null +++ b/Stripe/StripeiOSTests/STPGenericInputPickerFieldValidatorTest.swift @@ -0,0 +1,45 @@ +// +// STPGenericInputPickerFieldValidatorTest.swift +// StripeiOS Tests +// +// Created by Mel Ludowise on 2/9/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +final class STPGenericInputPickerFieldValidatorTest: XCTestCase { + + private var validator: STPGenericInputPickerField.Validator! + + override func setUp() { + super.setUp() + + validator = STPGenericInputPickerField.Validator() + } + + func testInitial() { + XCTAssertEqual(validator.validationState, .unknown) + } + + func testValidInput() { + validator.inputValue = "hello" + XCTAssertEqual(validator.validationState, .valid(message: nil)) + } + + func testEmptyInput() { + validator.inputValue = "" + XCTAssertEqual(validator.validationState, .incomplete(description: nil)) + } + + func testNilInput() { + validator.inputValue = nil + XCTAssertEqual(validator.validationState, .incomplete(description: nil)) + } +} diff --git a/Stripe/StripeiOSTests/STPGenericInputTextFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPGenericInputTextFieldSnapshotTests.swift new file mode 100644 index 00000000..8dc5e4ca --- /dev/null +++ b/Stripe/StripeiOSTests/STPGenericInputTextFieldSnapshotTests.swift @@ -0,0 +1,42 @@ +// +// STPGenericInputTextFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 12/2/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPGenericInputTextFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testEmpty() { + let field = STPGenericInputTextField(placeholder: "Empty") + field.sizeToFit() + field.frame.size.width = 200 + + STPSnapshotVerifyView(field) + } + + func testWithContent() { + let field = STPGenericInputTextField(placeholder: "Has Content") + field.sizeToFit() + field.frame.size.width = 200 + field.text = "Hello" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + +} diff --git a/Stripe/StripeiOSTests/STPImageLibraryTest.swift b/Stripe/StripeiOSTests/STPImageLibraryTest.swift new file mode 100644 index 00000000..ab800f44 --- /dev/null +++ b/Stripe/StripeiOSTests/STPImageLibraryTest.swift @@ -0,0 +1,371 @@ +// +// STPImageLibraryTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 4/7/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class STPImageLibraryTestSwift: XCTestCase { + + static let cardBrands: [STPCardBrand] = [ + .amex, + .cartesBancaires, + .dinersClub, + .discover, + .JCB, + .mastercard, + .unionPay, + .unknown, + .visa, + ] + + func testCardIconMethods() { + STPAssertEqualImages( + STPImageLibrary.applePayCardImage(), + STPImageLibrary.safeImageNamed("stp_card_applepay", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.amexCardImage(), + STPImageLibrary.safeImageNamed("stp_card_amex", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.dinersClubCardImage(), + STPImageLibrary.safeImageNamed("stp_card_diners", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.discoverCardImage(), + STPImageLibrary.safeImageNamed("stp_card_discover", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.jcbCardImage(), + STPImageLibrary.safeImageNamed("stp_card_jcb", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.mastercardCardImage(), + STPImageLibrary.safeImageNamed("stp_card_mastercard", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.unionPayCardImage(), + STPImageLibrary.safeImageNamed("stp_card_unionpay", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.visaCardImage(), + STPImageLibrary.safeImageNamed("stp_card_visa", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.unknownCardCardImage(), + STPImageLibrary.safeImageNamed("stp_card_unknown", templateIfAvailable: false) + ) + } + + func testBrandImageForCardBrand() { + for brand in Self.cardBrands { + let image = STPImageLibrary.brandImage(for: brand, template: false) + + switch brand { + case .visa: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_visa", templateIfAvailable: false) + ) + case .amex: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_amex", templateIfAvailable: false) + ) + case .mastercard: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_mastercard", + templateIfAvailable: false + ) + ) + case .discover: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_discover", templateIfAvailable: false) + ) + case .JCB: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_jcb", templateIfAvailable: false) + ) + case .dinersClub: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_diners", templateIfAvailable: false) + ) + case .unionPay: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_unionpay", templateIfAvailable: false) + ) + case .cartesBancaires: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_cartes_bancaires", templateIfAvailable: false) + ) + case .unknown: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_unknown", templateIfAvailable: false) + ) + } + } + } + + func testTemplatedBrandImageForCardBrand() { + for brand in Self.cardBrands { + let image = STPImageLibrary.templatedBrandImage(for: brand) + + switch brand { + case .visa: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_visa_template", + templateIfAvailable: true + ) + ) + case .amex: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_amex_template", + templateIfAvailable: true + ) + ) + case .mastercard: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_mastercard_template", + templateIfAvailable: true + ) + ) + case .discover: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_discover_template", + templateIfAvailable: true + ) + ) + case .JCB: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_jcb_template", + templateIfAvailable: true + ) + ) + case .dinersClub: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_diners_template", + templateIfAvailable: true + ) + ) + case .unionPay: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_unionpay_template", + templateIfAvailable: true + ) + ) + case .cartesBancaires: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_cartes_bancaires_template", + templateIfAvailable: true + ) + ) + case .unknown: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_unknown", templateIfAvailable: true) + ) + } + } + } + + func testCVCImageForCardBrand() { + for brand in Self.cardBrands { + let image = STPImageLibrary.cvcImage(for: brand) + + switch brand { + case .amex: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_cvc_amex", templateIfAvailable: false) + ) + default: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_cvc", templateIfAvailable: false) + ) + } + } + } + + func testErrorImageForCardBrand() { + for brand in Self.cardBrands { + let image = STPImageLibrary.errorImage(for: brand) + + switch brand { + case .amex: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed( + "stp_card_error_amex", + templateIfAvailable: false + ) + ) + default: + STPAssertEqualImages( + image, + STPImageLibrary.safeImageNamed("stp_card_error", templateIfAvailable: false) + ) + } + } + } + + func testMiscImages() { + STPAssertEqualImages( + STPLegacyImageLibrary.addIcon(), + STPLegacyImageLibrary.safeImageNamed("stp_icon_add", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPImageLibrary.bankIcon(), + STPImageLibrary.safeImageNamed("stp_icon_bank", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPLegacyImageLibrary.checkmarkIcon(), + STPLegacyImageLibrary.safeImageNamed("stp_icon_checkmark", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPLegacyImageLibrary.largeCardFrontImage(), + STPLegacyImageLibrary.safeImageNamed("stp_card_form_front", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPLegacyImageLibrary.largeCardBackImage(), + STPLegacyImageLibrary.safeImageNamed("stp_card_form_back", templateIfAvailable: false) + ) + STPAssertEqualImages( + STPLegacyImageLibrary.largeCardAmexCVCImage(), + STPLegacyImageLibrary.safeImageNamed( + "stp_card_form_amex_cvc", + templateIfAvailable: false + ) + ) + STPAssertEqualImages( + STPLegacyImageLibrary.largeShippingImage(), + STPLegacyImageLibrary.safeImageNamed("stp_shipping_form", templateIfAvailable: false) + ) + } + + func testFPXImages() { + // Probably better to make STPFPXBankBrand conform to CaseIterable, + // but let's not change behavior of a legacy product just for this test. + for i in 0...(STPFPXBankBrand.unknown.rawValue - 1) { + let brand = STPFPXBankBrand(rawValue: i)! + let bankIdentifier = STPFPXBank.identifierFrom(brand)! + let bankImageName = "stp_bank_fpx_" + bankIdentifier + STPAssertEqualImages( + STPLegacyImageLibrary.fpxBrandImage(for: brand), + STPLegacyImageLibrary.safeImageNamed(bankImageName, templateIfAvailable: false) + ) + + } + } + + func testBankIconCodeImagesExist() { + for iconCode in PaymentSheetImageLibrary.BankIconCodeRegexes.keys { + XCTAssertNotNil( + PaymentSheetImageLibrary.bankIcon(for: iconCode), + "Missing image for \(iconCode)" + ) + } + } + + func testBankNameToIconCode() { + // bank of america + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "bank of america"), "boa") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "BANK of AMERICA"), "boa") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "BANKof AMERICA"), "default") + + // capital one + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "capital one"), "capitalone") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Capital One"), "capitalone") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Capital One"), "default") + + // citibank + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "citibank"), "citibank") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Citibank"), "citibank") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Citi Bank"), "default") + + // compass + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "bbva"), "compass") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "BBVA"), "compass") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "compass"), "compass") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "b b v a"), "default") + + // morganchase + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Morgan Chase"), "morganchase") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "morgan chase"), "morganchase") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "jp morgan"), "morganchase") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "JP Morgan"), "morganchase") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Chase"), "morganchase") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "chase"), "morganchase") + + // pnc + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "pncbank"), "pnc") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "PNCBANK"), "pnc") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "pnc bank"), "pnc") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "PNC Bank"), "pnc") + + // suntrust + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "suntrust"), "suntrust") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "SUNTRUST"), "suntrust") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "suntrust bank"), "suntrust") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Suntrust Bank"), "suntrust") + + // svb + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Silicon Valley Bank"), "svb") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "SILICON VALLEY BANK"), "svb") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "SILICONVALLEYBANK"), "default") + + // usaa + XCTAssertEqual( + PaymentSheetImageLibrary.bankIconCode(for: "USAA Federal Savings Bank"), + "usaa" + ) + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "USAA Bank"), "usaa") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "USAA Savings Bank"), "default") + + // usbank + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "US Bank"), "usbank") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "U.S. Bank"), "usbank") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "u.s. Bank"), "usbank") + + // wellsfargo + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Wells Fargo"), "wellsfargo") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "WELLS FARGO"), "wellsfargo") + XCTAssertEqual(PaymentSheetImageLibrary.bankIconCode(for: "Well's Fargo"), "default") + } + +} diff --git a/Stripe/StripeiOSTests/STPInputTextFieldFormatterTests.swift b/Stripe/StripeiOSTests/STPInputTextFieldFormatterTests.swift new file mode 100644 index 00000000..8be567ca --- /dev/null +++ b/Stripe/StripeiOSTests/STPInputTextFieldFormatterTests.swift @@ -0,0 +1,81 @@ +// +// STPInputTextFieldFormatterTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPInputTextFieldFormatterTests: XCTestCase { + + func testAllowsDeletion() { + let formatter = STPInputTextFieldFormatter() + let textField = UITextField() + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 2), + replacementString: "" + ), + "Should allow deletion on empty" + ) + textField.text = "Hi" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 2), + replacementString: "" + ), + "Should allow full deletion" + ) + textField.text = "Hello" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 4, length: 1), + replacementString: "" + ), + "Should allow partial deletion at end" + ) + textField.text = "Hello" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 3, length: 1), + replacementString: "" + ), + "Should allow partial deletion in middle" + ) + textField.text = "Hello" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 1), + replacementString: "" + ), + "Should allow partial deletion at beginning" + ) + } + + func testAllowsInitialSpaceForAutofill() { + let formatter = STPInputTextFieldFormatter() + let textField = UITextField() + textField.textContentType = .nickname + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 0), + replacementString: " " + ) + ) + } + +} diff --git a/Stripe/StripeiOSTests/STPInputTextFieldValidatorTests.swift b/Stripe/StripeiOSTests/STPInputTextFieldValidatorTests.swift new file mode 100644 index 00000000..542ff6b3 --- /dev/null +++ b/Stripe/StripeiOSTests/STPInputTextFieldValidatorTests.swift @@ -0,0 +1,64 @@ +// +// STPInputTextFieldValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPInputTextFieldValidatorTests: XCTestCase { + + class ObserverWithExpectation: NSObject, STPFormInputValidationObserver { + + let expectation: XCTestExpectation + init( + _ expectation: XCTestExpectation + ) { + self.expectation = expectation + super.init() + } + + func validationDidUpdate( + to state: STPValidatedInputState, + from previousState: STPValidatedInputState, + for unformattedInput: String?, + in input: STPFormInput + ) { + expectation.fulfill() + } + + } + + func testUpdatingObservers() { + let textField = STPInputTextField( + formatter: STPInputTextFieldFormatter(), + validator: STPInputTextFieldValidator() + ) + let expectationForNewValue = expectation(description: "Receives expectation with new value") + let observerForNewValue = ObserverWithExpectation(expectationForNewValue) + let validator = textField.validator + + validator.addObserver(observerForNewValue) + validator.validationState = STPValidatedInputState.valid(message: nil) + wait(for: [expectationForNewValue], timeout: 1) + validator.removeObserver(observerForNewValue) + + let expectationForSameValue = expectation( + description: "Receives expectation with same value" + ) + let observerForSameValue = ObserverWithExpectation(expectationForSameValue) + validator.validationState = .incomplete(description: nil) + validator.addObserver(observerForSameValue) + validator.validationState = .incomplete(description: nil) + wait(for: [expectationForSameValue], timeout: 1) + } + +} diff --git a/Stripe/StripeiOSTests/STPIntentActionAlipayHandleRedirectTest.swift b/Stripe/StripeiOSTests/STPIntentActionAlipayHandleRedirectTest.swift new file mode 100644 index 00000000..73a55ab7 --- /dev/null +++ b/Stripe/StripeiOSTests/STPIntentActionAlipayHandleRedirectTest.swift @@ -0,0 +1,55 @@ +// +// STPIntentActionAlipayHandleRedirectTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 12/2/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPIntentActionAlipayHandleRedirectTest: XCTestCase { + func testMarlinReturnURL() throws { + let testJSONString = """ + { + "alipay_handle_redirect": { + "native_url": "alipay://alipayclient/?%7B%22dataString%22%3A%22_input_charset=utf-8%26app_pay=Y%26currency=USD%26forex_biz=FP%26notify_url=https%253A%252F%252Fhooks.stripe.com%252Falipay%252Falipay%252Fhook%252REDACTED%252Fsrc_REDACTED%26out_trade_no=src_REDACTED%26partner=REDACTED%26payment_type=1%26product_code=NEW_WAP_OVERSEAS_SELLER%26return_url=https%253A%252F%252Fhooks.stripe.com%252Fadapter%252Falipay%252Fredirect%252Fcomplete%252Fsrc_REDACTED%252Fsrc_client_secret_REDACTED%26secondary_merchant_id=acct_REDACTED%26secondary_merchant_industry=5734%26secondary_merchant_name=Yuki-Test%26sendFormat=normal%26service=create_forex_trade_wap%26sign=REDACTED%26sign_type=MD5%26subject=Yuki-Test%26supplier=Yuki-Test%26timeout_rule=20m%26total_fee=1.00%26bizcontext=%7B%5C%22av%5C%22%3A%5C%221.0%5C%22%2C%5C%22ty%5C%22%3A%5C%22ios_lite%5C%22%2C%5C%22appkey%5C%22%3A%5C%22123456789%5C%22%2C%5C%22sv%5C%22%3A%5C%22h.a.3.2.5%5C%22%2C%5C%22an%5C%22%3A%5C%22com.stripe.CustomSDKExample%5C%22%7D%22%2C%22fromAppUrlScheme%22%3A%22payments-example%22%2C%22requestType%22%3A%22SafePay%22%7D", + "return_url": "payments-example://safepay/", + "url": "https://hooks.stripe.com/redirect/authenticate/src_REDACTED?client_secret=src_client_secret_REDACTED" + }, + "type": "alipay_handle_redirect" + } + """ + guard + let testJSONData = testJSONString.data(using: .utf8), + let json = try? JSONSerialization.jsonObject( + with: testJSONData, + options: .allowFragments + ) as? [AnyHashable: Any], + let nextAction = STPIntentAction.decodedObject(fromAPIResponse: json), + let alipayRedirect = nextAction.alipayHandleRedirect + else { + XCTFail() + return + } + XCTAssertEqual( + alipayRedirect.nativeURL, + URL( + string: + "alipay://alipayclient/?%7B%22dataString%22%3A%22_input_charset=utf-8%26app_pay=Y%26currency=USD%26forex_biz=FP%26notify_url=https%253A%252F%252Fhooks.stripe.com%252Falipay%252Falipay%252Fhook%252REDACTED%252Fsrc_REDACTED%26out_trade_no=src_REDACTED%26partner=REDACTED%26payment_type=1%26product_code=NEW_WAP_OVERSEAS_SELLER%26return_url=https%253A%252F%252Fhooks.stripe.com%252Fadapter%252Falipay%252Fredirect%252Fcomplete%252Fsrc_REDACTED%252Fsrc_client_secret_REDACTED%26secondary_merchant_id=acct_REDACTED%26secondary_merchant_industry=5734%26secondary_merchant_name=Yuki-Test%26sendFormat=normal%26service=create_forex_trade_wap%26sign=REDACTED%26sign_type=MD5%26subject=Yuki-Test%26supplier=Yuki-Test%26timeout_rule=20m%26total_fee=1.00%26bizcontext=%7B%5C%22av%5C%22%3A%5C%221.0%5C%22%2C%5C%22ty%5C%22%3A%5C%22ios_lite%5C%22%2C%5C%22appkey%5C%22%3A%5C%22123456789%5C%22%2C%5C%22sv%5C%22%3A%5C%22h.a.3.2.5%5C%22%2C%5C%22an%5C%22%3A%5C%22com.stripe.CustomSDKExample%5C%22%7D%22%2C%22fromAppUrlScheme%22%3A%22payments-example%22%2C%22requestType%22%3A%22SafePay%22%7D" + ) + ) + XCTAssertEqual(alipayRedirect.returnURL, URL(string: "payments-example://safepay/")) + XCTAssertEqual( + alipayRedirect.marlinReturnURL, + URL( + string: + "https://hooks.stripe.com/adapter/alipay/redirect/complete/src_REDACTED/src_client_secret_REDACTED" + ) + ) + } +} diff --git a/Stripe/StripeiOSTests/STPIntentActionTest.m b/Stripe/StripeiOSTests/STPIntentActionTest.m new file mode 100644 index 00000000..88d50903 --- /dev/null +++ b/Stripe/StripeiOSTests/STPIntentActionTest.m @@ -0,0 +1,93 @@ +// +// STPIntentActionTest.m +// StripeiOS Tests +// +// Created by Daniel Jackson on 11/7/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPIntentActionTest : XCTestCase + +@end + +@implementation STPIntentActionTest + +- (void)testDecodedObjectFromAPIResponseRedirectToURL { + + STPIntentAction *(^decode)(NSDictionary *) = ^STPIntentAction *(NSDictionary * dict) { + return [STPIntentAction decodedObjectFromAPIResponse:dict]; + }; + + XCTAssertNil(decode(nil)); + XCTAssertNil(decode(@{})); + XCTAssertNil(decode(@{ @"redirect_to_url": @{@"url": @"http://stripe.com"} }), + @"fails without type"); + + STPIntentAction *missingDetails = decode(@{ + @"type": @"redirect_to_url" + }); + XCTAssertNotNil(missingDetails); + XCTAssertEqual(missingDetails.type, STPIntentActionTypeUnknown, + @"Type becomes unknown if the redirect_to_url details are missing"); + + STPIntentAction *badURL = decode(@{ + @"type": @"redirect_to_url", + @"redirect_to_url": @{ + @"url": @"not a url" + } + }); + XCTAssertNotNil(badURL); + XCTAssertEqual(badURL.type, STPIntentActionTypeUnknown, + @"Type becomes unknown if the redirect_to_url details don't have a valid URL"); + + STPIntentAction *missingReturnURL = decode(@{ + @"type": @"redirect_to_url", + @"redirect_to_url": @{ + @"url": @"https://stripe.com/" + } + }); + XCTAssertNotNil(missingReturnURL); + XCTAssertEqual(missingReturnURL.type, STPIntentActionTypeRedirectToURL, + @"Missing return_url won't prevent it from decoding"); + XCTAssertNotNil(missingReturnURL.redirectToURL.url); + XCTAssertEqualObjects(missingReturnURL.redirectToURL.url, + [NSURL URLWithString:@"https://stripe.com/"]); + XCTAssertNil(missingReturnURL.redirectToURL.returnURL); + + STPIntentAction *badReturnURL = decode(@{ + @"type": @"redirect_to_url", + @"redirect_to_url": @{ + @"url": @"https://stripe.com/", + @"return_url": @"not a url" + } + }); + XCTAssertNotNil(badReturnURL); + XCTAssertEqual(badReturnURL.type, STPIntentActionTypeRedirectToURL, + @"invalid return_url won't prevent it from decoding"); + XCTAssertNotNil(badReturnURL.redirectToURL.url); + XCTAssertEqualObjects(badReturnURL.redirectToURL.url, + [NSURL URLWithString:@"https://stripe.com/"]); + XCTAssertNil(badReturnURL.redirectToURL.returnURL); + + + STPIntentAction *complete = decode(@{ + @"type": @"redirect_to_url", + @"redirect_to_url": @{ + @"url": @"https://stripe.com/", + @"return_url": @"my-app://payment-complete" + } + }); + XCTAssertNotNil(complete); + XCTAssertEqual(complete.type, STPIntentActionTypeRedirectToURL); + XCTAssertNotNil(complete.redirectToURL.url); + XCTAssertEqualObjects(complete.redirectToURL.url, + [NSURL URLWithString:@"https://stripe.com/"]); + XCTAssertNotNil(complete.redirectToURL.returnURL); + XCTAssertEqualObjects(complete.redirectToURL.returnURL, + [NSURL URLWithString:@"my-app://payment-complete"]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPIntentActionTypeTest.swift b/Stripe/StripeiOSTests/STPIntentActionTypeTest.swift new file mode 100644 index 00000000..e1879c45 --- /dev/null +++ b/Stripe/StripeiOSTests/STPIntentActionTypeTest.swift @@ -0,0 +1,48 @@ +// +// STPIntentActionTypeTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 9/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPIntentActionTypeTest: XCTestCase { + + func testTypeFromString() { + XCTAssertEqual( + STPIntentActionType(string: "redirect_to_url"), + STPIntentActionType.redirectToURL + ) + XCTAssertEqual( + STPIntentActionType(string: "REDIRECT_TO_URL"), + STPIntentActionType.redirectToURL + ) + + XCTAssertEqual( + STPIntentActionType(string: "use_stripe_sdk"), + STPIntentActionType.useStripeSDK + ) + XCTAssertEqual( + STPIntentActionType(string: "USE_STRIPE_SDK"), + STPIntentActionType.useStripeSDK + ) + + XCTAssertEqual( + STPIntentActionType(string: "garbage"), + STPIntentActionType.unknown + ) + XCTAssertEqual( + STPIntentActionType(string: "GARBAGE"), + STPIntentActionType.unknown + ) + } + +} diff --git a/Stripe/StripeiOSTests/STPIntentActionWeChatPayRedirectToAppTest.swift b/Stripe/StripeiOSTests/STPIntentActionWeChatPayRedirectToAppTest.swift new file mode 100644 index 00000000..5e476ad1 --- /dev/null +++ b/Stripe/StripeiOSTests/STPIntentActionWeChatPayRedirectToAppTest.swift @@ -0,0 +1,44 @@ +// +// STPIntentActionWeChatPayRedirectToAppTest.swift +// StripeiOS Tests +// +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPIntentActionWeChatPayRedirectToAppTest: XCTestCase { + func testActionNativeURL() throws { + let testJSONString = """ + { + "wechat_pay_redirect_to_ios_app": { + "native_url": "weixin://app/value:wx12345a1234b1234c/pay/?package=Sign=WXPay&appid=wx12345a1234b1234c&partnerid=123456789&prepayid=wx12345a1234b1234c&noncestr=12345×tamp=12345&sign=12341234", + }, + "type": "wechat_pay_redirect_to_ios_app" + } + """ + guard + let testJSONData = testJSONString.data(using: .utf8), + let json = try? JSONSerialization.jsonObject( + with: testJSONData, + options: .allowFragments + ) as? [AnyHashable: Any], + let nextAction = STPIntentAction.decodedObject(fromAPIResponse: json), + let weChatPayRedirectToApp = nextAction.weChatPayRedirectToApp + else { + XCTFail() + return + } + XCTAssertEqual( + weChatPayRedirectToApp.nativeURL, + URL( + string: + "weixin://app/value:wx12345a1234b1234c/pay/?package=Sign=WXPay&appid=wx12345a1234b1234c&partnerid=123456789&prepayid=wx12345a1234b1234c&noncestr=12345×tamp=12345&sign=12341234" + ) + ) + } +} diff --git a/Stripe/StripeiOSTests/STPIntentWithPreferencesTest.swift b/Stripe/StripeiOSTests/STPIntentWithPreferencesTest.swift new file mode 100644 index 00000000..12f1490e --- /dev/null +++ b/Stripe/StripeiOSTests/STPIntentWithPreferencesTest.swift @@ -0,0 +1,154 @@ +// +// STPIntentWithPreferencesTest.swift +// StripeiOS Tests +// +// Created by Jaime Park on 6/23/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) @_spi(ExperimentalPaymentSheetDecouplingAPI) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPIntentWithPreferencesTest: XCTestCase { + private let paymentIntentClientSecret = + "pi_1H5J4RFY0qyl6XeWFTpgue7g_secret_1SS59M0x65qWMaX2wEB03iwVE" + private let setupIntentClientSecret = + "seti_1GGCuIFY0qyl6XeWVfbQK6b3_secret_GnoX2tzX2JpvxsrcykRSVna2lrYLKew" + + func testPaymentIntentWithPreferences() { + let expectation = XCTestExpectation(description: "Retrieve Payment Intent With Preferences") + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + client.retrievePaymentIntentWithPreferences(withClientSecret: paymentIntentClientSecret) { + result in + switch result { + case .success(let paymentIntentWithPreferences): + expectation.fulfill() + // Check for required PI fields + XCTAssertEqual(paymentIntentWithPreferences.stripeId, "pi_1H5J4RFY0qyl6XeWFTpgue7g") + XCTAssertEqual( + paymentIntentWithPreferences.clientSecret, + self.paymentIntentClientSecret + ) + XCTAssertEqual(paymentIntentWithPreferences.amount, 2000) + XCTAssertEqual(paymentIntentWithPreferences.currency, "usd") + XCTAssertEqual( + paymentIntentWithPreferences.status, + STPPaymentIntentStatus.succeeded + ) + XCTAssertEqual(paymentIntentWithPreferences.livemode, false) + XCTAssertEqual( + paymentIntentWithPreferences.paymentMethodTypes, + STPPaymentMethod.types(from: ["card"]) + ) + // Check for ordered payment method types + XCTAssertNotNil(paymentIntentWithPreferences.orderedPaymentMethodTypes) + XCTAssertEqual( + paymentIntentWithPreferences.orderedPaymentMethodTypes, + [STPPaymentMethodType.card] + ) + case .failure(let error): + print(error) + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testSetupIntentWithPreferences() { + let expectation = XCTestExpectation(description: "Retrieve Setup Intent With Preferences") + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + client.retrieveSetupIntentWithPreferences(withClientSecret: setupIntentClientSecret) { + result in + switch result { + case .success(let setupIntentWithPreferences): + expectation.fulfill() + // Check required SI fields + XCTAssertEqual(setupIntentWithPreferences.stripeID, "seti_1GGCuIFY0qyl6XeWVfbQK6b3") + XCTAssertEqual( + setupIntentWithPreferences.clientSecret, + self.setupIntentClientSecret + ) + XCTAssertEqual(setupIntentWithPreferences.status, .requiresPaymentMethod) + XCTAssertEqual( + setupIntentWithPreferences.paymentMethodTypes, + STPPaymentMethod.types(from: ["card"]) + ) + // Check for ordered payment method types + XCTAssertNotNil(setupIntentWithPreferences.orderedPaymentMethodTypes) + XCTAssertEqual( + setupIntentWithPreferences.orderedPaymentMethodTypes, + [STPPaymentMethodType.card] + ) + case .failure(let error): + print(error) + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testRetrieveElementSession_deferredPayment() { + let expectation = XCTestExpectation(description: "Retrieve ElementsSession") + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let intentConfig = PaymentSheet.IntentConfiguration(mode: .payment(amount: 2000, + currency: "USD", + setupFutureUsage: .onSession), + paymentMethodTypes: ["card", "cashapp"], + confirmHandler: { _, _ in }) + + client.retrieveElementsSession(withIntentConfig: intentConfig) { result in + switch result { + case .success(let elementsSession): + XCTAssertNotNil(elementsSession) + XCTAssertEqual(elementsSession.countryCode, "US") + XCTAssertNotNil(elementsSession.linkSettings) + XCTAssertNotNil(elementsSession.paymentMethodSpecs) + XCTAssertEqual( + elementsSession.orderedPaymentMethodTypes, + [STPPaymentMethodType.card, STPPaymentMethodType.cashApp] + ) + + expectation.fulfill() + case .failure(let error): + print(error) + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } + + func testRetrieveElementSession_deferredSetup() { + let expectation = XCTestExpectation(description: "Retrieve ElementsSession") + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let intentConfig = PaymentSheet.IntentConfiguration(mode: .setup(currency: "USD", + setupFutureUsage: .offSession), + paymentMethodTypes: ["card", "cashapp"], + confirmHandler: { _, _ in }) + + client.retrieveElementsSession(withIntentConfig: intentConfig) { result in + switch result { + case .success(let elementsSession): + XCTAssertNotNil(elementsSession) + XCTAssertEqual(elementsSession.countryCode, "US") + XCTAssertNotNil(elementsSession.linkSettings) + XCTAssertNotNil(elementsSession.paymentMethodSpecs) + XCTAssertEqual( + elementsSession.orderedPaymentMethodTypes, + [STPPaymentMethodType.card, STPPaymentMethodType.cashApp] + ) + + expectation.fulfill() + case .failure(let error): + print(error) + } + } + wait(for: [expectation], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPLabeledFormTextFieldViewSnapshotTests.m b/Stripe/StripeiOSTests/STPLabeledFormTextFieldViewSnapshotTests.m new file mode 100644 index 00000000..cd3266a8 --- /dev/null +++ b/Stripe/StripeiOSTests/STPLabeledFormTextFieldViewSnapshotTests.m @@ -0,0 +1,36 @@ +// +// STPLabeledFormTextFieldViewSnapshotTests.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + + +#import "STPTestUtils.h" + +@import iOSSnapshotTestCaseCore; + +@interface STPLabeledFormTextFieldViewSnapshotTests : FBSnapshotTestCase + +@end + +@implementation STPLabeledFormTextFieldViewSnapshotTests + +//- (void)setUp { +// [super setUp]; +// +// self.recordMode = YES; +//} + +- (void)testAppearance { + STPFormTextField *formTextField = [[STPFormTextField alloc] init]; + formTextField.placeholder = @"A placeholder"; + formTextField.placeholderColor = [UIColor lightGrayColor]; + STPLabeledFormTextFieldView *labeledFormField = [[STPLabeledFormTextFieldView alloc] initWithFormLabel:@"Test Label" textField:formTextField]; + labeledFormField.formBackgroundColor = [UIColor whiteColor]; + labeledFormField.frame = CGRectMake(0.f, 0.f, 320.f, 44.f); + STPSnapshotVerifyView(labeledFormField, @"STPLabeledFormTextFieldView.defaultAppearance"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPLabeledMultiFormTextFieldViewSnapshotTests.swift b/Stripe/StripeiOSTests/STPLabeledMultiFormTextFieldViewSnapshotTests.swift new file mode 100644 index 00000000..2a154fbe --- /dev/null +++ b/Stripe/StripeiOSTests/STPLabeledMultiFormTextFieldViewSnapshotTests.swift @@ -0,0 +1,44 @@ +// +// STPLabeledMultiFormTextFieldViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPLabeledMultiFormTextFieldViewSnapshotTests: FBSnapshotTestCase { + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func testAppearance() { + let formTextField1 = STPFormTextField() + formTextField1.placeholder = "Placeholder 1" + formTextField1.placeholderColor = UIColor.lightGray + + let formTextField2 = STPFormTextField() + formTextField2.placeholder = "Placeholder 2" + formTextField2.placeholderColor = UIColor.lightGray + + let labeledFormField = STPLabeledMultiFormTextFieldView( + formLabel: "Test Label", + firstTextField: formTextField1, + secondTextField: formTextField2 + ) + labeledFormField.formBackgroundColor = UIColor.white + labeledFormField.frame = CGRect(x: 0.0, y: 0.0, width: 320.0, height: 62.0) + STPSnapshotVerifyView( + labeledFormField, + identifier: "STPLabeledMultiFormTextFieldView.defaultAppearance" + ) + } +} diff --git a/Stripe/StripeiOSTests/STPMandateCustomerAcceptanceParamsTest.swift b/Stripe/StripeiOSTests/STPMandateCustomerAcceptanceParamsTest.swift new file mode 100644 index 00000000..66f1e0d5 --- /dev/null +++ b/Stripe/StripeiOSTests/STPMandateCustomerAcceptanceParamsTest.swift @@ -0,0 +1,44 @@ +// +// STPMandateCustomerAcceptanceParamsTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/18/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPMandateCustomerAcceptanceParamsTest: XCTestCase { + func testRootObjectName() { + XCTAssertEqual(STPMandateCustomerAcceptanceParams.rootObjectName(), "customer_acceptance") + } + + func testEncoding() { + let onlineParams = STPMandateOnlineParams(ipAddress: "", userAgent: "") + onlineParams.inferFromClient = NSNumber(value: true) + var params = STPMandateCustomerAcceptanceParams(type: .online, onlineParams: onlineParams)! + + var paramsAsDict = STPFormEncoder.dictionary(forObject: params) + var expected = [ + "customer_acceptance": [ + "type": "online", + "online": [ + "infer_from_client": NSNumber(value: true) + ], + ], + ] + XCTAssertEqual(paramsAsDict as NSDictionary, expected as NSDictionary) + + params = STPMandateCustomerAcceptanceParams(type: .offline, onlineParams: nil)! + paramsAsDict = STPFormEncoder.dictionary(forObject: params) + expected = [ + "customer_acceptance": [ + "type": "offline" + ], + ] + XCTAssertEqual(paramsAsDict as NSDictionary, expected as NSDictionary) + } +} diff --git a/Stripe/StripeiOSTests/STPMandateDataParamsTest.swift b/Stripe/StripeiOSTests/STPMandateDataParamsTest.swift new file mode 100644 index 00000000..587d8f17 --- /dev/null +++ b/Stripe/StripeiOSTests/STPMandateDataParamsTest.swift @@ -0,0 +1,42 @@ +// +// STPMandateDataParamsTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/18/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPMandateDataParamsTest: XCTestCase { + func testRootObjectName() { + XCTAssertEqual(STPMandateDataParams.rootObjectName(), "mandate_data") + } + + func testEncoding() { + let onlineParams = STPMandateOnlineParams(ipAddress: "", userAgent: "") + onlineParams.inferFromClient = NSNumber(value: true) + let customerAcceptanceParams = STPMandateCustomerAcceptanceParams( + type: .online, + onlineParams: onlineParams + )! + + let params = STPMandateDataParams(customerAcceptance: customerAcceptanceParams) + + let paramsAsDict = STPFormEncoder.dictionary(forObject: params) + let expected = [ + "mandate_data": [ + "customer_acceptance": [ + "type": "online", + "online": [ + "infer_from_client": true + ], + ], + ], + ] + XCTAssertEqual(paramsAsDict as NSDictionary, expected as NSDictionary) + } +} diff --git a/Stripe/StripeiOSTests/STPMandateOnlineParamsTest.swift b/Stripe/StripeiOSTests/STPMandateOnlineParamsTest.swift new file mode 100644 index 00000000..990e02b2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPMandateOnlineParamsTest.swift @@ -0,0 +1,40 @@ +// +// STPMandateOnlineParamsTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/18/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPMandateOnlineParamsTest: XCTestCase { + func testRootObjectName() { + XCTAssertEqual(STPMandateOnlineParams.rootObjectName(), "online") + } + + func testEncoding() { + var params = STPMandateOnlineParams(ipAddress: "test_ip_address", userAgent: "a_user_agent") + var paramsAsDict = STPFormEncoder.dictionary(forObject: params) + var expected: [String: AnyHashable] = [ + "online": [ + "ip_address": "test_ip_address", + "user_agent": "a_user_agent", + ], + ] + XCTAssertEqual(paramsAsDict as NSDictionary, expected as NSDictionary) + + params = STPMandateOnlineParams(ipAddress: "", userAgent: "") + params.inferFromClient = NSNumber(value: true) + paramsAsDict = STPFormEncoder.dictionary(forObject: params) + expected = [ + "online": [ + "infer_from_client": NSNumber(value: true) + ], + ] + XCTAssertEqual(paramsAsDict as NSDictionary, expected as NSDictionary) + } +} diff --git a/Stripe/StripeiOSTests/STPMocks.h b/Stripe/StripeiOSTests/STPMocks.h new file mode 100644 index 00000000..1b782caa --- /dev/null +++ b/Stripe/StripeiOSTests/STPMocks.h @@ -0,0 +1,34 @@ +// +// STPMocks.h +// Stripe +// +// Created by Ben Guo on 4/5/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import +#import +@import Stripe; + +@interface STPMocks : NSObject + +/** + A stateless customer context that always retrieves the same customer object. + */ ++ (STPCustomerContext *)staticCustomerContext; + +/** + A static customer context that always retrieves the given customer and the given payment methods. + Selecting a default source and attaching a source have no effect. + */ ++ (STPCustomerContext *)staticCustomerContextWithCustomer:(STPCustomer *)customer paymentMethods:(NSArray *)paymentMethods; + +/** + A PaymentConfiguration object with a fake publishable key and a fake apple + merchant identifier that ignores the true value of [StripeAPI deviceSupportsApplePay] + and bases its `applePayEnabled` value solely on what is set + in `additionalPaymentOptions` + */ ++ (STPPaymentConfiguration *)paymentConfigurationWithApplePaySupportingDevice; + +@end diff --git a/Stripe/StripeiOSTests/STPMocks.m b/Stripe/StripeiOSTests/STPMocks.m new file mode 100644 index 00000000..24522d85 --- /dev/null +++ b/Stripe/StripeiOSTests/STPMocks.m @@ -0,0 +1,59 @@ +// +// STPMocks.m +// Stripe +// +// Created by Ben Guo on 4/5/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import "STPMocks.h" + +#import "STPFixtures.h" +#import "StripeiOS_Tests-Swift.h" + +@interface STPPaymentConfiguration (STPMocks) + +/** + Mock apple pay enabled response to just be based on setting and not hardware + capability. + + `paymentConfigurationWithApplePaySupportingDevice` forwards calls to the + real method to this stub + */ +- (BOOL)stpmock_applePayEnabled; + +@end + +@implementation STPMocks + ++ (STPCustomerContext *)staticCustomerContext { + return [self staticCustomerContextWithCustomer:[STPFixtures customerWithSingleCardTokenSource] + paymentMethods:@[[STPFixtures paymentMethod]]]; +} + ++ (STPCustomerContext *)staticCustomerContextWithCustomer:(STPCustomer *)customer paymentMethods:(NSArray *)paymentMethods { + if (@available(iOS 13.0, *)) { + return [[Testing_StaticCustomerContext_Objc alloc] initWithCustomer:customer paymentMethods:paymentMethods]; + } else { + return nil; + } +} + ++ (STPPaymentConfiguration *)paymentConfigurationWithApplePaySupportingDevice { + STPPaymentConfiguration *config = [STPFixtures paymentConfiguration]; + config.appleMerchantIdentifier = @"fake_apple_merchant_id"; + id partialMock = OCMPartialMock(config); + OCMStub([partialMock applePayEnabled]).andCall(partialMock, @selector(stpmock_applePayEnabled)); + return partialMock; +} + +@end + +@implementation STPPaymentConfiguration (STPMocks) + +- (BOOL)stpmock_applePayEnabled { + return self.applePayEnabled; +} + +@end + diff --git a/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.h b/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.h new file mode 100644 index 00000000..730b6eb7 --- /dev/null +++ b/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.h @@ -0,0 +1,24 @@ +// +// STPNetworkStubbingTestCase.h +// StripeiOS Tests +// +// Created by Jack Flintermann on 11/24/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/** + Test cases that subclass `STPNetworkStubbingTestCase` will automatically capture all network traffic when run with `recordingMode = YES` and save it to disk. When run with `recordingMode = NO`, they will use the persisted request/response pairs, and raise an exception if an unexpected HTTP request is made. + */ +@interface STPNetworkStubbingTestCaseObjc : XCTestCase +/// Set this to YES to record all traffic during this test. The test will then fail, to remind you to set this back to NO before pushing. +@property (nonatomic) BOOL recordingMode; + +/// Implement this, returning the URLSessionConfiguration from the API client to record. +@property (nonatomic) NSURLSessionConfiguration *urlSessionConfig; +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.m b/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.m new file mode 100644 index 00000000..200871e7 --- /dev/null +++ b/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.m @@ -0,0 +1,104 @@ +// +// STPNetworkStubbingTestCase.m +// StripeiOS Tests +// +// Created by Jack Flintermann on 11/24/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import "STPNetworkStubbingTestCase.h" +#import "SWHttpTrafficRecorder.h" + +@import OHHTTPStubs; + +@implementation STPNetworkStubbingTestCaseObjc + +- (void)setUp { + [super setUp]; + + // [self name] returns a string like `-[STPMyTestCase testThing]` - this transforms it into the recorded path `recorded_network_traffic/STPMyTestCase/testThing`. + NSMutableArray *rawComponents = [[[self name] componentsSeparatedByString:@" "] mutableCopy]; + NSCAssert(rawComponents.count == 2, @"Invalid format received from XCTest#name: %@", [self name]); + NSMutableArray *components = [NSMutableArray array]; + [rawComponents enumerateObjectsUsingBlock:^(NSString *component, NSUInteger idx, __unused BOOL *stop) { + components[idx] = [[component componentsSeparatedByCharactersInSet:[[NSCharacterSet alphanumericCharacterSet] invertedSet]] componentsJoinedByString:@""]; + }]; + + NSString *testClass = components[0]; + NSString *testMethod = components[1]; + NSString *relativePath = [@"recorded_network_traffic" stringByAppendingPathComponent:[testClass stringByAppendingPathComponent:testMethod]]; + + if (self.recordingMode) { +#if TARGET_OS_SIMULATOR +#else + // Must be in the simulator, so that we can write recorded traffic into the repo. + NSCAssert(NO, @"Tests executed in recording mode must be run in the simulator."); +#endif + NSURLSessionConfiguration *config = [self urlSessionConfig]; + SWHttpTrafficRecorder *recorder = [SWHttpTrafficRecorder sharedRecorder]; + + // Creates filenames like `post_v1_tokens_0.tail`. + __block int count = 0; + [recorder setFileNamingBlock:^NSString *(NSURLRequest *request, __unused NSURLResponse *response, __unused NSString *defaultName) { + NSString *method = [request.HTTPMethod lowercaseString]; + NSString *urlPath = [request.URL.path stringByReplacingOccurrencesOfString:@"/" withString:@"_"]; + NSString *fileName = [NSString stringWithFormat:@"%@%@_%d", method, urlPath, count]; + fileName = [fileName stringByAppendingPathExtension:@"tail"]; + count++; + return fileName; + }]; + + // The goal is for `basePath` to be e.g. `~/stripe/stripe-ios/Tests` + // A little gross/hardcoded (but it works fine); feel free to improve this... + NSString *testDirectoryName = @"stripe-ios/Tests"; + NSString *basePath = [NSString stringWithFormat:@"%s", __FILE__]; + while (![basePath hasSuffix:testDirectoryName]) { + NSCAssert([basePath containsString:testDirectoryName], @"Not in a subdirectory of %@: %s", testDirectoryName, __FILE__); + basePath = [basePath stringByDeletingLastPathComponent]; + } + + NSString *recordingPath = [basePath stringByAppendingPathComponent:relativePath]; + // Delete existing stubs + [[NSFileManager defaultManager] removeItemAtPath:recordingPath error:nil]; + NSError *recordingError; + BOOL success = [[SWHttpTrafficRecorder sharedRecorder] startRecordingAtPath:recordingPath forSessionConfiguration:config error:&recordingError]; + NSCAssert(success, @"Error recording requests: %@", recordingError); + + // Make sure to fail, to remind ourselves to turn this off + __weak typeof(self) weakSelf = self; + [self addTeardownBlock:^{ + // Like XCTFail, but avoiding a retain cycle + _XCTPrimitiveFail(weakSelf, @"Network traffic for %@ has been recorded - re-run with self.recordingMode = NO for this test to succeed", [weakSelf name]); + }]; + } else { + // Stubs are evaluated in the reverse order that they are added, so if the network is hit and no other stub is matched, raise an exception + [HTTPStubs stubRequestsPassingTest:^BOOL(__unused NSURLRequest * _Nonnull request) { + return YES; + } withStubResponse:^HTTPStubsResponse * _Nonnull(NSURLRequest * _Nonnull request) { + NSCAssert(NO, @"Attempted to hit the live network at %@", request.URL.path); + return nil; + }]; + + // Note: in order to make this work, the stub files (end in .tail) must be added to the test bundle during Build Phases/Copy Resources Step. + NSBundle *bundle = [NSBundle bundleForClass:self.class]; + NSURL *url = [bundle URLForResource:relativePath withExtension:nil]; + if (url) { + NSError *stubError; + [HTTPStubs stubRequestsUsingMocktailsAtPath:relativePath inBundle:bundle error:&stubError]; + NSCAssert(!stubError, @"Error stubbing requests: %@", stubError); + } else { + NSLog(@"No stubs found - all network access will raise an exception."); + } + } +} + +- (void)tearDown { + [super tearDown]; + // Additional calls to `setFileNamingBlock` will be ignored if you don't do this + [[SWHttpTrafficRecorder sharedRecorder] stopRecording]; + + // Don't accidentally keep any stubs around during the next test run + [HTTPStubs removeAllStubs]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.swift b/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.swift new file mode 100644 index 00000000..2b9f3afd --- /dev/null +++ b/Stripe/StripeiOSTests/STPNetworkStubbingTestCase.swift @@ -0,0 +1,147 @@ +// +// STPNetworkStubbingTestCase.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 11/24/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +import OHHTTPStubs +@_spi(STP) import StripeCore + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +/// Test cases that subclass `STPNetworkStubbingTestCase` will automatically capture all network traffic when run with `recordingMode = YES` and save it to disk. When run with `recordingMode = NO`, they will use the persisted request/response pairs, and raise an exception if an unexpected HTTP request is made. +/// ⚠️ Warning: `STPAPIClient`s created before `setUp` is called are not recorded! +class STPNetworkStubbingTestCase: XCTestCase { + /// Set this to YES to record all traffic during this test. The test will then fail, to remind you to set this back to NO before pushing. + var recordingMode = false + + override func setUp() { + super.setUp() + + // Set the STPTestingAPIClient to use the sharedURLSessionConfig so that we can intercept requests from it too + STPTestingAPIClient.shared().sessionConfig = + StripeAPIConfiguration.sharedUrlSessionConfiguration + + // [self name] returns a string like `-[STPMyTestCase testThing]` - this transforms it into the recorded path `recorded_network_traffic/STPMyTestCase/testThing`. + let rawComponents = name.components(separatedBy: " ") + assert(rawComponents.count == 2, "Invalid format received from XCTest#name: \(name)") + var components: [AnyHashable] = [] + (rawComponents as NSArray).enumerateObjects({ component, _, _ in + components.append( + (component as! NSString).components( + separatedBy: CharacterSet.alphanumerics.inverted + ) + .joined() + ) + }) + + let testClass = components[0] as! NSString + let testMethod = components[1] as! String + let relativePath = ("recorded_network_traffic" as NSString).appendingPathComponent( + testClass.appendingPathComponent(testMethod) + ) + + if recordingMode { + #if targetEnvironment(simulator) + #else + // Must be in the simulator, so that we can write recorded traffic into the repo. + assert(false, "Tests executed in recording mode must be run in the simulator.") + #endif + let config = StripeAPIConfiguration.sharedUrlSessionConfiguration + let recorder = SWHttpTrafficRecorder.shared() + + // Creates filenames like `post_v1_tokens_0.tail`. + var count = 0 + recorder?.fileNamingBlock = { request, _, _ in + let method = request!.httpMethod?.lowercased() + let urlPath = request!.url?.path.replacingOccurrences(of: "/", with: "_") + var fileName = "\(method ?? "")\(urlPath ?? "")_\(count)" + fileName = + URL(fileURLWithPath: fileName).appendingPathExtension("tail").lastPathComponent + count += 1 + return fileName + } + + // The goal is for `basePath` to be e.g. `~/stripe-ios/Stripe/StripeiOSTests` + // A little gross/hardcoded (but it works fine); feel free to improve this... + let testDirectoryName = "stripe-ios/Stripe/StripeiOSTests" + var basePath = "\(#file)" + while !basePath.hasSuffix(testDirectoryName) { + assert( + basePath.contains(testDirectoryName), + "Not in a subdirectory of \(testDirectoryName): \(#file)" + ) + basePath = URL(fileURLWithPath: basePath).deletingLastPathComponent().path + } + + let recordingPath = URL(fileURLWithPath: basePath) + .appendingPathComponent("Resources") + .appendingPathComponent(relativePath) + .path + // Delete existing stubs + do { + try FileManager.default.removeItem(atPath: recordingPath) + } catch { + } + guard + (try? SWHttpTrafficRecorder.shared().startRecording( + atPath: recordingPath, + for: config + )) != nil + else { + assert(false, "Error recording requests") + return + } + + // Make sure to fail, to remind ourselves to turn this off + addTeardownBlock { + XCTFail( + "Network traffic has been recorded - re-run with self.recordingMode = NO for this test to succeed" + ) + } + } else { + // Stubs are evaluated in the reverse order that they are added, so if the network is hit and no other stub is matched, raise an exception + HTTPStubs.stubRequests( + passingTest: { _ in + return true + }, + withStubResponse: { request in + XCTFail("Attempted to hit the live network at \(request.url?.path ?? "")") + return HTTPStubsResponse() + } + ) + + // Note: in order to make this work, the stub files (end in .tail) must be added to the test bundle during Build Phases/Copy Resources Step. + let bundle = Bundle(for: STPNetworkStubbingTestCase.self) + let url = bundle.url(forResource: relativePath, withExtension: nil) + if url != nil { + var stubError: NSError? + HTTPStubs.stubRequestsUsingMocktails( + atPath: relativePath, + in: bundle, + error: &stubError + ) + if let stubError = stubError { + XCTFail("Error stubbing requests: \(stubError)") + } + } else { + print("No stubs found - all network access will raise an exception.") + } + } + } + + override func tearDown() { + super.tearDown() + // Additional calls to `setFileNamingBlock` will be ignored if you don't do this + SWHttpTrafficRecorder.shared().stopRecording() + + // Don't accidentally keep any stubs around during the next test run + HTTPStubs.removeAllStubs() + } +} diff --git a/Stripe/StripeiOSTests/STPNumericDigitInputTextFormatterTests.swift b/Stripe/StripeiOSTests/STPNumericDigitInputTextFormatterTests.swift new file mode 100644 index 00000000..ea41c2f3 --- /dev/null +++ b/Stripe/StripeiOSTests/STPNumericDigitInputTextFormatterTests.swift @@ -0,0 +1,157 @@ +// +// STPNumericDigitInputTextFormatterTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPNumericDigitInputTextFormatterTests: XCTestCase { + + func testDisallowsNonDigits() { + let formatter = STPNumericDigitInputTextFormatter() + XCTAssertFalse( + formatter.isAllowedInput("a", to: "", at: NSRange(location: 0, length: 1)), + "Shouldn't allow non-digit in empty string" + ) + XCTAssertFalse( + formatter.isAllowedInput("1a", to: "", at: NSRange(location: 0, length: 1)), + "Shouldn't allow digit + non-digit in empty string" + ) + XCTAssertFalse( + formatter.isAllowedInput("a", to: "1", at: NSRange(location: 0, length: 1)), + "Shouldn't allow non-digit in digit string" + ) + XCTAssertFalse( + formatter.isAllowedInput("1a", to: "1", at: NSRange(location: 0, length: 1)), + "Shouldn't allow digit + non-digit in digit string" + ) + XCTAssertFalse( + formatter.isAllowedInput(" ", to: "1", at: NSRange(location: 0, length: 1)), + "Shouldn't allow spaces" + ) + // for now we only validate the input, not the result + XCTAssertTrue( + formatter.isAllowedInput("1", to: "a", at: NSRange(location: 0, length: 1)), + "Should allow digit added to non-digit string" + ) + } + + func testAllowsDigits() { + let formatter = STPNumericDigitInputTextFormatter() + XCTAssertTrue( + formatter.isAllowedInput("1", to: "", at: NSRange(location: 0, length: 1)), + "Should allow digit in empty string" + ) + XCTAssertTrue( + formatter.isAllowedInput("2", to: "1", at: NSRange(location: 0, length: 1)), + "Should allow digit insert at beginning of string" + ) + XCTAssertTrue( + formatter.isAllowedInput("3", to: "1", at: NSRange(location: 1, length: 1)), + "Should allow digit insert at end of string" + ) + XCTAssertTrue( + formatter.isAllowedInput("45", to: "1", at: NSRange(location: 0, length: 1)), + "Should allow multi-digit insert" + ) + } + + func testFormattingCharacterSet() { + let formatter = STPNumericDigitInputTextFormatter( + allowedFormattingCharacterSet: CharacterSet(charactersIn: "xy") + ) + XCTAssertTrue( + formatter.isAllowedInput("x", to: "", at: NSRange(location: 0, length: 1)), + "Should allow formatting character in empty string" + ) + XCTAssertFalse( + formatter.isAllowedInput("xa", to: "", at: NSRange(location: 0, length: 1)), + "Shouldn't allow formatting + non-formatting in empty string" + ) + XCTAssertTrue( + formatter.isAllowedInput("x", to: "1", at: NSRange(location: 0, length: 1)), + "Should allow formatting character in digit string" + ) + XCTAssertTrue( + formatter.isAllowedInput("1x", to: "1", at: NSRange(location: 0, length: 1)), + "Should allow digit + formatting in digit string" + ) + XCTAssertTrue( + formatter.isAllowedInput("xxxxyyy", to: "1", at: NSRange(location: 0, length: 6)), + "Should allow multiple formatting in digit string" + ) + } + + // MARK: - Inherited Tests + func testAllowsDeletion() { + let formatter = STPNumericDigitInputTextFormatter() + let textField = UITextField() + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 2), + replacementString: "" + ), + "Should allow deletion on empty" + ) + textField.text = "12" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 2), + replacementString: "" + ), + "Should allow full deletion" + ) + textField.text = "12345" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 4, length: 1), + replacementString: "" + ), + "Should allow partial deletion at end" + ) + textField.text = "12345" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 3, length: 1), + replacementString: "" + ), + "Should allow partial deletion in middle" + ) + textField.text = "12345" + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 1), + replacementString: "" + ), + "Should allow partial deletion at beginning" + ) + } + + func testAllowsInitialSpaceForAutofill() { + let formatter = STPNumericDigitInputTextFormatter() + let textField = UITextField() + textField.textContentType = .nickname + XCTAssertTrue( + formatter.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 0), + replacementString: " " + ) + ) + } + +} diff --git a/Stripe/StripeiOSTests/STPNumericStringValidatorTests.swift b/Stripe/StripeiOSTests/STPNumericStringValidatorTests.swift new file mode 100644 index 00000000..8f5cdccd --- /dev/null +++ b/Stripe/StripeiOSTests/STPNumericStringValidatorTests.swift @@ -0,0 +1,49 @@ +// +// STPNumericStringValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPNumericStringValidatorTests: XCTestCase { + func testNumberSanitization() { + let tests = [ + ["4242424242424242", "4242424242424242"], + ["XXXXXX", ""], + ["424242424242424X", "424242424242424"], + ["X4242", "4242"], + ["4242 4242 4242 4242", "4242424242424242"], + ["123-456-", "123456"], + ] + for test in tests { + XCTAssertEqual(STPNumericStringValidator.sanitizedNumericString(for: test[0]), test[1]) + } + } + + func testIsStringNumeric() { + let tests = [ + ["4242424242424242", NSNumber(value: true)], + ["XXXXXX", NSNumber(value: false)], + ["424242424242424X", NSNumber(value: false)], + ["X4242", NSNumber(value: false)], + ["4242 4242 4242 4242", NSNumber(value: false)], + ["123-456-", NSNumber(value: false)], + [" 1", NSNumber(value: false)], + ["", NSNumber(value: true)], + ] + for test in tests { + let first = STPNumericStringValidator.isStringNumeric(test[0] as! String) + let second = (test[1] as! NSNumber).boolValue + XCTAssertEqual(first, second) + } + } +} diff --git a/Stripe/StripeiOSTests/STPPIIFunctionalTest.m b/Stripe/StripeiOSTests/STPPIIFunctionalTest.m new file mode 100644 index 00000000..107dc644 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPIIFunctionalTest.m @@ -0,0 +1,51 @@ +// +// STPPIIFunctionalTest.m +// Stripe +// +// Created by Charles Scalesse on 1/8/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPIIFunctionalTest : XCTestCase +@end + +@implementation STPPIIFunctionalTest + +- (void)testCreatePersonallyIdentifiableInformationToken { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"PII creation"]; + + [client createTokenWithPersonalIDNumber:@"0123456789" completion:^(STPToken * _Nullable token, NSError * _Nullable error) { + [expectation fulfill]; + XCTAssertNil(error, @"error should be nil %@", error.localizedDescription); + XCTAssertNotNil(token, @"token should not be nil"); + XCTAssertNotNil(token.tokenId); + XCTAssertEqual(token.type, STPTokenTypePII); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testSSNLast4Token { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"PII creation"]; + + [client createTokenWithSSNLast4:@"1234" completion:^(STPToken * _Nullable token, NSError * _Nullable error) { + [expectation fulfill]; + XCTAssertNil(error, @"error should be nil %@", error.localizedDescription); + XCTAssertNotNil(token, @"token should not be nil"); + XCTAssertNotNil(token.tokenId); + XCTAssertEqual(token.type, STPTokenTypePII); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentCardTextFieldTest.m b/Stripe/StripeiOSTests/STPPaymentCardTextFieldTest.m new file mode 100644 index 00000000..48e5167c --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentCardTextFieldTest.m @@ -0,0 +1,1191 @@ +// +// STPPaymentCardTextFieldTest.m +// Stripe +// +// Created by Jack Flintermann on 8/26/15. +// Copyright (c) 2015 Stripe, Inc. All rights reserved. +// + +@import UIKit; +@import XCTest; +@import OCMock; +@import StripeCoreTestUtils; + +#import "STPFixtures.h" + + +#import "STPTestingAPIClient.h" + +@interface STPPaymentCardTextField (Testing) +@property (nonatomic, readwrite, weak) UIImageView *brandImageView; +@property (nonatomic, readwrite, weak) STPFormTextField *numberField; +@property (nonatomic, readwrite, weak) STPFormTextField *expirationField; +@property (nonatomic, readwrite, weak) STPFormTextField *cvcField; +@property (nonatomic, readwrite, weak) STPFormTextField *postalCodeField; +@property (nonatomic, readonly, weak) STPFormTextField *currentFirstResponderField; +@property (nonatomic, readwrite, strong) STPPaymentCardTextFieldViewModel *viewModel; +@property (nonatomic, copy) NSNumber *focusedTextFieldForLayout; ++ (UIImage *)cvcImageForCardBrand:(STPCardBrand)cardBrand; ++ (UIImage *)brandImageForCardBrand:(STPCardBrand)cardBrand; +@end + +/** + Class that implements STPPaymentCardTextFieldDelegate and uses a block for each delegate method. + */ +@interface PaymentCardTextFieldBlockDelegate: NSObject +@property (nonatomic, strong, nullable) void (^didChange)(STPPaymentCardTextField *); +@property (nonatomic, strong, nullable) void (^willEndEditingForReturn)(STPPaymentCardTextField *); +@property (nonatomic, strong, nullable) void (^didEndEditing)(STPPaymentCardTextField *); +// add more properties for other delegate methods as this test needs them +@end +@implementation PaymentCardTextFieldBlockDelegate +- (void)paymentCardTextFieldDidChange:(STPPaymentCardTextField *)textField { + if (self.didChange) { + self.didChange(textField); + } +} +- (void)paymentCardTextFieldWillEndEditingForReturn:(STPPaymentCardTextField *)textField { + if (self.willEndEditingForReturn) { + self.willEndEditingForReturn(textField); + } +} +- (void)paymentCardTextFieldDidEndEditing:(STPPaymentCardTextField *)textField { + if (self.didEndEditing) { + self.didEndEditing(textField); + } +} +@end + + +@interface STPPaymentCardTextFieldTest : XCTestCase +@end + +// N.B. It is eexpected for setting the card params to generate API response errors +// because we are calling to the card metadata service without configuration STPAPIClient +@implementation STPPaymentCardTextFieldTest + ++ (void)setUp { + [super setUp]; + [[STPAPIClient sharedClient] setPublishableKey:STPTestingDefaultPublishableKey]; +} + +- (void)testIntrinsicContentSize { + STPPaymentCardTextField *textField = [STPPaymentCardTextField new]; + + UIFont *iOS8SystemFont = [UIFont fontWithName:@"HelveticaNeue" size:18]; + textField.font = iOS8SystemFont; + XCTAssertEqualWithAccuracy(textField.intrinsicContentSize.height, 44, 0.1); + XCTAssertEqualWithAccuracy(textField.intrinsicContentSize.width, 247, 0.1); + + UIFont *iOS9SystemFont = [UIFont systemFontOfSize:18];; + textField.font = iOS9SystemFont; + XCTAssertEqualWithAccuracy(textField.intrinsicContentSize.height, 44, 0.1); + XCTAssertEqualWithAccuracy(textField.intrinsicContentSize.width, 259, 1.0); + + textField.font = [UIFont fontWithName:@"Avenir" size:44]; + if (@available(iOS 13.0, *)) { + XCTAssertEqualWithAccuracy(textField.intrinsicContentSize.height, 62, 0.1); + } else { + XCTAssertEqualWithAccuracy(textField.intrinsicContentSize.height, 61, 0.1); + } + XCTAssertEqualWithAccuracy(textField.intrinsicContentSize.width, 478, 0.1); +} + +- (void)testSetCard_numberUnknown { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"1"; + card.number = number; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField errorImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); +} + +- (void)testSetCard_expiration { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + card.expMonth = @(10); + card.expYear = @(99); + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_CVC { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *cvc = @"123"; + card.cvc = cvc; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_updatesCVCValidity { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.numberField.text = @"378282246310005"; + sut.cvcField.text = @"1234"; + sut.expirationField.text = @"10/99"; + XCTAssertTrue(sut.cvcField.validText); + sut.numberField.text = @"4242424242424242"; + XCTAssertFalse(sut.cvcField.validText); +} + +- (void)testSetCard_numberVisa { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"424242"; + card.number = number; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.placeholder, @"CVC"); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_numberVisaInvalid { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242111111111111"; + card.number = number; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField errorImageForCardBrand:STPCardBrandVisa]); + + XCTAssertTrue([expectedImgData isEqualToData:imgData]); +} + +- (void)testSetCard_numberAmex { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"378282"; + card.number = number; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandAmex]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.placeholder, @"CVV"); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_numberAmexInvalid { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"378282246311111"; + card.number = number; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField errorImageForCardBrand:STPCardBrandAmex]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); +} + +- (void)testSetCard_numberAndExpiration { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_partialNumberAndExpiration { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"424242"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_numberAndCVC { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"378282246310005"; + NSString *cvc = @"123"; + card.number = number; + card.cvc = cvc; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandAmex]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_expirationAndCVC { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *cvc = @"123"; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_completeCardCountryWithoutPostal { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.countryCode = @"BZ"; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + NSString *cvc = @"123"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertTrue(sut.isValid); +} + +- (void)testSetCard_completeCardNoPostal { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.postalCodeEntryEnabled = NO; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + NSString *cvc = @"123"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertTrue(sut.isValid); +} + +- (void)testSetCard_completeCard { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + NSString *cvc = @"123"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + + STPPaymentMethodBillingDetails *billingDetails = [[STPPaymentMethodBillingDetails alloc] init]; + billingDetails.address = [[STPPaymentMethodAddress alloc] init]; + billingDetails.address.postalCode = @"90210"; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:billingDetails metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertTrue(sut.isValid); +} + +- (void)testSetCard_empty { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.numberField.text = @"4242424242424242"; + sut.cvcField.text = @"123"; + sut.expirationField.text = @"10/99"; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + STPPaymentMethodParams *params = [[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]; + [sut setPaymentMethodParams:params]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + +#pragma clang diagnostic pop + +- (void)testSettingTextUpdatesViewModelText { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.numberField.text = @"4242424242424242"; + XCTAssertEqualObjects(sut.viewModel.cardNumber, sut.numberField.text); + + sut.cvcField.text = @"123"; + XCTAssertEqualObjects(sut.viewModel.cvc, sut.cvcField.text); + + sut.expirationField.text = @"10/99"; + XCTAssertEqualObjects(sut.viewModel.rawExpiration, sut.expirationField.text); + XCTAssertEqualObjects(sut.viewModel.expirationMonth, @"10"); + XCTAssertEqualObjects(sut.viewModel.expirationYear, @"99"); +} + +- (void)testSettingTextUpdatesCardParams { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.numberField.text = @"4242424242424242"; + sut.cvcField.text = @"123"; + sut.expirationField.text = @"10/99"; + sut.postalCodeField.text = @"90210"; + + STPPaymentMethodCardParams *card = sut.paymentMethodParams.card; + XCTAssertNotNil(card); + XCTAssertEqualObjects(card.number, @"4242424242424242"); + XCTAssertEqualObjects(card.cvc, @"123"); + XCTAssertEqual(card.expMonth.integerValue, 10); + XCTAssertEqual(card.expYear.integerValue, 99); + XCTAssertEqualObjects(sut.paymentMethodParams.billingDetails.address.postalCode, @"90210"); +} + +- (void)testSettingBillingDetailsRetainsBillingDetails { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *params = [STPPaymentMethodCardParams new]; + STPPaymentMethodBillingDetails *billingDetails = [[STPPaymentMethodBillingDetails alloc] init]; + billingDetails.name = @"Test test"; + + sut.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:params billingDetails:billingDetails metadata:nil]; + STPPaymentMethodParams *actual = sut.paymentMethodParams; + + XCTAssertEqualObjects(@"Test test", actual.billingDetails.name); +} + + +- (void)testSettingMetadataRetainsMetadata { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *params = [STPPaymentMethodCardParams new]; + sut.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:params billingDetails:nil metadata:@{@"hello": @"test"}]; + STPPaymentMethodParams *actual = sut.paymentMethodParams; + + XCTAssertEqualObjects(@{@"hello": @"test"}, actual.metadata); +} + + +- (void)testSettingPostalCodeUpdatesCardParams { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.numberField.text = @"4242424242424242"; + sut.cvcField.text = @"123"; + sut.expirationField.text = @"10/99"; + sut.postalCodeField.text = @"90210"; + + STPPaymentMethodCardParams *params = sut.paymentMethodParams.card; + XCTAssertNotNil(params); + XCTAssertEqualObjects(params.number, @"4242424242424242"); + XCTAssertEqualObjects(params.cvc, @"123"); + XCTAssertEqual(params.expMonth.integerValue, 10); + XCTAssertEqual(params.expYear.integerValue, 99); +} + +- (void)testEmptyPostalCodeVendsNilAddress { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.numberField.text = @"4242424242424242"; + sut.cvcField.text = @"123"; + sut.expirationField.text = @"10/99"; + + XCTAssertNil(sut.paymentMethodParams.billingDetails.address.postalCode); + STPPaymentMethodCardParams *params = sut.paymentMethodParams.card; + XCTAssertNotNil(params); + XCTAssertEqualObjects(params.number, @"4242424242424242"); + XCTAssertEqualObjects(params.cvc, @"123"); + XCTAssertEqual(params.expMonth.integerValue, 10); + XCTAssertEqual(params.expYear.integerValue, 99); +} + + +- (void)testAccessingCardParamsDuringSettingCardParams { + PaymentCardTextFieldBlockDelegate *delegate = [PaymentCardTextFieldBlockDelegate new]; + delegate.didChange = ^(STPPaymentCardTextField *textField) { + // delegate reads the `cardParams` for any reason it wants + [[textField paymentMethodParams] card]; + }; + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.delegate = delegate; + + STPPaymentMethodCardParams *params = [STPPaymentMethodCardParams new]; + params.number = @"4242424242424242"; + params.cvc = @"123"; + + sut.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:params billingDetails:nil metadata:nil]; + STPPaymentMethodCardParams *actual = sut.paymentMethodParams.card; + + XCTAssertEqualObjects(@"4242424242424242", actual.number); + XCTAssertEqualObjects(@"123", actual.cvc); +} + +- (void)testSetCardParamsCopiesObject { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *params = [STPPaymentMethodCardParams new]; + + params.number = @"4242424242424242"; // legit + sut.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:params billingDetails:nil metadata:nil]; + + // fetching `sut.cardParams` returns a copy, so edits happen to caller's copy + sut.paymentMethodParams.card.number = @"number 1"; + + // `sut` copied `params` (& `params.address`) when set, so edits to original don't show up + params.number = @"number 2"; + + XCTAssertEqualObjects(@"4242424242424242", sut.paymentMethodParams.card.number, @"set via setCardParams:"); + + XCTAssertNotEqualObjects(@"number 1", sut.paymentMethodParams.card.number, @"return value from cardParams cannot be edited inline"); + + XCTAssertNotEqualObjects(@"number 2", sut.paymentMethodParams.card.number, @"caller changed their copy after setCardParams:"); +} + + +// MARK: - paymentMethodParams + +- (void)testSetCard_numberUnknown_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"1"; + card.number = number; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField errorImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); +} + +- (void)testSetCard_expiration_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + card.expMonth = @(10); + card.expYear = @(99); + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_CVC_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *cvc = @"123"; + card.cvc = cvc; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_numberVisa_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"424242"; + card.number = number; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.placeholder, @"CVC"); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_numberVisaInvalid_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242111111111111"; + card.number = number; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField errorImageForCardBrand:STPCardBrandVisa]); + + XCTAssertTrue([expectedImgData isEqualToData:imgData]); +} + +- (void)testSetCard_numberAmex_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"378282"; + card.number = number; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandAmex]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.placeholder, @"CVV"); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_numberAmexInvalid_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"378282246311111"; + card.number = number; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField errorImageForCardBrand:STPCardBrandAmex]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); +} + +- (void)testSetCard_numberAndExpiration_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_partialNumberAndExpiration_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"424242"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_numberAndCVC_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"378282246310005"; + NSString *cvc = @"123"; + card.number = number; + card.cvc = cvc; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandAmex]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_expirationAndCVC_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *cvc = @"123"; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +- (void)testSetCard_completeCardCountryWithoutPostal_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.countryCode = @"BZ"; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + NSString *cvc = @"123"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertTrue(sut.isValid); +} + +- (void)testSetCard_completeCardNoPostal_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.postalCodeEntryEnabled = NO; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + NSString *cvc = @"123"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertTrue(sut.isValid); +} + +- (void)testSetCard_completeCard_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + NSString *cvc = @"123"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:[[STPPaymentMethodBillingDetails alloc] initWithPostalCode:@"90210" countryCode:@"US"] metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(sut.numberField.text, number); + XCTAssertEqualObjects(sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(sut.cvcField.text, cvc); + XCTAssertNil(sut.currentFirstResponderField); + BOOL isvalid = sut.isValid; + XCTAssertTrue(isvalid); + + + STPPaymentMethodParams *paymentMethodParams = sut.paymentMethodParams; + XCTAssertNotNil(paymentMethodParams); + + STPPaymentMethodCardParams *sutCardParams = paymentMethodParams.card; + XCTAssertNotNil(sutCardParams); + + XCTAssertEqualObjects(sutCardParams.number, card.number); + XCTAssertEqualObjects(sutCardParams.expMonth, card.expMonth); + XCTAssertEqualObjects(sutCardParams.expYear, card.expYear); + XCTAssertEqualObjects(sutCardParams.cvc, card.cvc); + + STPPaymentMethodBillingDetails *sutBillingDetails = paymentMethodParams.billingDetails; + XCTAssertNotNil(sutBillingDetails); + + STPPaymentMethodAddress *sutAddress = sutBillingDetails.address; + XCTAssertNotNil(sutAddress); + + XCTAssertEqualObjects(sutAddress.postalCode, @"90210"); + XCTAssertEqualObjects(sutAddress.country, @"US"); +} + +- (void)testSetCard_empty_pm { + STPPaymentCardTextField *sut = [STPPaymentCardTextField new]; + sut.numberField.text = @"4242424242424242"; + sut.cvcField.text = @"123"; + sut.expirationField.text = @"10/99"; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + [sut setPaymentMethodParams:[[STPPaymentMethodParams alloc] initWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(sut.focusedTextFieldForLayout); + XCTAssertTrue(sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(sut.cvcField.text.length, (NSUInteger)0); + XCTAssertNil(sut.currentFirstResponderField); + XCTAssertFalse(sut.isValid); +} + +@end + +@interface STPPaymentCardTextFieldUITests : XCTestCase +@property (nonatomic) UIWindow *window; +@property (nonatomic) STPPaymentCardTextField *sut; +@end + +@implementation STPPaymentCardTextFieldUITests + ++ (void)setUp { + [super setUp]; + [[STPAPIClient sharedClient] setPublishableKey:STPTestingDefaultPublishableKey]; +} + +- (void)setUp { + [super setUp]; + self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds]; + STPPaymentCardTextField *textField = [[STPPaymentCardTextField alloc] initWithFrame:self.window.bounds]; + [self.window addSubview:textField]; + XCTAssertTrue([textField.numberField canBecomeFirstResponder], @"text field cannot become first responder"); + self.sut = textField; +} + +#pragma mark - UI Tests + +- (void)testSetCard_allFields_whileEditingNumber { + XCTAssertTrue([self.sut.numberField becomeFirstResponder], @"text field is not first responder"); + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + STPPaymentMethodBillingDetails *billingDetails = [[STPPaymentMethodBillingDetails alloc] init]; + billingDetails.address = [[STPPaymentMethodAddress alloc] init]; + billingDetails.address.postalCode = @"90210"; + NSString *number = @"4242424242424242"; + NSString *cvc = @"123"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + card.cvc = cvc; + [self.sut setPaymentMethodParams:[STPPaymentMethodParams paramsWithCard:card billingDetails:billingDetails metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(self.sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNil(self.sut.focusedTextFieldForLayout); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(self.sut.numberField.text, number); + XCTAssertEqualObjects(self.sut.expirationField.text, @"10/99"); + XCTAssertEqualObjects(self.sut.cvcField.text, cvc); + XCTAssertEqualObjects(self.sut.postalCode, @"90210"); + XCTAssertFalse([self.sut isFirstResponder], @"after `setCardParams:`, if all fields are valid, should resign firstResponder"); + XCTAssertTrue(self.sut.isValid); +} + +- (void)testSetCard_partialNumberAndExpiration_whileEditingExpiration { + XCTAssertTrue([self.sut.expirationField becomeFirstResponder], @"text field is not first responder"); + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"42"; + card.number = number; + card.expMonth = @(10); + card.expYear = @(99); + [self.sut setPaymentMethodParams:[STPPaymentMethodParams paramsWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(self.sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField cvcImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(self.sut.focusedTextFieldForLayout); + XCTAssertTrue(self.sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeCVC); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(self.sut.numberField.text, number); + XCTAssertEqualObjects(self.sut.expirationField.text, @"10/99"); + XCTAssertEqual(self.sut.cvcField.text.length, (NSUInteger)0); + XCTAssertTrue([self.sut.cvcField isFirstResponder], @"after `setCardParams:`, when firstResponder becomes valid, first invalid field should become firstResponder"); + XCTAssertFalse(self.sut.isValid); +} + +- (void)testSetCard_number_whileEditingCVC { + XCTAssertTrue([self.sut.cvcField becomeFirstResponder], @"text field is not first responder"); + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + NSString *number = @"4242424242424242"; + card.number = number; + [self.sut setPaymentMethodParams:[STPPaymentMethodParams paramsWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(self.sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField cvcImageForCardBrand:STPCardBrandVisa]); + + XCTAssertNotNil(self.sut.focusedTextFieldForLayout); + XCTAssertTrue(self.sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeCVC); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqualObjects(self.sut.numberField.text, number); + XCTAssertEqual(self.sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(self.sut.cvcField.text.length, (NSUInteger)0); + XCTAssertTrue([self.sut.cvcField isFirstResponder], @"after `setCardParams:`, if firstResponder is invalid, it should remain firstResponder"); + XCTAssertFalse(self.sut.isValid); +} + +- (void)testSetCard_empty_whileEditingNumber { + self.sut.numberField.text = @"4242424242424242"; + self.sut.cvcField.text = @"123"; + self.sut.expirationField.text = @"10/99"; + XCTAssertTrue([self.sut.numberField becomeFirstResponder], @"text field is not first responder"); + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + [self.sut setPaymentMethodParams:[STPPaymentMethodParams paramsWithCard:card billingDetails:nil metadata:nil]]; + + NSData *imgData = UIImagePNGRepresentation(self.sut.brandImageView.image); + NSData *expectedImgData = UIImagePNGRepresentation([STPPaymentCardTextField brandImageForCardBrand:STPCardBrandUnknown]); + + XCTAssertNotNil(self.sut.focusedTextFieldForLayout); + XCTAssertTrue(self.sut.focusedTextFieldForLayout.integerValue == STPCardFieldTypeNumber); + XCTAssertTrue([expectedImgData isEqualToData:imgData]); + XCTAssertEqual(self.sut.numberField.text.length, (NSUInteger)0); + XCTAssertEqual(self.sut.expirationField.text.length, (NSUInteger)0); + XCTAssertEqual(self.sut.cvcField.text.length, (NSUInteger)0); + XCTAssertTrue([self.sut.numberField isFirstResponder], @"after `setCardParams:` that clears the text fields, the first invalid field should become firstResponder"); + XCTAssertFalse(self.sut.isValid); +} + +- (void)testIsValidKVO { + id observer = OCMClassMock([UIViewController class]); + self.sut.numberField.text = @"4242424242424242"; + self.sut.expirationField.text = @"10/99"; + self.sut.postalCodeField.text = @"90210"; + XCTAssertFalse(self.sut.isValid); + + NSString *expectedKeyPath = @"sut.isValid"; + [self addObserver:observer forKeyPath:expectedKeyPath options:NSKeyValueObservingOptionNew context:nil]; + XCTestExpectation *exp = [self expectationWithDescription:@"observeValue"]; + OCMStub([observer observeValueForKeyPath:[OCMArg any] ofObject:[OCMArg any] change:[OCMArg any] context:nil]) + .andDo(^(NSInvocation *invocation) { + NSString *keyPath; + NSDictionary *change; + [invocation getArgument:&keyPath atIndex:2]; + [invocation getArgument:&change atIndex:4]; + if ([keyPath isEqualToString:expectedKeyPath]) { + if ([change[@"new"] boolValue]) { + [exp fulfill]; + [self removeObserver:observer forKeyPath:@"sut.isValid"]; + } + } + }); + + self.sut.cvcField.text = @"123"; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testBecomeFirstResponder { + self.sut.postalCodeEntryEnabled = NO; + XCTAssertTrue([self.sut canBecomeFirstResponder]); + XCTAssertTrue([self.sut becomeFirstResponder]); + XCTAssertTrue(self.sut.isFirstResponder); + + XCTAssertEqual(self.sut.numberField, self.sut.currentFirstResponderField); + + [self.sut becomeFirstResponder]; + XCTAssertEqual(self.sut.numberField, self.sut.currentFirstResponderField, + @"Repeated calls to becomeFirstResponder should not change the firstResponder"); + + self.sut.numberField.text = @"4242" "4242" "4242" "4242"; + + // Don't unit test auto-advance from number field here because we don't know the cache state + + XCTAssertTrue([self.sut.cvcField becomeFirstResponder]); + XCTAssertEqual(self.sut.cvcField, self.sut.currentFirstResponderField, + @"We don't block other fields from becoming firstResponder"); + + XCTAssertTrue([self.sut becomeFirstResponder]); + XCTAssertEqual(self.sut.cvcField, self.sut.currentFirstResponderField, + @"Calling becomeFirstResponder does not change the currentFirstResponder"); + + self.sut.expirationField.text = @"10/99"; + self.sut.cvcField.text = @"123"; + + [self.sut resignFirstResponder]; + XCTAssertTrue([self.sut canBecomeFirstResponder]); + XCTAssertTrue([self.sut becomeFirstResponder]); + + XCTAssertEqual(self.sut.cvcField, self.sut.currentFirstResponderField, + @"When all fields are valid, the last one should be the preferred firstResponder"); + + self.sut.postalCodeEntryEnabled = YES; + XCTAssertFalse(self.sut.isValid); + + [self.sut resignFirstResponder]; + XCTAssertTrue([self.sut becomeFirstResponder]); + XCTAssertEqual(self.sut.postalCodeField, self.sut.currentFirstResponderField, + @"When postalCodeEntryEnabled=YES, it should become firstResponder after other fields are valid"); + + self.sut.expirationField.text = @""; + [self.sut resignFirstResponder]; + XCTAssertTrue([self.sut becomeFirstResponder]); + XCTAssertEqual(self.sut.expirationField, self.sut.currentFirstResponderField, + @"Moves firstResponder back to expiration, because it's not valid anymore"); + + self.sut.expirationField.text = @"10/99"; + self.sut.postalCodeField.text = @"90210"; + + [self.sut resignFirstResponder]; + XCTAssertTrue([self.sut becomeFirstResponder]); + XCTAssertEqual(self.sut.postalCodeField, self.sut.currentFirstResponderField, + @"When all fields are valid, the last one should be the preferred firstResponder"); +} + +- (void)testShouldReturnCyclesThroughFields { + PaymentCardTextFieldBlockDelegate *delegate = [PaymentCardTextFieldBlockDelegate new]; + delegate.willEndEditingForReturn = ^(__unused STPPaymentCardTextField *textField) { + XCTFail(@"Did not expect editing to end in this test"); + }; + self.sut.delegate = delegate; + + [self.sut becomeFirstResponder]; + XCTAssertTrue(self.sut.numberField.isFirstResponder); + + XCTAssertFalse([self.sut.numberField.delegate textFieldShouldReturn:self.sut.numberField], @"shouldReturn = NO"); + XCTAssertTrue(self.sut.expirationField.isFirstResponder, @"with side effect to move 1st responder to next field"); + + XCTAssertFalse([self.sut.expirationField.delegate textFieldShouldReturn:self.sut.expirationField], @"shouldReturn = NO"); + XCTAssertTrue(self.sut.cvcField.isFirstResponder, @"with side effect to move 1st responder to next field"); + + XCTAssertFalse([self.sut.cvcField.delegate textFieldShouldReturn:self.sut.cvcField], @"shouldReturn = NO"); + XCTAssertTrue(self.sut.postalCodeField.isFirstResponder, @"with side effect to move 1st responder to next field"); + + XCTAssertFalse([self.sut.postalCodeField.delegate textFieldShouldReturn:self.sut.postalCodeField], @"shouldReturn = NO"); + XCTAssertTrue(self.sut.numberField.isFirstResponder, @"with side effect to move 1st responder from last field to first invalid field"); +} + +- (void)testShouldReturnCyclesThroughFieldsWithoutPostal { + PaymentCardTextFieldBlockDelegate *delegate = [PaymentCardTextFieldBlockDelegate new]; + delegate.willEndEditingForReturn = ^(__unused STPPaymentCardTextField *textField) { + XCTFail(@"Did not expect editing to end in this test"); + }; + self.sut.delegate = delegate; + self.sut.postalCodeEntryEnabled = NO; + + [self.sut becomeFirstResponder]; + XCTAssertTrue(self.sut.numberField.isFirstResponder); + + XCTAssertFalse([self.sut.numberField.delegate textFieldShouldReturn:self.sut.numberField], @"shouldReturn = NO"); + XCTAssertTrue(self.sut.expirationField.isFirstResponder, @"with side effect to move 1st responder to next field"); + + XCTAssertFalse([self.sut.expirationField.delegate textFieldShouldReturn:self.sut.expirationField], @"shouldReturn = NO"); + XCTAssertTrue(self.sut.cvcField.isFirstResponder, @"with side effect to move 1st responder to next field"); + + XCTAssertFalse([self.sut.cvcField.delegate textFieldShouldReturn:self.sut.cvcField], @"shouldReturn = NO"); + XCTAssertTrue(self.sut.numberField.isFirstResponder, @"with side effect to move 1st responder from last field to first invalid field"); +} + +- (void)testShouldReturnDismissesWhenValidNoPostalCode { + __block BOOL hasReturned = NO; + __block BOOL didEnd = NO; + + self.sut.postalCodeEntryEnabled = NO; + [self.sut setPaymentMethodParams:[STPPaymentMethodParams paramsWithCard:[STPFixtures paymentMethodCardParams] billingDetails:nil metadata:nil]]; + + PaymentCardTextFieldBlockDelegate *delegate = [PaymentCardTextFieldBlockDelegate new]; + delegate.willEndEditingForReturn = ^(__unused STPPaymentCardTextField *textField) { + XCTAssertFalse(didEnd, @"willEnd is called before didEnd"); + XCTAssertFalse(hasReturned, @"willEnd is only called once"); + hasReturned = YES; + }; + delegate.didEndEditing = ^(__unused STPPaymentCardTextField *textField) { + XCTAssertTrue(hasReturned, @"didEndEditing should be called after willEnd"); + XCTAssertFalse(didEnd, @"didEnd is only called once"); + didEnd = YES; + }; + + self.sut.delegate = delegate; + [self.sut becomeFirstResponder]; + XCTAssertTrue(self.sut.cvcField.isFirstResponder, @"when textfield is filled out, default first responder is the last field"); + + XCTAssertFalse(hasReturned, @"willEndEditingForReturn delegate method should not have been called yet"); + XCTAssertFalse([self.sut.cvcField.delegate textFieldShouldReturn:self.sut.cvcField], @"shouldReturn = NO"); + + XCTAssertNil(self.sut.currentFirstResponderField, @"Should have resigned first responder"); + XCTAssertTrue(hasReturned, @"delegate method has been invoked"); + XCTAssertTrue(didEnd, @"delegate method has been invoked"); +} + +- (void)testShouldReturnDismissesWhenValid { + __block BOOL hasReturned = NO; + __block BOOL didEnd = NO; + + [self.sut setPaymentMethodParams:[STPPaymentMethodParams paramsWithCard:[STPFixtures paymentMethodCardParams] billingDetails:nil metadata:nil]]; + self.sut.postalCodeField.text = @"90210"; + PaymentCardTextFieldBlockDelegate *delegate = [PaymentCardTextFieldBlockDelegate new]; + delegate.willEndEditingForReturn = ^(__unused STPPaymentCardTextField *textField) { + XCTAssertFalse(didEnd, @"willEnd is called before didEnd"); + XCTAssertFalse(hasReturned, @"willEnd is only called once"); + hasReturned = YES; + }; + delegate.didEndEditing = ^(__unused STPPaymentCardTextField *textField) { + XCTAssertTrue(hasReturned, @"didEndEditing should be called after willEnd"); + XCTAssertFalse(didEnd, @"didEnd is only called once"); + didEnd = YES; + }; + + self.sut.delegate = delegate; + [self.sut becomeFirstResponder]; + XCTAssertTrue(self.sut.postalCodeField.isFirstResponder, @"when textfield is filled out, default first responder is the last field"); + + XCTAssertFalse(hasReturned, @"willEndEditingForReturn delegate method should not have been called yet"); + XCTAssertFalse([self.sut.postalCodeField.delegate textFieldShouldReturn:self.sut.postalCodeField], @"shouldReturn = NO"); + + XCTAssertNil(self.sut.currentFirstResponderField, @"Should have resigned first responder"); + XCTAssertTrue(hasReturned, @"delegate method has been invoked"); + XCTAssertTrue(didEnd, @"delegate method has been invoked"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentCardTextFieldTestsSwift.swift b/Stripe/StripeiOSTests/STPPaymentCardTextFieldTestsSwift.swift new file mode 100644 index 00000000..c914d4dd --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentCardTextFieldTestsSwift.swift @@ -0,0 +1,67 @@ +// +// STPPaymentCardTextFieldTestsSwift.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 8/24/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentCardTextFieldTestsSwift: XCTestCase { + + func testClearMaintainsPostalCodeEntryEnabled() { + let textField = STPPaymentCardTextField() + let postalCodeEntryDefaultEnabled = textField.postalCodeEntryEnabled + textField.clear() + XCTAssertEqual( + postalCodeEntryDefaultEnabled, + textField.postalCodeEntryEnabled, + "clear overrode default postalCodeEntryEnabled value" + ) + + // -- + textField.postalCodeEntryEnabled = false + textField.clear() + XCTAssertFalse( + textField.postalCodeEntryEnabled, + "clear overrode custom postalCodeEntryEnabled false value" + ) + + // -- + textField.postalCodeEntryEnabled = true + // The ORs in this test are to handle if these tests are run in an environment + // where the locale doesn't require postal codes, in which case the calculated + // value for postalCodeEntryEnabled can be different than the value set + // (this is a legacy API). + let stillTrueOrRequestedButNoPostal = + textField.postalCodeEntryEnabled + || (textField.viewModel.postalCodeRequested + && STPPostalCodeValidator.postalCodeIsRequired( + forCountryCode: textField.viewModel.postalCodeCountryCode + )) + XCTAssertTrue( + stillTrueOrRequestedButNoPostal, + "clear overrode custom postalCodeEntryEnabled true value" + ) + + } + + func testPostalCodeIsValidWhenExpirationIsNot() { + let cardTextField = STPPaymentCardTextField() + + // Old expiration date + cardTextField.expirationField.text = "10/10" + XCTAssertFalse(cardTextField.expirationField.validText) + + cardTextField.postalCode = "10001" + cardTextField.formTextFieldTextDidChange(cardTextField.postalCodeField) + XCTAssertTrue(cardTextField.postalCodeField.validText) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentCardTextFieldViewModelTest.swift b/Stripe/StripeiOSTests/STPPaymentCardTextFieldViewModelTest.swift new file mode 100644 index 00000000..bf18859d --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentCardTextFieldViewModelTest.swift @@ -0,0 +1,112 @@ +// +// STPPaymentCardTextFieldViewModelTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 7/16/15. +// Copyright (c) 2015 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentCardTextFieldViewModelTest: XCTestCase { + var viewModel: STPPaymentCardTextFieldViewModel? + + override func setUp() { + super.setUp() + viewModel = STPPaymentCardTextFieldViewModel() + } + + func testCardNumber() { + let tests = [ + ["", ""], + ["4242", "4242"], + ["4242424242424242", "4242424242424242"], + ["4242 4242 4242 4242", "4242424242424242"], + ["4242xxx4242", "42424242"], + ["12345678901234567890", "1234567890123456789"], + ] + for test in tests { + viewModel?.cardNumber = test[0] + XCTAssertEqual(viewModel?.cardNumber, test[1]) + } + } + + func testRawExpiration() { + // swiftlint:disable:next large_tuple + let tests: [(String, String, String, String, STPCardValidationState)] = [ + ("", "", "", "", .incomplete), + ("12/23", "12/23", "12", "23", .valid), + ("1223", "12/23", "12", "23", .valid), + ("1", "1", "1", "", .incomplete), + ("2", "02/", "02", "", .incomplete), + ("12", "12/", "12", "", .incomplete), + ("12/2", "12/2", "12", "2", .incomplete), + ("99/23", "99", "99", "23", .invalid), + ("10/12", "10/12", "10", "12", .invalid), + ("12*23", "12/23", "12", "23", .valid), + ("12/*", "12/", "12", "", .incomplete), + ("*", "", "", "", .incomplete), + ] + for test in tests { + viewModel?.rawExpiration = test.0 + XCTAssertEqual(viewModel?.rawExpiration, test.1) + XCTAssertEqual(viewModel?.expirationMonth, test.2) + XCTAssertEqual(viewModel?.expirationYear, test.3) + XCTAssertEqual(viewModel?.validationStateForExpiration(), test.4) + } + } + + func testCVC() { + let tests = [["1", "1"], ["1234", "1234"], ["12345", "1234"], ["1x", "1"]] + for test in tests { + viewModel?.cvc = test[0] + XCTAssertEqual(viewModel?.cvc, test[1]) + } + } + + func testValidity() { + viewModel?.cardNumber = "4242424242424242" + viewModel?.rawExpiration = "12/24" + viewModel?.cvc = "123" + XCTAssertTrue(viewModel!.isValid) + + viewModel?.cvc = "12" + XCTAssertFalse(viewModel!.isValid) + } + + func testCompressedCardNumber() { + viewModel?.cardNumber = nil + // Should use default placeholder + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "4242") + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: "1234567812345678"), "5678") + + viewModel?.cardNumber = "424212345678" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "5678") + viewModel?.cardNumber = "42421234567" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "567") + viewModel?.cardNumber = "4242123456" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "56") + viewModel?.cardNumber = "424212345" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "5") + viewModel?.cardNumber = "42421234" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "1234") + + viewModel?.cardNumber = "12" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "12") + + viewModel?.cardNumber = "36227206271667" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "1667") + viewModel?.cardNumber = "3622720627166" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "166") + viewModel?.cardNumber = "36227206271" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "1") + viewModel?.cardNumber = "3622720627" + XCTAssertEqual(viewModel?.compressedCardNumber(withPlaceholder: nil), "720627") + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentConfigurationTest.m b/Stripe/StripeiOSTests/STPPaymentConfigurationTest.m new file mode 100644 index 00000000..48dd733a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentConfigurationTest.m @@ -0,0 +1,137 @@ +// +// STPPaymentConfigurationTest.m +// Stripe +// +// Created by Joey Dong on 7/18/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import +#import + + +@import StripeCore; + + + + +@interface STPPaymentConfigurationTest : XCTestCase + +@end + +@implementation STPPaymentConfigurationTest + +- (void)testSharedConfiguration { + XCTAssertEqual([STPPaymentConfiguration sharedConfiguration], [STPPaymentConfiguration sharedConfiguration]); +} + +- (void)testInit { + STPPaymentConfiguration *paymentConfiguration = [[STPPaymentConfiguration alloc] init]; + + XCTAssertFalse(paymentConfiguration.fpxEnabled); + XCTAssertEqual(paymentConfiguration.requiredBillingAddressFields, STPBillingAddressFieldsPostalCode); + XCTAssertNil(paymentConfiguration.requiredShippingAddressFields); + XCTAssert(paymentConfiguration.verifyPrefilledShippingAddress); + XCTAssertEqual(paymentConfiguration.shippingType, STPShippingTypeShipping); + XCTAssertEqualObjects(paymentConfiguration.companyName, @"xctest"); + XCTAssertNil(paymentConfiguration.appleMerchantIdentifier); + XCTAssert(paymentConfiguration.canDeletePaymentOptions); + XCTAssertFalse(paymentConfiguration.cardScanningEnabled); +} + +- (void)testApplePayEnabledSatisfied { + id stripeMock = OCMClassMock([StripeAPI class]); + OCMStub([stripeMock deviceSupportsApplePay]).andReturn(YES); + + STPPaymentConfiguration *paymentConfiguration = [[STPPaymentConfiguration alloc] init]; + paymentConfiguration.appleMerchantIdentifier = @"appleMerchantIdentifier"; + + XCTAssert([paymentConfiguration applePayEnabled]); +} + +- (void)testApplePayEnabledMissingAppleMerchantIdentifier { + id stripeMock = OCMClassMock([StripeAPI class]); + OCMStub([stripeMock deviceSupportsApplePay]).andReturn(YES); + + STPPaymentConfiguration *paymentConfiguration = [[STPPaymentConfiguration alloc] init]; + paymentConfiguration.appleMerchantIdentifier = nil; + + XCTAssertFalse([paymentConfiguration applePayEnabled]); +} + +- (void)testApplePayEnabledDisallowAdditionalPaymentOptions { + id stripeMock = OCMClassMock([StripeAPI class]); + OCMStub([stripeMock deviceSupportsApplePay]).andReturn(YES); + + STPPaymentConfiguration *paymentConfiguration = [[STPPaymentConfiguration alloc] init]; + paymentConfiguration.appleMerchantIdentifier = @"appleMerchantIdentifier"; + paymentConfiguration.applePayEnabled = false; + + XCTAssertFalse([paymentConfiguration applePayEnabled]); +} + +- (void)testApplePayEnabledMisisngDeviceSupport { + id paymentAuthControllerMock = OCMClassMock([PKPaymentAuthorizationController class]); + OCMStub([paymentAuthControllerMock canMakePaymentsUsingNetworks:[OCMArg any]]).andReturn(NO); + + STPPaymentConfiguration *paymentConfiguration = [[STPPaymentConfiguration alloc] init]; + paymentConfiguration.appleMerchantIdentifier = @"appleMerchantIdentifier"; + + XCTAssertFalse([paymentConfiguration applePayEnabled]); + [paymentAuthControllerMock stopMocking]; +} + +#pragma mark - Description + +- (void)testDescription { + STPPaymentConfiguration *paymentConfiguration = [[STPPaymentConfiguration alloc] init]; + XCTAssert(paymentConfiguration.description); +} + +#pragma mark - NSCopying + +- (void)testCopyWithZone { + NSSet *allFields = [NSSet setWithArray:@[STPContactField.postalAddress, + STPContactField.emailAddress, + STPContactField.phoneNumber, + STPContactField.name]]; + + STPPaymentConfiguration *paymentConfigurationA = [[STPPaymentConfiguration alloc] init]; +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + paymentConfigurationA.publishableKey = @"publishableKey"; + paymentConfigurationA.stripeAccount = @"stripeAccount"; +#pragma clang diagnostic pop + paymentConfigurationA.applePayEnabled = YES; + paymentConfigurationA.requiredBillingAddressFields = STPBillingAddressFieldsFull; + paymentConfigurationA.requiredShippingAddressFields = allFields; + paymentConfigurationA.verifyPrefilledShippingAddress = NO; + paymentConfigurationA.availableCountries = [NSSet setWithArray:@[@"US", @"CA", @"BT"]]; + paymentConfigurationA.shippingType = STPShippingTypeDelivery; + paymentConfigurationA.companyName = @"companyName"; + paymentConfigurationA.appleMerchantIdentifier = @"appleMerchantIdentifier"; + paymentConfigurationA.canDeletePaymentOptions = NO; + paymentConfigurationA.cardScanningEnabled = NO; + + STPPaymentConfiguration *paymentConfigurationB = [paymentConfigurationA copy]; + XCTAssertNotEqual(paymentConfigurationA, paymentConfigurationB); + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertEqualObjects(paymentConfigurationB.publishableKey, @"publishableKey"); + XCTAssertEqualObjects(paymentConfigurationB.stripeAccount, @"stripeAccount"); +#pragma clang diagnostic pop + XCTAssertTrue(paymentConfigurationB.applePayEnabled); + XCTAssertEqual(paymentConfigurationB.requiredBillingAddressFields, STPBillingAddressFieldsFull); + XCTAssertEqualObjects(paymentConfigurationB.requiredShippingAddressFields, allFields); + XCTAssertFalse(paymentConfigurationB.verifyPrefilledShippingAddress); + XCTAssertEqual(paymentConfigurationB.shippingType, STPShippingTypeDelivery); + XCTAssertEqualObjects(paymentConfigurationB.companyName, @"companyName"); + XCTAssertEqualObjects(paymentConfigurationB.appleMerchantIdentifier, @"appleMerchantIdentifier"); + NSSet *availableCountries = [NSSet setWithArray:@[@"US", @"CA", @"BT"]]; + XCTAssertEqualObjects(paymentConfigurationB.availableCountries, availableCountries); + XCTAssertEqual(paymentConfigurationA.canDeletePaymentOptions, paymentConfigurationB.canDeletePaymentOptions); + XCTAssertEqual(paymentConfigurationA.cardScanningEnabled, paymentConfigurationB.cardScanningEnabled); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentContextApplePayTest.swift b/Stripe/StripeiOSTests/STPPaymentContextApplePayTest.swift new file mode 100644 index 00000000..a3c6d13d --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentContextApplePayTest.swift @@ -0,0 +1,204 @@ +// +// STPPaymentContextApplePayTest.swift +// StripeiOS Tests +// +// Created by Brian Dorfman on 8/1/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +/// These tests cover STPPaymentContext's Apple Pay specific behavior: +/// - building a PKPaymentRequest +/// - determining paymentSummaryItems +class STPPaymentContextApplePayTest: XCTestCase { + func buildPaymentContext() -> STPPaymentContext { + let config = STPFixtures.paymentConfiguration() + config.appleMerchantIdentifier = "fake_merchant_id" + let theme = STPTheme.defaultTheme + let customerContext = Testing_StaticCustomerContext() + let paymentContext = STPPaymentContext( + customerContext: customerContext, + configuration: config, + theme: theme + ) + return paymentContext + } + + // MARK: - buildPaymentRequest + func testBuildPaymentRequest_totalAmount() { + let context = buildPaymentContext() + context.paymentAmount = 150 + let request = context.buildPaymentRequest() + + XCTAssertTrue( + (request?.paymentSummaryItems.last?.amount == NSDecimalNumber(string: "1.50")), + "PKPayment total is not equal to STPPaymentContext amount" + ) + } + + func testBuildPaymentRequest_USDDefault() { + let context = buildPaymentContext() + context.paymentAmount = 100 + let request = context.buildPaymentRequest() + + XCTAssertTrue( + (request?.currencyCode == "USD"), + "Default PKPaymentRequest currency code is not USD" + ) + } + + func testBuildPaymentRequest_currency() { + let context = buildPaymentContext() + context.paymentAmount = 100 + context.paymentCurrency = "GBP" + let request = context.buildPaymentRequest() + + XCTAssertTrue( + (request?.currencyCode == "GBP"), + "PKPaymentRequest currency code is not equal to STPPaymentContext currency" + ) + } + + func testBuildPaymentRequest_uppercaseCurrency() { + let context = buildPaymentContext() + context.paymentAmount = 100 + context.paymentCurrency = "eur" + let request = context.buildPaymentRequest() + + XCTAssertTrue( + (request?.currencyCode == "EUR"), + "PKPaymentRequest currency code is not uppercased" + ) + } + + func testSummaryItems() -> [PKPaymentSummaryItem]? { + return [ + PKPaymentSummaryItem( + label: "First item", + amount: NSDecimalNumber(mantissa: 20, exponent: 0, isNegative: false) + ), + PKPaymentSummaryItem( + label: "Second item", + amount: NSDecimalNumber(mantissa: 90, exponent: 0, isNegative: false) + ), + PKPaymentSummaryItem( + label: "Discount", + amount: NSDecimalNumber(mantissa: 10, exponent: 0, isNegative: true) + ), + PKPaymentSummaryItem( + label: "Total", + amount: NSDecimalNumber(mantissa: 100, exponent: 0, isNegative: false) + ), + ] + } + + func testBuildPaymentRequest_summaryItems() { + let context = buildPaymentContext() + context.paymentSummaryItems = testSummaryItems()! + let request = context.buildPaymentRequest() + + XCTAssertTrue((request?.paymentSummaryItems == context.paymentSummaryItems)) + } + + // MARK: - paymentSummaryItems + func testSetPaymentAmount_generateSummaryItems() { + let context = buildPaymentContext() + context.paymentAmount = 10000 + context.paymentCurrency = "USD" + let itemTotalAmount = context.paymentSummaryItems.last?.amount + let correctTotalAmount = NSDecimalNumber.stp_decimalNumber( + withAmount: context.paymentAmount, + currency: context.paymentCurrency + ) + + XCTAssertTrue((itemTotalAmount == correctTotalAmount)) + } + + func testSetPaymentAmount_generateSummaryItemsShippingMethod() { + let context = buildPaymentContext() + context.paymentAmount = 100 + context.configuration.companyName = "Foo Company" + let method = PKShippingMethod() + method.amount = NSDecimalNumber(string: "5.99") + method.label = "FedEx" + method.detail = "foo" + method.identifier = "123" + context.selectedShippingMethod = method + + let items = context.paymentSummaryItems + XCTAssertEqual(Int(items.count), 2) + let item1 = items[0] + XCTAssertEqual(item1.label, "FedEx") + XCTAssertEqual(item1.amount, NSDecimalNumber(string: "5.99")) + let item2 = items[1] + XCTAssertEqual(item2.label, "Foo Company") + XCTAssertEqual(item2.amount, NSDecimalNumber(string: "6.99")) + } + + func testSummaryItemsToSummaryItems_shippingMethod() { + let context = buildPaymentContext() + let item1 = PKPaymentSummaryItem() + item1.amount = NSDecimalNumber(string: "1.00") + item1.label = "foo" + let item2 = PKPaymentSummaryItem() + item2.amount = NSDecimalNumber(string: "9.00") + item2.label = "bar" + let item3 = PKPaymentSummaryItem() + item3.amount = NSDecimalNumber(string: "10.00") + item3.label = "baz" + context.paymentSummaryItems = [item1, item2, item3] + let method = PKShippingMethod() + method.amount = NSDecimalNumber(string: "5.99") + method.label = "FedEx" + method.detail = "foo" + method.identifier = "123" + context.selectedShippingMethod = method + + let items = context.paymentSummaryItems + XCTAssertEqual(Int(items.count), 4) + let resultItem1 = items[0] + XCTAssertEqual(resultItem1.label, "foo") + XCTAssertEqual(resultItem1.amount, NSDecimalNumber(string: "1.00")) + let resultItem2 = items[1] + XCTAssertEqual(resultItem2.label, "bar") + XCTAssertEqual(resultItem2.amount, NSDecimalNumber(string: "9.00")) + let resultItem3 = items[2] + XCTAssertEqual(resultItem3.label, "FedEx") + XCTAssertEqual(resultItem3.amount, NSDecimalNumber(string: "5.99")) + let resultItem4 = items[3] + XCTAssertEqual(resultItem4.label, "baz") + XCTAssertEqual(resultItem4.amount, NSDecimalNumber(string: "15.99")) + } + + func testAmountToAmount_shippingMethod_usd() { + let context = buildPaymentContext() + context.paymentAmount = 100 + let method = PKShippingMethod() + method.amount = NSDecimalNumber(string: "5.99") + method.label = "FedEx" + method.detail = "foo" + method.identifier = "123" + context.selectedShippingMethod = method + let amount = context.paymentAmount + XCTAssertEqual(amount, 699) + } + + func testSummaryItems_generateAmountDecimalCurrency() { + let context = buildPaymentContext() + context.paymentSummaryItems = testSummaryItems()! + context.paymentCurrency = "USD" + XCTAssertTrue(context.paymentAmount == 10000) + } + + func testSummaryItems_generateAmountNoDecimalCurrency() { + let context = buildPaymentContext() + context.paymentSummaryItems = testSummaryItems()! + context.paymentCurrency = "JPY" + XCTAssertTrue(context.paymentAmount == 100) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentContextSnapshotTests.m b/Stripe/StripeiOSTests/STPPaymentContextSnapshotTests.m new file mode 100644 index 00000000..e9d49298 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentContextSnapshotTests.m @@ -0,0 +1,111 @@ +// +// STPPaymentContextSnapshotTests.m +// StripeiOS Tests +// +// Created by Ben Guo on 12/13/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import +#import "StripeiOS_Tests-Swift.h" + +#import "STPFixtures.h" +#import "STPMocks.h" +#import "STPTestUtils.h" + +@import iOSSnapshotTestCaseCore; + +@interface STPPaymentContextSnapshotTests : FBSnapshotTestCase + +@property (nonatomic, strong) STPCustomerContext *customerContext; +@property (nonatomic, strong) STPPaymentConfiguration *config; +@property (nonatomic, strong) UINavigationController *hostViewController; +@property (nonatomic, strong) STPPaymentContext *paymentContext; + +@end + +@implementation STPPaymentContextSnapshotTests + +- (void)setUp { + [super setUp]; + STPPaymentConfiguration *config = [STPFixtures paymentConfiguration]; + config.companyName = @"Test Company"; + config.requiredBillingAddressFields = STPBillingAddressFieldsFull; + config.shippingType = STPShippingTypeShipping; + self.config = config; + STPCustomerContext *customerContext = nil; + if (@available(iOS 13.0, *)) { + customerContext = [[Testing_StaticCustomerContext_Objc alloc] initWithCustomer:[STPFixtures customerWithCardTokenAndSourceSources] paymentMethods:@[[STPFixtures paymentMethod], [STPFixtures paymentMethod]]]; + } else { + customerContext = [STPMocks staticCustomerContextWithCustomer:[STPFixtures customerWithCardTokenAndSourceSources] paymentMethods:@[[STPFixtures paymentMethod], [STPFixtures paymentMethod]]]; + } + self.customerContext = customerContext; + + UIViewController *viewController = [UIViewController new]; + self.hostViewController = [self stp_navigationControllerForSnapshotTestWithRootVC:viewController]; + +// self.recordMode = YES; +} + +- (void)buildPaymentContext { + STPPaymentContext *context = [[STPPaymentContext alloc] initWithCustomerContext:self.customerContext]; + context.hostViewController = self.hostViewController; + context.configuration.requiredShippingAddressFields = [NSSet setWithArray:@[STPContactField.emailAddress]]; + self.paymentContext = context; +} + +- (void)testPushPaymentOptionsSmallTitle { + if (@available(iOS 12.0, *)) { + [self buildPaymentContext]; + + self.hostViewController.navigationBar.prefersLargeTitles = NO; + self.paymentContext.largeTitleDisplayMode = UINavigationItemLargeTitleDisplayModeAutomatic; + [self.paymentContext pushPaymentOptionsViewController]; + UIView *view = [self stp_preparedAndSizedViewForSnapshotTestFromNavigationController:self.hostViewController]; + STPSnapshotVerifyView(view, nil); + } +} + +// This test renders at a slightly larger size half the time. +// We're deprecating Basic Integration soon, and we've spent enough time on this, +// so these tests are being disabled for now. +//- (void)testPushPaymentOptionsLargeTitle { +// if (@available(iOS 12.0, *)) { +// [self buildPaymentContext]; +// +// self.hostViewController.navigationBar.prefersLargeTitles = YES; +// self.paymentContext.largeTitleDisplayMode = UINavigationItemLargeTitleDisplayModeAutomatic; +// [self.paymentContext pushPaymentOptionsViewController]; +// UIView *view = [self stp_preparedAndSizedViewForSnapshotTestFromNavigationController:self.hostViewController]; +// STPSnapshotVerifyView(view, nil); +// } +//} + +- (void)testPushShippingAddressSmallTitle { + if (@available(iOS 12.0, *)) { + [self buildPaymentContext]; + + self.hostViewController.navigationBar.prefersLargeTitles = NO; + self.paymentContext.largeTitleDisplayMode = UINavigationItemLargeTitleDisplayModeAutomatic; + [self.paymentContext pushShippingViewController]; + UIView *view = [self stp_preparedAndSizedViewForSnapshotTestFromNavigationController:self.hostViewController]; + STPSnapshotVerifyView(view, nil); + } +} + +// This test renders at a slightly larger size half the time. +// We're deprecating Basic Integration soon, and we've spent enough time on this, +// so these tests are being disabled for now. +//- (void)testPushShippingAddressLargeTitle { +// if (@available(iOS 12.0, *)) { +// [self buildPaymentContext]; +// +// self.hostViewController.navigationBar.prefersLargeTitles = YES; +// self.paymentContext.largeTitleDisplayMode = UINavigationItemLargeTitleDisplayModeAutomatic; +// [self.paymentContext pushShippingViewController]; +// UIView *view = [self stp_preparedAndSizedViewForSnapshotTestFromNavigationController:self.hostViewController]; +// STPSnapshotVerifyView(view, nil); +// } +//} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentHandlerFunctionalTest.m b/Stripe/StripeiOSTests/STPPaymentHandlerFunctionalTest.m new file mode 100644 index 00000000..119d1568 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentHandlerFunctionalTest.m @@ -0,0 +1,76 @@ +// +// STPPaymentHandlerFunctionalTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 5/14/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import Stripe; +#import +#import + +#import "STPTestingAPIClient.h" + + +@interface STPPaymentHandlerFunctionalTest : XCTestCase +@property (nonatomic) id presentingViewController; +@end + +@interface STPPaymentHandler (Test) +- (BOOL)_canPresentWithAuthenticationContext:(id)authenticationContext error:(NSError **)error; +@end + +@implementation STPPaymentHandlerFunctionalTest + +- (void)setUp { + self.presentingViewController = OCMClassMock([UIViewController class]); + [STPAPIClient sharedClient].publishableKey = STPTestingDefaultPublishableKey; +} + +// N.B. Test mode alipay PaymentIntent's never have a native redirect so we can't test that here +- (void)testAlipayOpensWebviewAfterNativeURLUnavailable { + + __block NSString *clientSecret = @"pi_1GiohpFY0qyl6XeWw09oKwWi_secret_Co4Etlq8YhmB6p07LQTP1Yklg"; + id applicationMock = OCMClassMock([UIApplication class]); + OCMStub([applicationMock sharedApplication]).andReturn(applicationMock); + // Simulate the customer not having the Alipay app installed + OCMStub([applicationMock openURL:[OCMArg any] + options:[OCMArg any] + completionHandler:([OCMArg invokeBlockWithArgs:@NO, nil])]); + + id paymentHandler = OCMPartialMock(STPPaymentHandler.sharedHandler); + + // Simulate the safari VC finishing after presenting it + OCMStub([self.presentingViewController presentViewController:[OCMArg any] animated:YES completion:[OCMArg any]]).andDo(^(__unused NSInvocation *_) { + [paymentHandler safariViewControllerDidFinish:self.presentingViewController]; + }); + + STPPaymentIntentParams *confirmParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + confirmParams.paymentMethodOptions = [STPConfirmPaymentMethodOptions new]; + confirmParams.paymentMethodOptions.alipayOptions = [STPConfirmAlipayOptions new]; + confirmParams.paymentMethodParams = [STPPaymentMethodParams paramsWithAlipay:[STPPaymentMethodAlipayParams new] billingDetails:nil metadata:nil]; + confirmParams.returnURL = @"foo://bar"; + + XCTestExpectation *e = [self expectationWithDescription:@""]; + [paymentHandler confirmPayment:confirmParams withAuthenticationContext:self completion:^(STPPaymentHandlerActionStatus status, STPPaymentIntent * __unused paymentIntent, __unused NSError * _Nullable error) { + // ...shouldn't attempt to open the native URL (ie the alipay app) + OCMReject([applicationMock openURL:[OCMArg any] + options:[OCMArg any] + completionHandler:[OCMArg isNotNil]]); + // ...and then open UIViewController + OCMVerify([self.presentingViewController presentViewController:[OCMArg any] animated:YES completion:[OCMArg any]]); + + // ...and since we didn't actually authenticate, the final state is canceled + XCTAssertEqual(status, STPPaymentHandlerActionStatusCanceled); + [e fulfill]; + }]; + [self waitForExpectationsWithTimeout:4 handler:nil]; +} + +- (UIViewController *)authenticationPresentingViewController { + return self.presentingViewController; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentHandlerStubbedMockedFilesTests.swift b/Stripe/StripeiOSTests/STPPaymentHandlerStubbedMockedFilesTests.swift new file mode 100644 index 00000000..e930d71f --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentHandlerStubbedMockedFilesTests.swift @@ -0,0 +1,628 @@ +// +// STPPaymentHandlerStubbedMockedFilesTests.swift +// StripeiOS Tests +// +// Copyright © 2022 Stripe, Inc. All rights reserved. +// +import OHHTTPStubs +import OHHTTPStubsSwift +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeApplePay +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentHandlerStubbedMockedFilesTests: APIStubbedTestCase, STPAuthenticationContext { + func testCallConfirmAfterpay_Redirect_thenCanceled() { + let nextActionData = """ + { + "redirect_to_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/afterpay_clearpay/acct_123/pa_nonce_321/redirect" + }, + "type": "redirect_to_url" + } + """ + let paymentMethod = """ + { + "id": "pm_123123123123123", + "object": "payment_method", + "afterpay_clearpay": {}, + "billing_details": { + "address": { + "city": "San Francisco", + "country": "AT", + "line1": "510 Townsend St.", + "line2": "", + "postal_code": "94102", + "state": null + }, + "email": "foo@bar.com", + "name": "Jane Doe", + "phone": null + }, + "created": 1658187899, + "customer": null, + "livemode": false, + "type": "afterpay_clearpay" + } + """ + let paymentHandler = stubbedPaymentHandler(formSpecProvider: formSpecProvider()) + stubConfirm( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethod, + "": "\"requires_action\"", + ] + ) + } + ) + + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "pi_123456_secret_654321") + paymentIntentParams.returnURL = "payments-example://stripe-redirect" + paymentIntentParams.paymentMethodParams = STPPaymentMethodParams( + afterpayClearpay: STPPaymentMethodAfterpayClearpayParams(), + billingDetails: STPPaymentMethodBillingDetails(), + metadata: nil + ) + paymentIntentParams.paymentMethodParams?.afterpayClearpay = + STPPaymentMethodAfterpayClearpayParams() + let didRedirect = expectation(description: "didRedirect") + paymentHandler._redirectShim = { redirectTo, returnToURL, isStandardRedirect in + XCTAssertEqual( + redirectTo.absoluteString, + "https://hooks.stripe.com/afterpay_clearpay/acct_123/pa_nonce_321/redirect" + ) + XCTAssertEqual(returnToURL?.absoluteString, "payments-example://stripe-redirect") + XCTAssert(isStandardRedirect) + didRedirect.fulfill() + } + let expectConfirmWasCanceled = expectation(description: "didCancel") + paymentHandler.confirmPayment(paymentIntentParams, with: self) { + status, + _, + _ in + if case .canceled = status { + expectConfirmWasCanceled.fulfill() + } + } + guard XCTWaiter.wait(for: [didRedirect], timeout: 2.0) != .timedOut else { + XCTFail("Unable to redirect") + return + } + + // Test the cancel case + stubRetrievePaymentIntent( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethod, + "": "\"requires_action\"", + ] + ) + } + ) + paymentHandler._retrieveAndCheckIntentForCurrentAction() + wait(for: [expectConfirmWasCanceled], timeout: 2.0) + } + + func testCallConfirmAfterpay_Redirect_thenSucceeded() { + let nextActionData = """ + { + "redirect_to_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/afterpay_clearpay/acct_123/pa_nonce_321/redirect" + }, + "type": "redirect_to_url" + } + """ + let paymentMethodData = """ + { + "id": "pm_123123123123123", + "object": "payment_method", + "afterpay_clearpay": {}, + "billing_details": { + "address": { + "city": "San Francisco", + "country": "AT", + "line1": "510 Townsend St.", + "line2": "", + "postal_code": "94102", + "state": null + }, + "email": "foo@bar.com", + "name": "Jane Doe", + "phone": null + }, + "created": 1658187899, + "customer": null, + "livemode": false, + "type": "afterpay_clearpay" + } + """ + let paymentHandler = stubbedPaymentHandler(formSpecProvider: formSpecProvider()) + stubConfirm( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethodData, + "": "\"requires_action\"", + ] + ) + } + ) + + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "pi_123456_secret_654321") + paymentIntentParams.returnURL = "payments-example://stripe-redirect" + paymentIntentParams.paymentMethodParams = STPPaymentMethodParams( + afterpayClearpay: STPPaymentMethodAfterpayClearpayParams(), + billingDetails: STPPaymentMethodBillingDetails(), + metadata: nil + ) + paymentIntentParams.paymentMethodParams?.afterpayClearpay = + STPPaymentMethodAfterpayClearpayParams() + let didRedirect = expectation(description: "didRedirect") + paymentHandler._redirectShim = { redirectTo, returnToURL, isStandardRedirect in + XCTAssertEqual( + redirectTo.absoluteString, + "https://hooks.stripe.com/afterpay_clearpay/acct_123/pa_nonce_321/redirect" + ) + XCTAssertEqual(returnToURL?.absoluteString, "payments-example://stripe-redirect") + XCTAssert(isStandardRedirect) + didRedirect.fulfill() + } + confirmPaymentWithSucceed(nextActionData: nextActionData, + paymentMethodData: paymentMethodData, + didRedirect: didRedirect, + paymentHandler: paymentHandler, + paymentIntentParams: paymentIntentParams) + } + + func testRedirectStrategy_external_browser() { + let nextActionData = """ + { + "redirect_to_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/affirm/acct_123/pa_nonce_321/redirect" + }, + "type": "redirect_to_url" + } + """ + let paymentMethodData = """ + { + "id": "pm_123123123123123", + "object": "payment_method", + "affirm": {}, + "created": 1658187899, + "customer": null, + "livemode": false, + "type": "affirm" + } + """ + + let formSpecProvider = formSpecProvider() + let paymentHandler = stubbedPaymentHandler(formSpecProvider: formSpecProvider) + stubConfirm( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethodData, + "": "\"requires_action\"", + ] + ) + } + ) + XCTAssertTrue(formSpecProvider.loadFrom(affirmSpec(redirectStrategy: "external_browser"))) + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "pi_123456_secret_654321") + paymentIntentParams.returnURL = "payments-example://stripe-redirect" + paymentIntentParams.paymentMethodParams = STPPaymentMethodParams(affirm: STPPaymentMethodAffirmParams(), + metadata: nil) + let didRedirect = expectation(description: "didRedirect") + paymentHandler._redirectShim = { redirectTo, returnToURL, isStandardRedirect in + XCTAssertEqual( + redirectTo.absoluteString, + "https://hooks.stripe.com/affirm/acct_123/pa_nonce_321/redirect" + ) + XCTAssertEqual(returnToURL?.absoluteString, "payments-example://stripe-redirect") + XCTAssertFalse(isStandardRedirect) + didRedirect.fulfill() + } + confirmPaymentWithSucceed(nextActionData: nextActionData, + paymentMethodData: paymentMethodData, + didRedirect: didRedirect, + paymentHandler: paymentHandler, + paymentIntentParams: paymentIntentParams) + } + + func testRedirectStrategy_follow_redirects() { + let nextActionData = """ + { + "redirect_to_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/affirm/acct_123/pa_nonce_321/redirect" + }, + "type": "redirect_to_url" + } + """ + let paymentMethodData = """ + { + "id": "pm_123123123123123", + "object": "payment_method", + "affirm": {}, + "created": 1658187899, + "customer": null, + "livemode": false, + "type": "affirm" + } + """ + let formSpecProvider = formSpecProvider() + let paymentHandler = stubbedPaymentHandler(formSpecProvider: formSpecProvider) + stubConfirm( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethodData, + "": "\"requires_action\"", + ] + ) + } + ) + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/affirm/acct_123") ?? false + } response: { _ in + let data = "<>".data(using: .utf8)! + return HTTPStubsResponse(data: data, statusCode: 302, headers: ["Location": "https://www.financial-partner.com/"]) + } + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("financial-partner.com") ?? false + } response: { _ in + let data = "".data(using: .utf8)! + return HTTPStubsResponse(data: data, statusCode: 200, headers: nil) + } + + XCTAssertTrue(formSpecProvider.loadFrom(affirmSpec(redirectStrategy: "follow_redirects"))) + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "pi_123456_secret_654321") + paymentIntentParams.returnURL = "payments-example://stripe-redirect" + paymentIntentParams.paymentMethodParams = STPPaymentMethodParams(affirm: STPPaymentMethodAffirmParams(), + metadata: nil) + let didRedirect = expectation(description: "didRedirect") + paymentHandler._redirectShim = { redirectTo, returnToURL, isStandardRedirect in + XCTAssertEqual(redirectTo.absoluteString, "https://www.financial-partner.com/") + XCTAssertEqual(returnToURL?.absoluteString, "payments-example://stripe-redirect") + XCTAssert(isStandardRedirect) + didRedirect.fulfill() + } + confirmPaymentWithSucceed(nextActionData: nextActionData, + paymentMethodData: paymentMethodData, + didRedirect: didRedirect, + paymentHandler: paymentHandler, + paymentIntentParams: paymentIntentParams) + } + + func testCallConfirmAfterpay_Redirect_thenSucceeded_withoutNextActionSpec() { + let formSpecProvider = formSpecProvider() + let paymentHandler = stubbedPaymentHandler(formSpecProvider: formSpecProvider) + // Validate affirm is read in with next action spec + guard let affirm = formSpecProvider.formSpec(for: "affirm"), + affirm.fields.count == 1, + affirm.fields.first == .affirm_header, + case .redirect_to_url = affirm.nextActionSpec?.confirmResponseStatusSpecs[ + "requires_action" + ]?.type, + case .finished = affirm.nextActionSpec?.postConfirmHandlingPiStatusSpecs?["succeeded"]? + .type, + case .canceled = affirm.nextActionSpec?.postConfirmHandlingPiStatusSpecs?[ + "requires_action" + ]?.type + else { + XCTFail() + return + } + // Override it with a spec that doesn't define a next action so that we force the SDK to default behavior + let updatedSpecJson = + """ + [{ + "type": "affirm", + "async": false, + "fields": [ + { + "type": "name" + } + ] + }] + """.data(using: .utf8)! + let formSpec = try! JSONSerialization.jsonObject(with: updatedSpecJson) as! [NSDictionary] + XCTAssert(formSpecProvider.loadFrom(formSpec)) + guard let affirmUpdated = formSpecProvider.formSpec(for: "affirm") else { + XCTFail() + return + } + XCTAssertNil(affirmUpdated.nextActionSpec) + + let nextActionData = """ + { + "redirect_to_url": { + "return_url": "payments-example://stripe-redirect", + "url": "https://hooks.stripe.com/affirm/acct_123/pa_nonce_321/redirect" + }, + "type": "redirect_to_url" + } + """ + let paymentMethodData = """ + { + "id": "pm_123123123123123", + "object": "payment_method", + "afterpay_clearpay": {}, + "billing_details": { + "address": { + "city": "San Francisco", + "country": "AT", + "line1": "510 Townsend St.", + "line2": "", + "postal_code": "94102", + "state": null + }, + "email": "foo@bar.com", + "name": "Jane Doe", + "phone": null + }, + "created": 1658187899, + "customer": null, + "livemode": false, + "type": "afterpay_clearpay" + } + """ + stubConfirm( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethodData, + "": "\"requires_action\"", + ] + ) + } + ) + + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "pi_123456_secret_654321") + paymentIntentParams.returnURL = "payments-example://stripe-redirect" + paymentIntentParams.paymentMethodParams = STPPaymentMethodParams( + afterpayClearpay: STPPaymentMethodAfterpayClearpayParams(), + billingDetails: STPPaymentMethodBillingDetails(), + metadata: nil + ) + paymentIntentParams.paymentMethodParams?.afterpayClearpay = + STPPaymentMethodAfterpayClearpayParams() + let didRedirect = expectation(description: "didRedirect") + paymentHandler._redirectShim = { redirectTo, returnToURL, isStandardRedirect in + XCTAssertEqual( + redirectTo.absoluteString, + "https://hooks.stripe.com/affirm/acct_123/pa_nonce_321/redirect" + ) + XCTAssertEqual(returnToURL?.absoluteString, "payments-example://stripe-redirect") + XCTAssert(isStandardRedirect) + didRedirect.fulfill() + } + confirmPaymentWithSucceed(nextActionData: nextActionData, + paymentMethodData: paymentMethodData, + didRedirect: didRedirect, + paymentHandler: paymentHandler, + paymentIntentParams: paymentIntentParams) + } + + func testCallConfirmBlikSucceeds() { + let nextActionData = """ + { + "type": "blik_authorize" + } + """ + let paymentMethodData = """ + { + "id": "pm_123123123123123", + "object": "payment_method", + "billing_details": { + "address": { + "city": null, + "country": null, + "line1": null, + "line2": null, + "postal_code": null, + "state": null + }, + "email": null, + "name": null, + "phone": null + }, + "blik": {}, + "created": 1658187899, + "customer": null, + "livemode": false, + "type": "blik" + } + """ + let paymentHandler = stubbedPaymentHandler(formSpecProvider: formSpecProvider()) + let paymentIntentParams = STPPaymentIntentParams(clientSecret: "pi_123456_secret_654321") + paymentIntentParams.returnURL = "payments-example://stripe-redirect" + paymentIntentParams.paymentMethodParams = STPPaymentMethodParams( + blik: STPPaymentMethodBLIKParams(), + billingDetails: nil, + metadata: nil + ) + + stubRetrievePaymentIntent( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethodData, + "": "\"requires_action\"", + ] + ) + } + ) + + let expectConfirmSucceeded = expectation(description: "didSucceed") + paymentHandler.confirmPayment( + paymentIntentParams, + with: self) { status, _, _ in + if case .succeeded = status { + expectConfirmSucceeded.fulfill() + } + } + waitForExpectations(timeout: 2.0) + } + + private func confirmPaymentWithSucceed( + nextActionData: String, + paymentMethodData: String, + didRedirect: XCTestExpectation, + paymentHandler: STPPaymentHandler, + paymentIntentParams: STPPaymentIntentParams + ) { + let expectConfirmSucceeded = expectation(description: "didSucceed") + paymentHandler.confirmPayment(paymentIntentParams, with: self) { + status, + _, + _ in + if case .succeeded = status { + expectConfirmSucceeded.fulfill() + } + } + + guard XCTWaiter.wait(for: [didRedirect], timeout: 2.0) != .timedOut else { + XCTFail("Unable to redirect") + return + } + + // Test status as succeeded + stubRetrievePaymentIntent( + fileMock: .paymentIntentResponse, + responseCallback: { data in + self.replaceData( + data: data, + variables: [ + "": nextActionData, + "": paymentMethodData, + "": "\"succeeded\"", + ] + ) + } + ) + paymentHandler._retrieveAndCheckIntentForCurrentAction() + wait(for: [expectConfirmSucceeded], timeout: 2.0) + } + + private func formSpecProvider() -> FormSpecProvider { + let expectation = expectation(description: "Load Specs") + let formSpecProvider = FormSpecProvider() + formSpecProvider.load { _ in + expectation.fulfill() + } + wait(for: [expectation], timeout: 5.0) + return formSpecProvider + } + + private func stubbedPaymentHandler(formSpecProvider: FormSpecProvider) -> STPPaymentHandler { + let stubbedAPIClient = stubbedAPIClient() + let paymentSheetFormSpecHandler = PaymentSheetFormSpecPaymentHandler(urlSession: stubbedAPIClient.urlSession, + formSpecProvider: formSpecProvider) + return STPPaymentHandler(apiClient: stubbedAPIClient, + formSpecPaymentHandler: paymentSheetFormSpecHandler) + } + + private func replaceData(data: Data, variables: [String: String]) -> Data { + var template = String(data: data, encoding: .utf8)! + for (templateKey, templateValue) in variables { + let translated = template.replacingOccurrences(of: templateKey, with: templateValue) + template = translated + } + return template.data(using: .utf8)! + } + + private func stubConfirm(fileMock: FileMock, responseCallback: ((Data) -> Data)? = nil) { + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/confirm") ?? false + } response: { _ in + let mockResponseData = try! fileMock.data() + let data = responseCallback?(mockResponseData) ?? mockResponseData + return HTTPStubsResponse(data: data, statusCode: 200, headers: nil) + } + } + private func stubRetrievePaymentIntent( + fileMock: FileMock, + responseCallback: ((Data) -> Data)? = nil + ) { + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/payment_intents") ?? false + } response: { _ in + let mockResponseData = try! fileMock.data() + let data = responseCallback?(mockResponseData) ?? mockResponseData + return HTTPStubsResponse(data: data, statusCode: 200, headers: nil) + } + } + private func affirmSpec(redirectStrategy: String) -> [NSDictionary] { + let formSpec = + """ + [{ + "type": "affirm", + "async": false, + "fields": [ + { + "type": "affirm_header" + } + ], + "next_action_spec": { + "confirm_response_status_specs": { + "requires_action": { + "type": "redirect_to_url", + "native_mobile_redirect_strategy": "\(redirectStrategy)" + } + }, + "post_confirm_handling_pi_status_specs": { + "succeeded": { + "type": "finished" + } + } + } + }] + """.data(using: .utf8)! + return try! JSONSerialization.jsonObject(with: formSpec) as! [NSDictionary] + } +} +extension STPPaymentHandlerStubbedMockedFilesTests { + func authenticationPresentingViewController() -> UIViewController { + return UIViewController() + } +} + +public class ClassForBundle {} +@_spi(STP) public enum FileMock: String, MockData { + public typealias ResponseType = StripeFile + public var bundle: Bundle { return Bundle(for: ClassForBundle.self) } + + case paymentIntentResponse = "MockFiles/paymentIntentResponse" +} diff --git a/Stripe/StripeiOSTests/STPPaymentHandlerTests.swift b/Stripe/StripeiOSTests/STPPaymentHandlerTests.swift new file mode 100644 index 00000000..4fee56e4 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentHandlerTests.swift @@ -0,0 +1,259 @@ +// +// STPPaymentHandlerTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/8/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import OHHTTPStubs +import OHHTTPStubsSwift +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import Stripe3DS2 +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentHandlerStubbedTests: STPNetworkStubbingTestCase { + override func setUp() { + self.recordingMode = false + super.setUp() + } + + func testCanPresentErrorsAreReported() { + let createPaymentIntentExpectation = expectation( + description: "createPaymentIntentExpectation" + ) + var retrievedClientSecret: String? + STPTestingAPIClient.shared().createPaymentIntent(withParams: nil) { + (createdPIClientSecret, _) in + if let createdPIClientSecret = createdPIClientSecret { + retrievedClientSecret = createdPIClientSecret + createPaymentIntentExpectation.fulfill() + } else { + XCTFail() + } + } + wait(for: [createPaymentIntentExpectation], timeout: 8) // STPTestingNetworkRequestTimeout + guard let clientSecret = retrievedClientSecret, + let currentYear = Calendar.current.dateComponents([.year], from: Date()).year + else { + XCTFail() + return + } + + let expiryYear = NSNumber(value: currentYear + 2) + let expiryMonth = NSNumber(1) + + let cardParams = STPPaymentMethodCardParams() + cardParams.number = "4000000000003220" + cardParams.expYear = expiryYear + cardParams.expMonth = expiryMonth + cardParams.cvc = "123" + + let address = STPPaymentMethodAddress() + address.postalCode = "12345" + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.address = address + + let paymentMethodParams = STPPaymentMethodParams.paramsWith( + card: cardParams, + billingDetails: billingDetails, + metadata: nil + ) + + let paymentIntentParams = STPPaymentIntentParams(clientSecret: clientSecret) + paymentIntentParams.paymentMethodParams = paymentMethodParams + + // STPTestingDefaultPublishableKey + STPAPIClient.shared.publishableKey = "pk_test_ErsyMEOTudSjQR8hh0VrQr5X008sBXGOu6" + + let paymentHandlerExpectation = expectation(description: "paymentHandlerExpectation") + STPPaymentHandler.shared().checkCanPresentInTest = true + STPPaymentHandler.shared().confirmPayment(paymentIntentParams, with: self) { + (status, paymentIntent, error) in + XCTAssertTrue(status == .failed) + XCTAssertNotNil(paymentIntent) + XCTAssertNotNil(error) + XCTAssertEqual( + error?.userInfo[STPError.errorMessageKey] as? String, + "authenticationPresentingViewController is not in the window hierarchy. You should probably return the top-most view controller instead." + ) + paymentHandlerExpectation.fulfill() + } + // 2*STPTestingNetworkRequestTimeout payment handler needs to make an ares for this + // test in addition to fetching the payment intent + wait(for: [paymentHandlerExpectation], timeout: 2 * 8) + } +} + +class STPPaymentHandlerTests: APIStubbedTestCase { + + func testPaymentHandlerRetriesWithBackoff() { + STPPaymentHandler.sharedHandler.apiClient = stubbedAPIClient() + + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("3ds2/authenticate") ?? false + } response: { _ in + let jsonText = """ + { + "state": "challenge_required", + "livemode": "false", + "ares" : { + "dsTransID": "4e4750e7-6ab5-45a4-accf-9c668ed3b5a7", + "acsTransID": "fa695a82-a48c-455d-9566-a652058dda27", + "p_messageVersion": "1.0.5", + "acsOperatorID": "acsOperatorUL", + "sdkTransID": "D77EB83F-F317-4E29-9852-EBAAB55515B7", + "eci": "00", + "dsReferenceNumber": "3DS_LOA_DIS_PPFU_020100_00010", + "acsReferenceNumber": "3DS_LOA_ACS_PPFU_020100_00009", + "threeDSServerTransID": "fc7a39de-dc41-4b65-ba76-a322769b2efc", + "messageVersion": "2.1.0", + "authenticationValue": "AABBCCDDEEFFAABBCCDDEEFFAAA=", + "messageType": "pArs", + "transStatus": "C", + "acsChallengeMandated": "NO" + } + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("3ds2/challenge_complete") ?? false + } response: { _ in + let errorResponse = [ + "error": + [ + "message": "This is intentionally failing for this test.", + "type": "invalid_request_error", + ], + ] + return HTTPStubsResponse(jsonObject: errorResponse, statusCode: 400, headers: nil) + } + + let paymentHandlerExpectation = expectation( + description: "paymentHandlerFinished" + ) + var inProgress = true + + // Meaningless cert, generated for this test + // Expires 3/2/2121: Apologies to future engineers! + let cert = """ + MIIBijCB9AIBATANBgkqhkiG9w0BAQUFADANMQswCQYDVQQGEwJVUzAgFw0yMTAz + MjYxODQyNDVaGA8yMTIxMDMwMjE4NDI0NVowDTELMAkGA1UEBhMCVVMwgZ8wDQYJ + KoZIhvcNAQEBBQADgY0AMIGJAoGBAL6rIW6t+8eo1exqhvYt8H1vM+TyHNNychlD + hILw745yXZQAy9ByRG3euYEydE3SFINgWBCUuwWmkNfsZUW7Uci1PBMglBFHJrE8 + 8ZvtuJgnPkqmu97a9JkyROiaqAmqoMDP95HiZG5i3a1E/QPpPyYA3VJ/El17Qqkl + aHN32qzjAgMBAAEwDQYJKoZIhvcNAQEFBQADgYEAUhxbGQ5sQMDUqFTvibU7RzqL + dTaFhdjTDBu5YeIbXXUrJSG2AydXRq7OacRksnQhvNYXimfcgfse46XQG7rKUCfj + kbazRiRxMZylTz8zbePAFcVq6zxJ+RBVrv51D+/JgbCcQ50nZiocllR0J9UL8CKZ + obaUC2OjBbSuCZwF8Ig= + """ + let rootCA = """ + MIIBkDCB+gIJAJ3pmjFOkxTXMA0GCSqGSIb3DQEBBQUAMA0xCzAJBgNVBAYTAlVT + MB4XDTIxMDMyNjE4NDEzMVoXDTIyMDMyNjE4NDEzMVowDTELMAkGA1UEBhMCVVMw + gZ8wDQYJKoZIhvcNAQEBBQADgY0AMIGJAoGBAKmFDGPV77Fk/wgUMwbxjQk+bpUY + cTjNBsjK3xMaUWeE17Sry6IguO1iWaXVey9YJ1Dm83PNO/5i9nHh3gmFhEJmc55T + g+0tZQigjTcs5/BfmWtrfPYIWqKvIJqkkHrIEJnwavAS5OFGyDArHLwUtsgJbDmW + tIeQg3EH/8BSWR0BAgMBAAEwDQYJKoZIhvcNAQEFBQADgYEATY2aQvZZJLPgUr1/ + oDvRy6KZ6p7n3+jXF8DNvVOIaQRD4Ndk5NfStteIT5XvzfmD6QqpG3nlJ6Wy3oSP + 03KvO4GWIyP9cuP/QLaEmxJIYKwPrdxLkUHFfzyy8tN54xOWPxN4Up9gVN6pSdVk + KWrsPfhPs3G57wir370Q69lV/8A= + """ + let iauss = STPIntentActionUseStripeSDK( + encryptionInfo: [ + "certificate": cert, + "directory_server_id": "0000000000", + "root_certificate_authorities": [rootCA], + ], + directoryServerName: "none", + directoryServerKeyID: "none", + serverTransactionID: "none", + threeDSSourceID: "none", + publishableKeyOverride: nil, + threeDS2IntentOverride: nil, + allResponseFields: [:] + ) + let action = STPIntentAction( + type: .useStripeSDK, + redirectToURL: nil, + alipayHandleRedirect: nil, + useStripeSDK: iauss, + oxxoDisplayDetails: nil, + weChatPayRedirectToApp: nil, + boletoDisplayDetails: nil, + verifyWithMicrodeposits: nil, + cashAppRedirectToApp: nil, + allResponseFields: [:] + ) + let setupIntent = STPSetupIntent( + stripeID: "test", + clientSecret: "test", + created: Date(), + countryCode: "US", + customerID: nil, + stripeDescription: nil, + linkSettings: nil, + livemode: false, + nextAction: action, + orderedPaymentMethodTypes: [], + paymentMethodID: "test", + paymentMethod: nil, + paymentMethodOptions: nil, + paymentMethodTypes: [], + status: .requiresAction, + usage: .none, + lastSetupError: nil, + allResponseFields: [:], + unactivatedPaymentMethodTypes: [] + ) + + // We expect this request to retry a few times with exponential backoff before calling the completion handler. + STPPaymentHandler.sharedHandler._handleNextAction( + for: setupIntent, + with: self, + returnURL: nil + ) { (status, _, _) in + XCTAssertEqual(status, .failed) + inProgress = false + paymentHandlerExpectation.fulfill() + } + + let checkedStillInProgress = expectation( + description: "Checked that we're still in progress after 2s" + ) + DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 2.0) { + // Make sure we're still in progress after 2 seconds + // This shows that we're retrying the 3DS2 request a few times + // while applying an appropriate amount of backoff. + XCTAssertEqual(inProgress, true) + checkedStillInProgress.fulfill() + } + + wait(for: [paymentHandlerExpectation, checkedStillInProgress], timeout: 30) + STPPaymentHandler.sharedHandler.apiClient = STPAPIClient.shared + } +} + +extension STPPaymentHandlerTests: STPAuthenticationContext { + func authenticationPresentingViewController() -> UIViewController { + return UIViewController() + } +} + +extension STPPaymentHandlerStubbedTests: STPAuthenticationContext { + func authenticationPresentingViewController() -> UIViewController { + return UIViewController() + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentIntentEnumsTest.swift b/Stripe/StripeiOSTests/STPPaymentIntentEnumsTest.swift new file mode 100644 index 00000000..eecef5c8 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentIntentEnumsTest.swift @@ -0,0 +1,184 @@ +// +// STPPaymentIntentEnumsTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 9/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentIntentEnumsTest: XCTestCase { + + func textStatusFromString() { + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "requires_payment_method"), + .requiresPaymentMethod + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "REQUIRES_PAYMENT_METHOD"), + .requiresPaymentMethod + ) + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "requires_confirmation"), + .requiresConfirmation + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "REQUIRES_CONFIRMATION"), + .requiresConfirmation + ) + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "requires_action"), + .requiresAction + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "REQUIRES_ACTION"), + .requiresAction + ) + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "processing"), + .processing + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "PROCESSING"), + .processing + ) + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "succeeded"), + .succeeded + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "SUCCEEDED"), + .succeeded + ) + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "requires_capture"), + .requiresCapture + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "REQUIRES_CAPTURE"), + .requiresCapture + ) + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "canceled"), + .canceled + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "CANCELED"), + .canceled + ) + + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "garbage"), + .unknown + ) + XCTAssertEqual( + STPPaymentIntentStatus.status(from: "GARBAGE"), + .unknown + ) + } + + func testCaptureMethodFromString() { + XCTAssertEqual( + STPPaymentIntentCaptureMethod.captureMethod(from: "manual"), + .manual + ) + XCTAssertEqual( + STPPaymentIntentCaptureMethod.captureMethod(from: "MANUAL"), + .manual + ) + + XCTAssertEqual( + STPPaymentIntentCaptureMethod.captureMethod(from: "automatic"), + .automatic + ) + XCTAssertEqual( + STPPaymentIntentCaptureMethod.captureMethod(from: "AUTOMATIC"), + .automatic + ) + + XCTAssertEqual( + STPPaymentIntentCaptureMethod.captureMethod(from: "garbage"), + .unknown + ) + XCTAssertEqual( + STPPaymentIntentCaptureMethod.captureMethod(from: "GARBAGE"), + .unknown + ) + } + + func testConfirmationMethodFromString() { + XCTAssertEqual( + STPPaymentIntentConfirmationMethod.confirmationMethod(from: "automatic"), + .automatic + ) + XCTAssertEqual( + STPPaymentIntentConfirmationMethod.confirmationMethod(from: "AUTOMATIC"), + .automatic + ) + + XCTAssertEqual( + STPPaymentIntentConfirmationMethod.confirmationMethod(from: "manual"), + .manual + ) + XCTAssertEqual( + STPPaymentIntentConfirmationMethod.confirmationMethod(from: "MANUAL"), + .manual + ) + + XCTAssertEqual( + STPPaymentIntentConfirmationMethod.confirmationMethod(from: "garbage"), + .unknown + ) + XCTAssertEqual( + STPPaymentIntentConfirmationMethod.confirmationMethod(from: "GARBAGE"), + .unknown + ) + } + + func testSetupFutureUsageFromString() { + XCTAssertEqual( + STPPaymentIntentSetupFutureUsage(string: "on_session"), + .onSession + ) + XCTAssertEqual( + STPPaymentIntentSetupFutureUsage(string: "ON_SESSION"), + .onSession + ) + + XCTAssertEqual( + STPPaymentIntentSetupFutureUsage(string: "off_session"), + .offSession + ) + XCTAssertEqual( + STPPaymentIntentSetupFutureUsage(string: "OFF_SESSION"), + .offSession + ) + + XCTAssertEqual( + STPPaymentIntentSetupFutureUsage(string: "garbage"), + .unknown + ) + XCTAssertEqual( + STPPaymentIntentSetupFutureUsage(string: "GARBAGE"), + .unknown + ) + } + + func testStringConvertible() { + XCTAssertEqual(String(describing: STPPaymentIntentStatus.requiresAction), "requiresAction") + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.m b/Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.m new file mode 100644 index 00000000..276d363d --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.m @@ -0,0 +1,1058 @@ +// +// STPPaymentIntentFunctionalTest.m +// StripeiOS Tests +// +// Created by Daniel Jackson on 6/27/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +@import Stripe; + + +#import "STPTestingAPIClient.h" + + +@interface STPPaymentIntentFunctionalTest : XCTestCase +@end + +@implementation STPPaymentIntentFunctionalTest + +- (void)testCreatePaymentIntentWithTestingServer { + XCTestExpectation *expectation = [self expectationWithDescription:@"PaymentIntent create."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil + completion:^(NSString * _Nullable clientSecret, NSError * _Nullable error) { + XCTAssertNotNil(clientSecret); + XCTAssertNil(error); + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreatePaymentIntentWithInvalidCurrency { + XCTestExpectation *expectation = [self expectationWithDescription:@"PaymentIntent create."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{@"payment_method_types": @[@"bancontact"]} completion:^(NSString * _Nullable clientSecret, NSError * _Nullable error) { + XCTAssertNil(clientSecret); + XCTAssertNotNil(error); + XCTAssertTrue([error.userInfo[[STPError errorMessageKey]] hasPrefix:@"Error creating PaymentIntent: The currency provided (usd) is invalid. Payments with bancontact support the following currencies: eur."]); + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testRetrievePreviousCreatedPaymentIntent { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent retrieve"]; + + [client retrievePaymentIntentWithClientSecret:@"pi_1GGCGfFY0qyl6XeWbSAsh2hn_secret_jbhwsI0DGWhKreJs3CCrluUGe" + completion:^(STPPaymentIntent *paymentIntent, NSError *error) { + XCTAssertNil(error); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, @"pi_1GGCGfFY0qyl6XeWbSAsh2hn"); + XCTAssertEqual(paymentIntent.amount, 100); + XCTAssertEqualObjects(paymentIntent.currency, @"usd"); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNil(paymentIntent.sourceId); + XCTAssertNil(paymentIntent.paymentMethodId); + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusCanceled); + XCTAssertEqual(paymentIntent.setupFutureUsage, STPPaymentIntentSetupFutureUsageNone); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(paymentIntent.nextSourceAction); +#pragma clang diagnostic pop + XCTAssertNil(paymentIntent.nextAction); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testRetrieveWithWrongSecret { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent retrieve"]; + + [client retrievePaymentIntentWithClientSecret:@"pi_1GGCGfFY0qyl6XeWbSAsh2hn_secret_bad-secret" + completion:^(STPPaymentIntent *paymentIntent, NSError *error) { + XCTAssertNil(paymentIntent); + + XCTAssertNotNil(error); + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + XCTAssertEqual(error.code, STPInvalidRequestError); + XCTAssertEqualObjects(error.userInfo[[STPError errorParameterKey]], + @"clientSecret"); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testRetrieveMismatchedPublishableKey { + // Given an API Client with a publishable key for a test account A... + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_test_51JtgfQKG6vc7r7YCU0qQNOkDaaHrEgeHgGKrJMNfuWwaKgXMLzPUA1f8ZlCNPonIROLOnzpUnJK1C1xFH3M3Mz8X00Q6O4GfUt"]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent retrieve"]; + + // ...retrieving a PI attached to a *different* account + [client retrievePaymentIntentWithClientSecret:@"pi_1GGCGfFY0qyl6XeWbSAsh2hn_secret_jbhwsI0DGWhKreJs3CCrluUGe" + completion:^(STPPaymentIntent *paymentIntent, NSError *error) { + // ...should fail. + XCTAssertNil(paymentIntent); + + XCTAssertNotNil(error); + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + XCTAssertEqual(error.code, STPInvalidRequestError); + XCTAssertEqualObjects(error.userInfo[[STPError errorParameterKey]], + @"intent"); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmCanceledPaymentIntentFails { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:@"pi_1GGCGfFY0qyl6XeWbSAsh2hn_secret_jbhwsI0DGWhKreJs3CCrluUGe"]; + params.sourceParams = [self cardSourceParams]; + [client confirmPaymentIntentWithParams:params + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(paymentIntent); + + XCTAssertNotNil(error); + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + XCTAssertEqual(error.code, STPInvalidRequestError); + XCTAssertTrue([error.userInfo[[STPError errorMessageKey]] hasPrefix:@"This PaymentIntent's source could not be updated because it has a status of canceled. You may only update the source of a PaymentIntent with one of the following statuses: requires_payment_method, requires_confirmation, requires_action."], + @"Expected error message to complain about status being canceled. Actual msg: `%@`", error.userInfo[[STPError errorMessageKey]]); + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmPaymentIntentWith3DSCardSucceeds { + + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + params.sourceParams = [self cardSourceParams]; + // returnURL must be passed in while confirming (not creation time) + params.returnURL = @"example-app-scheme://authorized"; + [client confirmPaymentIntentWithParams:params + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, params.stripeId); + XCTAssertFalse(paymentIntent.livemode); + + // sourceParams is the 3DS-required test card + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + + // STPRedirectContext is relying on receiving returnURL + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + // Test deprecated property still works too +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNotNil(paymentIntent.nextSourceAction.authorizeWithURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextSourceAction.authorizeWithURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmPaymentIntentWith3DSCardPaymentMethodSucceeds { + + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodCardParams *cardParams = [STPPaymentMethodCardParams new]; + cardParams.number = @"4000000000003063"; + cardParams.expMonth = @(7); + cardParams.expYear = @([[NSCalendar currentCalendar] component:NSCalendarUnitYear fromDate:[NSDate date]] + 5); + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:cardParams + billingDetails:billingDetails + metadata:nil]; + // returnURL must be passed in while confirming (not creation time) + params.returnURL = @"example-app-scheme://authorized"; + [client confirmPaymentIntentWithParams:params + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, params.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // sourceParams is the 3DS-required test card + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + + // STPRedirectContext is relying on receiving returnURL + + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + // Going to log all the fields so that you, the developer manually running this test, can inspect them + NSLog(@"Confirmed PaymentIntent: %@", paymentIntent.allResponseFields); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmPaymentIntentWithShippingDetailsSucceeds { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodCardParams *cardParams = [STPPaymentMethodCardParams new]; + cardParams.number = @"4242424242424242"; + cardParams.expMonth = @(7); + cardParams.expYear = @([[NSCalendar currentCalendar] component:NSCalendarUnitYear fromDate:[NSDate date]] + 5); + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:cardParams + billingDetails:billingDetails + metadata:nil]; + + STPPaymentIntentShippingDetailsAddressParams *addressParams = [[STPPaymentIntentShippingDetailsAddressParams alloc] initWithLine1:@"123 Main St"]; + addressParams.line2 = @"Apt 2"; + addressParams.city = @"San Francisco"; + addressParams.state = @"CA"; + addressParams.country = @"US"; + addressParams.postalCode = @"94106"; + params.shipping = [[STPPaymentIntentShippingDetailsParams alloc] initWithAddress:addressParams name:@"Jane"]; + params.shipping.carrier = @"UPS"; + params.shipping.phone = @"555-555-5555"; + params.shipping.trackingNumber = @"123abc"; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + [client confirmPaymentIntentWithParams:params + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, params.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // Address + XCTAssertEqualObjects(paymentIntent.shipping.address.line1, @"123 Main St"); + XCTAssertEqualObjects(paymentIntent.shipping.address.line2, @"Apt 2"); + XCTAssertEqualObjects(paymentIntent.shipping.address.city, @"San Francisco"); + XCTAssertEqualObjects(paymentIntent.shipping.address.state, @"CA"); + XCTAssertEqualObjects(paymentIntent.shipping.address.country, @"US"); + XCTAssertEqualObjects(paymentIntent.shipping.address.postalCode, @"94106"); + + XCTAssertEqualObjects(paymentIntent.shipping.name, @"Jane"); + XCTAssertEqualObjects(paymentIntent.shipping.carrier, @"UPS"); + XCTAssertEqualObjects(paymentIntent.shipping.phone, @"555-555-5555"); + XCTAssertEqualObjects(paymentIntent.shipping.trackingNumber, @"123abc"); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmCardWithoutNetworkParam { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodCardParams *cardParams = [STPPaymentMethodCardParams new]; + cardParams.number = @"4242424242424242"; + cardParams.expMonth = @(7); + cardParams.expYear = @([[NSCalendar currentCalendar] component:NSCalendarUnitYear fromDate:[NSDate date]] + 5); + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:cardParams + billingDetails:billingDetails + metadata:nil]; + + [client confirmPaymentIntentWithParams:params + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, params.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusSucceeded); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmCardWithNetworkParam { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodCardParams *cardParams = [STPPaymentMethodCardParams new]; + cardParams.number = @"4242424242424242"; + cardParams.expMonth = @(7); + cardParams.expYear = @([[NSCalendar currentCalendar] component:NSCalendarUnitYear fromDate:[NSDate date]] + 5); + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:cardParams + billingDetails:billingDetails + metadata:nil]; + + STPConfirmCardOptions *cardOptions = [[STPConfirmCardOptions alloc] init]; + cardOptions.network = @"visa"; + STPConfirmPaymentMethodOptions *paymentMethodOptions = [[STPConfirmPaymentMethodOptions alloc] init]; + paymentMethodOptions.cardOptions = cardOptions; + params.paymentMethodOptions = paymentMethodOptions; + + [client confirmPaymentIntentWithParams:params + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, params.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusSucceeded); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmCardWithInvalidNetworkParam { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:nil completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodCardParams *cardParams = [STPPaymentMethodCardParams new]; + cardParams.number = @"4242424242424242"; + cardParams.expMonth = @(7); + cardParams.expYear = @([[NSCalendar currentCalendar] component:NSCalendarUnitYear fromDate:[NSDate date]] + 5); + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + + params.paymentMethodParams = [STPPaymentMethodParams paramsWithCard:cardParams + billingDetails:billingDetails + metadata:nil]; + + STPConfirmCardOptions *cardOptions = [[STPConfirmCardOptions alloc] init]; + cardOptions.network = @"fake_network"; + STPConfirmPaymentMethodOptions *paymentMethodOptions = [[STPConfirmPaymentMethodOptions alloc] init]; + paymentMethodOptions.cardOptions = cardOptions; + params.paymentMethodOptions = paymentMethodOptions; + + [client confirmPaymentIntentWithParams:params + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNotNil(error, @"Confirming with invalid network should result in an error"); + + XCTAssertNil(paymentIntent); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - giropay + +- (void)testConfirmPaymentIntentWithGiropay { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"payment_method_types": @[@"giropay"], + @"currency": @"eur", + } + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodGiropayParams *giropayParams = [STPPaymentMethodGiropayParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithGiropay:giropayParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + paymentIntentParams.returnURL = @"example-app-scheme://authorized"; + + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // giropay requires a redirect + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - AU BECS Debit + +- (void)testConfirmAUBECSDebitPaymentIntent { + + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"currency": @"aud", + @"amount": @(2000), + @"payment_method_types": @[@"au_becs_debit"], + } + account:@"au" + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPPaymentMethodAUBECSDebitParams *becsParams = [STPPaymentMethodAUBECSDebitParams new]; + becsParams.bsbNumber = @"000000"; // Stripe test bank + becsParams.accountNumber = @"000123456"; // test account + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + billingDetails.email = @"jrosen@example.com"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithAUBECSDebit:becsParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + paymentIntentParams.paymentMethodParams = params; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingAUPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // AU BECS Debit should be in Processing + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusProcessing); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - Przelewy24 + +- (void)testConfirmPaymentIntentWithPrzelewy24 { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"payment_method_types": @[@"p24"], + @"currency": @"eur", + } + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodPrzelewy24Params *przelewy24Params = [STPPaymentMethodPrzelewy24Params new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.email = @"email@email.com"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithPrzelewy24:przelewy24Params + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + paymentIntentParams.returnURL = @"example-app-scheme://authorized"; + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // Przelewy24 requires a redirect + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - Bancontact + +- (void)testConfirmPaymentIntentWithBancontact { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"payment_method_types": @[@"bancontact"], + @"currency": @"eur", + } + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodBancontactParams *bancontact = [STPPaymentMethodBancontactParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jane Doe"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithBancontact:bancontact + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + paymentIntentParams.returnURL = @"example-app-scheme://authorized"; + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // Bancontact requires a redirect + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - OXXO + +- (void)testConfirmPaymentIntentWithOXXO { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"payment_method_types": @[@"oxxo"], + @"amount": @(2000), + @"currency": @"mxn", + } + account:@"mex" + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingMEXPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodOXXOParams *oxxo = [STPPaymentMethodOXXOParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jane Doe"; + billingDetails.email = @"email@email.com"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithOXXO:oxxo + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // OXXO requires display the voucher as next step + NSDictionary *oxxoDisplayDetails = [paymentIntent.nextAction.allResponseFields objectForKey:@"oxxo_display_details"]; + XCTAssertNotNil([oxxoDisplayDetails objectForKey:@"expires_after"]); + XCTAssertNotNil([oxxoDisplayDetails objectForKey:@"number"]); + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + + +#pragma mark - EPS + +- (void)testConfirmPaymentIntentWithEPS { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"payment_method_types": @[@"eps"], + @"currency": @"eur", + } + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodEPSParams *epsParams = [STPPaymentMethodEPSParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithEPS:epsParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + paymentIntentParams.returnURL = @"example-app-scheme://authorized"; + + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // EPS requires a redirect + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - Alipay + +- (void)testConfirmAlipayPaymentIntent { + __block NSString *clientSecret; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"currency": @"usd", + @"amount": @(2000), + @"payment_method_types": @[@"alipay"], + } + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithAlipay:[STPPaymentMethodAlipayParams new] billingDetails:nil metadata:nil]; + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + paymentIntentParams.paymentMethodParams = params; + paymentIntentParams.returnURL = @"foo://bar"; + paymentIntentParams.paymentMethodOptions = [STPConfirmPaymentMethodOptions new]; + paymentIntentParams.paymentMethodOptions.alipayOptions = [STPConfirmAlipayOptions new]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertEqual(paymentIntent.nextAction.type, STPIntentActionTypeAlipayHandleRedirect); + XCTAssertNotNil(paymentIntent.nextAction); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - GrabPay + +- (void)testConfirmPaymentIntentWithGrabPay { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"payment_method_types": @[@"grabpay"], + @"currency": @"sgd", + } + account:@"sg" + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingSGPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodGrabPayParams *grabpay = [STPPaymentMethodGrabPayParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithGrabPay:grabpay + billingDetails:billingDetails + metadata:nil]; + paymentIntentParams.returnURL = @"example-app-scheme://authorized"; + + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // GrabPay requires a redirect + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - PayPal + +- (void)testConfirmPaymentIntentWithPayPal { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createPaymentIntentWithParams:@{ + @"payment_method_types": @[@"paypal"], + @"currency": @"eur", + } + account:@"be" + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingBEPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodPayPalParams *payPal = [STPPaymentMethodPayPalParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jane Doe"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithPayPal:payPal + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + paymentIntentParams.returnURL = @"example-app-scheme://authorized"; + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // PayPal requires a redirect + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(paymentIntent.nextAction.redirectToURL.returnURL, + [NSURL URLWithString:@"example-app-scheme://authorized"]); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - BLIK + +- (void)testConfirmPaymentIntentWithBLIK { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] + createPaymentIntentWithParams: @{ + @"payment_method_types": @[@"blik"], + @"currency": @"pln", + @"amount": @1000, + } + account: @"be" + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingBEPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodBLIKParams *blik = [STPPaymentMethodBLIKParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jane Doe"; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithBLIK:blik + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + STPConfirmPaymentMethodOptions *options = [STPConfirmPaymentMethodOptions new]; + options.blikOptions = [[STPConfirmBLIKOptions alloc] initWithCode:@"123456"]; + paymentIntentParams.paymentMethodOptions = options; + paymentIntentParams.returnURL = @"example-app-scheme://unused"; + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + // Blik transitions to requires_action until the customer authorizes the transaction or 1 minute passes + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertEqual(paymentIntent.nextAction.type, STPIntentActionTypeBLIKAuthorize); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - Affirm + +- (void)testConfirmPaymentIntentWithAffirm { + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] + createPaymentIntentWithParams: @{ + @"payment_method_types": @[@"affirm"], + @"currency": @"usd", + @"amount": @6000, + } + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Intent confirm"]; + + STPPaymentIntentParams *paymentIntentParams = [[STPPaymentIntentParams alloc] initWithClientSecret:clientSecret]; + STPPaymentMethodAffirmParams *affirm = [STPPaymentMethodAffirmParams new]; + + paymentIntentParams.paymentMethodParams = [STPPaymentMethodParams paramsWithAffirm:affirm + metadata:@{@"test_key": @"test_value"}]; + + STPPaymentIntentShippingDetailsAddressParams *addressParams = [[STPPaymentIntentShippingDetailsAddressParams alloc] initWithLine1:@"123 Main St"]; + addressParams.line2 = @"Apt 2"; + addressParams.city = @"San Francisco"; + addressParams.state = @"CA"; + addressParams.country = @"US"; + addressParams.postalCode = @"94106"; + paymentIntentParams.shipping = [[STPPaymentIntentShippingDetailsParams alloc] initWithAddress:addressParams name:@"Jane"]; + + STPConfirmPaymentMethodOptions *options = [STPConfirmPaymentMethodOptions new]; + paymentIntentParams.paymentMethodOptions = options; + paymentIntentParams.returnURL = @"example-app-scheme://unused"; + [client confirmPaymentIntentWithParams:paymentIntentParams + completion:^(STPPaymentIntent * _Nullable paymentIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(paymentIntent); + XCTAssertEqualObjects(paymentIntent.stripeId, paymentIntentParams.stripeId); + XCTAssertFalse(paymentIntent.livemode); + XCTAssertNotNil(paymentIntent.paymentMethodId); + + XCTAssertEqual(paymentIntent.status, STPPaymentIntentStatusRequiresAction); + XCTAssertEqual(paymentIntent.nextAction.type, STPIntentActionTypeRedirectToURL); + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - Test Objective-C setupFutureUsage + +- (void)testObjectiveCSetupFutureUsage { + STPPaymentIntentParams *params = [[STPPaymentIntentParams alloc] init]; + params.setupFutureUsage = @(STPPaymentIntentSetupFutureUsageOnSession); + XCTAssertEqualObjects(params.setupFutureUsageRawString, @"on_session"); +} + +#pragma mark - Helpers + +- (STPSourceParams *)cardSourceParams { + STPCardParams *card = [[STPCardParams alloc] init]; + card.number = @"4000 0000 0000 3063"; // Test 3DS required card + card.expMonth = 7; + card.expYear = [[NSCalendar currentCalendar] component:NSCalendarUnitYear fromDate:[NSDate date]] + 5; + card.currency = @"usd"; + + return [STPSourceParams cardParamsWithCard:card]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.swift b/Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.swift new file mode 100644 index 00000000..7a11b760 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentIntentFunctionalTest.swift @@ -0,0 +1,152 @@ +// +// STPPaymentIntentFunctionalTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/2/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentIntentFunctionalTestSwift: XCTestCase { + + // MARK: - US Bank Account + func createAndConfirmPaymentIntentWithUSBankAccount( + paymentMethodOptions: STPConfirmUSBankAccountOptions? = nil, + completion: @escaping (String?) -> Void + ) { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + var clientSecret: String? + let createPIExpectation = expectation(description: "Create PaymentIntent") + STPTestingAPIClient.shared().createPaymentIntent( + withParams: [ + "payment_method_types": ["us_bank_account"], + "currency": "usd", + "amount": 1000, + ], + account: nil + ) { intentClientSecret, error in + XCTAssertNil(error) + XCTAssertNotNil(intentClientSecret) + clientSecret = intentClientSecret + createPIExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + guard let clientSecret = clientSecret else { + XCTFail("Failed to create PaymentIntent") + return + } + + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountType = .checking + usBankAccountParams.accountHolderType = .individual + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let paymentMethodParams = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let paymentIntentParams = STPPaymentIntentParams(clientSecret: clientSecret) + paymentIntentParams.paymentMethodParams = paymentMethodParams + if let paymentMethodOptions = paymentMethodOptions { + let pmo = STPConfirmPaymentMethodOptions() + pmo.usBankAccountOptions = paymentMethodOptions + paymentIntentParams.paymentMethodOptions = pmo + } + + let confirmPIExpectation = expectation(description: "Confirm PaymentIntent") + client.confirmPaymentIntent(with: paymentIntentParams, expand: ["payment_method"]) { + paymentIntent, + error in + XCTAssertNil(error) + XCTAssertNotNil(paymentIntent) + XCTAssertNotNil(paymentIntent?.paymentMethod) + XCTAssertNotNil(paymentIntent?.paymentMethod?.usBankAccount) + XCTAssertEqual(paymentIntent?.paymentMethod?.usBankAccount?.last4, "6789") + XCTAssertEqual(paymentIntent?.status, .requiresAction) + XCTAssertEqual(paymentIntent?.nextAction?.type, .verifyWithMicrodeposits) + if let paymentMethodOptions = paymentMethodOptions { + XCTAssertEqual( + paymentIntent?.paymentMethodOptions?.usBankAccount?.setupFutureUsage, + paymentMethodOptions.setupFutureUsage + ) + } + confirmPIExpectation.fulfill() + } + + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + completion(clientSecret) + } + + func testConfirmPaymentIntentWithUSBankAccount_verifyWithAmounts() { + createAndConfirmPaymentIntentWithUSBankAccount { [self] clientSecret in + guard let clientSecret = clientSecret else { + XCTFail("Failed to create PaymentIntent") + return + } + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let verificationExpectation = expectation(description: "Verify with microdeposits") + client.verifyPaymentIntentWithMicrodeposits( + clientSecret: clientSecret, + firstAmount: 32, + secondAmount: 45 + ) { paymentIntent, error in + XCTAssertNil(error) + XCTAssertNotNil(paymentIntent) + XCTAssertEqual(paymentIntent?.status, .processing) + verificationExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + } + + func testConfirmPaymentIntentWithUSBankAccount_verifyWithDescriptorCode() { + createAndConfirmPaymentIntentWithUSBankAccount { [self] clientSecret in + guard let clientSecret = clientSecret else { + XCTFail("Failed to create PaymentIntent") + return + } + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let verificationExpectation = expectation(description: "Verify with microdeposits") + client.verifyPaymentIntentWithMicrodeposits( + clientSecret: clientSecret, + descriptorCode: "SM11AA" + ) { paymentIntent, error in + XCTAssertNil(error) + XCTAssertNotNil(paymentIntent) + XCTAssertEqual(paymentIntent?.status, .processing) + verificationExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + } + + func testConfirmUSBankAccountWithPaymentMethodOptions() { + createAndConfirmPaymentIntentWithUSBankAccount( + paymentMethodOptions: STPConfirmUSBankAccountOptions(setupFutureUsage: .offSession) + ) { clientSecret in + XCTAssertNotNil(clientSecret) + } + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentIntentLastPaymentErrorTest.swift b/Stripe/StripeiOSTests/STPPaymentIntentLastPaymentErrorTest.swift new file mode 100644 index 00000000..208b9a2d --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentIntentLastPaymentErrorTest.swift @@ -0,0 +1,60 @@ +// +// STPPaymentIntentLastPaymentErrorTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 9/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentIntentLastPaymentErrorTest: XCTestCase { + + func testErrorType() { + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "api_connection_error"), + .apiConnection + ) + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "API_CONNECTION_ERROR"), + .apiConnection + ) + XCTAssertEqual(STPPaymentIntentLastPaymentErrorType(string: "api_error"), .api) + XCTAssertEqual(STPPaymentIntentLastPaymentErrorType(string: "API_ERROR"), .api) + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "authentication_error"), + .authentication + ) + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "AUTHENTICATION_ERROR"), + .authentication + ) + XCTAssertEqual(STPPaymentIntentLastPaymentErrorType(string: "card_error"), .card) + XCTAssertEqual(STPPaymentIntentLastPaymentErrorType(string: "CARD_ERROR"), .card) + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "idempotency_error"), + .idempotency + ) + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "IDEMPOTENCY_ERROR"), + .idempotency + ) + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "invalid_request_error"), + .invalidRequest + ) + XCTAssertEqual( + STPPaymentIntentLastPaymentErrorType(string: "INVALID_REQUEST_ERROR"), + .invalidRequest + ) + XCTAssertEqual(STPPaymentIntentLastPaymentErrorType(string: "rate_limit_error"), .rateLimit) + XCTAssertEqual(STPPaymentIntentLastPaymentErrorType(string: "RATE_LIMIT_ERROR"), .rateLimit) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentIntentParamsTest.swift b/Stripe/StripeiOSTests/STPPaymentIntentParamsTest.swift new file mode 100644 index 00000000..333101f3 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentIntentParamsTest.swift @@ -0,0 +1,218 @@ +// +// STPPaymentIntentParamsTest.swift +// StripeiOS Tests +// +// Created by Daniel Jackson on 7/5/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentIntentParamsTest: XCTestCase { + func testInit() { + for params in [ + STPPaymentIntentParams(clientSecret: "secret"), + STPPaymentIntentParams(), + STPPaymentIntentParams(), + ] { + XCTAssertNotNil(params) + XCTAssertNotNil(params.clientSecret) + XCTAssertNotNil(params.additionalAPIParameters) + XCTAssertEqual(params.additionalAPIParameters.count, 0) + + XCTAssertNil(params.stripeId, "invalid secrets, no stripeId") + XCTAssertNil(params.sourceParams) + XCTAssertNil(params.sourceId) + XCTAssertNil(params.receiptEmail) + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(params.saveSourceToCustomer) + // #pragma clang diagnostic pop + XCTAssertNil(params.savePaymentMethod) + XCTAssertNil(params.returnURL) + XCTAssertNil(params.setupFutureUsage) + XCTAssertNil(params.useStripeSDK) + XCTAssertNil(params.mandateData) + XCTAssertNil(params.paymentMethodOptions) + XCTAssertNil(params.shipping) + } + } + + func testDescription() { + let params = STPPaymentIntentParams() + XCTAssertNotNil(params.description) + } + + // MARK: Deprecated Property + + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wdeprecated" + func testReturnURLRenaming() { + let params = STPPaymentIntentParams() + + XCTAssertNil(params.returnURL) + XCTAssertNil(params.returnUrl) + + params.returnURL = "set via new name" + XCTAssertEqual(params.returnUrl, "set via new name") + + params.returnUrl = "set via old name" + XCTAssertEqual(params.returnURL, "set via old name") + } + + func testSaveSourceToCustomerRenaming() { + let params = STPPaymentIntentParams() + + XCTAssertNil(params.saveSourceToCustomer) + XCTAssertNil(params.savePaymentMethod) + + params.savePaymentMethod = NSNumber(value: false) + XCTAssertEqual(params.saveSourceToCustomer, NSNumber(value: false)) + + params.saveSourceToCustomer = NSNumber(value: true) + XCTAssertEqual(params.savePaymentMethod, NSNumber(value: true)) + } + + func testDefaultMandateData() { + let params = STPPaymentIntentParams() + + // no configuration should have no mandateData + XCTAssertNil(params.mandateData) + + params.paymentMethodParams = STPPaymentMethodParams() + + params.paymentMethodParams!.rawTypeString = "card" + // card type should have no default mandateData + XCTAssertNil(params.mandateData) + + for type in ["sepa_debit", "au_becs_debit", "bacs_debit"] { + params.mandateData = nil + params.paymentMethodParams!.rawTypeString = type + // Mandate-required type should have mandateData + XCTAssertNotNil(params.mandateData) + XCTAssertEqual( + params.mandateData!.customerAcceptance.onlineParams!.inferFromClient, + NSNumber(value: true) + ) + + params.mandateData = STPMandateDataParams( + customerAcceptance: STPMandateCustomerAcceptanceParams( + type: .offline, + onlineParams: nil + )! + ) + // Default behavior should not override custom setting + XCTAssertNotNil(params.mandateData) + XCTAssertNil(params.mandateData!.customerAcceptance.onlineParams) + } + } + + // #pragma clang diagnostic pop + + // MARK: STPFormEncodable Tests + func testRootObjectName() { + XCTAssertNil(STPPaymentIntentParams.rootObjectName()) + } + + func testPropertyNamesToFormFieldNamesMapping() { + let params = STPPaymentIntentParams() + + let mapping = STPPaymentIntentParams.propertyNamesToFormFieldNamesMapping() + + for propertyName in mapping.keys { + XCTAssertFalse(propertyName.contains(":")) + XCTAssert(params.responds(to: NSSelectorFromString(propertyName))) + } + + for formFieldName in mapping.values { + XCTAssert(formFieldName.count > 0) + } + + XCTAssertEqual( + mapping.values.count, + NSSet(array: (mapping as NSDictionary).allValues).count + ) + } + + func testCopy() { + let params = STPPaymentIntentParams(clientSecret: "test_client_secret") + params.paymentMethodParams = STPPaymentMethodParams() + params.paymentMethodId = "test_payment_method_id" + params.savePaymentMethod = NSNumber(value: true) + params.returnURL = "fake://testing_only" + params.setupFutureUsage = STPPaymentIntentSetupFutureUsage( + rawValue: Int(truncating: NSNumber(value: 1)) + ) + params.useStripeSDK = NSNumber(value: true) + params.mandateData = STPMandateDataParams( + customerAcceptance: STPMandateCustomerAcceptanceParams( + type: .offline, + onlineParams: nil + )! + ) + params.paymentMethodOptions = STPConfirmPaymentMethodOptions() + params.additionalAPIParameters = [ + "other_param": "other_value" + ] + params.shipping = STPPaymentIntentShippingDetailsParams( + address: STPPaymentIntentShippingDetailsAddressParams(line1: ""), + name: "" + ) + + let paramsCopy = params.copy() as! STPPaymentIntentParams + XCTAssertEqual(params.clientSecret, paramsCopy.clientSecret) + XCTAssertEqual(params.paymentMethodId, paramsCopy.paymentMethodId) + + // assert equal, not equal objects, because this is a shallow copy + XCTAssertEqual(params.paymentMethodParams, paramsCopy.paymentMethodParams) + XCTAssertEqual(params.mandateData, paramsCopy.mandateData) + XCTAssertEqual(params.shipping, paramsCopy.shipping) + + XCTAssertEqual(params.setupFutureUsage, STPPaymentIntentSetupFutureUsage.none) + XCTAssertEqual(params.savePaymentMethod, paramsCopy.savePaymentMethod) + XCTAssertEqual(params.returnURL, paramsCopy.returnURL) + XCTAssertEqual(params.useStripeSDK, paramsCopy.useStripeSDK) + XCTAssertEqual(params.paymentMethodOptions, paramsCopy.paymentMethodOptions) + XCTAssertEqual( + params.additionalAPIParameters as NSDictionary, + paramsCopy.additionalAPIParameters as NSDictionary + ) + + } + + func testClientSecretValidation() { + XCTAssertFalse( + STPPaymentIntentParams.isClientSecretValid("pi_12345"), + "'pi_12345' is not a valid client secret." + ) + XCTAssertFalse( + STPPaymentIntentParams.isClientSecretValid("pi_12345_secret_"), + "'pi_12345_secret_' is not a valid client secret." + ) + XCTAssertFalse( + STPPaymentIntentParams.isClientSecretValid( + "pi_a1b2c3_secret_x7y8z9pi_a1b2c3_secret_x7y8z9" + ), + "'pi_a1b2c3_secret_x7y8z9pi_a1b2c3_secret_x7y8z9' is not a valid client secret." + ) + XCTAssertFalse( + STPPaymentIntentParams.isClientSecretValid("seti_a1b2c3_secret_x7y8z9"), + "'seti_a1b2c3_secret_x7y8z9' is not a valid client secret." + ) + + XCTAssertTrue( + STPPaymentIntentParams.isClientSecretValid("pi_a1b2c3_secret_x7y8z9"), + "'pi_a1b2c3_secret_x7y8z9' is a valid client secret." + ) + XCTAssertTrue( + STPPaymentIntentParams.isClientSecretValid( + "pi_1CkiBMLENEVhOs7YMtUehLau_secret_s4O8SDh7s6spSmHDw1VaYPGZA" + ), + "'pi_1CkiBMLENEVhOs7YMtUehLau_secret_s4O8SDh7s6spSmHDw1VaYPGZA' is a valid client secret." + ) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentIntentTest.swift b/Stripe/StripeiOSTests/STPPaymentIntentTest.swift new file mode 100644 index 00000000..cea18e03 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentIntentTest.swift @@ -0,0 +1,190 @@ +// +// STPPaymentIntentTest.swift +// StripeiOS Tests +// +// Created by Daniel Jackson on 6/27/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentIntentTest: XCTestCase { + func testIdentifierFromSecret() { + XCTAssertEqual( + STPPaymentIntent.id(fromClientSecret: "pi_123_secret_XYZ"), + "pi_123" + ) + XCTAssertEqual( + STPPaymentIntent.id( + fromClientSecret: "pi_123_secret_RandomlyContains_secret_WhichIsFine" + ), + "pi_123" + ) + + XCTAssertNil(STPPaymentIntent.id(fromClientSecret: "")) + XCTAssertNil(STPPaymentIntent.id(fromClientSecret: "po_123_secret_HasBadPrefix")) + XCTAssertNil(STPPaymentIntent.id(fromClientSecret: "MissingSentinalForSplitting")) + } + + // MARK: - Description Tests + func testDescription() { + let paymentIntent = STPFixtures.paymentIntent() + + XCTAssertNotNil(paymentIntent) + let desc = paymentIntent.description + XCTAssertTrue(desc.contains(NSStringFromClass(type(of: paymentIntent).self))) + XCTAssertGreaterThan((desc.count), 500, "Custom description should be long") + } + + // MARK: - STPAPIResponseDecodable Tests + func testDecodedObjectFromAPIResponseRequiredFields() { + let fullJson = STPTestUtils.jsonNamed(STPTestJSONPaymentIntent) + + XCTAssertNotNil( + STPPaymentIntent.decodedObject(fromAPIResponse: fullJson), + "can decode with full json" + ) + + let requiredFields = ["id", "client_secret", "amount", "currency", "livemode", "status"] + + for field in requiredFields { + var partialJson = fullJson + + XCTAssertNotNil(partialJson?[field]) + partialJson?.removeValue(forKey: field) + + XCTAssertNil(STPPaymentIntent.decodedObject(fromAPIResponse: partialJson)) + } + } + + func testDecodedObjectFromAPIResponseMapping() { + let paymentIntentJson = STPTestUtils.jsonNamed("PaymentIntent")! + let orderedPaymentJson = ["card", "ideal", "sepa_debit"] + let paymentIntentResponse = + [ + "payment_intent": paymentIntentJson, + "ordered_payment_method_types": orderedPaymentJson, + ] as [String: Any] + let unactivatedPaymentMethodTypes = ["sepa_debit"] + let response = + [ + "payment_method_preference": paymentIntentResponse, + "unactivated_payment_method_types": unactivatedPaymentMethodTypes, + ] as [String: Any] + + let paymentIntent = STPPaymentIntent.decodedObject(fromAPIResponse: response)! + + XCTAssertEqual(paymentIntent.stripeId, "pi_1Cl15wIl4IdHmuTbCWrpJXN6") + XCTAssertEqual( + paymentIntent.clientSecret, + "pi_1Cl15wIl4IdHmuTbCWrpJXN6_secret_EkKtQ7Sg75hLDFKqFG8DtWcaK" + ) + XCTAssertEqual(paymentIntent.amount, 2345) + XCTAssertEqual(paymentIntent.canceledAt, Date(timeIntervalSince1970: 1_530_911_045)) + XCTAssertEqual(paymentIntent.captureMethod, .manual) + XCTAssertEqual(paymentIntent.confirmationMethod, .automatic) + XCTAssertEqual(paymentIntent.created, Date(timeIntervalSince1970: 1_530_911_040)) + XCTAssertEqual(paymentIntent.currency, "usd") + XCTAssertEqual(paymentIntent.stripeDescription, "My Sample PaymentIntent") + XCTAssertFalse(paymentIntent.livemode) + XCTAssertEqual(paymentIntent.receiptEmail, "danj@example.com") + + // Deprecated: `nextSourceAction` & `authorizeWithURL` should just be aliases for `nextAction` & `redirectToURL` + // #pragma clang diagnostic push + // #pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertEqual( + paymentIntent.nextAction, + paymentIntent.nextAction, + "Should be the same object." + ) + XCTAssertEqual( + paymentIntent.nextAction!.redirectToURL!, + paymentIntent.nextAction!.redirectToURL, + "Should be the same object." + ) + // #pragma clang diagnostic pop + + // nextAction + XCTAssertNotNil(paymentIntent.nextAction) + XCTAssertEqual(paymentIntent.nextAction!.type, .redirectToURL) + XCTAssertNotNil(paymentIntent.nextAction!.redirectToURL) + XCTAssertNotNil(paymentIntent.nextAction!.redirectToURL!.url) + let returnURL = paymentIntent.nextAction!.redirectToURL!.returnURL + XCTAssertNotNil(returnURL) + XCTAssertEqual(returnURL, URL(string: "payments-example://stripe-redirect")) + let url = paymentIntent.nextAction!.redirectToURL!.url + XCTAssertNotNil(url) + + XCTAssertEqual( + url, + URL( + string: + "https://hooks.stripe.com/redirect/authenticate/src_1Cl1AeIl4IdHmuTb1L7x083A?client_secret=src_client_secret_DBNwUe9qHteqJ8qQBwNWiigk" + ) + ) + XCTAssertEqual(paymentIntent.sourceId, "src_1Cl1AdIl4IdHmuTbseiDWq6m") + XCTAssertEqual(paymentIntent.status, .requiresAction) + XCTAssertEqual(paymentIntent.setupFutureUsage, .none) + + XCTAssertEqual( + paymentIntent.paymentMethodTypes, + [NSNumber(value: STPPaymentMethodType.card.rawValue)] + ) + + // lastPaymentError + + XCTAssertNotNil(paymentIntent.lastPaymentError) + XCTAssertEqual( + paymentIntent.lastPaymentError!.code, + "payment_intent_authentication_failure" + ) + XCTAssertEqual( + paymentIntent.lastPaymentError!.docURL, + "https://stripe.com/docs/error-codes#payment-intent-authentication-failure" + ) + XCTAssertEqual( + paymentIntent.lastPaymentError!.message, + "The provided PaymentMethod has failed authentication. You can provide payment_method_data or a new PaymentMethod to attempt to fulfill this PaymentIntent again." + ) + XCTAssertNotNil(paymentIntent.lastPaymentError!.paymentMethod) + XCTAssertEqual(paymentIntent.lastPaymentError!.type, .invalidRequest) + + // Shipping + XCTAssertNotNil(paymentIntent.shipping) + XCTAssertEqual(paymentIntent.shipping!.carrier, "USPS") + XCTAssertEqual(paymentIntent.shipping!.name, "Dan") + XCTAssertEqual(paymentIntent.shipping!.phone, "1-415-555-1234") + XCTAssertEqual(paymentIntent.shipping!.trackingNumber, "xyz123abc") + XCTAssertNotNil(paymentIntent.shipping!.address) + XCTAssertEqual(paymentIntent.shipping!.address!.city, "San Francisco") + XCTAssertEqual(paymentIntent.shipping!.address!.country, "USA") + XCTAssertEqual(paymentIntent.shipping!.address!.line1, "123 Main St") + XCTAssertEqual(paymentIntent.shipping!.address!.line2, "Apt 456") + XCTAssertEqual(paymentIntent.shipping!.address!.postalCode, "94107") + XCTAssertEqual(paymentIntent.shipping!.address!.state, "CA") + + // Ordered Payment Method Types + XCTAssertEqual( + paymentIntent.orderedPaymentMethodTypes.map({ $0.displayName }), + ["Card", "iDEAL", "SEPA Debit"] + ) + + // Unactivated Payment Method Types + XCTAssertEqual( + paymentIntent.unactivatedPaymentMethodTypes.map({ $0.displayName }), + ["SEPA Debit"] + ) + + var allResponseFields = paymentIntentJson + allResponseFields["ordered_payment_method_types"] = orderedPaymentJson + allResponseFields["unactivated_payment_method_types"] = unactivatedPaymentMethodTypes + XCTAssertEqual( + paymentIntent.allResponseFields as NSDictionary, + allResponseFields as NSDictionary + ) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitParamsTests.m new file mode 100644 index 00000000..93b504b2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitParamsTests.m @@ -0,0 +1,59 @@ +// +// STPPaymentMethodAUBECSDebitParamsTests.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/4/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodAUBECSDebitParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodAUBECSDebitParamsTests + +- (void)testCreateAUBECSPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingAUPublishableKey]; + STPPaymentMethodAUBECSDebitParams *becsParams = [STPPaymentMethodAUBECSDebitParams new]; + becsParams.bsbNumber = @"000000"; // Stripe test bank + becsParams.accountNumber = @"000123456"; // test account + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + billingDetails.email = @"jrosen@example.com"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithAUBECSDebit:becsParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method AU BECS Debit create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating AU BECS Debit PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create AU BECS Debit PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeAUBECSDebit, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.email, @"jrosen@example.com", @"Incorrect email"); + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + + // AU BECS Debit + XCTAssertEqualObjects(paymentMethod.auBECSDebit.bsbNumber, @"000000", @"Incorrect BSB Number"); + XCTAssertEqualObjects(paymentMethod.auBECSDebit.last4, @"3456", @"Incorrect last4"); + XCTAssertNotNil(paymentMethod.auBECSDebit.fingerprint, @"Missing fingerprint"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitTests.swift new file mode 100644 index 00000000..efd29c98 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodAUBECSDebitTests.swift @@ -0,0 +1,84 @@ +// +// STPPaymentMethodAUBECSDebitTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/4/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +private var kAUBECSDebitPaymentIntentClientSecret = + "pi_1GaRLjF7QokQdxByYgFPQEi0_secret_z76otRQH2jjOIEQYsA9vxhuKn" +class STPPaymentMethodAUBECSDebitTests: XCTestCase { + private(set) var auBECSDebitJSON: [AnyHashable: Any]? + + func _retrieveAUBECSDebitJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let auBECSDebitJSON = auBECSDebitJSON { + completion(auBECSDebitJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingAUPublishableKey) + client.retrievePaymentIntent( + withClientSecret: kAUBECSDebitPaymentIntentClientSecret, + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + auBECSDebitJSON = paymentIntent?.paymentMethod?.auBECSDebit?.allResponseFields + completion(auBECSDebitJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + _retrieveAUBECSDebitJSON({ json in + let auBECSDebit = STPPaymentMethodAUBECSDebit.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(auBECSDebit, "Failed to decode JSON") + retrieveJSON.fulfill() + }) + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } + + func testFailWithoutRequired() { + var retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + _retrieveAUBECSDebitJSON({ json in + var auBECSDebitJSON = json + auBECSDebitJSON?["bsb_number"] = nil + XCTAssertNil( + STPPaymentMethodAUBECSDebit.decodedObject(fromAPIResponse: auBECSDebitJSON), + "Should not intialize with missing `bsb_number`" + ) + retrieveJSON.fulfill() + }) + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + + retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + _retrieveAUBECSDebitJSON({ json in + var auBECSDebitJSON = json + auBECSDebitJSON?["last4"] = nil + XCTAssertNil( + STPPaymentMethodAUBECSDebit.decodedObject(fromAPIResponse: auBECSDebitJSON), + "Should not intialize with missing `last4`" + ) + retrieveJSON.fulfill() + }) + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + + retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + _retrieveAUBECSDebitJSON({ json in + var auBECSDebitJSON = json + auBECSDebitJSON?["fingerprint"] = nil + XCTAssertNil( + STPPaymentMethodAUBECSDebit.decodedObject(fromAPIResponse: auBECSDebitJSON), + "Should not intialize with missing `fingerprint`" + ) + retrieveJSON.fulfill() + }) + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodAddressTest.m b/Stripe/StripeiOSTests/STPPaymentMethodAddressTest.m new file mode 100644 index 00000000..f587fbde --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodAddressTest.m @@ -0,0 +1,44 @@ +// +// STPPaymentMethodAddressTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/6/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPTestUtils.h" +#import "STPFixtures.h" + +@interface STPPaymentMethodAddressTest : XCTestCase + +@end + +@implementation STPPaymentMethodAddressTest + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"billing_details"][@"address"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPPaymentMethodAddress decodedObjectFromAPIResponse:response]); + } + + XCTAssertNotNil([STPPaymentMethodAddress decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"billing_details"][@"address"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"billing_details"][@"address"]; + STPPaymentMethodAddress *address = [STPPaymentMethodAddress decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(address.city, @"München"); + XCTAssertEqualObjects(address.country, @"DE"); + XCTAssertEqualObjects(address.postalCode, @"80337"); + XCTAssertEqualObjects(address.line1, @"Marienplatz"); + XCTAssertEqualObjects(address.line2, @"8"); + XCTAssertEqualObjects(address.state, @"Bayern"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodAffirmParamsTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodAffirmParamsTest.swift new file mode 100644 index 00000000..29bb9990 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodAffirmParamsTest.swift @@ -0,0 +1,43 @@ +// +// STPPaymentMethodAffirmParamsTest.swift +// StripeiOS Tests +// +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodAffirmParamsTests: XCTestCase { + + func testCreateAffirmPaymentMethod() throws { + let affirmParams = STPPaymentMethodAffirmParams() + + let params = STPPaymentMethodParams( + affirm: affirmParams, + metadata: nil + ) + + let exp = expectation(description: "Payment Method Affirm create") + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + exp.fulfill() + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .affirm, "Incorrect PaymentMethod type") + XCTAssertNotNil(paymentMethod?.affirm, "The `affirm` property must be populated") + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodAffirmTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodAffirmTests.swift new file mode 100644 index 00000000..83bb542e --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodAffirmTests.swift @@ -0,0 +1,46 @@ +// +// STPPaymentMethodAffirmTests.swift +// StripeiOS Tests +// +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodAffirmTests: XCTestCase { + + static let affirmPaymentIntentClientSecret = + "pi_3KUFbTFY0qyl6XeW1oDBbiQk_secret_8kdpLx37oa5WMrI2xoXThCK9s" + + func _retrieveAffirmJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: Self.affirmPaymentIntentClientSecret, + expand: ["payment_method"] + ) { paymentIntent, _ in + let affirmJson = paymentIntent?.paymentMethod?.affirm?.allResponseFields + XCTAssertNotNil(paymentIntent?.paymentMethod?.affirm) + completion(affirmJson ?? [:]) + } + } + + func testObjectDecoding() { + let retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + + _retrieveAffirmJSON({ json in + let affirm = STPPaymentMethodAffirm.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(affirm, "Failed to decode JSON") + retrieveJSON.fulfill() + }) + + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayParamsTest.m b/Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayParamsTest.m new file mode 100644 index 00000000..3b0d534a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayParamsTest.m @@ -0,0 +1,65 @@ +// +// STPPaymentMethodAfterpayClearpayParamsTest.m +// StripeiOS Tests +// +// Created by Ali Riaz on 1/14/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodAfterpayClearpayParamsTest : XCTestCase + +@end + +@implementation STPPaymentMethodAfterpayClearpayParamsTest + +- (void)testCreateAfterpayClearpayPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodAfterpayClearpayParams *afterpayClearpayParams = [STPPaymentMethodAfterpayClearpayParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + billingDetails.email = @"jrosen@example.com"; + billingDetails.address = [STPPaymentMethodAddress new]; + billingDetails.address.line1 = @"510 Townsend St."; + billingDetails.address.postalCode = @"94102"; + billingDetails.address.country = @"US"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithAfterpayClearpay:afterpayClearpayParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method AfterpayClearpay create"]; + + [client createPaymentMethodWithParams:params completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating AfterpayClearpay Payment Method %@a", error); + XCTAssertNotNil(paymentMethod, @"Failed to create AfterpayClearpay PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeAfterpayClearpay, @"Incorrect PaymentMethod type"); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(paymentMethod.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.email, @"jrosen@example.com", @"Incorrect email"); + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.line1, @"510 Townsend St.", @"Incorrect address"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.postalCode, @"94102", @"Incorrect address"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.country, @"US", @"Incorrect address"); + + XCTAssertNotNil(paymentMethod.afterpayClearpay, @""); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayTest.m b/Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayTest.m new file mode 100644 index 00000000..ed3fced8 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodAfterpayClearpayTest.m @@ -0,0 +1,44 @@ +// +// STPPaymentMethodAfterpayClearpayTest.m +// StripeiOS Tests +// +// Created by Ali Riaz on 1/14/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" +@import Stripe; + +@interface STPPaymentMethodAfterpayClearpayTest : XCTestCase +@property (strong, nonatomic) NSDictionary *afterpayJSON; +@end + +@implementation STPPaymentMethodAfterpayClearpayTest + +- (void)_retrieveAfterpayJSON:(void (^)(NSDictionary *))completion { + if (self.afterpayJSON) { + completion(self.afterpayJSON); + } else { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + [client retrievePaymentIntentWithClientSecret:@"pi_1HbSAfFY0qyl6XeWRnlezJ7K_secret_t6Ju9Z0hxOvslawK34uC1Wm2b" + expand:@[@"payment_method"] completion:^(STPPaymentIntent * _Nullable paymentIntent, __unused NSError * _Nullable error) { + self.afterpayJSON = paymentIntent.paymentMethod.afterpayClearpay.allResponseFields; + completion(self.afterpayJSON); + }]; + } +} + +- (void)testCorrectParsing { + XCTestExpectation *jsonExpectation = [[XCTestExpectation alloc] initWithDescription:@"Fetch Afterpay Clearpay JSON"]; + [self _retrieveAfterpayJSON:^(NSDictionary *json) { + STPPaymentMethodAfterpayClearpay *afterpay = [STPPaymentMethodAfterpayClearpay decodedObjectFromAPIResponse:json]; + XCTAssertNotNil(afterpay, @"Failed to decode JSON"); + [jsonExpectation fulfill]; + }]; + [self waitForExpectations:@[jsonExpectation] timeout:TestConstants.STPTestingNetworkRequestTimeout]; +} + + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodBacsDebitTest.m b/Stripe/StripeiOSTests/STPPaymentMethodBacsDebitTest.m new file mode 100644 index 00000000..8da91a8a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodBacsDebitTest.m @@ -0,0 +1,46 @@ +// +// STPPaymentMethodBacsDebitTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 1/28/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPTestUtils.h" +#import "STPFixtures.h" + +@interface STPPaymentMethodBacsDebitTest : XCTestCase + +@end + +@implementation STPPaymentMethodBacsDebitTest + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSDictionary *paymentMethodJSON = [STPTestUtils jsonNamed:STPTestJSONPaymentMethodBacsDebit]; + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [paymentMethodJSON[@"bacs_debit"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPPaymentMethodBacsDebit decodedObjectFromAPIResponse:response]); + } + + STPPaymentMethod *paymentMethod = [STPPaymentMethod decodedObjectFromAPIResponse:paymentMethodJSON]; + XCTAssertNotNil(paymentMethod); + XCTAssertNotNil(paymentMethod.bacsDebit); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONPaymentMethodBacsDebit][@"bacs_debit"]; + STPPaymentMethodBacsDebit *bacs = [STPPaymentMethodBacsDebit decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(bacs.fingerprint, @"9eMbmctOrd8i7DYa"); + XCTAssertEqualObjects(bacs.last4, @"2345"); + XCTAssertEqualObjects(bacs.sortCode, @"108800"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodBancontactParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodBancontactParamsTests.m new file mode 100644 index 00000000..a6095ff1 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodBancontactParamsTests.m @@ -0,0 +1,53 @@ +// +// STPPaymentMethodBancontactParamsTests.m +// StripeiOS Tests +// +// Created by Vineet Shah on 4/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodBancontactParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodBancontactParamsTests + +- (void)testCreateBancontactPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodBancontactParams *bancontactParams = [STPPaymentMethodBancontactParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jane Doe"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithBancontact:bancontactParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method Bancontact create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating Bancontact PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create Bancontact PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeBancontact, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jane Doe", @"Incorrect name"); + + // Bancontact Details + XCTAssertNotNil(paymentMethod.bancontact, @"Missing Bancontact"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodBancontactTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodBancontactTests.swift new file mode 100644 index 00000000..16d299cd --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodBancontactTests.swift @@ -0,0 +1,44 @@ +// +// STPPaymentMethodBancontactTests.swift +// StripeiOS Tests +// +// Created by Vineet Shah on 4/29/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodBancontactTests: XCTestCase { + private(set) var bancontactJSON: [AnyHashable: Any]? + + func _retrieveBancontactJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let bancontactJSON = bancontactJSON { + completion(bancontactJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: "pi_1GdPnbFY0qyl6XeW8Ezvxe87_secret_Fxi2EZBQ0nInHumvvezcTRWF4", + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + bancontactJSON = paymentIntent?.paymentMethod?.bancontact?.allResponseFields + completion(bancontactJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let expectation = self.expectation(description: "Retrieve payment intent") + _retrieveBancontactJSON({ json in + let bancontact = STPPaymentMethodBancontact.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(bancontact, "Failed to decode JSON") + expectation.fulfill() + }) + waitForExpectations(timeout: STPTestingNetworkRequestTimeout, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTest.m b/Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTest.m new file mode 100644 index 00000000..15faf90b --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTest.m @@ -0,0 +1,44 @@ +// +// STPPaymentMethodBillingDetailsTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/6/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPPaymentMethodBillingDetailsTest : XCTestCase + +@end + +@implementation STPPaymentMethodBillingDetailsTest + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"billing_details"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPPaymentMethodBillingDetails decodedObjectFromAPIResponse:response]); + } + + XCTAssertNotNil([STPPaymentMethodBillingDetails decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"billing_details"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"billing_details"]; + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(billingDetails.email, @"jenny@example.com"); + XCTAssertEqualObjects(billingDetails.name, @"jenny"); + XCTAssertEqualObjects(billingDetails.phone, @"+15555555555"); + XCTAssertNotNil(billingDetails.address); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTests+Link.swift b/Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTests+Link.swift new file mode 100644 index 00000000..577a3d55 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodBillingDetailsTests+Link.swift @@ -0,0 +1,41 @@ +// +// STPPaymentMethodBillingDetailsTests.swift +// StripeiOSTests +// +// Created by Eduardo Urias on 2/27/23. +// + +@testable import StripePaymentSheet +import XCTest + +// Link mapping tests +final class STPPaymentMethodBillingDetailsTests: XCTestCase { + func testConsumersAPIParamsMapping() { + var billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "Name" + billingDetails.email = "Email" + billingDetails.phone = "Phone" + billingDetails.address = STPPaymentMethodAddress() + billingDetails.address?.line1 = "Line 1" + billingDetails.address?.line2 = "" + billingDetails.address?.city = "City" + billingDetails.address?.state = "State" + billingDetails.address?.country = "Country" + + let params = billingDetails.consumersAPIParams + XCTAssertEqual(params["name"] as? String, "Name") + XCTAssertEqual(params["line_1"] as? String, "Line 1") + XCTAssertNil(params["line_2"]) + XCTAssertEqual(params["locality"] as? String, "City") + XCTAssertEqual(params["administrative_area"] as? String, "State") + XCTAssertEqual(params["country_code"] as? String, "Country") + + XCTAssertNil(params["email"]) + XCTAssertNil(params["phone"]) + XCTAssertNil(params["line1"]) + XCTAssertNil(params["line2"]) + XCTAssertNil(params["city"]) + XCTAssertNil(params["state"]) + XCTAssertNil(params["country"]) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodBoletoParamsTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodBoletoParamsTests.swift new file mode 100644 index 00000000..91fe10a0 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodBoletoParamsTests.swift @@ -0,0 +1,71 @@ +// +// STPPaymentMethodBoletoParamsTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 9/9/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodBoletoParamsTests: XCTestCase { + + func testCreateBoletoPaymentMethod() throws { + let boletoParams = STPPaymentMethodBoletoParams() + boletoParams.taxID = "00.000.000/0001-91" + + let address = STPPaymentMethodAddress() + address.line1 = "Av. Do Brasil 1374" + address.city = "Sao Paulo" + address.state = "SP" + address.postalCode = "01310100" + address.country = "BR" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "Jane Diaz" + billingDetails.email = "jane@example.com" + billingDetails.address = address + + let params = STPPaymentMethodParams( + boleto: boletoParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method Boleto create") + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + exp.fulfill() + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .boleto, "Incorrect PaymentMethod type") + + XCTAssertEqual( + paymentMethod?.billingDetails?.name, + "Jane Diaz", + "Billing name should match the name provided during creation" + ) + + XCTAssertEqual( + paymentMethod?.billingDetails?.email, + "jane@example.com", + "Billing email should match the name provided during creation" + ) + + XCTAssertNotNil(paymentMethod?.boleto, "The `boleto` property must be populated") + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodBoletoTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodBoletoTests.swift new file mode 100644 index 00000000..ec93c6b9 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodBoletoTests.swift @@ -0,0 +1,53 @@ +// +// STPPaymentMethodBoletoTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 9/10/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodBoletoTests: XCTestCase { + + private(set) var boletoJSON: [AnyHashable: Any]? + + static let boletoPaymentIntentClientSecret = + "pi_3JYFj9JQVROkWvqT0d2HYaTk_secret_c2PniS4q2A7XhZ9mbFwOTpN08" + + func _retrieveBoletoJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let boletoJSON = boletoJSON { + completion(boletoJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingBRPublishableKey) + client.retrievePaymentIntent( + withClientSecret: Self.boletoPaymentIntentClientSecret, + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + boletoJSON = paymentIntent?.paymentMethod?.boleto?.allResponseFields + completion(boletoJSON ?? [:]) + } + } + } + + func testObjectDecoding() { + let retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + + _retrieveBoletoJSON({ json in + let boleto = STPPaymentMethodBoleto.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(boleto, "Failed to decode JSON") + XCTAssertEqual(boleto?.taxID, "00.000.000/0001-91", "It must properly decode `taxID`") + retrieveJSON.fulfill() + }) + + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCardChecksTest.m b/Stripe/StripeiOSTests/STPPaymentMethodCardChecksTest.m new file mode 100644 index 00000000..dd41da14 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCardChecksTest.m @@ -0,0 +1,48 @@ +// +// STPPaymentMethodCardChecksTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/5/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +@interface STPPaymentMethodCardChecksTest : XCTestCase + +@end + +@implementation STPPaymentMethodCardChecksTest + +- (void)testDecodedObjectFromAPIResponse { + NSDictionary *response = @{@"address_line1_check": [NSNull null], + @"address_postal_code_check": [NSNull null], + @"cvc_check": [NSNull null]}; + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *mutableResponse = [response mutableCopy]; + [mutableResponse removeObjectForKey:field]; + XCTAssertNil([STPPaymentMethodCardChecks decodedObjectFromAPIResponse:mutableResponse]); + } + STPPaymentMethodCardChecks *checks = [STPPaymentMethodCardChecks decodedObjectFromAPIResponse:response]; + XCTAssertNotNil(checks); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertEqual(checks.addressLine1Check, STPPaymentMethodCardCheckResultUnknown); + XCTAssertEqual(checks.addressPostalCodeCheck, STPPaymentMethodCardCheckResultUnknown); + XCTAssertEqual(checks.cvcCheck, STPPaymentMethodCardCheckResultUnknown); +#pragma clang diagnostic pop +} + +- (void)testCheckResultFromString { + XCTAssertEqual([STPPaymentMethodCardChecks checkResultFromString:@"pass"], STPPaymentMethodCardCheckResultPass); + XCTAssertEqual([STPPaymentMethodCardChecks checkResultFromString:@"failed"], STPPaymentMethodCardCheckResultFailed); + XCTAssertEqual([STPPaymentMethodCardChecks checkResultFromString:@"unavailable"], STPPaymentMethodCardCheckResultUnavailable); + XCTAssertEqual([STPPaymentMethodCardChecks checkResultFromString:@"unchecked"], STPPaymentMethodCardCheckResultUnchecked); + XCTAssertEqual([STPPaymentMethodCardChecks checkResultFromString:@"unknown_string"], STPPaymentMethodCardCheckResultUnknown); + XCTAssertEqual([STPPaymentMethodCardChecks checkResultFromString:nil], STPPaymentMethodCardCheckResultUnknown); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCardParamsTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodCardParamsTest.swift new file mode 100644 index 00000000..fcdadf81 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCardParamsTest.swift @@ -0,0 +1,68 @@ +// +// STPPaymentMethodCardParamsTest.swift +// StripeiOS Tests +// +// Created by David Estes on 2/10/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodCardParamsTest: XCTestCase { + func testEqualityCheck() { + let params1 = STPPaymentMethodCardParams() + params1.number = "4242424242424242" + params1.cvc = "123" + params1.expYear = 22 + params1.expMonth = 12 + let params2 = STPPaymentMethodCardParams() + params2.number = "4242424242424242" + params2.cvc = "123" + params2.expYear = 22 + params2.expMonth = 12 + XCTAssertEqual(params1, params2) + params1.additionalAPIParameters["test"] = "bla" + XCTAssertNotEqual(params1, params2) + params2.additionalAPIParameters["test"] = "bla" + XCTAssertEqual(params1, params2) + } + + func testCardParamsFromPaymentMethodParams() { + let pmCardParams = STPPaymentMethodCardParams() + pmCardParams.number = "4242424242424242" + pmCardParams.cvc = "123" + pmCardParams.expYear = 22 + pmCardParams.expMonth = 12 + let addressParams = STPPaymentMethodBillingDetails() + addressParams.name = "Tester McTestington" + let address = STPPaymentMethodAddress() + address.line1 = "123 Fake St" + address.line2 = "Apt 123" + address.city = "City" + address.state = "NY" + address.country = "US" + address.postalCode = "12345" + addressParams.address = address + let pmParams = STPPaymentMethodParams( + card: pmCardParams, + billingDetails: addressParams, + metadata: nil + ) + let cardParams = STPCardParams(paymentMethodParams: pmParams) + XCTAssertEqual(cardParams.number, "4242424242424242") + XCTAssertEqual(cardParams.cvc, "123") + XCTAssertEqual(cardParams.expYear, 22) + XCTAssertEqual(cardParams.expMonth, 12) + XCTAssertEqual(cardParams.name, "Tester McTestington") + XCTAssertEqual(cardParams.addressLine1, "123 Fake St") + XCTAssertEqual(cardParams.addressLine2, "Apt 123") + XCTAssertEqual(cardParams.addressCity, "City") + XCTAssertEqual(cardParams.addressState, "NY") + XCTAssertEqual(cardParams.addressCountry, "US") + XCTAssertEqual(cardParams.addressZip, "12345") + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCardTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodCardTest.swift new file mode 100644 index 00000000..b5a12964 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCardTest.swift @@ -0,0 +1,125 @@ +// +// STPPaymentMethodCardTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/6/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +private let kCardPaymentIntentClientSecret = + "pi_1H5J4RFY0qyl6XeWFTpgue7g_secret_1SS59M0x65qWMaX2wEB03iwVE" + +class STPPaymentMethodCardTest: XCTestCase { + private(set) var cardJSON: [AnyHashable: Any]? + + func _retrieveCardJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let cardJSON = cardJSON { + completion(cardJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: kCardPaymentIntentClientSecret, + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + cardJSON = paymentIntent?.paymentMethod?.card?.allResponseFields + completion(cardJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + _retrieveCardJSON({ json in + let card = STPPaymentMethodCard.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(card, "Failed to decode JSON") + retrieveJSON.fulfill() + XCTAssertEqual(card?.brand, .visa) + XCTAssertEqual(card?.country, "US") + XCTAssertNotNil(card?.checks) + XCTAssertEqual(card?.expMonth, 7) + XCTAssertEqual(card?.expYear, 2021) + XCTAssertEqual(card?.funding, "credit") + XCTAssertEqual(card?.last4, "4242") + XCTAssertNotNil(card?.threeDSecureUsage) + XCTAssertEqual(card?.threeDSecureUsage?.supported, true) + XCTAssertNotNil(card?.networks) + XCTAssertEqual(card?.networks?.available, ["visa"]) + XCTAssertNil(card?.networks?.preferred) + }) + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } + + func testDecodedObjectFromAPIResponseRequiredFields() { + let requiredFields: [String]? = [] + + for field in requiredFields ?? [] { + var response = + STPTestUtils.jsonNamed(STPTestJSONPaymentMethodCard)?["card"] as? [AnyHashable: Any] + response?.removeValue(forKey: field) + + XCTAssertNil(STPPaymentMethodCard.decodedObject(fromAPIResponse: response)) + } + let json = STPTestUtils.jsonNamed(STPTestJSONPaymentMethodCard)?["card"] + let decoded = STPPaymentMethodCard.decodedObject( + fromAPIResponse: json as? [AnyHashable: Any] + ) + XCTAssertNotNil(decoded) + } + + func testDecodedObjectFromAPIResponseMapping() { + let response = + STPTestUtils.jsonNamed(STPTestJSONPaymentMethodCard)?["card"] as? [AnyHashable: Any] + let card = STPPaymentMethodCard.decodedObject(fromAPIResponse: response) + XCTAssertEqual(card?.brand, .visa) + XCTAssertEqual(card?.country, "US") + XCTAssertNotNil(card?.checks) + XCTAssertEqual(card?.expMonth, 8) + XCTAssertEqual(card?.expYear, 2020) + XCTAssertEqual(card?.funding, "credit") + XCTAssertEqual(card?.last4, "4242") + XCTAssertEqual(card?.fingerprint, "6gVyxfIhqc8Z0g0X") + XCTAssertNotNil(card?.threeDSecureUsage) + XCTAssertEqual(card?.threeDSecureUsage?.supported, true) + XCTAssertNotNil(card?.wallet) + } + + func testBrandFromString() { + XCTAssertEqual(STPPaymentMethodCard.brand(from: "visa"), .visa) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "VISA"), .visa) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "amex"), .amex) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "AMEX"), .amex) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "american_express"), .amex) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "AMERICAN_EXPRESS"), .amex) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "mastercard"), .mastercard) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "MASTERCARD"), .mastercard) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "discover"), .discover) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "DISCOVER"), .discover) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "jcb"), .JCB) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "JCB"), .JCB) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "diners"), .dinersClub) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "DINERS"), .dinersClub) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "diners_club"), .dinersClub) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "DINERS_CLUB"), .dinersClub) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "unionpay"), .unionPay) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "UNIONPAY"), .unionPay) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "unknown"), .unknown) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "UNKNOWN"), .unknown) + + XCTAssertEqual(STPPaymentMethodCard.brand(from: "garbage"), .unknown) + XCTAssertEqual(STPPaymentMethodCard.brand(from: "GARBAGE"), .unknown) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCardWalletMasterpassTest.m b/Stripe/StripeiOSTests/STPPaymentMethodCardWalletMasterpassTest.m new file mode 100644 index 00000000..3bbf5ea6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCardWalletMasterpassTest.m @@ -0,0 +1,31 @@ +// +// STPPaymentMethodCardWalletMasterpassTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/9/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPPaymentMethodCardWalletMasterpassTest : XCTestCase + +@end + +@implementation STPPaymentMethodCardWalletMasterpassTest + +- (void)testDecodedObjectFromAPIResponseMapping { + // We reuse the visa checkout JSON because it's identical to the masterpass version + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"card"][@"wallet"][@"visa_checkout"]; + STPPaymentMethodCardWalletMasterpass *masterpass = [STPPaymentMethodCardWalletMasterpass decodedObjectFromAPIResponse:response]; + XCTAssertNotNil(masterpass); + XCTAssertEqualObjects(masterpass.name, @"Jenny"); + XCTAssertEqualObjects(masterpass.email, @"jenny@example.com"); + XCTAssertNotNil(masterpass.billingAddress); + XCTAssertNotNil(masterpass.shippingAddress); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCardWalletTest.m b/Stripe/StripeiOSTests/STPPaymentMethodCardWalletTest.m new file mode 100644 index 00000000..6fc0ceeb --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCardWalletTest.m @@ -0,0 +1,46 @@ +// +// STPPaymentMethodCardWalletTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/9/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPPaymentMethodCardWalletTest : XCTestCase + +@end + +@implementation STPPaymentMethodCardWalletTest + +#pragma mark - STPPaymentMethodCardWalletType Tests + +- (void)testTypeFromString { + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"amex_express_checkout"], STPPaymentMethodCardWalletTypeAmexExpressCheckout); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"AMEX_EXPRESS_CHECKOUT"], STPPaymentMethodCardWalletTypeAmexExpressCheckout); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"apple_pay"], STPPaymentMethodCardWalletTypeApplePay); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"APPLE_PAY"], STPPaymentMethodCardWalletTypeApplePay); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"google_pay"], STPPaymentMethodCardWalletTypeGooglePay); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"GOOGLE_PAY"], STPPaymentMethodCardWalletTypeGooglePay); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"masterpass"], STPPaymentMethodCardWalletTypeMasterpass); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"MASTERPASS"], STPPaymentMethodCardWalletTypeMasterpass); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"samsung_pay"], STPPaymentMethodCardWalletTypeSamsungPay); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"SAMSUNG_PAY"], STPPaymentMethodCardWalletTypeSamsungPay); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"visa_checkout"], STPPaymentMethodCardWalletTypeVisaCheckout); + XCTAssertEqual([STPPaymentMethodCardWallet typeFromString:@"VISA_CHECKOUT"], STPPaymentMethodCardWalletTypeVisaCheckout); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"card"][@"wallet"]; + STPPaymentMethodCardWallet *wallet = [STPPaymentMethodCardWallet decodedObjectFromAPIResponse:response]; + XCTAssertNotNil(wallet); + XCTAssertEqual(wallet.type, STPPaymentMethodCardWalletTypeVisaCheckout); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCardWalletVisaCheckoutTest.m b/Stripe/StripeiOSTests/STPPaymentMethodCardWalletVisaCheckoutTest.m new file mode 100644 index 00000000..816c4ecd --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCardWalletVisaCheckoutTest.m @@ -0,0 +1,30 @@ +// +// STPPaymentMethodCardWalletVisaCheckoutTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/9/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPPaymentMethodCardWalletVisaCheckoutTest : XCTestCase + +@end + +@implementation STPPaymentMethodCardWalletVisaCheckoutTest + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONPaymentMethodCard][@"card"][@"wallet"][@"visa_checkout"]; + STPPaymentMethodCardWalletVisaCheckout *visaCheckout = [STPPaymentMethodCardWalletVisaCheckout decodedObjectFromAPIResponse:response]; + XCTAssertNotNil(visaCheckout); + XCTAssertEqualObjects(visaCheckout.name, @"Jenny"); + XCTAssertEqualObjects(visaCheckout.email, @"jenny@example.com"); + XCTAssertNotNil(visaCheckout.billingAddress); + XCTAssertNotNil(visaCheckout.shippingAddress); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCashAppParamsTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodCashAppParamsTests.swift new file mode 100644 index 00000000..c9674abc --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCashAppParamsTests.swift @@ -0,0 +1,44 @@ +// +// STPPaymentMethodCashAppParamsTests.swift +// StripeiOSTests +// +// Created by Nick Porter on 1/4/23. +// + +import Foundation +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodCashAppParamsTests: XCTestCase { + + func testCreateCashAppPaymentMethod() throws { + let cashAppParams = STPPaymentMethodCashAppParams() + + let params = STPPaymentMethodParams( + cashApp: cashAppParams, + billingDetails: nil, + metadata: nil + ) + + let exp = expectation(description: "Payment Method Cash App create") + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + exp.fulfill() + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .cashApp, "Incorrect PaymentMethod type") + XCTAssertNotNil(paymentMethod?.cashApp, "The `cashApp` property must be populated") + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodCashAppTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodCashAppTests.swift new file mode 100644 index 00000000..36502dd6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodCashAppTests.swift @@ -0,0 +1,40 @@ +// +// STPPaymentMethodCashAppTests.swift +// StripeiOSTests +// +// Created by Nick Porter on 1/4/23. +// + +@testable import Stripe +import StripeCoreTestUtils +import XCTest + +class STPPaymentMethodCashAppTests: XCTestCase { + + static let cashAppPaymentIntentClientSecret = "pi_3MMa4NFY0qyl6XeW1FM3HOts_secret_b4HQ5YksK3mfe7zZaxBlWCark" + + func _retrieveCashAppJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: Self.cashAppPaymentIntentClientSecret, + expand: ["payment_method"] + ) { paymentIntent, _ in + XCTAssertNotNil(paymentIntent?.paymentMethod?.allResponseFields["cashapp"]) + let cashAppJson = try? XCTUnwrap(paymentIntent?.paymentMethod?.cashApp?.allResponseFields) + completion(cashAppJson) + } + } + + func testObjectDecoding() { + let retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + + _retrieveCashAppJSON({ json in + let cashApp = STPPaymentMethodCashApp.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(cashApp, "Failed to decode JSON") + retrieveJSON.fulfill() + }) + + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodEPSParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodEPSParamsTests.m new file mode 100644 index 00000000..7f7a6b60 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodEPSParamsTests.m @@ -0,0 +1,55 @@ +// +// STPPaymentMethodEPSParamsTests.m +// StripeiOS Tests +// +// Created by Shengwei Wu on 5/15/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import + +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" +@import StripeCore; + +@interface STPPaymentMethodEPSParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodEPSParamsTests + +- (void)testCreateEPSPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodEPSParams *epsParams = [STPPaymentMethodEPSParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithEPS:epsParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method EPS create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating EPS PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create EPS PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeEPS, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + + // EPS Details + XCTAssertNotNil(paymentMethod.eps, @"Missing eps"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodEPSTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodEPSTests.swift new file mode 100644 index 00000000..b5e4d814 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodEPSTests.swift @@ -0,0 +1,43 @@ +// +// STPPaymentMethodEPSTests.swift +// StripeiOS Tests +// +// Created by Shengwei Wu on 5/15/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodEPSTests: XCTestCase { + private(set) var epsJSON: [AnyHashable: Any]? + + func _retrieveEPSJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let epsJSON = epsJSON { + completion(epsJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: "pi_1Gj0rqFY0qyl6XeWrug30CPz_secret_tKyf8QOKtiIrE3NSEkWCkBbyy", + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + epsJSON = paymentIntent?.paymentMethod?.eps?.allResponseFields + completion(epsJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let jsonExpectation = XCTestExpectation(description: "Fetch EPS JSON") + _retrieveEPSJSON({ json in + let eps = STPPaymentMethodEPS.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(eps, "Failed to decode JSON") + jsonExpectation.fulfill() + }) + wait(for: [jsonExpectation], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodFPXTest.m b/Stripe/StripeiOSTests/STPPaymentMethodFPXTest.m new file mode 100644 index 00000000..54bdc121 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodFPXTest.m @@ -0,0 +1,42 @@ +// +// STPPaymentMethodFPXTest.m +// StripeiOS Tests +// +// Created by David Estes on 8/26/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +@interface STPPaymentMethodFPXTest : XCTestCase + +@end + +@implementation STPPaymentMethodFPXTest + +- (NSDictionary *)exampleJson { + return @{ + @"bank": @"maybank2u", + }; +} + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[self exampleJson] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPPaymentMethodFPX decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPPaymentMethodFPX decodedObjectFromAPIResponse:[self exampleJson]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [self exampleJson]; + STPPaymentMethodFPX *fpx = [STPPaymentMethodFPX decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(fpx.bankIdentifierCode, @"maybank2u"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodFunctionalTest.m b/Stripe/StripeiOSTests/STPPaymentMethodFunctionalTest.m new file mode 100644 index 00000000..05cebc9a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodFunctionalTest.m @@ -0,0 +1,167 @@ +// +// STPPaymentMethodFunctionalTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/6/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + + +@import Stripe; + +@interface STPPaymentMethodFunctionalTest : XCTestCase + +@end + +@implementation STPPaymentMethodFunctionalTest + +- (void)setUp { + [super setUp]; +} + +- (void)testCreateCardPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodCardParams *card = [STPPaymentMethodCardParams new]; + card.number = @"4242424242424242"; + card.expMonth = @(10); + card.expYear = @(2028); + card.cvc = @"100"; + + STPPaymentMethodAddress *billingAddress = [STPPaymentMethodAddress new]; + billingAddress.city = @"San Francisco"; + billingAddress.country = @"US"; + billingAddress.line1 = @"150 Townsend St"; + billingAddress.line2 = @"4th Floor"; + billingAddress.postalCode = @"94103"; + billingAddress.state = @"CA"; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.address = billingAddress; + billingDetails.email = @"email@email.com"; + billingDetails.name = @"Isaac Asimov"; + billingDetails.phone = @"555-555-5555"; + + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithCard:card + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method Card create"]; + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod *paymentMethod, NSError *error) { + XCTAssertNil(error); + XCTAssertNotNil(paymentMethod); + XCTAssertNotNil(paymentMethod.stripeId); + XCTAssertNotNil(paymentMethod.created); + XCTAssertFalse(paymentMethod.liveMode); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeCard); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.email, @"email@email.com"); + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Isaac Asimov"); + XCTAssertEqualObjects(paymentMethod.billingDetails.phone, @"555-555-5555"); + + // Billing Details Address + XCTAssertEqualObjects(paymentMethod.billingDetails.address.line1, @"150 Townsend St"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.line2, @"4th Floor"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.city, @"San Francisco"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.country, @"US"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.state, @"CA"); + XCTAssertEqualObjects(paymentMethod.billingDetails.address.postalCode, @"94103"); + + // Card + XCTAssertEqual(paymentMethod.card.brand, STPCardBrandVisa); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertEqual(paymentMethod.card.checks.cvcCheck, STPPaymentMethodCardCheckResultUnknown); + XCTAssertEqual(paymentMethod.card.checks.addressLine1Check, STPPaymentMethodCardCheckResultUnknown); + XCTAssertEqual(paymentMethod.card.checks.addressPostalCodeCheck, STPPaymentMethodCardCheckResultUnknown); +#pragma clang diagnostic pop + XCTAssertEqualObjects(paymentMethod.card.country, @"US"); + XCTAssertEqual(paymentMethod.card.expMonth, 10); + XCTAssertEqual(paymentMethod.card.expYear, 2028); + XCTAssertEqualObjects(paymentMethod.card.funding, @"credit"); + XCTAssertEqualObjects(paymentMethod.card.last4, @"4242"); + XCTAssertTrue(paymentMethod.card.threeDSecureUsage.supported); + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateBacsPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_test_z6Ct4bpx0NUjHii0rsi4XZBf00jmM8qA28"]; + + STPPaymentMethodBacsDebitParams *bacs = [STPPaymentMethodBacsDebitParams new]; + bacs.sortCode = @"108800"; + bacs.accountNumber = @"00012345"; + + STPPaymentMethodAddress *billingAddress = [STPPaymentMethodAddress new]; + billingAddress.city = @"London"; + billingAddress.country = @"GB"; + billingAddress.line1 = @"Stripe, 7th Floor The Bower Warehouse"; + billingAddress.postalCode = @"EC1V 9NR"; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.address = billingAddress; + billingDetails.email = @"email@email.com"; + billingDetails.name = @"Isaac Asimov"; + billingDetails.phone = @"555-555-5555"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithBacsDebit:bacs billingDetails:billingDetails metadata:nil]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method create"]; + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod *paymentMethod, NSError *error) { + XCTAssertNil(error); + XCTAssertNotNil(paymentMethod); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeBacsDebit); + + // Bacs Debit + XCTAssertEqualObjects(paymentMethod.bacsDebit.fingerprint, @"UkSG0HfCGxxrja1H"); + XCTAssertEqualObjects(paymentMethod.bacsDebit.last4, @"2345"); + XCTAssertEqualObjects(paymentMethod.bacsDebit.sortCode, @"108800"); + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:5 handler:nil]; +} + +- (void)testCreateAlipayPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_test_JBVAMwnBuzCdmsgN34jfxbU700LRiPqVit"]; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithAlipay:[STPPaymentMethodAlipayParams new] billingDetails:nil metadata:nil]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method create"]; + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod *paymentMethod, NSError *error) { + XCTAssertNil(error); + XCTAssertNotNil(paymentMethod); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeAlipay); + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:5 handler:nil]; +} + +- (void)testCreateBLIKPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithBLIK:[STPPaymentMethodBLIKParams new] billingDetails:nil metadata:nil]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method create"]; + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod *paymentMethod, NSError *error) { + XCTAssertNil(error); + XCTAssertNotNil(paymentMethod); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeBLIK); + XCTAssertNotNil(paymentMethod.blik); + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:5 handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodGiropayParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodGiropayParamsTests.m new file mode 100644 index 00000000..85f39519 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodGiropayParamsTests.m @@ -0,0 +1,53 @@ +// +// STPPaymentMethodGiropayParamsTests.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 4/21/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodGiropayParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodGiropayParamsTests + +- (void)testCreateGiropayPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodGiropayParams *giropayParams = [STPPaymentMethodGiropayParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithGiropay:giropayParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method giropay create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating giropay PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create giropay PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeGiropay, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + + // giropay Details + XCTAssertNotNil(paymentMethod.giropay, @"Missing giropay"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodGiropayTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodGiropayTests.swift new file mode 100644 index 00000000..5e421c4f --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodGiropayTests.swift @@ -0,0 +1,43 @@ +// +// STPPaymentMethodGiropayTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 4/21/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodGiropayTests: XCTestCase { + private(set) var giropayJSON: [AnyHashable: Any]? + + func _retrieveGiropayDebitJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let giropayJSON = giropayJSON { + completion(giropayJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: "pi_1GfsdtFY0qyl6XeWLnepIXCI_secret_bLJRSeSY7fBjDXnwh9BUKilMW", + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + giropayJSON = paymentIntent?.paymentMethod?.giropay?.allResponseFields + completion(giropayJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let jsonExpectation = XCTestExpectation(description: "Fetch Giropay JSON") + _retrieveGiropayDebitJSON({ json in + let giropay = STPPaymentMethodGiropay.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(giropay, "Failed to decode JSON") + jsonExpectation.fulfill() + }) + wait(for: [jsonExpectation], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodGrabPayParamsTest.m b/Stripe/StripeiOSTests/STPPaymentMethodGrabPayParamsTest.m new file mode 100644 index 00000000..2fedb8cb --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodGrabPayParamsTest.m @@ -0,0 +1,55 @@ +// +// STPPaymentMethodGrabPayParamsTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 7/21/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" +@import StripeCore; + +@interface STPPaymentMethodGrabPayParamsTest : XCTestCase + +@end + +@implementation STPPaymentMethodGrabPayParamsTest + + +- (void)testCreateGrabPayPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingSGPublishableKey]; + STPPaymentMethodGrabPayParams *grabPayParams = [STPPaymentMethodGrabPayParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithGrabPay:grabPayParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method GrabPay create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating GrabPay PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create GrabPay PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeGrabPay, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + + // GrabPay Details + XCTAssertNotNil(paymentMethod.grabPay, @"Missing grabPay"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodKlarnaParamsTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodKlarnaParamsTests.swift new file mode 100644 index 00000000..93a87e7c --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodKlarnaParamsTests.swift @@ -0,0 +1,71 @@ +// +// STPPaymentMethodKlarnaParamsTests.swift +// StripeiOS Tests +// +// Created by Nick Porter on 10/20/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodKlarnaParamsTests: XCTestCase { + + func testCreateKlarnaPaymentMethod() throws { + let klarnaParams = STPPaymentMethodKlarnaParams() + + let address = STPPaymentMethodAddress() + address.line1 = "55 John St" + address.line2 = "#3B" + address.city = "New York" + address.state = "NY" + address.postalCode = "10002" + address.country = "US" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "John Smith" + billingDetails.email = "foo@example.com" + billingDetails.address = address + + let params = STPPaymentMethodParams( + klarna: klarnaParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method Klarna create") + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + exp.fulfill() + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .klarna, "Incorrect PaymentMethod type") + + XCTAssertEqual( + paymentMethod?.billingDetails?.name, + "John Smith", + "Billing name should match the name provided during creation" + ) + + XCTAssertEqual( + paymentMethod?.billingDetails?.email, + "foo@example.com", + "Billing email should match the name provided during creation" + ) + + XCTAssertNotNil(paymentMethod?.klarna, "The `klarna` property must be populated") + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodKlarnaTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodKlarnaTests.swift new file mode 100644 index 00000000..ba626aa7 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodKlarnaTests.swift @@ -0,0 +1,46 @@ +// +// STPPaymentMethodKlarnaTests.swift +// StripeiOS Tests +// +// Created by Nick Porter on 10/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodKlarnaTests: XCTestCase { + + static let klarnaPaymentIntentClientSecret = + "pi_3Jn3kUFY0qyl6XeW0mCp95UD_secret_28aNjjd1zsySFWvGoSzgcR5Qw" + + func _retrieveKlarnaJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: Self.klarnaPaymentIntentClientSecret, + expand: ["payment_method"] + ) { paymentIntent, _ in + let klarnaJson = paymentIntent?.paymentMethod?.klarna?.allResponseFields + completion(klarnaJson ?? [:]) + } + } + + func testObjectDecoding() { + let retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + + _retrieveKlarnaJSON({ json in + let klarna = STPPaymentMethodKlarna.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(klarna, "Failed to decode JSON") + retrieveJSON.fulfill() + }) + + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodNetBankingParamsTest.m b/Stripe/StripeiOSTests/STPPaymentMethodNetBankingParamsTest.m new file mode 100644 index 00000000..ed1f02e4 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodNetBankingParamsTest.m @@ -0,0 +1,49 @@ +// +// STPPaymentMethodNetBankingParamsTest.m +// StripeiOS +// +// Created by Anirudh Bhargava on 11/19/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" +@import StripeCore; + +@interface STPPaymentMethodNetBankingParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodNetBankingParamsTests + +- (void)testCreateNetBankingPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingINPublishableKey]; + STPPaymentMethodNetBankingParams *netbankingParams = [STPPaymentMethodNetBankingParams new]; + netbankingParams.bank = @"icici"; + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithNetBanking:netbankingParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method NetBanking create"]; + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + XCTAssertNil(error, @"Unexpected error creating NetBanking PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create NetBanking PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeNetBanking, @"Incorrect PaymentMethod type"); + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + // UPI Details + XCTAssertNotNil(paymentMethod.netBanking, @"Missing NetBanking"); + XCTAssertEqualObjects(paymentMethod.netBanking.bank, @"icici", @"Incorrect bank value"); + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodNetBankingTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodNetBankingTests.swift new file mode 100644 index 00000000..d42ef55a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodNetBankingTests.swift @@ -0,0 +1,44 @@ +// +// STPPaymentMethodNetBankingTests.swift +// StripeiOS Tests +// +// Created by Anirudh Bhargava on 11/19/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodNetBankingTests: XCTestCase { + private(set) var netbankingJSON: [AnyHashable: Any]? + + func _retrieveNetBankingJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let netbankingJSON = netbankingJSON { + completion(netbankingJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingINPublishableKey) + client.retrievePaymentIntent( + withClientSecret: "pi_1HoPqsBte6TMTRd4jX0PwrFa_secret_ThiIwyssre9qjJ6gtmghC21fk", + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + netbankingJSON = paymentIntent?.paymentMethod?.netBanking?.allResponseFields + completion(netbankingJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let jsonExpectation = XCTestExpectation(description: "Fetch NetBanking JSON") + _retrieveNetBankingJSON({ json in + let netbanking = STPPaymentMethodNetBanking.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(netbanking, "Failed to decode JSON") + jsonExpectation.fulfill() + }) + wait(for: [jsonExpectation], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodOXXOParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodOXXOParamsTests.m new file mode 100644 index 00000000..8d532a9e --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodOXXOParamsTests.m @@ -0,0 +1,57 @@ +// +// STPPaymentMethodOXXOParamsTests.m +// StripeiOS Tests +// +// Created by Polo Li on 6/16/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +@import StripeCore; + +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodOXXOParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodOXXOParamsTests + +- (void)testCreateOXXOPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodOXXOParams *oxxoParams = [STPPaymentMethodOXXOParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jane Doe"; + billingDetails.email = @"test@test.com"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithOXXO:oxxoParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method OXXO create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating OXXO PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create OXXO PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeOXXO, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jane Doe", @"Incorrect name"); + XCTAssertEqualObjects(paymentMethod.billingDetails.email, @"test@test.com", @"Incorrect email"); + + // OXXO Details + XCTAssertNotNil(paymentMethod.oxxo, @"Missing OXXO"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodOXXOTests.m b/Stripe/StripeiOSTests/STPPaymentMethodOXXOTests.m new file mode 100644 index 00000000..fe9fcfb0 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodOXXOTests.m @@ -0,0 +1,46 @@ +// +// STPPaymentMethodOXXOTests.m +// StripeiOS Tests +// +// Created by Polo Li on 6/16/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +@import StripeCore; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodOXXOTests : XCTestCase + +@property (nonatomic, readonly) NSDictionary *oxxoJSON; + +@end + +@implementation STPPaymentMethodOXXOTests + +- (void)_retrieveOXXOJSON:(void (^)(NSDictionary *))completion { + if (self.oxxoJSON) { + completion(self.oxxoJSON); + } else { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingMEXPublishableKey]; + [client retrievePaymentIntentWithClientSecret:@"pi_1GvAdyHNG4o8pO5l0dr078gf_secret_h0tJE5mSX9BPEkmpKSh93jBXi" + expand:@[@"payment_method"] + completion:^(STPPaymentIntent * _Nullable paymentIntent, __unused NSError * _Nullable error) { + self->_oxxoJSON = paymentIntent.paymentMethod.oxxo.allResponseFields; + completion(self.oxxoJSON); + }]; + } +} + +- (void)testCorrectParsing { + XCTestExpectation *expectation = [self expectationWithDescription:@"Retrieve payment intent"]; + [self _retrieveOXXOJSON:^(NSDictionary *json) { + STPPaymentMethodOXXO *oxxo = [STPPaymentMethodOXXO decodedObjectFromAPIResponse:json]; + XCTAssertNotNil(oxxo, @"Failed to decode JSON"); + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodOptionsTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodOptionsTest.swift new file mode 100644 index 00000000..91478a8f --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodOptionsTest.swift @@ -0,0 +1,131 @@ +// +// STPPaymentMethodOptionsTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 4/8/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodOptionsTest: XCTestCase { + + func testUSBankAccountOptions_PaymentIntent() { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let verificationMethods = [ + "skip", + "automatic", + "instant", + "microdeposits", + "instant_or_skip", + ] + for verificationMethod in verificationMethods { + var clientSecret: String? + let createPIExpectation = expectation(description: "Create PaymentIntent") + STPTestingAPIClient.shared().createPaymentIntent( + withParams: [ + "payment_method_types": ["us_bank_account"], + "payment_method_options": [ + "us_bank_account": ["verification_method": verificationMethod] + ], + "currency": "usd", + "amount": 1000, + ], + account: nil + ) { intentClientSecret, error in + XCTAssertNil(error) + XCTAssertNotNil(intentClientSecret) + clientSecret = intentClientSecret + createPIExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + guard let clientSecret = clientSecret else { + XCTFail("Failed to create PaymentIntent") + continue + } + + let retrievePIExpectation = expectation(description: "Retrieve PaymentIntent") + client.retrievePaymentIntent(withClientSecret: clientSecret) { paymentIntent, error in + XCTAssertNil(error) + XCTAssertNotNil(paymentIntent) + + XCTAssertNotNil( + paymentIntent?.paymentMethodOptions?.usBankAccount?.verificationMethod + ) + XCTAssertEqual( + paymentIntent?.paymentMethodOptions?.usBankAccount?.verificationMethod, + STPPaymentMethodOptions.USBankAccount.VerificationMethod( + rawValue: verificationMethod + ) + ) + retrievePIExpectation.fulfill() + + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + + } + + func testUSBankAccountOptions_SetupIntent() { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let verificationMethods = [ + "skip", + "automatic", + "instant", + "microdeposits", + "instant_or_skip", + ] + for verificationMethod in verificationMethods { + var clientSecret: String? + let createPIExpectation = expectation(description: "Create SetupIntent") + STPTestingAPIClient.shared().createSetupIntent( + withParams: [ + "payment_method_types": ["us_bank_account"], + "payment_method_options": [ + "us_bank_account": ["verification_method": verificationMethod] + ], + ], + account: nil + ) { intentClientSecret, error in + XCTAssertNil(error) + XCTAssertNotNil(intentClientSecret) + clientSecret = intentClientSecret + createPIExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + guard let clientSecret = clientSecret else { + XCTFail("Failed to create SetupIntent") + continue + } + + let retrievePIExpectation = expectation(description: "Retrieve PaymentIntent") + client.retrieveSetupIntent(withClientSecret: clientSecret) { setupIntent, error in + XCTAssertNil(error) + XCTAssertNotNil(setupIntent) + + XCTAssertNotNil( + setupIntent?.paymentMethodOptions?.usBankAccount?.verificationMethod + ) + XCTAssertEqual( + setupIntent?.paymentMethodOptions?.usBankAccount?.verificationMethod, + STPPaymentMethodOptions.USBankAccount.VerificationMethod( + rawValue: verificationMethod + ) + ) + retrievePIExpectation.fulfill() + + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodParamsTest.m b/Stripe/StripeiOSTests/STPPaymentMethodParamsTest.m new file mode 100644 index 00000000..bbe71048 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodParamsTest.m @@ -0,0 +1,42 @@ +// +// STPPaymentMethodParamsTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/7/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPPaymentMethodParamsTest : XCTestCase + +@end + +@implementation STPPaymentMethodParamsTest + +#pragma mark STPFormEncodable Tests + +- (void)testRootObjectName { + XCTAssertNil([STPPaymentMethodParams rootObjectName]); +} + +- (void)testPropertyNamesToFormFieldNamesMapping { + STPPaymentMethodParams *params = [STPPaymentMethodParams new]; + + NSDictionary *mapping = [STPPaymentMethodParams propertyNamesToFormFieldNamesMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([params respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodPayPalParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodPayPalParamsTests.m new file mode 100644 index 00000000..1426a930 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodPayPalParamsTests.m @@ -0,0 +1,53 @@ +// +// STPPaymentMethodPayPalParamsTests.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/7/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodPayPalParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodPayPalParamsTests + +- (void)testCreatePayPalPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodPayPalParams *payPalParams = [STPPaymentMethodPayPalParams new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jane Doe"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithPayPal:payPalParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method PayPal create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating PayPal PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create PayPal PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypePayPal, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jane Doe", @"Incorrect name"); + + // PayPal Details + XCTAssertNotNil(paymentMethod.payPal, @"Missing PayPal"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodPayPalTests.m b/Stripe/StripeiOSTests/STPPaymentMethodPayPalTests.m new file mode 100644 index 00000000..3d232130 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodPayPalTests.m @@ -0,0 +1,45 @@ +// +// STPPaymentMethodPayPalTests.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/7/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodPayPalTests : XCTestCase + +@property (nonatomic) NSDictionary *payPalJSON; + +@end + +@implementation STPPaymentMethodPayPalTests + +- (void)_retrievePayPalJSON:(void (^)(NSDictionary *))completion { + if (self.payPalJSON) { + completion(self.payPalJSON); + } else { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + [client retrievePaymentIntentWithClientSecret:@"pi_1HcI17FY0qyl6XeWcFAAbZCw_secret_oAZ9OCoeyIg8EPeBEdF96ZJOT" + expand:@[@"payment_method"] + completion:^(STPPaymentIntent * _Nullable paymentIntent, __unused NSError * _Nullable error) { + self->_payPalJSON = paymentIntent.lastPaymentError.paymentMethod.payPal.allResponseFields; + completion(self.payPalJSON); + }]; + } +} + +- (void)testCorrectParsing { + XCTestExpectation *expectation = [self expectationWithDescription:@"Retrieve payment intent"]; + [self _retrievePayPalJSON:^(NSDictionary *json) { + STPPaymentMethodPayPal *payPal = [STPPaymentMethodPayPal decodedObjectFromAPIResponse:json]; + XCTAssertNotNil(payPal, @"Failed to decode JSON"); + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24ParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24ParamsTests.m new file mode 100644 index 00000000..d6897b28 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24ParamsTests.m @@ -0,0 +1,54 @@ +// +// STPPaymentMethodPrzelewy24ParamsTests.m +// StripeiOS Tests +// +// Created by Vineet Shah on 4/23/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCore; +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" + +@interface STPPaymentMethodPrzelewy24ParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodPrzelewy24ParamsTests + +- (void)testCreatePrzelewy24PaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodPrzelewy24Params *przelewy24Params = [STPPaymentMethodPrzelewy24Params new]; + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.email = @"email@email.com"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithPrzelewy24:przelewy24Params + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method Przelewy24 create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating Przelewy24 PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create Przelewy24 PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypePrzelewy24, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.email, @"email@email.com", @"Incorrect email"); + + // Przelewy24 Details + XCTAssertNotNil(paymentMethod.przelewy24, @"Missing Przelewy24"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24Tests.swift b/Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24Tests.swift new file mode 100644 index 00000000..933474ba --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodPrzelewy24Tests.swift @@ -0,0 +1,43 @@ +// +// STPPaymentMethodPrzelewy24Tests.swift +// StripeiOS Tests +// +// Created by Vineet Shah on 4/23/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodPrzelewy24Tests: XCTestCase { + private(set) var przelewy24JSON: [AnyHashable: Any]? + + func _retrievePrzelewy24JSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let przelewy24JSON = przelewy24JSON { + completion(przelewy24JSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: "pi_1GciHFFY0qyl6XeWp9RdhmFF_secret_rFeERcidL1O5o1lwQUcIjLEZz", + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + przelewy24JSON = paymentIntent?.paymentMethod?.przelewy24?.allResponseFields + completion(przelewy24JSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let expectation = self.expectation(description: "Retrieve payment intent") + _retrievePrzelewy24JSON({ json in + let przelewy24 = STPPaymentMethodPrzelewy24.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(przelewy24, "Failed to decode JSON") + expectation.fulfill() + }) + waitForExpectations(timeout: STPTestingNetworkRequestTimeout, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodSEPADebitTest.m b/Stripe/StripeiOSTests/STPPaymentMethodSEPADebitTest.m new file mode 100644 index 00000000..71cb392c --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodSEPADebitTest.m @@ -0,0 +1,50 @@ +// +// STPPaymentMethodSEPADebitTest.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/7/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPTestUtils.h" + +@interface STPPaymentMethodSEPADebitTest : XCTestCase + +@end + +@implementation STPPaymentMethodSEPADebitTest + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"SEPADebitSource"][@"sepa_debit"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPPaymentMethodSEPADebit decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPPaymentMethodSEPADebit decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"SEPADebitSource"][@"sepa_debit"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:@"SEPADebitSource"][@"sepa_debit"]; + STPPaymentMethodSEPADebit *sepaDebit = [STPPaymentMethodSEPADebit decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(sepaDebit.bankCode, @"37040044"); + XCTAssertEqualObjects(sepaDebit.branchCode, @"a_branch"); + XCTAssertEqualObjects(sepaDebit.country, @"DE"); + XCTAssertEqualObjects(sepaDebit.fingerprint, @"NxdSyRegc9PsMkWy"); + XCTAssertEqualObjects(sepaDebit.last4, @"3001"); + XCTAssertEqualObjects(sepaDebit.mandate, @"NXDSYREGC9PSMKWY"); + + XCTAssertNotEqual(sepaDebit.allResponseFields, response); + XCTAssertEqualObjects(sepaDebit.allResponseFields, response); +} + + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodSofortParamsTests.m b/Stripe/StripeiOSTests/STPPaymentMethodSofortParamsTests.m new file mode 100644 index 00000000..b46214ca --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodSofortParamsTests.m @@ -0,0 +1,55 @@ +// +// STPPaymentMethodSofortParamsTests.m +// StripeiOS Tests +// +// Created by David Estes on 8/7/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" +@import StripeCore; + +@interface STPPaymentMethodSofortParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodSofortParamsTests + +- (void)testCreateSofortPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + STPPaymentMethodSofortParams *sofortParams = [STPPaymentMethodSofortParams new]; + sofortParams.country = @"DE"; + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithSofort:sofortParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method Sofort create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating Sofort PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create Sofort PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeSofort, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + + // Sofort Details + XCTAssertNotNil(paymentMethod.sofort, @"Missing Sofort"); + XCTAssertEqualObjects(paymentMethod.sofort.country, @"DE", @"Incorrect country"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodSofortTests.swift b/Stripe/StripeiOSTests/STPPaymentMethodSofortTests.swift new file mode 100644 index 00000000..cce7b33d --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodSofortTests.swift @@ -0,0 +1,43 @@ +// +// STPPaymentMethodSofortTests.swift +// StripeiOS Tests +// +// Created by David Estes on 8/7/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodSofortTests: XCTestCase { + private(set) var sofortJSON: [AnyHashable: Any]? + + func _retrieveSofortJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let sofortJSON = sofortJSON { + completion(sofortJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: "pi_1HDdfSFY0qyl6XeWjk7ogYVV_secret_5ikjoct7F271A4Bp6t7HkHwUo", + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + sofortJSON = paymentIntent?.paymentMethod?.sofort?.allResponseFields + completion(sofortJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let jsonExpectation = XCTestExpectation(description: "Fetch Sofort JSON") + _retrieveSofortJSON({ json in + let sofort = STPPaymentMethodSofort.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(sofort, "Failed to decode JSON") + jsonExpectation.fulfill() + }) + wait(for: [jsonExpectation], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodTest.swift new file mode 100644 index 00000000..66ac5996 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodTest.swift @@ -0,0 +1,161 @@ +// +// STPPaymentMethodTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/6/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodTest: XCTestCase { + // MARK: - STPPaymentMethodType Tests + func testTypeFromString() { + XCTAssertEqual( + STPPaymentMethod.type(from: "au_becs_debit"), + STPPaymentMethodType.AUBECSDebit + ) + XCTAssertEqual( + STPPaymentMethod.type(from: "AU_BECS_DEBIT"), + STPPaymentMethodType.AUBECSDebit + ) + XCTAssertEqual(STPPaymentMethod.type(from: "BACS_DEBIT"), STPPaymentMethodType.bacsDebit) + XCTAssertEqual(STPPaymentMethod.type(from: "bacs_debit"), STPPaymentMethodType.bacsDebit) + XCTAssertEqual(STPPaymentMethod.type(from: "BACS_DEBIT"), STPPaymentMethodType.bacsDebit) + XCTAssertEqual(STPPaymentMethod.type(from: "card"), STPPaymentMethodType.card) + XCTAssertEqual(STPPaymentMethod.type(from: "CARD"), STPPaymentMethodType.card) + XCTAssertEqual(STPPaymentMethod.type(from: "ideal"), STPPaymentMethodType.iDEAL) + XCTAssertEqual(STPPaymentMethod.type(from: "IDEAL"), STPPaymentMethodType.iDEAL) + XCTAssertEqual(STPPaymentMethod.type(from: "fpx"), STPPaymentMethodType.FPX) + XCTAssertEqual(STPPaymentMethod.type(from: "FPX"), STPPaymentMethodType.FPX) + XCTAssertEqual(STPPaymentMethod.type(from: "sepa_debit"), STPPaymentMethodType.SEPADebit) + XCTAssertEqual(STPPaymentMethod.type(from: "SEPA_DEBIT"), STPPaymentMethodType.SEPADebit) + XCTAssertEqual( + STPPaymentMethod.type(from: "card_present"), + STPPaymentMethodType.cardPresent + ) + XCTAssertEqual( + STPPaymentMethod.type(from: "CARD_PRESENT"), + STPPaymentMethodType.cardPresent + ) + XCTAssertEqual(STPPaymentMethod.type(from: "unknown_string"), STPPaymentMethodType.unknown) + } + + func testTypesFromStrings() { + let rawTypes = [ + "card", + "ideal", + "card_present", + "fpx", + "sepa_debit", + "bacs_debit", + "au_becs_debit", + ] + let expectedTypes: [STPPaymentMethodType] = [ + .card, + .iDEAL, + .cardPresent, + .FPX, + .SEPADebit, + .bacsDebit, + .AUBECSDebit, + ] + XCTAssertEqual(STPPaymentMethod.paymentMethodTypes(from: rawTypes), expectedTypes) + } + + func testStringFromType() { + let values: [STPPaymentMethodType] = [ + .card, + .iDEAL, + .cardPresent, + .FPX, + .SEPADebit, + .bacsDebit, + .AUBECSDebit, + .OXXO, + .alipay, + .payPal, + .giropay, + .unknown, + ] + for type in values { + let string = STPPaymentMethod.string(from: type) + + switch type { + case .card: + XCTAssertEqual(string, "card") + case .iDEAL: + XCTAssertEqual(string, "ideal") + case .cardPresent: + XCTAssertEqual(string, "card_present") + case .FPX: + XCTAssertEqual(string, "fpx") + case .SEPADebit: + XCTAssertEqual(string, "sepa_debit") + case .bacsDebit: + XCTAssertEqual(string, "bacs_debit") + case .AUBECSDebit: + XCTAssertEqual(string, "au_becs_debit") + case .giropay: + XCTAssertEqual(string, "giropay") + case .przelewy24: + XCTAssertEqual(string, "p24") + case .bancontact: + XCTAssertEqual(string, "bancontact") + case .EPS: + XCTAssertEqual(string, "eps") + case .OXXO: + XCTAssertEqual(string, "oxxo") + case .sofort: + XCTAssertEqual(string, "sofort") + case .alipay: + XCTAssertEqual(string, "alipay") + case .payPal: + XCTAssertEqual(string, "paypal") + case .unknown: + XCTAssertNil(string) + case .grabPay: + XCTAssertEqual(string, "grabpay") + default: + break + } + } + } + + // MARK: - STPAPIResponseDecodable Tests + func testDecodedObjectFromAPIResponseRequiredFields() { + let fullJson = STPTestUtils.jsonNamed(STPTestJSONPaymentMethodCard) + + XCTAssertNotNil( + STPPaymentMethod.decodedObject(fromAPIResponse: fullJson), + "can decode with full json" + ) + + let requiredFields = ["id"] + + for field in requiredFields { + var partialJson = fullJson + + XCTAssertNotNil(partialJson?[field]) + partialJson?.removeValue(forKey: field) + + XCTAssertNil(STPPaymentIntent.decodedObject(fromAPIResponse: partialJson)) + } + } + + func testDecodedObjectFromAPIResponseMapping() { + let response = STPTestUtils.jsonNamed(STPTestJSONPaymentMethodCard) + let paymentMethod = STPPaymentMethod.decodedObject(fromAPIResponse: response) + XCTAssertEqual(paymentMethod?.stripeId, "pm_123456789") + XCTAssertEqual(paymentMethod?.created, Date(timeIntervalSince1970: 123_456_789)) + XCTAssertEqual(paymentMethod?.liveMode, false) + XCTAssertEqual(paymentMethod?.type, .card) + XCTAssertNotNil(paymentMethod?.billingDetails) + XCTAssertNotNil(paymentMethod?.card) + XCTAssertNil(paymentMethod?.customerId) + XCTAssertEqual(paymentMethod!.allResponseFields as NSDictionary, response! as NSDictionary) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodThreeDSecureUsageTest.m b/Stripe/StripeiOSTests/STPPaymentMethodThreeDSecureUsageTest.m new file mode 100644 index 00000000..953b2872 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodThreeDSecureUsageTest.m @@ -0,0 +1,31 @@ +// +// STPPaymentMethodThreeDSecureUsageTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/5/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +@interface STPPaymentMethodThreeDSecureUsageTest : XCTestCase + +@end + +@implementation STPPaymentMethodThreeDSecureUsageTest + +- (void)testDecodedObjectFromAPIResponse { + NSDictionary *response = @{@"supported": @YES}; + NSArray *requiredFields = @[@"supported"]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *mutableResponse = [response mutableCopy]; + [mutableResponse removeObjectForKey:field]; + + XCTAssertNil([STPPaymentMethodThreeDSecureUsage decodedObjectFromAPIResponse:mutableResponse]); + } + XCTAssertNotNil([STPPaymentMethodThreeDSecureUsage decodedObjectFromAPIResponse:response]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodUPIParamsTest.m b/Stripe/StripeiOSTests/STPPaymentMethodUPIParamsTest.m new file mode 100644 index 00000000..7451217c --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodUPIParamsTest.m @@ -0,0 +1,55 @@ +// +// STPPaymentMethodUPIParamsTest.m +// StripeiOS Tests +// +// Created by Anirudh Bhargava on 11/6/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +#import "STPTestingAPIClient.h" +@import StripeCore; + +@interface STPPaymentMethodUPIParamsTests : XCTestCase + +@end + +@implementation STPPaymentMethodUPIParamsTests + +- (void)testCreateUPIPaymentMethod { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingINPublishableKey]; + STPPaymentMethodUPIParams *upiParams = [STPPaymentMethodUPIParams new]; + upiParams.vpa = @"somevpa@hdfcbank"; + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithUPI:upiParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Payment Method UPI create"]; + + [client createPaymentMethodWithParams:params + completion:^(STPPaymentMethod * _Nullable paymentMethod, NSError * _Nullable error) { + [expectation fulfill]; + + XCTAssertNil(error, @"Unexpected error creating UPI PaymentMethod: %@", error); + XCTAssertNotNil(paymentMethod, @"Failed to create UPI PaymentMethod"); + XCTAssertNotNil(paymentMethod.stripeId, @"Missing stripeId"); + XCTAssertNotNil(paymentMethod.created, @"Missing created"); + XCTAssertFalse(paymentMethod.liveMode, @"Incorrect livemode"); + XCTAssertEqual(paymentMethod.type, STPPaymentMethodTypeUPI, @"Incorrect PaymentMethod type"); + + // Billing Details + XCTAssertEqualObjects(paymentMethod.billingDetails.name, @"Jenny Rosen", @"Incorrect name"); + + // UPI Details + XCTAssertNotNil(paymentMethod.upi, @"Missing UPI"); + XCTAssertEqualObjects(paymentMethod.upi.vpa, @"somevpa@hdfcbank", @"Incorrect vpa"); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentMethodUPITests.swift b/Stripe/StripeiOSTests/STPPaymentMethodUPITests.swift new file mode 100644 index 00000000..9340d4cd --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodUPITests.swift @@ -0,0 +1,44 @@ +// +// STPPaymentMethodUPITests.swift +// StripeiOS Tests +// +// Created by Anirudh Bhargava on 11/10/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodUPITests: XCTestCase { + private(set) var upiJSON: [AnyHashable: Any]? + + func _retrieveUPIJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + if let upiJSON = upiJSON { + completion(upiJSON) + } else { + let client = STPAPIClient(publishableKey: STPTestingINPublishableKey) + client.retrievePaymentIntent( + withClientSecret: "pi_1HlYxxBte6TMTRd48W66zjTJ_secret_TgB7p7e7aTRbr22UT6N6KNrSm", + expand: ["payment_method"] + ) { [self] paymentIntent, _ in + upiJSON = paymentIntent?.paymentMethod?.upi?.allResponseFields + completion(upiJSON ?? [:]) + } + } + } + + func testCorrectParsing() { + let jsonExpectation = XCTestExpectation(description: "Fetch UPI JSON") + _retrieveUPIJSON({ json in + let upi = STPPaymentMethodUPI.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(upi, "Failed to decode JSON") + jsonExpectation.fulfill() + }) + wait(for: [jsonExpectation], timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsStubbedTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsStubbedTest.swift new file mode 100644 index 00000000..e58023d6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsStubbedTest.swift @@ -0,0 +1,302 @@ +// +// STPPaymentMethodUSBankAccountParamsStubbedTest.swift +// StripeiOS Tests +// +// Created by John Woo on 3/24/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import OHHTTPStubs +import OHHTTPStubsSwift +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeApplePay +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet + +class STPPaymentMethodUSBankAccountParamsStubbedTest: APIStubbedTestCase { + func testus_bank_account_withoutNetworks() { + let stubbedApiClient = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/payment_methods") ?? false + } response: { _ in + let jsonText = """ + { + "id": "pm_1KgvOfFY0qyl6XeW7RfvDvxE", + "object": "payment_method", + "billing_details": { + "address": { + "city": null, + "country": null, + "line1": null, + "line2": null, + "postal_code": null, + "state": null + }, + "email": "tester@example.com", + "name": "iOS CI Tester", + "phone": null + }, + "created": 1648146513, + "customer": null, + "livemode": false, + "type": "us_bank_account", + "us_bank_account": { + "account_holder_type": "individual", + "account_type": "checking", + "bank_name": "STRIPE TEST BANK", + "fingerprint": "ickfX9sbxIyAlbuh", + "last4": "6789", + "linked_account": null, + "routing_number": "110000000" + } + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountType = .checking + usBankAccountParams.accountHolderType = .individual + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let params = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method US Bank Account create") + stubbedApiClient.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .USBankAccount, "Incorrect PaymentMethod type") + XCTAssertNotNil( + paymentMethod?.usBankAccount, + "The `usBankAccount` property must be populated" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountHolderType, + .individual, + "`accountHolderType` should be individual" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountType, + .checking, + "`accountType` should be checking" + ) + XCTAssertEqual(paymentMethod?.usBankAccount?.last4, "6789", "`last4` should be 6789") + XCTAssertNil(paymentMethod?.usBankAccount?.networks) + exp.fulfill() + + } + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + func testus_bank_account_networksInPayloadWithPreferred() { + + let stubbedApiClient = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/payment_methods") ?? false + } response: { _ in + let jsonText = """ + { + "id": "pm_1KgvOfFY0qyl6XeW7RfvDvxE", + "object": "payment_method", + "billing_details": { + "address": { + "city": null, + "country": null, + "line1": null, + "line2": null, + "postal_code": null, + "state": null + }, + "email": "tester@example.com", + "name": "iOS CI Tester", + "phone": null + }, + "created": 1648146513, + "customer": null, + "livemode": false, + "type": "us_bank_account", + "us_bank_account": { + "account_holder_type": "individual", + "account_type": "checking", + "bank_name": "STRIPE TEST BANK", + "fingerprint": "ickfX9sbxIyAlbuh", + "last4": "6789", + "linked_account": null, + "networks": { + "preferred": "ach", + "supported": [ + "ach" + ] + }, + "routing_number": "110000000" + } + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountType = .checking + usBankAccountParams.accountHolderType = .individual + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let params = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method US Bank Account create") + + stubbedApiClient.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .USBankAccount, "Incorrect PaymentMethod type") + XCTAssertNotNil( + paymentMethod?.usBankAccount, + "The `usBankAccount` property must be populated" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountHolderType, + .individual, + "`accountHolderType` should be individual" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountType, + .checking, + "`accountType` should be checking" + ) + XCTAssertEqual(paymentMethod?.usBankAccount?.last4, "6789", "`last4` should be 6789") + XCTAssertEqual(paymentMethod?.usBankAccount?.networks?.preferred, "ach") + XCTAssertEqual(paymentMethod?.usBankAccount?.networks?.supported.count, 1) + XCTAssertEqual(paymentMethod?.usBankAccount?.networks?.supported.first, "ach") + exp.fulfill() + + } + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + func testus_bank_account_networksInPayloadWithoutPreferred() { + + let stubbedApiClient = stubbedAPIClient() + stub { urlRequest in + return urlRequest.url?.absoluteString.contains("/payment_methods") ?? false + } response: { _ in + let jsonText = """ + { + "id": "pm_1KgvOfFY0qyl6XeW7RfvDvxE", + "object": "payment_method", + "billing_details": { + "address": { + "city": null, + "country": null, + "line1": null, + "line2": null, + "postal_code": null, + "state": null + }, + "email": "tester@example.com", + "name": "iOS CI Tester", + "phone": null + }, + "created": 1648146513, + "customer": null, + "livemode": false, + "type": "us_bank_account", + "us_bank_account": { + "account_holder_type": "individual", + "account_type": "checking", + "bank_name": "STRIPE TEST BANK", + "fingerprint": "ickfX9sbxIyAlbuh", + "last4": "6789", + "linked_account": null, + "networks": { + "supported": [ + "ach" + ] + }, + "routing_number": "110000000" + } + } + """ + return HTTPStubsResponse( + data: jsonText.data(using: .utf8)!, + statusCode: 200, + headers: nil + ) + } + + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountType = .checking + usBankAccountParams.accountHolderType = .individual + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let params = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method US Bank Account create") + + stubbedApiClient.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .USBankAccount, "Incorrect PaymentMethod type") + XCTAssertNotNil( + paymentMethod?.usBankAccount, + "The `usBankAccount` property must be populated" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountHolderType, + .individual, + "`accountHolderType` should be individual" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountType, + .checking, + "`accountType` should be checking" + ) + XCTAssertEqual(paymentMethod?.usBankAccount?.last4, "6789", "`last4` should be 6789") + XCTAssertNil(paymentMethod?.usBankAccount?.networks?.preferred) + XCTAssertEqual(paymentMethod?.usBankAccount?.networks?.supported.count, 1) + XCTAssertEqual(paymentMethod?.usBankAccount?.networks?.supported.first, "ach") + exp.fulfill() + + } + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsTest.swift new file mode 100644 index 00000000..dcbe4ef2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountParamsTest.swift @@ -0,0 +1,233 @@ +// +// STPPaymentMethodUSBankAccountParamsTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/2/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodUSBankAccountParamsTest: XCTestCase { + + let apiClient: STPAPIClient = { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + return client + }() + + func testCreateUSBankAccountPaymentMethod_checking_individual() { + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountType = .checking + XCTAssertEqual(usBankAccountParams.accountTypeString, "checking") + usBankAccountParams.accountHolderType = .individual + XCTAssertEqual(usBankAccountParams.accountHolderTypeString, "individual") + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let params = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method US Bank Account create") + + apiClient.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .USBankAccount, "Incorrect PaymentMethod type") + XCTAssertNotNil( + paymentMethod?.usBankAccount, + "The `usBankAccount` property must be populated" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountHolderType, + .individual, + "`accountHolderType` should be individual" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountType, + .checking, + "`accountType` should be checking" + ) + XCTAssertEqual(paymentMethod?.usBankAccount?.last4, "6789", "`last4` should be 6789") + + exp.fulfill() + + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + + func testCreateUSBankAccountPaymentMethod_savings_company() { + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountType = .savings + XCTAssertEqual(usBankAccountParams.accountTypeString, "savings") + usBankAccountParams.accountHolderType = .company + XCTAssertEqual(usBankAccountParams.accountHolderTypeString, "company") + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let params = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method US Bank Account create") + + apiClient.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .USBankAccount, "Incorrect PaymentMethod type") + XCTAssertNotNil( + paymentMethod?.usBankAccount, + "The `usBankAccount` property must be populated" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountHolderType, + .company, + "`accountHolderType` should be company" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountType, + .savings, + "`accountType` should be savings" + ) + XCTAssertEqual(paymentMethod?.usBankAccount?.last4, "6789", "`last4` should be 6789") + + exp.fulfill() + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + + func testCreateUSBankAccountPaymentMethod_checking_individual_set_with_string() { + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountTypeString = "checking" + XCTAssertEqual(usBankAccountParams.accountType, .checking) + usBankAccountParams.accountHolderTypeString = "individual" + XCTAssertEqual(usBankAccountParams.accountHolderType, .individual) + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let params = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method US Bank Account create") + + apiClient.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .USBankAccount, "Incorrect PaymentMethod type") + XCTAssertNotNil( + paymentMethod?.usBankAccount, + "The `usBankAccount` property must be populated" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountHolderType, + .individual, + "`accountHolderType` should be individual" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountType, + .checking, + "`accountType` should be checking" + ) + XCTAssertEqual(paymentMethod?.usBankAccount?.last4, "6789", "`last4` should be 6789") + + exp.fulfill() + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + + func testCreateUSBankAccountPaymentMethod_savings_company_set_with_string() { + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountTypeString = "savings" + XCTAssertEqual(usBankAccountParams.accountType, .savings) + usBankAccountParams.accountHolderTypeString = "company" + XCTAssertEqual(usBankAccountParams.accountHolderType, .company) + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let params = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let exp = expectation(description: "Payment Method US Bank Account create") + + apiClient.createPaymentMethod(with: params) { + (paymentMethod: STPPaymentMethod?, error: Error?) in + + XCTAssertNil(error) + XCTAssertNotNil(paymentMethod, "Payment method should be populated") + XCTAssertEqual(paymentMethod?.type, .USBankAccount, "Incorrect PaymentMethod type") + XCTAssertNotNil( + paymentMethod?.usBankAccount, + "The `usBankAccount` property must be populated" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountHolderType, + .company, + "`accountHolderType` should be company" + ) + XCTAssertEqual( + paymentMethod?.usBankAccount?.accountType, + .savings, + "`accountType` should be savings" + ) + XCTAssertEqual(paymentMethod?.usBankAccount?.last4, "6789", "`last4` should be 6789") + + exp.fulfill() + } + + self.waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + + func testEncodingWithLinkAccountSessionID() { + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.linkAccountSessionID = "las_test" + let encoded = STPFormEncoder.dictionary(forObject: usBankAccountParams) + XCTAssertEqual( + (encoded["us_bank_account"] as? [AnyHashable: Any])?["link_account_session"] as? String, + "las_test" + ) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountTest.swift b/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountTest.swift new file mode 100644 index 00000000..cd6fc6c5 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodUSBankAccountTest.swift @@ -0,0 +1,52 @@ +// +// STPPaymentMethodUSBankAccountTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/2/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentMethodUSBankAccountTest: XCTestCase { + + static let usBankAccountPaymentIntentClientSecret = + "pi_3KhHLqFY0qyl6XeW1X2ZMsOT_secret_k5bOLoKJEW8ZhQFpokL0OrpbU" + + func retrieveUSBankAccountJSON(_ completion: @escaping ([AnyHashable: Any]?) -> Void) { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + client.retrievePaymentIntent( + withClientSecret: Self.usBankAccountPaymentIntentClientSecret, + expand: ["payment_method"] + ) { paymentIntent, _ in + let klarnaJson = paymentIntent?.paymentMethod?.usBankAccount?.allResponseFields + completion(klarnaJson ?? [:]) + } + } + + func testObjectDecoding() { + let retrieveJSON = XCTestExpectation(description: "Retrieve JSON") + + retrieveUSBankAccountJSON({ json in + let usBankAccount = STPPaymentMethodUSBankAccount.decodedObject(fromAPIResponse: json) + XCTAssertNotNil(usBankAccount, "Failed to decode JSON") + XCTAssertEqual(usBankAccount?.last4, "6789") + XCTAssertEqual(usBankAccount?.routingNumber, "110000000") + XCTAssertEqual(usBankAccount?.bankName, "STRIPE TEST BANK") + XCTAssertEqual(usBankAccount?.accountHolderType, .individual) + XCTAssertEqual(usBankAccount?.accountType, .checking) + XCTAssertNotNil(usBankAccount?.fingerprint) + retrieveJSON.fulfill() + }) + + wait(for: [retrieveJSON], timeout: STPTestingNetworkRequestTimeout) + } + +} diff --git a/Stripe/StripeiOSTests/STPPaymentMethodiDEALTest.m b/Stripe/StripeiOSTests/STPPaymentMethodiDEALTest.m new file mode 100644 index 00000000..e4c3992a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentMethodiDEALTest.m @@ -0,0 +1,45 @@ +// +// STPPaymentMethodiDEALTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 3/9/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPPaymentMethodiDEALTest : XCTestCase + +@end + +@implementation STPPaymentMethodiDEALTest + +- (NSDictionary *)exampleJson { + return @{ + @"bank": @"Rabobank", + @"bic": @"RABONL2U", + }; +} + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[self exampleJson] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPPaymentMethodiDEAL decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPPaymentMethodiDEAL decodedObjectFromAPIResponse:[self exampleJson]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [self exampleJson]; + STPPaymentMethodiDEAL *ideal = [STPPaymentMethodiDEAL decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(ideal.bankName, @"Rabobank"); + XCTAssertEqualObjects(ideal.bankIdentifierCode, @"RABONL2U"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerLocalizationTests.swift b/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerLocalizationTests.swift new file mode 100644 index 00000000..442ce9e6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerLocalizationTests.swift @@ -0,0 +1,106 @@ +// +// STPPaymentOptionsViewControllerLocalizationTests.swift +// StripeiOS Tests +// +// Created by Brian Dorfman on 10/17/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class MockSTPPaymentOptionsViewControllerDelegate: NSObject, STPPaymentOptionsViewControllerDelegate +{ + func paymentOptionsViewController( + _ paymentOptionsViewController: STPPaymentOptionsViewController, + didFailToLoadWithError error: Error + ) { + } + + func paymentOptionsViewControllerDidFinish( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) { + } + + func paymentOptionsViewControllerDidCancel( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) { + } + +} + +class STPPaymentOptionsViewControllerLocalizationTests: FBSnapshotTestCase { + override func setUp() { + super.setUp() + + // self.recordMode = true; + } + + func performSnapshotTest(forLanguage language: String?) { + let config = STPFixtures.paymentConfiguration() + config.companyName = "Test Company" + config.requiredBillingAddressFields = .full + let theme = STPTheme.defaultTheme + let paymentMethods = [STPFixtures.paymentMethod(), STPFixtures.paymentMethod()] + let customerContext = Testing_StaticCustomerContext.init( + customer: STPFixtures.customerWithCardTokenAndSourceSources(), + paymentMethods: paymentMethods + ) + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + STPLocalizationUtils.overrideLanguage(to: language) + let paymentOptionsVC = STPPaymentOptionsViewController( + configuration: config, + theme: theme, + customerContext: customerContext, + delegate: delegate + ) + let didLoadExpectation = expectation(description: "VC did load") + + paymentOptionsVC.loadingPromise?.onSuccess({ (_) in + didLoadExpectation.fulfill() + }) + wait(for: [didLoadExpectation].compactMap { $0 }, timeout: 2) + + let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: paymentOptionsVC)! + + STPSnapshotVerifyView(viewToTest, identifier: nil) + STPLocalizationUtils.overrideLanguage(to: nil) + } + + func testGerman() { + performSnapshotTest(forLanguage: "de") + } + + func testEnglish() { + performSnapshotTest(forLanguage: "en") + } + + func testSpanish() { + performSnapshotTest(forLanguage: "es") + } + + func testFrench() { + performSnapshotTest(forLanguage: "fr") + } + + func testItalian() { + performSnapshotTest(forLanguage: "it") + } + + func testJapanese() { + performSnapshotTest(forLanguage: "ja") + } + + func testDutch() { + performSnapshotTest(forLanguage: "nl") + } + + func testChinese() { + performSnapshotTest(forLanguage: "zh-Hans") + } +} diff --git a/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerTest.swift b/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerTest.swift new file mode 100644 index 00000000..205ab728 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerTest.swift @@ -0,0 +1,351 @@ +// +// STPPaymentOptionsViewControllerTest.swift +// StripeiOS Tests +// +// Created by Brian Dorfman on 10/10/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import OCMock + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPaymentOptionsViewControllerTest: XCTestCase { + class MockSTPPaymentOptionsViewControllerDelegate: NSObject, + STPPaymentOptionsViewControllerDelegate + { + var didFail = false + func paymentOptionsViewController( + _ paymentOptionsViewController: STPPaymentOptionsViewController, + didFailToLoadWithError error: Error + ) { + didFail = true + } + + var didFinish = false + func paymentOptionsViewControllerDidFinish( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) { + didFinish = true + } + + var didCancel = false + func paymentOptionsViewControllerDidCancel( + _ paymentOptionsViewController: STPPaymentOptionsViewController + ) { + didCancel = true + } + + var didSelect = false + func paymentOptionsViewController( + _ paymentOptionsViewController: STPPaymentOptionsViewController, + didSelect paymentOption: STPPaymentOption + ) { + didSelect = true + } + } + + func buildViewController( + with customer: STPCustomer, + paymentMethods: [STPPaymentMethod], + configuration config: STPPaymentConfiguration, + delegate: STPPaymentOptionsViewControllerDelegate + ) -> STPPaymentOptionsViewController { + let mockCustomerContext = Testing_StaticCustomerContext( + customer: customer, + paymentMethods: paymentMethods + ) + return buildViewController( + with: mockCustomerContext, + configuration: config, + delegate: delegate + ) + } + + func buildViewController( + with customerContext: STPCustomerContext, + configuration config: STPPaymentConfiguration, + delegate: STPPaymentOptionsViewControllerDelegate + ) -> STPPaymentOptionsViewController { + let theme = STPTheme.defaultTheme + let vc = STPPaymentOptionsViewController( + configuration: config, + theme: theme, + customerContext: customerContext, + delegate: delegate + ) + let didLoadExpectation = expectation(description: "VC did load") + vc.loadingPromise?.onSuccess({ (_) in + didLoadExpectation.fulfill() + }) + + wait(for: [didLoadExpectation], timeout: 2) + + return vc + } + + /// When the customer has no sources, and card is the sole available payment + /// method, STPAddCardViewController should be shown. + func testInitWithNoSourcesAndConfigWithUseSourcesOffAndCardAvailable() { + let customer = STPFixtures.customerWithNoSources() + let config = STPFixtures.paymentConfiguration() + config.applePayEnabled = false + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: [], + configuration: config, + delegate: delegate + ) + XCTAssertTrue((sut.internalViewController is STPAddCardViewController)) + } + + /// When the customer has a single card token source and the available payment methods + /// are card and apple pay, STPPaymentOptionsInternalVC should be shown. + func testInitWithSingleCardTokenSourceAndCardAvailable() { + let customer = STPFixtures.customerWithSingleCardTokenSource() + let paymentMethods = [STPFixtures.paymentMethod()] + let config = STPFixtures.paymentConfiguration() + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: paymentMethods.compactMap { $0 }, + configuration: config, + delegate: delegate + ) + XCTAssertTrue((sut.internalViewController is STPPaymentOptionsInternalViewController)) + } + + /// When the customer has a single card source source and the available payment methods + /// are card only, STPPaymentOptionsInternalVC should be shown. + func testInitWithSingleCardSourceSourceAndCardAvailable() { + let customer = STPFixtures.customerWithSingleCardSourceSource() + let paymentMethods = [STPFixtures.paymentMethod()] + let config = STPFixtures.paymentConfiguration() + config.applePayEnabled = false + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: paymentMethods.compactMap { $0 }, + configuration: config, + delegate: delegate + ) + XCTAssertTrue((sut.internalViewController is STPPaymentOptionsInternalViewController)) + } + + /// Tapping cancel in an internal AddCard view controller should result in a call to + /// didCancel: + func testAddCardCancelForwardsToDelegate() { + let customer = STPFixtures.customerWithNoSources() + let config = STPFixtures.paymentConfiguration() + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: [], + configuration: config, + delegate: delegate + ) + XCTAssertTrue((sut.internalViewController is STPAddCardViewController)) + let cancelButton = sut.internalViewController?.navigationItem.leftBarButtonItem + cancelButton?.target?.perform(cancelButton?.action, with: cancelButton) + + XCTAssertTrue(delegate.didCancel) + } + + /// Tapping cancel in an internal PaymentOptionsInternal view controller should + /// result in a call to didCancel: + func testInternalCancelForwardsToDelegate() { + let customer = STPFixtures.customerWithSingleCardTokenSource() + let paymentMethods = [STPFixtures.paymentMethod()] + let config = STPFixtures.paymentConfiguration() + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: paymentMethods.compactMap { $0 }, + configuration: config, + delegate: delegate + ) + XCTAssertTrue((sut.internalViewController is STPPaymentOptionsInternalViewController)) + let cancelButton = sut.internalViewController?.navigationItem.leftBarButtonItem + _ = cancelButton?.target?.perform(cancelButton?.action, with: cancelButton) + + XCTAssertTrue(delegate.didCancel) + } + + /// When an AddCard view controller creates a card payment method, it should be attached to the + /// customer and the correct delegate methods should be called. + func testAddCardAttachesToCustomerAndFinishes() { + let config = STPFixtures.paymentConfiguration() + let customer = STPFixtures.customerWithNoSources() + let mockCustomerContext = Testing_StaticCustomerContext( + customer: customer, + paymentMethods: [] + ) + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: mockCustomerContext, + configuration: config, + delegate: delegate + ) + XCTAssertNotNil(sut.view) + XCTAssertTrue((sut.internalViewController is STPAddCardViewController)) + + let internalVC = sut.internalViewController as? STPAddCardViewController + let exp = expectation(description: "completion") + let expectedPaymentMethod = STPFixtures.paymentMethod() + internalVC?.delegate?.addCardViewController( + internalVC!, + didCreatePaymentMethod: expectedPaymentMethod + ) { error in + XCTAssertNil(error) + exp.fulfill() + } + + let _: ((Any?) -> Bool)? = { obj in + let paymentMethod = obj as? STPPaymentMethod + return paymentMethod?.stripeId == expectedPaymentMethod.stripeId + } + XCTAssertTrue(mockCustomerContext.didAttach) + XCTAssertTrue(delegate.didSelect) + XCTAssertTrue(delegate.didFinish) + waitForExpectations(timeout: 2, handler: nil) + } + + // Tests for race condition where the promise for fetching payment methods + // finishes in the context of intializing the sut, and `addCardViewControllerFooterView` + // is set directly after init, while internalViewController is `STPAddCardViewController` + func testSetAfterInit_addCardViewControllerFooterView_STPAddCardViewController() { + let customer = STPFixtures.customerWithNoSources() + let config = STPFixtures.paymentConfiguration() + config.applePayEnabled = false + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: [], + configuration: config, + delegate: delegate + ) + sut.addCardViewControllerFooterView = UIView() + guard let payMethodsInternal = sut.internalViewController as? STPAddCardViewController else { + XCTFail() + return + } + XCTAssertNotNil(payMethodsInternal.customFooterView) + } + + // Tests for race condition where the promise for fetching payment methods + // finishes in the context of intializing the sut, and the `paymentOptionsViewControllerFooterView` + // is set directly after init, while internalViewController is `STPPaymentOptionsInternalViewController` + func testSetAfterInit_paymentOptionsViewControllerFooterView_STPPaymentOptionsInternalViewController() { + let customer = STPFixtures.customerWithSingleCardTokenSource() + let paymentMethods = [STPFixtures.paymentMethod()] + let config = STPFixtures.paymentConfiguration() + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: paymentMethods.compactMap { $0 }, + configuration: config, + delegate: delegate + ) + sut.paymentOptionsViewControllerFooterView = UIView() + guard let payMethodsInternal = sut.internalViewController as? STPPaymentOptionsInternalViewController else { + XCTFail() + return + } +#if compiler(>=5.7) + XCTAssertNotNil(payMethodsInternal.customFooterView) +#endif + } + + // Tests for race condition where the promise for fetching payment methods + // finishes in the context of init the sut, and the `addCardViewControllerFooterView` + // is set directly after init, while internalViewController is `STPPaymentOptionsInternalViewController` + func testSetAfterInit_addCardViewControllerFooterView_STPPaymentOptionsInternalViewController() { + let customer = STPFixtures.customerWithSingleCardTokenSource() + let paymentMethods = [STPFixtures.paymentMethod()] + let config = STPFixtures.paymentConfiguration() + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: paymentMethods.compactMap { $0 }, + configuration: config, + delegate: delegate + ) + sut.addCardViewControllerFooterView = UIView() + guard let payMethodsInternal = sut.internalViewController as? STPPaymentOptionsInternalViewController else { + XCTFail() + return + } +#if compiler(>=5.7) + XCTAssertNotNil(payMethodsInternal.addCardViewControllerCustomFooterView) +#endif + } + + // Tests for race condition where the promise for fetching payment methods + // finishes in the context of init the sut, and the `prefilledInformation` + // is set directly after init, while internalViewController is `STPPaymentOptionsInternalViewController` + func testSetAfterInit_prefilledInformation_STPPaymentOptionsInternalViewController() { + let customer = STPFixtures.customerWithSingleCardTokenSource() + let paymentMethods = [STPFixtures.paymentMethod()] + let config = STPFixtures.paymentConfiguration() + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: paymentMethods.compactMap { $0 }, + configuration: config, + delegate: delegate + ) + let userInformation = STPUserInformation() + let address = STPAddress() + address.name = "John Doe" + address.line1 = "123 Main" + address.city = "Seattle" + address.state = "Washington" + address.postalCode = "98104" + address.phone = "2065551234" + userInformation.billingAddress = address + sut.prefilledInformation = userInformation + guard let payMethodsInternal = sut.internalViewController as? STPPaymentOptionsInternalViewController else { + XCTFail() + return + } +#if compiler(>=5.7) + XCTAssertNotNil(payMethodsInternal.prefilledInformation) +#endif + } + + // Tests for race condition where the promise for fetching payment methods + // finishes in the context of init the sut, and the `prefilledInformation` + // is set directly after init, while internalViewController is `STPAddCardViewController` + func testSetAfterInit_prefilledInformation_STPAddCardViewController() { + let customer = STPFixtures.customerWithNoSources() + let config = STPFixtures.paymentConfiguration() + config.applePayEnabled = false + let delegate = MockSTPPaymentOptionsViewControllerDelegate() + let sut = buildViewController( + with: customer, + paymentMethods: [], + configuration: config, + delegate: delegate + ) + let userInformation = STPUserInformation() + let address = STPAddress() + address.name = "John Doe" + address.line1 = "123 Main" + address.city = "Seattle" + address.state = "Washington" + address.postalCode = "98104" + address.phone = "2065551234" + userInformation.billingAddress = address + sut.prefilledInformation = userInformation + guard let payMethodsInternal = sut.internalViewController as? STPAddCardViewController else { + XCTFail() + return + } + XCTAssertNotNil(payMethodsInternal.prefilledInformation) + } +} diff --git a/Stripe/StripeiOSTests/STPPhoneNumberValidatorTest.swift b/Stripe/StripeiOSTests/STPPhoneNumberValidatorTest.swift new file mode 100644 index 00000000..a1baab25 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPhoneNumberValidatorTest.swift @@ -0,0 +1,137 @@ +// +// STPPhoneNumberValidatorTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 3/22/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +private let kUSCountryCode = "US" +private let kUKCountryCode = "UK" +class STPPhoneNumberValidatorTest: XCTestCase { + func testValidPhoneNumbers() { + XCTAssertTrue( + STPPhoneNumberValidator.stringIsValidPhoneNumber( + "555-555-5555", + forCountryCode: kUSCountryCode + ) + ) + XCTAssertTrue( + STPPhoneNumberValidator.stringIsValidPhoneNumber( + "5555555555", + forCountryCode: kUSCountryCode + ) + ) + XCTAssertTrue( + STPPhoneNumberValidator.stringIsValidPhoneNumber( + "(555) 555-5555", + forCountryCode: kUSCountryCode + ) + ) + } + + func testInvalidPhoneNumbers() { + XCTAssertFalse( + STPPhoneNumberValidator.stringIsValidPhoneNumber("", forCountryCode: kUSCountryCode) + ) + XCTAssertFalse( + STPPhoneNumberValidator.stringIsValidPhoneNumber( + "555-555-555", + forCountryCode: kUSCountryCode + ) + ) + XCTAssertFalse( + STPPhoneNumberValidator.stringIsValidPhoneNumber( + "555-555-A555", + forCountryCode: kUSCountryCode + ) + ) + XCTAssertTrue( + STPPhoneNumberValidator.stringIsValidPhoneNumber( + "55555555555", + forCountryCode: kUSCountryCode + ) + ) + } + + func testFormattedSanitizedPhoneNumberForString() { + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "55", + forCountryCode: kUSCountryCode + ), + "55" + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "555", + forCountryCode: kUSCountryCode + ), + "(555) " + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "55555", + forCountryCode: kUSCountryCode + ), + "(555) 55" + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "A-55555", + forCountryCode: kUSCountryCode + ), + "(555) 55" + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "5555555", + forCountryCode: kUSCountryCode + ), + "(555) 555-5" + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "5555555555", + forCountryCode: kUSCountryCode + ), + "(555) 555-5555" + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "5555555555123", + forCountryCode: kUSCountryCode + ), + "(555) 555-5555" + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedSanitizedPhoneNumber( + for: "5555555555123", + forCountryCode: kUKCountryCode + ), + "5555555555123" + ) + } + + func testFormattedRedactedPhoneNumberForString() { + XCTAssertEqual( + STPPhoneNumberValidator.formattedRedactedPhoneNumber( + for: "+1******1234", + forCountryCode: kUSCountryCode + ), + "+1 (•••) •••-1234" + ) + XCTAssertEqual( + STPPhoneNumberValidator.formattedRedactedPhoneNumber( + for: "+86******1234", + forCountryCode: kUKCountryCode + ), + "+86 ••••••1234" + ) + } +} diff --git a/Stripe/StripeiOSTests/STPPinManagementServiceFunctionalTest.swift b/Stripe/StripeiOSTests/STPPinManagementServiceFunctionalTest.swift new file mode 100644 index 00000000..23a221e2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPinManagementServiceFunctionalTest.swift @@ -0,0 +1,103 @@ +// +// STPPinManagementServiceFunctionalTest.swift +// StripeiOS Tests +// +// Created by Arnaud Cavailhez on 4/29/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +import PassKit +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class TestEphemeralKeyProvider: NSObject, STPIssuingCardEphemeralKeyProvider { + func createIssuingCardKey( + withAPIVersion apiVersion: String, + completion: STPJSONResponseCompletionBlock + ) { + print("apiVersion \(apiVersion)") + let response = + [ + "id": "ephkey_token", + "object": "ephemeral_key", + "associated_objects": [ + [ + "type": "issuing.card", + "id": "ic_token", + ], + ], + "created": NSNumber(value: 1_556_656_558), + "expires": NSNumber(value: 1_556_660_158), + "livemode": NSNumber(value: true), + "secret": "ek_live_secret", + ] as [String: Any] + completion(response, nil) + } +} + +class STPPinManagementServiceFunctionalTest: STPNetworkStubbingTestCase { + override func setUp() { + // self.recordingMode = YES; + super.setUp() + } + + func testRetrievePin() { + let keyProvider = TestEphemeralKeyProvider() + let service = STPPinManagementService(keyProvider: keyProvider) + + let expectation = self.expectation(description: "Received PIN") + + service.retrievePin( + "ic_token", + verificationId: "iv_token", + oneTimeCode: "123456" + ) { cardPin, status, error in + if error == nil && status == .success && (cardPin?.pin == "2345") { + expectation.fulfill() + } + } + waitForExpectations(timeout: 5.0, handler: nil) + } + + func testUpdatePin() { + let keyProvider = TestEphemeralKeyProvider() + let service = STPPinManagementService(keyProvider: keyProvider) + + let expectation = self.expectation(description: "Received PIN") + + service.updatePin( + "ic_token", + newPin: "3456", + verificationId: "iv_token", + oneTimeCode: "123-456" + ) { cardPin, status, error in + if error == nil && status == .success && (cardPin?.pin == "3456") { + expectation.fulfill() + } + } + waitForExpectations(timeout: 5.0, handler: nil) + } + + func testRetrievePinWithError() { + let keyProvider = TestEphemeralKeyProvider() + let service = STPPinManagementService(keyProvider: keyProvider) + + let expectation = self.expectation(description: "Received Error") + + service.retrievePin( + "ic_token", + verificationId: "iv_token", + oneTimeCode: "123456" + ) { _, status, _ in + if status == .errorVerificationAlreadyRedeemed { + expectation.fulfill() + } + } + waitForExpectations(timeout: 5.0, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldFormatterTests.swift b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldFormatterTests.swift new file mode 100644 index 00000000..5a664d9e --- /dev/null +++ b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldFormatterTests.swift @@ -0,0 +1,58 @@ +// +// STPPostalCodeInputTextFieldFormatterTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/30/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPostalCodeInputTextFieldFormatterTests: XCTestCase { + + func testIsAllowedInput() { + let formatter = STPPostalCodeInputTextFieldFormatter() + formatter.countryCode = "US" + XCTAssertTrue(formatter.isAllowedInput("10002", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("21218", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertFalse(formatter.isAllowedInput("10002-1234", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertFalse(formatter.isAllowedInput("100021234", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertFalse(formatter.isAllowedInput("ABC10002", to: "", at: NSRange(location: 0, length: 0))) + + XCTAssertFalse(formatter.isAllowedInput("1", to: "100021234", at: NSRange(location: 10, length: 0))) + XCTAssertFalse(formatter.isAllowedInput("1", to: "10002", at: NSRange(location: 4, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("1", to: "1000", at: NSRange(location: 4, length: 0))) + + formatter.countryCode = "UK" + XCTAssertTrue(formatter.isAllowedInput("10002-1234", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("100021234", to: "", at: NSRange(location: 0, length: 0))) + XCTAssertTrue(formatter.isAllowedInput("ABC10002", to: "", at: NSRange(location: 0, length: 0))) + } + + func testFormattedString() { + let formatter = STPPostalCodeInputTextFieldFormatter() + formatter.countryCode = "US" + + XCTAssertEqual( + NSAttributedString(string: ""), + formatter.formattedText(from: "- ", with: [:]) + ) + XCTAssertEqual( + NSAttributedString(string: "10002"), + formatter.formattedText(from: "10002-1234", with: [:]) + ) + + formatter.countryCode = "UK" + XCTAssertEqual( + NSAttributedString(string: "A B C D E F G"), + formatter.formattedText(from: " a b c d e f g", with: [:]) + ) + } + +} diff --git a/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldSnapshotTests.swift b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldSnapshotTests.swift new file mode 100644 index 00000000..63540878 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldSnapshotTests.swift @@ -0,0 +1,77 @@ +// +// STPPostalCodeInputTextFieldSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/30/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPostalCodeInputTextFieldSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func testEmpty() { + let field = STPPostalCodeInputTextField(postalCodeRequirement: .standard) + field.sizeToFit() + field.frame.size.width = 200 + + STPSnapshotVerifyView(field) + } + + func testIncomplete() { + let field = STPPostalCodeInputTextField(postalCodeRequirement: .standard) + field.sizeToFit() + field.frame.size.width = 200 + field.countryCode = "US" + field.text = "1" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + func testValidUS() { + let field = STPPostalCodeInputTextField(postalCodeRequirement: .standard) + field.sizeToFit() + field.frame.size.width = 200 + field.countryCode = "US" + field.text = "12345" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + func testValidUK() { + let field = STPPostalCodeInputTextField(postalCodeRequirement: .standard) + field.sizeToFit() + field.frame.size.width = 200 + field.countryCode = "UK" + field.text = "abcdef" + field.textDidChange() + + STPSnapshotVerifyView(field) + } + + func testInvalid() { + let field = STPPostalCodeInputTextField(postalCodeRequirement: .standard) + field.sizeToFit() + field.frame.size.width = 200 + field.countryCode = "US" + field.text = "12-3456789" + field.textDidChange() + // manually set because the formatter prevents setting invalid text + field.validator.validationState = .invalid(errorMessage: nil) + + STPSnapshotVerifyView(field) + } +} diff --git a/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldTests.swift b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldTests.swift new file mode 100644 index 00000000..d9ce772a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldTests.swift @@ -0,0 +1,69 @@ +// +// STPPostalCodeInputTextFieldTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 9/3/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPostalCodeInputTextFieldTests: XCTestCase { + + func testClearingInvalidPostalCodeAfterCountryChange() { + let postalCodeField = STPPostalCodeInputTextField(postalCodeRequirement: .standard) + postalCodeField.countryCode = "UK" + postalCodeField.text = "DL12" // valid UK post code, invalid US ZIP Code + + // Change country + postalCodeField.countryCode = "US" + + XCTAssertEqual( + postalCodeField.text, + "", + "Postal code field should clear its value if no longer valid after country change" + ) + } + + func testPreservingValidPostalCodeAfterCountryChange() { + let postalCodeField = STPPostalCodeInputTextField(postalCodeRequirement: .standard) + postalCodeField.countryCode = "US" + postalCodeField.text = "10010" // valid US and HR ZIP/postal code + + // Change country + postalCodeField.countryCode = "HR" + + XCTAssertEqual( + postalCodeField.text, + "10010", + "Postal code field should preserve its value if it is still valid after country change" + ) + } + + func testChangeToNonRequiredPostalCodeIsValid() { + let postalCodeField = STPPostalCodeInputTextField(postalCodeRequirement: .upe) + // given that the postal code field is empty... + + // when + postalCodeField.countryCode = "US" + if case .incomplete = postalCodeField.validationState { + // pass + } else { + XCTFail("Empty postal code should be incomplete for US") + } + + // when + postalCodeField.countryCode = "FR" + if case .valid = postalCodeField.validationState { + // pass + } else { + XCTFail("Empty postal code should be valid for non-required country") + } + } +} diff --git a/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldValidatorTests.swift b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldValidatorTests.swift new file mode 100644 index 00000000..edee1d92 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPostalCodeInputTextFieldValidatorTests.swift @@ -0,0 +1,79 @@ +// +// STPPostalCodeInputTextFieldValidatorTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/30/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPostalCodeInputTextFieldValidatorTests: XCTestCase { + + func testValidation() { + let validator = STPPostalCodeInputTextFieldValidator(postalCodeRequirement: .standard) + validator.countryCode = "US" + + validator.inputValue = nil + XCTAssertEqual( + STPValidatedInputState.incomplete(description: nil), + validator.validationState + ) + + validator.inputValue = "" + XCTAssertEqual( + STPValidatedInputState.incomplete(description: nil), + validator.validationState + ) + + validator.inputValue = "1234" + XCTAssertEqual( + STPValidatedInputState.incomplete(description: "Your ZIP is incomplete."), + validator.validationState + ) + + validator.inputValue = "12345" + XCTAssertEqual(STPValidatedInputState.valid(message: nil), validator.validationState) + + validator.inputValue = "12345678" + XCTAssertEqual( + STPValidatedInputState.incomplete(description: "Your ZIP is incomplete."), + validator.validationState + ) + + validator.inputValue = "123456789" + XCTAssertEqual(STPValidatedInputState.valid(message: nil), validator.validationState) + + validator.inputValue = "12345-6789" + XCTAssertEqual(STPValidatedInputState.valid(message: nil), validator.validationState) + + validator.inputValue = "12-3456789" + XCTAssertEqual( + STPValidatedInputState.invalid(errorMessage: "Your ZIP is invalid."), + validator.validationState + ) + + validator.inputValue = "12345-" + XCTAssertEqual( + STPValidatedInputState.incomplete(description: "Your ZIP is incomplete."), + validator.validationState + ) + + validator.inputValue = "hi" + XCTAssertEqual( + STPValidatedInputState.invalid(errorMessage: "Your ZIP is invalid."), + validator.validationState + ) + + validator.countryCode = "UK" + validator.inputValue = "hi" + XCTAssertEqual(STPValidatedInputState.valid(message: nil), validator.validationState) + } + +} diff --git a/Stripe/StripeiOSTests/STPPostalCodeValidatorTest.swift b/Stripe/StripeiOSTests/STPPostalCodeValidatorTest.swift new file mode 100644 index 00000000..04c47d73 --- /dev/null +++ b/Stripe/StripeiOSTests/STPPostalCodeValidatorTest.swift @@ -0,0 +1,103 @@ +// +// STPPostalCodeValidatorTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 4/14/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPostalCodeValidatorTest: XCTestCase { + func testValidUSPostalCodes() { + let codes = ["10002", "10002-1234", "100021234", "21218"] + for code in codes { + XCTAssertEqual( + STPPostalCodeValidator.validationState( + forPostalCode: code, + countryCode: "US" + ), + .valid + ) + } + } + + func testInvalidUSPostalCodes() { + let codes = ["100A03", "12345-12345", "1234512345", "$$$$$", "foo"] + for code in codes { + XCTAssertEqual( + STPPostalCodeValidator.validationState( + forPostalCode: code, + countryCode: "US" + ), + .invalid + ) + } + } + + func testIncompleteUSPostalCodes() { + let codes = ["", "123", "12345-", "12345-12"] + for code in codes { + XCTAssertEqual( + STPPostalCodeValidator.validationState( + forPostalCode: code, + countryCode: "US" + ), + .incomplete + ) + } + } + + func testValidGenericPostalCodes() { + let codes = ["ABC10002", "10002-ABCD", "ABCDE"] + for code in codes { + XCTAssertEqual( + STPPostalCodeValidator.validationState( + forPostalCode: code, + countryCode: "UK" + ), + .valid + ) + } + } + + func testIncompleteGenericPostalCodes() { + let codes = [""] + for code in codes { + XCTAssertEqual( + STPPostalCodeValidator.validationState( + forPostalCode: code, + countryCode: "UK" + ), + .incomplete + ) + } + } + + func testPostalCodeIsRequiredForUPE_nil() { + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequiredForUPE(forCountryCode: nil)) + } + + func testPostalCodeIsRequiredForUPE_empty() { + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequiredForUPE(forCountryCode: "")) + } + + func testPostalCodeIsRequiredForUPE_CA() { + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequiredForUPE(forCountryCode: "CA")) + } + + func testPostalCodeIsRequiredForUPE_GB() { + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequiredForUPE(forCountryCode: "GB")) + } + + func testPostalCodeIsRequiredForUPE_US() { + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequiredForUPE(forCountryCode: "CA")) + } + + func testPostalCodeIsRequiredForUPE_DK() { + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequiredForUPE(forCountryCode: "DK")) + } +} diff --git a/Stripe/StripeiOSTests/STPPushProvisioningDetailsFunctionalTest.swift b/Stripe/StripeiOSTests/STPPushProvisioningDetailsFunctionalTest.swift new file mode 100644 index 00000000..f1dfd54a --- /dev/null +++ b/Stripe/StripeiOSTests/STPPushProvisioningDetailsFunctionalTest.swift @@ -0,0 +1,56 @@ +// +// STPPushProvisioningDetailsFunctionalTest.swift +// StripeiOS Tests +// +// Created by Jack Flintermann on 11/30/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +import Stripe + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPPushProvisioningDetailsFunctionalTest: STPNetworkStubbingTestCase { + override func setUp() { + super.setUp() + } + + func testRetrievePushProvisioningDetails() { + // this API requires a secret key - replace the key below if you need to re-record the network traffic. + let client = STPAPIClient(publishableKey: "pk_test_REPLACEME") + let cardId = "ic_1C0Xig4JYtv6MPZK91WoXa9u" + let cert1 = + "MIID/TCCA6OgAwIBAgIIGM2CpiS9WyYwCgYIKoZIzj0EAwIwgYAxNDAyBgNVBAMMK0FwcGxlIFdvcmxkd2lkZSBEZXZlbG9wZXIgUmVsYXRpb25zIENBIC0gRzIxJjAkBgNVBAsMHUFwcGxlIENlcnRpZmljYXRpb24gQXV0aG9yaXR5MRMwEQYDVQQKDApBcHBsZSBJbmMuMQswCQYDVQQGEwJVUzAeFw0xODA2MDEyMjE0MTVaFw0yMDA2MzAyMjE0MTVaMGwxMjAwBgNVBAMMKWVjYy1jcnlwdG8tc2VydmljZXMtZW5jaXBoZXJtZW50X1VDNi1QUk9EMRQwEgYDVQQLDAtpT1MgU3lzdGVtczETMBEGA1UECgwKQXBwbGUgSW5jLjELMAkGA1UEBhMCVVMwWTATBgcqhkjOPQIBBggqhkjOPQMBBwNCAASzCVyQGX3syyW2aI6nyfNQe+vjjzjU4rLO0ZiWiVZZSmEzYfACFI8tuDFiDLv9XWrHEeX0/yNtGVjwAzpanWb/o4ICGDCCAhQwDAYDVR0TAQH/BAIwADAfBgNVHSMEGDAWgBSEtoTMOoZichZZlOgao71I3zrfCzBHBggrBgEFBQcBAQQ7MDkwNwYIKwYBBQUHMAGGK2h0dHA6Ly9vY3NwLmFwcGxlLmNvbS9vY3NwMDMtYXBwbGV3d2RyY2EyMDUwggEdBgNVHSAEggEUMIIBEDCCAQwGCSqGSIb3Y2QFATCB/jCBwwYIKwYBBQUHAgIwgbYMgbNSZWxpYW5jZSBvbiB0aGlzIGNlcnRpZmljYXRlIGJ5IGFueSBwYXJ0eSBhc3N1bWVzIGFjY2VwdGFuY2Ugb2YgdGhlIHRoZW4gYXBwbGljYWJsZSBzdGFuZGFyZCB0ZXJtcyBhbmQgY29uZGl0aW9ucyBvZiB1c2UsIGNlcnRpZmljYXRlIHBvbGljeSBhbmQgY2VydGlmaWNhdGlvbiBwcmFjdGljZSBzdGF0ZW1lbnRzLjA2BggrBgEFBQcCARYqaHR0cDovL3d3dy5hcHBsZS5jb20vY2VydGlmaWNhdGVhdXRob3JpdHkvMDYGA1UdHwQvMC0wK6ApoCeGJWh0dHA6Ly9jcmwuYXBwbGUuY29tL2FwcGxld3dkcmNhMi5jcmwwHQYDVR0OBBYEFI5aYtQKaJCRpvI1Dgh+Ra4x2iCrMA4GA1UdDwEB/wQEAwIDKDASBgkqhkiG92NkBicBAf8EAgUAMAoGCCqGSM49BAMCA0gAMEUCIAY/9gwN/KAAw3EtW3NyeX1UVM3fO+wVt0cbeHL8eM/mAiEAppLm5O/2Ox8uHkxI4U/kU5vDhJA21DRbzm2rsYN+EcQ=" + let cert2 = + "MIIC9zCCAnygAwIBAgIIb+/Y9emjp+4wCgYIKoZIzj0EAwIwZzEbMBkGA1UEAwwSQXBwbGUgUm9vdCBDQSAtIEczMSYwJAYDVQQLDB1BcHBsZSBDZXJ0aWZpY2F0aW9uIEF1dGhvcml0eTETMBEGA1UECgwKQXBwbGUgSW5jLjELMAkGA1UEBhMCVVMwHhcNMTQwNTA2MjM0MzI0WhcNMjkwNTA2MjM0MzI0WjCBgDE0MDIGA1UEAwwrQXBwbGUgV29ybGR3aWRlIERldmVsb3BlciBSZWxhdGlvbnMgQ0EgLSBHMjEmMCQGA1UECwwdQXBwbGUgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkxEzARBgNVBAoMCkFwcGxlIEluYy4xCzAJBgNVBAYTAlVTMFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE3fC3BkvP3XMEE8RDiQOTgPte9nStQmFSWAImUxnIYyIHCVJhysTZV+9tJmiLdJGMxPmAaCj8CWjwENrp0C7JGqOB9zCB9DBGBggrBgEFBQcBAQQ6MDgwNgYIKwYBBQUHMAGGKmh0dHA6Ly9vY3NwLmFwcGxlLmNvbS9vY3NwMDQtYXBwbGVyb290Y2FnMzAdBgNVHQ4EFgQUhLaEzDqGYnIWWZToGqO9SN863wswDwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBS7sN6hWDOImqSKmd6+veuv2sskqzA3BgNVHR8EMDAuMCygKqAohiZodHRwOi8vY3JsLmFwcGxlLmNvbS9hcHBsZXJvb3RjYWczLmNybDAOBgNVHQ8BAf8EBAMCAQYwEAYKKoZIhvdjZAYCDwQCBQAwCgYIKoZIzj0EAwIDaQAwZgIxANmxxzHGI/ZPTdDZR8V9GGkRh3En02it4Jtlmr5s3z9GppAJvm6hOyywUYlBPIfSvwIxAPxkUolLPF2/axzCiZgvcq61m6oaCyNUd1ToFUOixRLal1BzfF7QbrJcYlDXUfE6Wg==" + let nonce = "ea85a73a" + let nonceSignature = + "QBfCqTvDhmRcwqxJF3fDqzhXezIpwrpHFcOMw7/DvGVBwpfCuicwwqHCmMKYMD06w754wrjChcObwqjDr8K9wqxxUydQaMOyfsKGZMK4AcKMwqNfwoHDlcKLHsO5w7JqQiHDln7Du8KUNMOnwqpGwq/CqcKswo1Lw7s=" + let certs: [Data] = [ + Data(base64Encoded: cert1, options: [])!, + Data(base64Encoded: cert2, options: [])!, + ] + let expectation = self.expectation(description: "Push provisioning details") + let params = STPPushProvisioningDetailsParams( + cardId: "ic_1C0Xig4JYtv6MPZK91WoXa9u", + certificates: certs, + nonce: Data(base64Encoded: nonce, options: [])!, + nonceSignature: Data(base64Encoded: nonceSignature, options: [])! + ) + // To re-record this test, get an ephemeral key for the above Issuing card and pass that instead of [STPFixtures ephemeralKey] + let ephemeralKey = STPFixtures.ephemeralKey() + client.retrievePushProvisioningDetails(with: params, ephemeralKey: ephemeralKey) { + details, + error in + expectation.fulfill() + XCTAssertNil(error) + XCTAssert((details?.cardId == cardId)) + XCTAssertEqual(details, details) + } + waitForExpectations(timeout: 5.0, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPRadarSessionFunctionalTest.swift b/Stripe/StripeiOSTests/STPRadarSessionFunctionalTest.swift new file mode 100644 index 00000000..52f94747 --- /dev/null +++ b/Stripe/StripeiOSTests/STPRadarSessionFunctionalTest.swift @@ -0,0 +1,56 @@ +// +// STPRadarSessionFunctionalTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 5/20/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPRadarSessionFunctionalTest: XCTestCase { + func testCreateWithoutInitialFraudDetection() { + // When fraudDetectionData is empty... + FraudDetectionData.shared.reset() + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + let exp1 = expectation(description: "Create RadarSession") + client.createRadarSession { session, error in + // ...creates a Radar Session + XCTAssertNil(error) + guard let session = session else { + XCTFail() + return + } + XCTAssertTrue(session.id.count > 0) + exp1.fulfill() + } + + waitForExpectations(timeout: 10, handler: nil) + + // Now that fraudDetectionData is populated... + XCTAssertNotNil(FraudDetectionData.shared.sid) + XCTAssertNotNil(FraudDetectionData.shared.muid) + XCTAssertNotNil(FraudDetectionData.shared.guid) + + let exp2 = expectation(description: "Create RadarSession again") + client.createRadarSession { session, error in + // ...still creates a Radar Session + XCTAssertNil(error) + guard let session = session else { + XCTFail() + return + } + XCTAssertTrue(session.id.count > 0) + exp2.fulfill() + } + + waitForExpectations(timeout: 10, handler: nil) + } +} diff --git a/Stripe/StripeiOSTests/STPRedirectContextTest.m b/Stripe/StripeiOSTests/STPRedirectContextTest.m new file mode 100644 index 00000000..533748d4 --- /dev/null +++ b/Stripe/StripeiOSTests/STPRedirectContextTest.m @@ -0,0 +1,698 @@ +// +// STPRedirectContextTest.m +// Stripe +// +// Created by Ben Guo on 4/6/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +#import +#import + + + +#import "STPFixtures.h" + +#import "STPTestUtils.h" +@import StripeCore; + +@interface STPRedirectContext (Testing) +- (void)unsubscribeFromNotifications; +- (void)dismissPresentedViewController; +- (void)handleRedirectCompletionWithError:(nullable NSError *)error + shouldDismissViewController:(BOOL)shouldDismissViewController; +@end + +@interface STPRedirectContextTest : XCTestCase +@property (nonatomic, weak) STPRedirectContext *weak_sut; +@end + +/* + NOTE: + + If you are adding a test make sure your context unsubscribes from notifications + before your test ends. Otherwise notifications fired from other tests can cause + a reaction in an earlier, completed test and cause strange failures. + + Possible ways to do this: + 1. Your sut should already be calling unsubscribe, verified by OCMVerify + - you're good + 2. Your sut doesn't call unsubscribe as part of the test but it's not explicitly + disallowed - call [sut unsubscribeFromNotifications] at the end of your test + 3. Your sut doesn't call unsubscribe and you explicitly OCMReject it firing + - call [self unsubscribeContext:context] at the end of your test (use + the original context object here and _NOT_ the sut or it will not work). + */ +@implementation STPRedirectContextTest + + +/** + Use this to unsubscribe a context from notifications without calling + `sut.unsubscrbeFromNotifications` if you have OCMReject'd that method and thus + can't call it. + + Note: You MUST pass in the actual context object here and not the mock or the + unsubscibe will silently fail. + */ +- (void)unsubscribeContext:(STPRedirectContext *)context { + [[NSNotificationCenter defaultCenter] removeObserver:context + name:UIApplicationDidBecomeActiveNotification + object:nil]; + [[STPURLCallbackHandler shared] unregisterListener:(id)context]; +} + +- (void)testInitWithNonRedirectSourceReturnsNil { + STPSource *source = [STPFixtures cardSource]; + STPRedirectContext *sut = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion was called"); + }]; + XCTAssertNil(sut); +} + +- (void)testInitWithConsumedSourceReturnsNil { + NSMutableDictionary *json = [[STPTestUtils jsonNamed:STPTestJSONSourceCard] mutableCopy]; + json[@"status"] = @"consumed"; + STPSource *source = [STPSource decodedObjectFromAPIResponse:json]; + STPRedirectContext *sut = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion was called"); + }]; + XCTAssertNil(sut); +} + +- (void)testInitWithSource { + STPSource *source = [STPFixtures iDEALSource]; + __block BOOL completionCalled = NO; + NSError *fakeError = [[NSError alloc] initWithDomain:NSCocoaErrorDomain code:0 userInfo:nil]; + + STPRedirectContext *sut = [[STPRedirectContext alloc] initWithSource:source completion:^(NSString * _Nonnull sourceID, NSString * _Nullable clientSecret, NSError * _Nullable error) { + XCTAssertEqualObjects(source.stripeID, sourceID); + XCTAssertEqualObjects(source.clientSecret, clientSecret); + XCTAssertEqual(error, fakeError, @"Should be the same NSError object passed to completion() below"); + completionCalled = YES; + }]; + + // Make sure the initWithSource: method pulled out the right values from the Source + XCTAssertNil(sut.nativeRedirectURL); + XCTAssertEqualObjects(sut.redirectURL, source.redirect.url); + XCTAssertEqualObjects(sut.returnURL, source.redirect.returnURL); + + // and make sure the completion calls the completion block above + sut.completion(fakeError); + XCTAssertTrue(completionCalled); +} + +- (void)testInitWithSourceWithNativeURL { + STPSource *source = [STPFixtures alipaySourceWithNativeURL]; + __block BOOL completionCalled = NO; + NSURL *nativeURL = [NSURL URLWithString:source.details[@"native_url"]]; + NSError *fakeError = [[NSError alloc] initWithDomain:NSCocoaErrorDomain code:0 userInfo:nil]; + + STPRedirectContext *sut = [[STPRedirectContext alloc] initWithSource:source completion:^(NSString * _Nonnull sourceID, NSString * _Nullable clientSecret, NSError * _Nullable error) { + XCTAssertEqualObjects(source.stripeID, sourceID); + XCTAssertEqualObjects(source.clientSecret, clientSecret); + XCTAssertEqual(error, fakeError, @"Should be the same NSError object passed to completion() below"); + completionCalled = YES; + }]; + + // Make sure the initWithSource: method pulled out the right values from the Source + XCTAssertEqualObjects(sut.nativeRedirectURL, nativeURL); + XCTAssertEqualObjects(sut.redirectURL, source.redirect.url); + XCTAssertEqualObjects(sut.returnURL, source.redirect.returnURL); + + // and make sure the completion calls the completion block above + sut.completion(fakeError); + XCTAssertTrue(completionCalled); +} + +- (void)testInitWithPaymentIntent { + STPPaymentIntent *paymentIntent = [STPFixtures paymentIntent]; + __block BOOL completionCalled = NO; + NSError *fakeError = [[NSError alloc] initWithDomain:NSCocoaErrorDomain code:0 userInfo:nil]; + + STPRedirectContext *sut = [[STPRedirectContext alloc] initWithPaymentIntent:paymentIntent completion:^(NSString * _Nonnull clientSecret, NSError * _Nullable error) { + XCTAssertEqualObjects(paymentIntent.clientSecret, clientSecret); + XCTAssertEqual(error, fakeError, @"Should be the same NSError object passed to completion() below"); + completionCalled = YES; + }]; + + // Make sure the initWithPaymentIntent: method pulled out the right values from the PaymentIntent + XCTAssertNil(sut.nativeRedirectURL); + XCTAssertEqualObjects(sut.redirectURL.absoluteString, + @"https://hooks.stripe.com/redirect/authenticate/src_1Cl1AeIl4IdHmuTb1L7x083A?client_secret=src_client_secret_DBNwUe9qHteqJ8qQBwNWiigk"); + + // `nextSourceAction` & `authorizeWithURL` should just be aliases for `nextAction` & `redirectToURL`, already tested in `STPPaymentIntentTest` + XCTAssertNotNil(paymentIntent.nextAction.redirectToURL.returnURL); + XCTAssertEqualObjects(sut.returnURL, paymentIntent.nextAction.redirectToURL.returnURL); + + // and make sure the completion calls the completion block above + XCTAssertNotNil(sut); + if (sut.completion) { + sut.completion(fakeError); + } + XCTAssertTrue(completionCalled); +} + +- (void)testInitWithPaymentIntentFailures { + // Note next_action has been renamed to next_source_action in the API, but both still get sent down in the 2015-10-12 API + NSMutableDictionary *json = [[STPTestUtils jsonNamed:STPTestJSONPaymentIntent] mutableCopy]; + json[@"next_action"] = [json[@"next_action"] mutableCopy]; + json[@"next_action"][@"redirect_to_url"] = [json[@"next_action"][@"redirect_to_url"] mutableCopy]; + + void (^unusedCompletion)(NSString *, NSError *) = ^(__unused NSString * _Nonnull clientSecret, __unused NSError * _Nullable error) { + XCTFail(@"should not be constructed, definitely not completed"); + }; + + STPRedirectContext *(^create)(void) = ^{ + STPPaymentIntent *paymentIntent = [STPPaymentIntent decodedObjectFromAPIResponse:json]; + return [[STPRedirectContext alloc] initWithPaymentIntent:paymentIntent + completion:unusedCompletion]; + }; + + XCTAssertNotNil(create(), @"before mutation of json, creation should succeed"); + + json[@"status"] = @"processing"; + XCTAssertNil(create(), @"not created with wrong status"); + json[@"status"] = @"requires_action"; + + json[@"next_action"][@"type"] = @"not_redirect_to_url"; + XCTAssertNil(create(), @"not created with wrong next_action.type"); + json[@"next_action"][@"type"] = @"redirect_to_url"; + + NSString *correctURL = json[@"next_action"][@"redirect_to_url"][@"url"]; + json[@"next_action"][@"redirect_to_url"][@"url"] = @"not a valid URL"; + XCTAssertNil(create(), @"not created with an invalid URL in next_action.redirect_to_url.url"); + json[@"next_action"][@"redirect_to_url"][@"url"] = correctURL; + + NSString *correctReturnURL = json[@"next_action"][@"redirect_to_url"][@"return_url"]; + json[@"next_action"][@"redirect_to_url"][@"return_url"] = @"not a url"; + XCTAssertNil(create(), @"not created with invalid returnUrl"); + json[@"next_action"][@"redirect_to_url"][@"return_url"] = correctReturnURL; + + XCTAssertNotNil(create(), @"works again when everything is back to normal"); +} + +/** + After starting a SafariViewController redirect flow, + when a DidBecomeActive notification is posted, RedirectContext's completion + block and dismiss method should _NOT_ be called. + */ +- (void)testSafariViewControllerRedirectFlow_activeNotification { + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + id sut = OCMPartialMock(context); + + OCMReject([sut unsubscribeFromNotifications]); + OCMReject([sut dismissPresentedViewController]); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + [[NSNotificationCenter defaultCenter] postNotificationName:UIApplicationDidBecomeActiveNotification object:nil]; + + BOOL(^checker)(id) = ^BOOL(id vc) { + if ([vc isKindOfClass:[SFSafariViewController class]]) { + return YES; + } + return NO; + }; + + OCMVerify([mockVC presentViewController:[OCMArg checkWithBlock:checker] + animated:YES + completion:[OCMArg any]]); + [self unsubscribeContext:context]; +} + + +/** + After starting a SafariViewController redirect flow, + when the shared URLCallbackHandler is called with a valid URL, + RedirectContext's completion block and dismiss method should be called. + */ +- (void)testSafariViewControllerRedirectFlow_callbackHandlerCalledValidURL { + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + XCTestExpectation *exp = [self expectationWithDescription:@"completion"]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(NSString *sourceID, NSString *clientSecret, NSError *error) { + XCTAssertEqualObjects(sourceID, source.stripeID); + XCTAssertEqualObjects(clientSecret, source.clientSecret); + XCTAssertNil(error); + [exp fulfill]; + }]; + XCTAssertEqualObjects(source.redirect.returnURL, context.returnURL); + id sut = OCMPartialMock(context); + + OCMStub([sut handleRedirectCompletionWithError:[OCMArg any] shouldDismissViewController:YES]).andForwardToRealObject().andDo(^(__unused NSInvocation *invocation) { + [context safariViewControllerDidCompleteDismissal:[[SFSafariViewController alloc] initWithURL:[NSURL URLWithString:@"https://www.stripe.com"]]]; + }); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + BOOL(^checker)(id) = ^BOOL(id vc) { + if ([vc isKindOfClass:[SFSafariViewController class]]) { + NSURL *url = source.redirect.returnURL; + NSURLComponents *comps = [NSURLComponents componentsWithURL:url resolvingAgainstBaseURL:NO]; + [comps setStp_queryItemsDictionary:@{@"source": source.stripeID, + @"client_secret": source.clientSecret}]; + [[STPURLCallbackHandler shared] handleURLCallback:comps.URL]; + return YES; + } + return NO; + }; + + OCMVerify([mockVC presentViewController:[OCMArg checkWithBlock:checker] + animated:YES + completion:[OCMArg any]]); + OCMVerify([sut unsubscribeFromNotifications]); + OCMVerify([sut dismissPresentedViewController]); + + [self waitForExpectationsWithTimeout:2 handler:nil]; +} + +/** + After starting a SafariViewController redirect flow, + when the shared URLCallbackHandler is called with an invalid URL, + RedirectContext's completion block and dismiss method should not be called. + */ +- (void)testSafariViewControllerRedirectFlow_callbackHandlerCalledInvalidURL { + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + id sut = OCMPartialMock(context); + + OCMReject([sut unsubscribeFromNotifications]); + OCMReject([sut dismissPresentedViewController]); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + + BOOL(^checker)(id) = ^BOOL(id vc) { + if ([vc isKindOfClass:[SFSafariViewController class]]) { + NSURL *url = [NSURL URLWithString:@"my-app://some_path"]; + XCTAssertNotEqualObjects(url, context.returnURL); + [[STPURLCallbackHandler shared] handleURLCallback:url]; + return YES; + } + return NO; + }; + OCMVerify([mockVC presentViewController:[OCMArg checkWithBlock:checker] + animated:YES + completion:[OCMArg any]]); + + + [self unsubscribeContext:context]; +} + +/** + After starting a SafariViewController redirect flow, + when SafariViewController finishes, RedirectContext's completion block + should be called. + */ +- (void)testSafariViewControllerRedirectFlow_didFinish { + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + + XCTestExpectation *exp = [self expectationWithDescription:@"completion"]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(NSString *sourceID, NSString *clientSecret, NSError *error) { + XCTAssertEqualObjects(sourceID, source.stripeID); + XCTAssertEqualObjects(clientSecret, source.clientSecret); + // because we are manually invoking the dismissal, we report this as a cancelation + XCTAssertEqualObjects(error.domain, [STPError stripeDomain]); + XCTAssertEqual(error.code, STPCancellationError); + [exp fulfill]; + }]; + id sut = OCMPartialMock(context); + + // dismiss should not be called – SafariVC dismisses itself when Done is tapped + OCMReject([sut dismissPresentedViewController]); + + OCMStub([sut handleRedirectCompletionWithError:[OCMArg any] shouldDismissViewController:NO]).andForwardToRealObject().andDo(^(__unused NSInvocation *invocation) { + [context safariViewControllerDidCompleteDismissal:[[SFSafariViewController alloc] initWithURL:[NSURL URLWithString:@"https://www.stripe.com"]]]; + }); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + + BOOL(^checker)(id) = ^BOOL(id vc) { + if ([vc isKindOfClass:[SFSafariViewController class]]) { + SFSafariViewController *sfvc = (SFSafariViewController *)vc; + [sfvc.delegate safariViewControllerDidFinish:sfvc]; + return YES; + } + return NO; + }; + OCMVerify([mockVC presentViewController:[OCMArg checkWithBlock:checker] + animated:YES + completion:[OCMArg any]]); + OCMVerify([sut unsubscribeFromNotifications]); + [self waitForExpectationsWithTimeout:2 handler:nil]; +} + +/** + After starting a SafariViewController redirect flow, + when SafariViewController fails to load the initial page (on iOS 11+ & without redirects), + RedirectContext's completion block and dismiss method should be called. + */ +- (void)testSafariViewControllerRedirectFlow_failedInitialLoad_iOS11Plus { + + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + XCTestExpectation *exp = [self expectationWithDescription:@"completion"]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(NSString *sourceID, NSString *clientSecret, NSError *error) { + XCTAssertEqualObjects(sourceID, source.stripeID); + XCTAssertEqualObjects(clientSecret, source.clientSecret); + NSError *expectedError = [NSError stp_genericConnectionError]; + XCTAssertEqualObjects(error, expectedError); + [exp fulfill]; + }]; + id sut = OCMPartialMock(context); + + OCMStub([sut handleRedirectCompletionWithError:[OCMArg any] shouldDismissViewController:YES]).andForwardToRealObject().andDo(^(__unused NSInvocation *invocation) { + [context safariViewControllerDidCompleteDismissal:[[SFSafariViewController alloc] initWithURL:[NSURL URLWithString:@"https://www.stripe.com"]]]; + }); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + + BOOL(^checker)(id) = ^BOOL(id vc) { + if ([vc isKindOfClass:[SFSafariViewController class]]) { + SFSafariViewController *sfvc = (SFSafariViewController *)vc; + [sfvc.delegate safariViewController:sfvc didCompleteInitialLoad:NO]; + return YES; + } + return NO; + }; + OCMVerify([mockVC presentViewController:[OCMArg checkWithBlock:checker] + animated:YES + completion:[OCMArg any]]); + + OCMVerify([sut unsubscribeFromNotifications]); + OCMVerify([sut dismissPresentedViewController]); + + [self waitForExpectationsWithTimeout:2 handler:nil]; +} + +/** + After starting a SafariViewController redirect flow, + when SafariViewController fails to load the initial page (on iOS 11+ after redirecting to non-Stripe page), + RedirectContext's completion block should not be called (SFVC keeps loading) + */ + +- (void)testSafariViewControllerRedirectFlow_failedInitialLoadAfterRedirect_iOS11Plus { + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + id sut = OCMPartialMock(context); + + OCMReject([sut unsubscribeFromNotifications]); + OCMReject([sut dismissPresentedViewController]); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + + BOOL(^checker)(id) = ^BOOL(id vc) { + if ([vc isKindOfClass:[SFSafariViewController class]]) { + SFSafariViewController *sfvc = (SFSafariViewController *)vc; + // before initial load is done, SFVC was redirected to a non-stripe.com domain + [sfvc.delegate safariViewController:sfvc + initialLoadDidRedirectToURL:[NSURL URLWithString:@"https://girogate.de"]]; + // Tell the delegate that the initial load failed. + // on iOS 11, with the redirect, this is a no-op + [sfvc.delegate safariViewController:sfvc didCompleteInitialLoad:NO]; + return YES; + } + return NO; + }; + OCMVerify([mockVC presentViewController:[OCMArg checkWithBlock:checker] + animated:YES + completion:[OCMArg any]]); + + [self unsubscribeContext:context]; +} + +/** + After starting a SafariViewController redirect flow, + when the RedirectContext is cancelled, its dismiss method should be called. + */ +- (void)testSafariViewControllerRedirectFlow_cancel { + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + id sut = OCMPartialMock(context); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + [sut cancel]; + + OCMVerify([mockVC presentViewController:[OCMArg isKindOfClass:[SFSafariViewController class]] + animated:YES + completion:[OCMArg any]]); + OCMVerify([sut unsubscribeFromNotifications]); + OCMVerify([sut dismissPresentedViewController]); +} + +/** + After starting a SafariViewController redirect flow, + if no action is taken, nothing should be called. + */ +- (void)testSafariViewControllerRedirectFlow_noAction { + id mockVC = OCMClassMock([UIViewController class]); + STPSource *source = [STPFixtures iDEALSource]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + id sut = OCMPartialMock(context); + + OCMReject([sut unsubscribeFromNotifications]); + OCMReject([sut dismissPresentedViewController]); + + [sut startSafariViewControllerRedirectFlowFromViewController:mockVC]; + + OCMVerify([mockVC presentViewController:[OCMArg isKindOfClass:[SFSafariViewController class]] + animated:YES + completion:[OCMArg any]]); + + [self unsubscribeContext:context]; +} + +/** + After starting a Safari app redirect flow, + when a DidBecomeActive notification is posted, RedirectContext's completion + block and dismiss method should be called. + */ +- (void)testSafariAppRedirectFlow_activeNotification { + __block id sut; + + STPSource *source = [STPFixtures iDEALSource]; + XCTestExpectation *exp = [self expectationWithDescription:@"completion"]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(NSString *sourceID, NSString *clientSecret, NSError *error) { + XCTAssertEqualObjects(sourceID, source.stripeID); + XCTAssertEqualObjects(clientSecret, source.clientSecret); + XCTAssertNil(error); + + OCMVerify([sut unsubscribeFromNotifications]); + + [exp fulfill]; + }]; + sut = OCMPartialMock(context); + + [sut startSafariAppRedirectFlow]; + [[NSNotificationCenter defaultCenter] postNotificationName:UIApplicationDidBecomeActiveNotification object:nil]; + + [self waitForExpectationsWithTimeout:2 handler:nil]; +} + +/** + After starting a Safari app redirect flow, + if no notification is posted, nothing should be called. + */ +- (void)testSafariAppRedirectFlow_noNotification { + STPSource *source = [STPFixtures iDEALSource]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + id sut = OCMPartialMock(context); + + OCMReject([sut unsubscribeFromNotifications]); + OCMReject([sut dismissPresentedViewController]); + + [sut startSafariAppRedirectFlow]; + + [self unsubscribeContext:context]; +} + +/** + If a source type that supports native redirect is used and it contains a native + url, an app to app redirect should attempt to be initiated. + */ +- (void)testNativeRedirectSupportingSourceFlow_validNativeURL { + STPSource *source = [STPFixtures alipaySourceWithNativeURL]; + NSURL *sourceURL = [NSURL URLWithString:source.details[@"native_url"]]; + + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source + completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + + XCTAssertNotNil(context.nativeRedirectURL); + XCTAssertEqualObjects(context.nativeRedirectURL, sourceURL); + + id sut = OCMPartialMock(context); + + id applicationMock = OCMClassMock([UIApplication class]); + OCMStub([applicationMock sharedApplication]).andReturn(applicationMock); + OCMStub([applicationMock openURL:[OCMArg any] + options:[OCMArg any] + completionHandler:([OCMArg invokeBlockWithArgs:@YES, nil])]); + + OCMReject([sut startSafariViewControllerRedirectFlowFromViewController:[OCMArg any]]); + OCMReject([sut startSafariAppRedirectFlow]); + + id mockVC = OCMClassMock([UIViewController class]); + [sut startRedirectFlowFromViewController:mockVC]; + + OCMVerify([applicationMock openURL:[OCMArg isEqual:sourceURL] + options:[OCMArg isEqual:@{}] + completionHandler:[OCMArg isNotNil]]); + + [sut unsubscribeFromNotifications]; +} + +/** + If a source type that supports native redirect is used and it does not + contain a native url, standard web view redirect should be attempted + */ +- (void)testNativeRedirectSupportingSourceFlow_invalidNativeURL { + STPSource *source = [STPFixtures alipaySource]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source + completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + XCTAssertNil(context.nativeRedirectURL); + + id sut = OCMPartialMock(context); + + id applicationMock = OCMClassMock([UIApplication class]); + OCMStub([applicationMock sharedApplication]).andReturn(applicationMock); + + OCMReject([applicationMock openURL:[OCMArg any] + options:[OCMArg any] + completionHandler:[OCMArg any]]); + + id mockVC = OCMClassMock([UIViewController class]); + [sut startRedirectFlowFromViewController:mockVC]; + + OCMVerify([sut startSafariViewControllerRedirectFlowFromViewController:[OCMArg isEqual:mockVC]]); + + OCMVerify([mockVC presentViewController:[OCMArg isKindOfClass:[SFSafariViewController class]] + animated:YES + completion:[OCMArg isNil]]); + XCTestExpectation *expectation = [self expectationWithDescription:@"Waiting 100ms for SafariServices"]; + + // Hack: Wait ~100ms to call sut back before unsubscribing from notifications. Otherwise the Safari thread doesn't get the unsubscribe request in time and calls the deallocated sut, crashing the app. + dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ + [expectation fulfill]; + }); + + [self waitForExpectationsWithTimeout:1 handler:nil]; + [sut unsubscribeFromNotifications]; +} + +#pragma mark - WeChat Pay + +/** + If a WeChat source type is used, we should attempt an app redirect. + */ +- (void)testWeChatPaySource_appRedirectSucceeds { + STPSource *source = [STPFixtures weChatPaySource]; + NSURL *sourceURL = [NSURL URLWithString:source.weChatPayDetails.weChatAppURL]; + + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source + completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTFail(@"completion called"); + }]; + + XCTAssertNotNil(context.nativeRedirectURL); + XCTAssertEqualObjects(context.nativeRedirectURL, sourceURL); + XCTAssertNil(context.redirectURL); + XCTAssertNotNil(context.returnURL); + + id sut = OCMPartialMock(context); + + id applicationMock = OCMClassMock([UIApplication class]); + OCMStub([applicationMock sharedApplication]).andReturn(applicationMock); + OCMStub([applicationMock openURL:[OCMArg any] + options:[OCMArg any] + completionHandler:([OCMArg invokeBlockWithArgs:@YES, nil])]); + + OCMReject([sut startSafariViewControllerRedirectFlowFromViewController:[OCMArg any]]); + OCMReject([sut startSafariAppRedirectFlow]); + + id mockVC = OCMClassMock([UIViewController class]); + [sut startRedirectFlowFromViewController:mockVC]; + + OCMVerify([applicationMock openURL:[OCMArg isEqual:sourceURL] + options:[OCMArg isEqual:@{}] + completionHandler:[OCMArg isNotNil]]); + XCTestExpectation *expectation = [self expectationWithDescription:@"Waiting 100ms for SafariServices"]; + + // Hack: Wait ~100ms to call sut back before unsubscribing from notifications. Otherwise the Safari thread doesn't get the unsubscribe request in time and calls the deallocated sut, crashing the app. + dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ + [expectation fulfill]; + }); + + [self waitForExpectationsWithTimeout:1 handler:nil]; + [sut unsubscribeFromNotifications]; +} + +/** + If a WeChat source type is used, we should attempt an app redirect. + If app redirect fails, expect an error. + */ +- (void)testWeChatPaySource_appRedirectFails { + STPSource *source = [STPFixtures weChatPaySource]; + NSURL *sourceURL = [NSURL URLWithString:source.weChatPayDetails.weChatAppURL]; + + XCTestExpectation *expectation = [self expectationWithDescription:@"Completion block called"]; + STPRedirectContext *context = [[STPRedirectContext alloc] initWithSource:source + completion:^(__unused NSString *sourceID, __unused NSString *clientSecret, __unused NSError *error) { + XCTAssertNotNil(error); + XCTAssertEqualObjects(error.domain, STPRedirectContext.STPRedirectContextErrorDomain); + XCTAssertEqual(error.code, STPRedirectContextAppRedirectError); + [expectation fulfill]; + }]; + + XCTAssertNotNil(context.nativeRedirectURL); + XCTAssertEqualObjects(context.nativeRedirectURL, sourceURL); + XCTAssertNil(context.redirectURL); + XCTAssertNotNil(context.returnURL); + + id sut = OCMPartialMock(context); + + id applicationMock = OCMClassMock([UIApplication class]); + OCMStub([applicationMock sharedApplication]).andReturn(applicationMock); + OCMStub([applicationMock openURL:[OCMArg any] + options:[OCMArg any] + completionHandler:([OCMArg invokeBlockWithArgs:@NO, nil])]); + + OCMReject([sut startSafariViewControllerRedirectFlowFromViewController:[OCMArg any]]); + OCMReject([sut startSafariAppRedirectFlow]); + + id mockVC = OCMClassMock([UIViewController class]); + [sut startRedirectFlowFromViewController:mockVC]; + + OCMVerify([applicationMock openURL:[OCMArg isEqual:sourceURL] + options:[OCMArg isEqual:@{}] + completionHandler:[OCMArg isNotNil]]); + XCTestExpectation *safariWaitExpectation = [self expectationWithDescription:@"Waiting 100ms for SafariServices"]; + + // Hack: Wait ~100ms to call sut back before unsubscribing from notifications. Otherwise the Safari thread doesn't get the unsubscribe request in time and calls the deallocated sut, crashing the app. + dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ + [safariWaitExpectation fulfill]; + }); + [self waitForExpectationsWithTimeout:10 handler:nil]; +} + + +@end diff --git a/Stripe/StripeiOSTests/STPSTPViewWithSeparatorSnapshotTests.m b/Stripe/StripeiOSTests/STPSTPViewWithSeparatorSnapshotTests.m new file mode 100644 index 00000000..a86a3de2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSTPViewWithSeparatorSnapshotTests.m @@ -0,0 +1,38 @@ +// +// STPSTPViewWithSeparatorSnapshotTests.m +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/13/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import "STPTestUtils.h" + +@import iOSSnapshotTestCaseCore; + +@interface STPSTPViewWithSeparatorSnapshotTests : FBSnapshotTestCase + +@end + +@implementation STPSTPViewWithSeparatorSnapshotTests + +//- (void)setUp { +// [super setUp]; +// +// self.recordMode = YES; +//} + +- (void)testDefaultAppearance { + STPViewWithSeparator *view = [[STPViewWithSeparator alloc] initWithFrame:CGRectMake(0.f, 0.f, 320.f, 44.f)]; + view.backgroundColor = [UIColor whiteColor]; + STPSnapshotVerifyView(view, @"STPViewWithSeparator.defaultAppearance"); +} + +- (void)testHiddenTopSeparator { + STPViewWithSeparator *view = [[STPViewWithSeparator alloc] initWithFrame:CGRectMake(0.f, 0.f, 320.f, 44.f)]; + view.backgroundColor = [UIColor whiteColor]; + view.topSeparatorHidden = YES; + STPSnapshotVerifyView(view, @"STPViewWithSeparator.hiddenTopSeparator"); +} + +@end diff --git a/Stripe/StripeiOSTests/STPSetupIntentConfirmParamsTest.swift b/Stripe/StripeiOSTests/STPSetupIntentConfirmParamsTest.swift new file mode 100644 index 00000000..788be811 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSetupIntentConfirmParamsTest.swift @@ -0,0 +1,156 @@ +// +// STPSetupIntentConfirmParamsTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 7/15/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPSetupIntentConfirmParamsTest: XCTestCase { + func testInit() { + for params in [ + STPSetupIntentConfirmParams(clientSecret: "secret"), + STPSetupIntentConfirmParams(), + STPSetupIntentConfirmParams(), + ] { + XCTAssertNotNil(params) + XCTAssertNotNil(params.clientSecret) + XCTAssertNotNil(params.additionalAPIParameters) + XCTAssertEqual(params.additionalAPIParameters.count, 0) + XCTAssertNil(params.paymentMethodID) + XCTAssertNil(params.returnURL) + XCTAssertNil(params.useStripeSDK) + XCTAssertNil(params.mandateData) + } + } + + func testDescription() { + let params = STPSetupIntentConfirmParams() + XCTAssertNotNil(params.description) + } + + func testDefaultMandateData() { + let params = STPSetupIntentConfirmParams() + + // no configuration should have no mandateData + XCTAssertNil(params.mandateData) + + params.paymentMethodParams = STPPaymentMethodParams() + + params.paymentMethodParams?.rawTypeString = "card" + // card type should have no default mandateData + XCTAssertNil(params.mandateData) + + for type in ["sepa_debit", "au_becs_debit", "bacs_debit"] { + params.mandateData = nil + params.paymentMethodParams?.rawTypeString = type + // Mandate-required type should have mandateData + XCTAssertNotNil(params.mandateData) + XCTAssertEqual( + params.mandateData?.customerAcceptance.onlineParams?.inferFromClient, + NSNumber(value: true) + ) + + let customerAcceptance = STPMandateCustomerAcceptanceParams( + type: .offline, + onlineParams: nil + ) + params.mandateData = STPMandateDataParams(customerAcceptance: customerAcceptance!) + // Default behavior should not override custom setting + XCTAssertNotNil(params.mandateData) + XCTAssertNil(params.mandateData?.customerAcceptance.onlineParams) + } + } + + // MARK: STPFormEncodable Tests + func testRootObjectName() { + XCTAssertNil(STPSetupIntentConfirmParams.rootObjectName()) + } + + func testPropertyNamesToFormFieldNamesMapping() { + let params = STPSetupIntentConfirmParams() + + let mapping = STPSetupIntentConfirmParams.propertyNamesToFormFieldNamesMapping() + + for propertyName in mapping.keys { + XCTAssertFalse(propertyName.contains(":")) + XCTAssert(params.responds(to: NSSelectorFromString(propertyName))) + } + + for formFieldName in mapping.values { + XCTAssert(formFieldName.count > 0) + } + + XCTAssertEqual(mapping.values.count, Set(mapping.values).count) + } + + func testCopy() { + let params = STPSetupIntentConfirmParams(clientSecret: "test_client_secret") + params.paymentMethodParams = STPPaymentMethodParams() + params.paymentMethodID = "test_payment_method_id" + params.returnURL = "fake://testing_only" + params.useStripeSDK = NSNumber(value: true) + params.mandateData = STPMandateDataParams( + customerAcceptance: STPMandateCustomerAcceptanceParams( + type: .offline, + onlineParams: nil + )! + ) + params.additionalAPIParameters = [ + "other_param": "other_value" + ] + + let paramsCopy = params.copy() as! STPSetupIntentConfirmParams + XCTAssertEqual(params.clientSecret, paramsCopy.clientSecret) + XCTAssertEqual(params.paymentMethodID, paramsCopy.paymentMethodID) + + // assert equal, not equal objects, because this is a shallow copy + XCTAssertEqual(params.paymentMethodParams, paramsCopy.paymentMethodParams) + XCTAssertEqual(params.mandateData, paramsCopy.mandateData) + + XCTAssertEqual(params.returnURL, paramsCopy.returnURL) + XCTAssertEqual(params.useStripeSDK, paramsCopy.useStripeSDK) + XCTAssertEqual( + params.additionalAPIParameters as NSDictionary, + paramsCopy.additionalAPIParameters as NSDictionary + ) + + } + + func testClientSecretValidation() { + XCTAssertFalse( + STPSetupIntentConfirmParams.isClientSecretValid("seti_12345"), + "'seti_12345' is not a valid client secret." + ) + XCTAssertFalse( + STPSetupIntentConfirmParams.isClientSecretValid("seti_12345_secret_"), + "'seti_12345_secret_' is not a valid client secret." + ) + XCTAssertFalse( + STPSetupIntentConfirmParams.isClientSecretValid( + "seti_a1b2c3_secret_x7y8z9seti_a1b2c3_secret_x7y8z9" + ), + "'seti_a1b2c3_secret_x7y8z9seti_a1b2c3_secret_x7y8z9' is not a valid client secret." + ) + XCTAssertFalse( + STPSetupIntentConfirmParams.isClientSecretValid("pi_a1b2c3_secret_x7y8z9"), + "'pi_a1b2c3_secret_x7y8z9' is not a valid client secret." + ) + + XCTAssertTrue( + STPSetupIntentConfirmParams.isClientSecretValid("seti_a1b2c3_secret_x7y8z9"), + "'seti_a1b2c3_secret_x7y8z9' is a valid client secret." + ) + XCTAssertTrue( + STPSetupIntentConfirmParams.isClientSecretValid( + "seti_1Eq5kyGMT9dGPIDGxiSp4cce_secret_FKlHb3yTI0YZWe4iqghS8ZXqwwMoMmy" + ), + "'seti_1Eq5kyGMT9dGPIDGxiSp4cce_secret_FKlHb3yTI0YZWe4iqghS8ZXqwwMoMmy' is a valid client secret." + ) + } +} diff --git a/Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.m b/Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.m new file mode 100644 index 00000000..84485abd --- /dev/null +++ b/Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.m @@ -0,0 +1,154 @@ +// +// STPSetupIntentFunctionalTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/28/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import +@import StripeCoreTestUtils; +@import StripeCore; +#import "STPTestingAPIClient.h" + +@import Stripe; + +@interface STPSetupIntentFunctionalTest : XCTestCase + +@end + +@implementation STPSetupIntentFunctionalTest + +- (void)testCreateSetupIntentWithTestingServer { + XCTestExpectation *expectation = [self expectationWithDescription:@"SetupIntent create."]; + [[STPTestingAPIClient sharedClient] createSetupIntentWithParams:nil + completion:^(NSString * _Nullable clientSecret, NSError * _Nullable error) { + XCTAssertNotNil(clientSecret); + XCTAssertNil(error); + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testRetrieveSetupIntentSucceeds { + // Tests retrieving a previously created SetupIntent succeeds + NSString *setupIntentClientSecret = @"seti_1GGCuIFY0qyl6XeWVfbQK6b3_secret_GnoX2tzX2JpvxsrcykRSVna2lrYLKew"; + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Setup Intent retrieve"]; + + [client retrieveSetupIntentWithClientSecret:setupIntentClientSecret + completion:^(STPSetupIntent *setupIntent, NSError *error) { + XCTAssertNil(error); + + XCTAssertNotNil(setupIntent); + XCTAssertEqualObjects(setupIntent.stripeID, @"seti_1GGCuIFY0qyl6XeWVfbQK6b3"); + XCTAssertEqualObjects(setupIntent.clientSecret, setupIntentClientSecret); + XCTAssertEqualObjects(setupIntent.created, [NSDate dateWithTimeIntervalSince1970:1582673622]); + XCTAssertNil(setupIntent.customerID); + XCTAssertNil(setupIntent.stripeDescription); + XCTAssertFalse(setupIntent.livemode); + XCTAssertNil(setupIntent.nextAction); + XCTAssertNil(setupIntent.paymentMethodID); + XCTAssertEqualObjects(setupIntent.paymentMethodTypes, @[@(STPPaymentMethodTypeCard)]); + XCTAssertEqual(setupIntent.status, STPSetupIntentStatusRequiresPaymentMethod); + XCTAssertEqual(setupIntent.usage, STPSetupIntentUsageOffSession); + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testConfirmSetupIntentSucceeds { + + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create SetupIntent."]; + [[STPTestingAPIClient sharedClient] createSetupIntentWithParams:nil completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"SetupIntent confirm"]; + STPSetupIntentConfirmParams *params = [[STPSetupIntentConfirmParams alloc] initWithClientSecret:clientSecret]; + params.returnURL = @"example-app-scheme://authorized"; + // Confirm using a card requiring 3DS1 authentication (ie requires next steps) + params.paymentMethodID = @"pm_card_authenticationRequired"; + [client confirmSetupIntentWithParams:params + completion:^(STPSetupIntent *setupIntent, NSError *error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(setupIntent); + XCTAssertEqualObjects(setupIntent.stripeID, [STPSetupIntent idFromClientSecret:params.clientSecret]); + XCTAssertEqualObjects(setupIntent.clientSecret, clientSecret); + XCTAssertFalse(setupIntent.livemode); + + XCTAssertEqual(setupIntent.status, STPSetupIntentStatusRequiresAction); + XCTAssertNotNil(setupIntent.nextAction); + XCTAssertEqual(setupIntent.nextAction.type, STPIntentActionTypeRedirectToURL); + XCTAssertEqualObjects(setupIntent.nextAction.redirectToURL.returnURL, [NSURL URLWithString:@"example-app-scheme://authorized"]); + XCTAssertNotNil(setupIntent.paymentMethodID); + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +#pragma mark - AU BECS Debit + +- (void)testConfirmAUBECSDebitSetupIntent { + + __block NSString *clientSecret = nil; + XCTestExpectation *createExpectation = [self expectationWithDescription:@"Create PaymentIntent."]; + [[STPTestingAPIClient sharedClient] createSetupIntentWithParams:@{ + @"payment_method_types": @[@"au_becs_debit"], + } + account:@"au" + completion:^(NSString * _Nullable createdClientSecret, NSError * _Nullable creationError) { + XCTAssertNotNil(createdClientSecret); + XCTAssertNil(creationError); + [createExpectation fulfill]; + clientSecret = [createdClientSecret copy]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; + XCTAssertNotNil(clientSecret); + + + STPPaymentMethodAUBECSDebitParams *becsParams = [STPPaymentMethodAUBECSDebitParams new]; + becsParams.bsbNumber = @"000000"; // Stripe test bank + becsParams.accountNumber = @"000123456"; // test account + + STPPaymentMethodBillingDetails *billingDetails = [STPPaymentMethodBillingDetails new]; + billingDetails.name = @"Jenny Rosen"; + billingDetails.email = @"jrosen@example.com"; + + + + STPPaymentMethodParams *params = [STPPaymentMethodParams paramsWithAUBECSDebit:becsParams + billingDetails:billingDetails + metadata:@{@"test_key": @"test_value"}]; + + + STPSetupIntentConfirmParams *setupIntentParams = [[STPSetupIntentConfirmParams alloc] initWithClientSecret:clientSecret]; + setupIntentParams.paymentMethodParams = params; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingAUPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Setup Intent confirm"]; + + [client confirmSetupIntentWithParams:setupIntentParams + completion:^(STPSetupIntent * _Nullable setupIntent, NSError * _Nullable error) { + XCTAssertNil(error, @"With valid key + secret, should be able to confirm the intent"); + + XCTAssertNotNil(setupIntent); + XCTAssertEqualObjects(setupIntent.stripeID, [STPSetupIntent idFromClientSecret:setupIntentParams.clientSecret]); + XCTAssertNotNil(setupIntent.paymentMethodID); + XCTAssertEqual(setupIntent.status, STPSetupIntentStatusSucceeded); + + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.swift b/Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.swift new file mode 100644 index 00000000..b350ed2a --- /dev/null +++ b/Stripe/StripeiOSTests/STPSetupIntentFunctionalTest.swift @@ -0,0 +1,126 @@ +// +// STPSetupIntentFunctionalTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 3/2/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +@_spi(STP) import StripeCore +import StripeCoreTestUtils +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPSetupIntentFunctionalTestSwift: XCTestCase { + + // MARK: - US Bank Account + func createAndConfirmSetupIntentWithUSBankAccount(completion: @escaping (String?) -> Void) { + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + var clientSecret: String? + let createSIExpectation = expectation(description: "Create SetupIntent") + STPTestingAPIClient.shared().createSetupIntent( + withParams: ["payment_method_types": ["us_bank_account"]], + account: nil + ) { intentClientSecret, error in + XCTAssertNil(error) + XCTAssertNotNil(intentClientSecret) + clientSecret = intentClientSecret + createSIExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + guard let clientSecret = clientSecret else { + XCTFail("Failed to create SetupIntent") + return + } + + let usBankAccountParams = STPPaymentMethodUSBankAccountParams() + usBankAccountParams.accountType = .checking + usBankAccountParams.accountHolderType = .individual + usBankAccountParams.accountNumber = "000123456789" + usBankAccountParams.routingNumber = "110000000" + + let billingDetails = STPPaymentMethodBillingDetails() + billingDetails.name = "iOS CI Tester" + billingDetails.email = "tester@example.com" + + let paymentMethodParams = STPPaymentMethodParams( + usBankAccount: usBankAccountParams, + billingDetails: billingDetails, + metadata: nil + ) + + let setupIntentParams = STPSetupIntentConfirmParams(clientSecret: clientSecret) + setupIntentParams.paymentMethodParams = paymentMethodParams + + let confirmSIExpectation = expectation(description: "Confirm SetupIntent") + client.confirmSetupIntent(with: setupIntentParams, expand: ["payment_method"]) { + setupIntent, + error in + XCTAssertNil(error) + XCTAssertNotNil(setupIntent) + XCTAssertNotNil(setupIntent?.paymentMethod) + XCTAssertNotNil(setupIntent?.paymentMethod?.usBankAccount) + XCTAssertEqual(setupIntent?.paymentMethod?.usBankAccount?.last4, "6789") + XCTAssertEqual(setupIntent?.status, .requiresAction) + XCTAssertEqual(setupIntent?.nextAction?.type, .verifyWithMicrodeposits) + confirmSIExpectation.fulfill() + } + + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + completion(clientSecret) + } + + func testConfirmSetupIntentWithUSBankAccount_verifyWithAmounts() { + createAndConfirmSetupIntentWithUSBankAccount { [self] clientSecret in + guard let clientSecret = clientSecret else { + XCTFail("Failed to create SetupIntent") + return + } + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let verificationExpectation = expectation(description: "Verify with microdeposits") + client.verifySetupIntentWithMicrodeposits( + clientSecret: clientSecret, + firstAmount: 32, + secondAmount: 45 + ) { setupIntent, error in + XCTAssertNil(error) + XCTAssertNotNil(setupIntent) + XCTAssertEqual(setupIntent?.status, .succeeded) + verificationExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + } + + func testConfirmSetupIntentWithUSBankAccount_verifyWithDescriptorCode() { + createAndConfirmSetupIntentWithUSBankAccount { [self] clientSecret in + guard let clientSecret = clientSecret else { + XCTFail("Failed to create SetupIntent") + return + } + + let client = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + + let verificationExpectation = expectation(description: "Verify with microdeposits") + client.verifySetupIntentWithMicrodeposits( + clientSecret: clientSecret, + descriptorCode: "SM11AA" + ) { setupIntent, error in + XCTAssertNil(error) + XCTAssertNotNil(setupIntent) + XCTAssertEqual(setupIntent?.status, .succeeded) + verificationExpectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout) + } + } + +} diff --git a/Stripe/StripeiOSTests/STPSetupIntentLastSetupErrorTest.m b/Stripe/StripeiOSTests/STPSetupIntentLastSetupErrorTest.m new file mode 100644 index 00000000..e509ab15 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSetupIntentLastSetupErrorTest.m @@ -0,0 +1,47 @@ +// +// STPSetupIntentLastSetupErrorTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 8/9/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +@import Stripe; +#import "STPTestUtils.h" +#import "STPFixtures.h" + +@interface STPSetupIntentLastSetupError (Testing) ++ (STPSetupIntentLastSetupErrorType)typeFromString:(NSString *)string; +@end + +@interface STPSetupIntentLastSetupErrorTest : XCTestCase + +@end + +@implementation STPSetupIntentLastSetupErrorTest + +- (void)testTypeFromString { + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"api_connection_error"], STPSetupIntentLastSetupErrorTypeAPIConnection); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"API_CONNECTION_ERROR"], STPSetupIntentLastSetupErrorTypeAPIConnection); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"api_error"], STPSetupIntentLastSetupErrorTypeAPI); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"API_ERROR"], STPSetupIntentLastSetupErrorTypeAPI); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"authentication_error"], STPSetupIntentLastSetupErrorTypeAuthentication); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"AUTHENTICATION_ERROR"], STPSetupIntentLastSetupErrorTypeAuthentication); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"card_error"], STPSetupIntentLastSetupErrorTypeCard); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"CARD_ERROR"], STPSetupIntentLastSetupErrorTypeCard); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"idempotency_error"], STPSetupIntentLastSetupErrorTypeIdempotency); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"IDEMPOTENCY_ERROR"], STPSetupIntentLastSetupErrorTypeIdempotency); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"invalid_request_error"], STPSetupIntentLastSetupErrorTypeInvalidRequest); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"INVALID_REQUEST_ERROR"], STPSetupIntentLastSetupErrorTypeInvalidRequest); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"rate_limit_error"], STPSetupIntentLastSetupErrorTypeRateLimit); + XCTAssertEqual([STPSetupIntentLastSetupError typeFromString:@"RATE_LIMIT_ERROR"], STPSetupIntentLastSetupErrorTypeRateLimit); +} + +#pragma mark - STPAPIResponseDecodable Tests + +// STPSetupIntentLastError is a sub-object of STPSetupIntent, see STPSetupIntentTest + + +@end diff --git a/Stripe/StripeiOSTests/STPSetupIntentTest.m b/Stripe/StripeiOSTests/STPSetupIntentTest.m new file mode 100644 index 00000000..ed40eb17 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSetupIntentTest.m @@ -0,0 +1,108 @@ +// +// STPSetupIntentTest.m +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/27/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +#import + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPSetupIntentTest : XCTestCase + +@end + +@implementation STPSetupIntentTest + +#pragma mark - Description Tests + +- (void)testDescription { + STPSetupIntent *setupIntent = [STPFixtures setupIntent]; + + XCTAssertNotNil(setupIntent); + NSString *desc = setupIntent.description; + XCTAssertTrue([desc containsString:NSStringFromClass([setupIntent class])]); + XCTAssertGreaterThan(desc.length, 500UL, @"Custom description should be long"); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSDictionary *fullJson = [STPTestUtils jsonNamed:STPTestJSONSetupIntent]; + + XCTAssertNotNil([STPSetupIntent decodedObjectFromAPIResponse:fullJson], @"can decode with full json"); + + NSArray *requiredFields = @[ + @"id", + @"client_secret", + @"livemode", + @"status", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *partialJson = [fullJson mutableCopy]; + + XCTAssertNotNil(partialJson[field], @"json should contain %@", field); + [partialJson removeObjectForKey:field]; + + XCTAssertNil([STPSetupIntent decodedObjectFromAPIResponse:partialJson], @"should not decode without %@", field); + } +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *setupIntentJson = [STPTestUtils jsonNamed:@"SetupIntent"]; + NSArray *orderedPaymentJson = @[@"card", @"ideal", @"sepa_debit"]; + NSDictionary *setupIntentResponse = @{@"setup_intent": setupIntentJson, + @"ordered_payment_method_types": orderedPaymentJson + }; + NSArray *unactivatedPaymentMethodTypes = @[@"sepa_debit"]; + NSDictionary *response = @{@"payment_method_preference": setupIntentResponse, + @"unactivated_payment_method_types": unactivatedPaymentMethodTypes}; + + STPSetupIntent *setupIntent = [STPSetupIntent decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(setupIntent.stripeID, @"seti_123456789"); + XCTAssertEqualObjects(setupIntent.clientSecret, @"seti_123456789_secret_123456789"); + XCTAssertEqualObjects(setupIntent.created, [NSDate dateWithTimeIntervalSince1970:123456789]); + XCTAssertEqualObjects(setupIntent.customerID, @"cus_123456"); + XCTAssertEqualObjects(setupIntent.paymentMethodID, @"pm_123456"); + XCTAssertEqualObjects(setupIntent.stripeDescription, @"My Sample SetupIntent"); + XCTAssertFalse(setupIntent.livemode); + // nextAction + XCTAssertNotNil(setupIntent.nextAction); + XCTAssertEqual(setupIntent.nextAction.type, STPIntentActionTypeRedirectToURL); + XCTAssertNotNil(setupIntent.nextAction.redirectToURL); + XCTAssertNotNil(setupIntent.nextAction.redirectToURL.url); + NSURL *returnURL = setupIntent.nextAction.redirectToURL.returnURL; + XCTAssertNotNil(returnURL); + XCTAssertEqualObjects(returnURL, [NSURL URLWithString:@"payments-example://stripe-redirect"]); + NSURL *url = setupIntent.nextAction.redirectToURL.url; + XCTAssertNotNil(url); + + XCTAssertEqualObjects(url, [NSURL URLWithString:@"https://hooks.stripe.com/redirect/authenticate/src_1Cl1AeIl4IdHmuTb1L7x083A?client_secret=src_client_secret_DBNwUe9qHteqJ8qQBwNWiigk"]); + XCTAssertEqualObjects(setupIntent.paymentMethodID, @"pm_123456"); + XCTAssertEqual(setupIntent.status, STPSetupIntentStatusRequiresAction); + XCTAssertEqual(setupIntent.usage, STPSetupIntentUsageOffSession); + + XCTAssertEqualObjects(setupIntent.paymentMethodTypes, @[@(STPPaymentMethodTypeCard)]); + + // lastSetupError + + XCTAssertNotNil(setupIntent.lastSetupError); + XCTAssertEqualObjects(setupIntent.lastSetupError.code, @"setup_intent_authentication_failure"); + XCTAssertEqualObjects(setupIntent.lastSetupError.docURL, @"https://stripe.com/docs/error-codes#setup-intent-authentication-failure"); + XCTAssertEqualObjects(setupIntent.lastSetupError.message, @"The latest attempt to set up the payment method has failed because authentication failed."); + XCTAssertNotNil(setupIntent.lastSetupError.paymentMethod); + XCTAssertEqual(setupIntent.lastSetupError.type, STPSetupIntentLastSetupErrorTypeInvalidRequest); + + // Hack to test internal variable, should be re-written in Swift with @testable + XCTAssertTrue([setupIntent.description containsString:@"unactivatedPaymentMethodTypes = [sepa_debit]"]); + + XCTAssertNotEqual(setupIntent.allResponseFields, response, @"should have own copy of fields"); +} + + +@end diff --git a/Stripe/StripeiOSTests/STPShippingAddressViewControllerLocalizationTests.swift b/Stripe/StripeiOSTests/STPShippingAddressViewControllerLocalizationTests.swift new file mode 100644 index 00000000..b071a8e6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPShippingAddressViewControllerLocalizationTests.swift @@ -0,0 +1,116 @@ +// +// STPShippingAddressViewControllerLocalizationTests.swift +// StripeiOS Tests +// +// Created by Ben Guo on 11/3/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPShippingAddressViewControllerLocalizationTests: FBSnapshotTestCase { + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func performSnapshotTest( + forLanguage language: String?, + shippingType: STPShippingType, + contact: Bool + ) { + var identifier = (shippingType == .shipping) ? "shipping" : "delivery" + let config = STPFixtures.paymentConfiguration() + config.companyName = "Test Company" + config.requiredShippingAddressFields = Set([ + .postalAddress, + .emailAddress, + .phoneNumber, + .name, + ]) + if contact { + config.requiredShippingAddressFields = Set([.emailAddress]) + identifier = "contact" + } + config.shippingType = shippingType + + STPLocalizationUtils.overrideLanguage(to: language) + let info = STPUserInformation() + info.billingAddress = STPAddress() + info.billingAddress!.email = "@" // trigger "use billing address" button + + let shippingVC = STPShippingAddressViewController( + configuration: config, + theme: STPTheme.defaultTheme, + currency: nil, + shippingAddress: nil, + selectedShippingMethod: nil, + prefilledInformation: info + ) + + /// This method rejects nil or empty country codes to stop strange looking behavior + /// when scrolling to the top "unset" position in the picker, so put in + /// an invalid country code instead to test seeing the "Country" placeholder + shippingVC.addressViewModel.addressFieldTableViewCountryCode = "INVALID" + + let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: shippingVC)! + + STPSnapshotVerifyView(viewToTest, identifier: identifier) + + STPLocalizationUtils.overrideLanguage(to: nil) + } + + func testGerman() { + performSnapshotTest(forLanguage: "de", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "de", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "de", shippingType: .delivery, contact: false) + } + + func testEnglish() { + performSnapshotTest(forLanguage: "en", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "en", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "en", shippingType: .delivery, contact: false) + } + + func testSpanish() { + performSnapshotTest(forLanguage: "es", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "es", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "es", shippingType: .delivery, contact: false) + } + + func testFrench() { + performSnapshotTest(forLanguage: "fr", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "fr", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "fr", shippingType: .delivery, contact: false) + } + + func testItalian() { + performSnapshotTest(forLanguage: "it", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "it", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "it", shippingType: .delivery, contact: false) + } + + func testJapanese() { + performSnapshotTest(forLanguage: "ja", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "ja", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "ja", shippingType: .delivery, contact: false) + } + + func testDutch() { + performSnapshotTest(forLanguage: "nl", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "nl", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "nl", shippingType: .delivery, contact: false) + } + + func testChinese() { + performSnapshotTest(forLanguage: "zh-Hans", shippingType: .shipping, contact: false) + performSnapshotTest(forLanguage: "zh-Hans", shippingType: .shipping, contact: true) + performSnapshotTest(forLanguage: "zh-Hans", shippingType: .delivery, contact: false) + } +} diff --git a/Stripe/StripeiOSTests/STPShippingAddressViewControllerTest.swift b/Stripe/StripeiOSTests/STPShippingAddressViewControllerTest.swift new file mode 100644 index 00000000..c2c0e3f4 --- /dev/null +++ b/Stripe/StripeiOSTests/STPShippingAddressViewControllerTest.swift @@ -0,0 +1,114 @@ +// +// STPShippingAddressViewControllerTest.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 8/7/18. +// Copyright © 2018 Stripe, Inc. All rights reserved. +// + +import Stripe + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPShippingAddressViewControllerTest: XCTestCase { + func testPrefilledBillingAddress_removeAddress() { + let config = STPFixtures.paymentConfiguration() + config.requiredShippingAddressFields = Set([.postalAddress]) + + let address = STPAddress() + address.name = "John Smith Doe" + address.phone = "8885551212" + address.email = "foo@example.com" + address.line1 = "55 John St" + address.city = "Harare" + address.postalCode = "10002" + // Zimbabwe does not require zip codes, while the default locale for tests (US) does + address.country = "ZW" + // Sanity checks + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) + + let sut = STPShippingAddressViewController( + configuration: config, + theme: STPTheme.defaultTheme, + currency: nil, + shippingAddress: address, + selectedShippingMethod: nil, + prefilledInformation: nil + ) + + XCTAssertNoThrow(sut.loadView()) + XCTAssertNoThrow(sut.viewDidLoad()) + } + + func testPrefilledBillingAddress_addAddressWithLimitedCountries() { + // Zimbabwe does not require zip codes, while the default locale for tests (US) does + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_ZW") as Locale) { + // Sanity checks + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) + let config = STPFixtures.paymentConfiguration() + config.requiredShippingAddressFields = Set([.postalAddress]) + config.availableCountries = Set(["CA", "BT"]) + + let address = STPAddress() + address.name = "John Smith Doe" + address.phone = "8885551212" + address.email = "foo@example.com" + address.line1 = "55 John St" + address.city = "New York" + address.state = "NY" + address.postalCode = "10002" + address.country = "US" + + let sut = STPShippingAddressViewController( + configuration: config, + theme: STPTheme.defaultTheme, + currency: nil, + shippingAddress: address, + selectedShippingMethod: nil, + prefilledInformation: nil + ) + + XCTAssertNoThrow(sut.loadView()) + XCTAssertNoThrow(sut.viewDidLoad()) + } + } + + func testPrefilledBillingAddress_addAddress() { + // Zimbabwe does not require zip codes, while the default locale for tests (US) does + NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_ZW") as Locale) { + // Sanity checks + XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) + XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) + let config = STPFixtures.paymentConfiguration() + config.requiredShippingAddressFields = Set([.postalAddress]) + + let address = STPAddress() + address.name = "John Smith Doe" + address.phone = "8885551212" + address.email = "foo@example.com" + address.line1 = "55 John St" + address.city = "New York" + address.state = "NY" + address.postalCode = "10002" + address.country = "US" + + let sut = STPShippingAddressViewController( + configuration: config, + theme: STPTheme.defaultTheme, + currency: nil, + shippingAddress: address, + selectedShippingMethod: nil, + prefilledInformation: nil + ) + + XCTAssertNoThrow(sut.loadView()) + XCTAssertNoThrow(sut.viewDidLoad()) + } + } +} diff --git a/Stripe/StripeiOSTests/STPShippingMethodsViewControllerLocalizationTests.swift b/Stripe/StripeiOSTests/STPShippingMethodsViewControllerLocalizationTests.swift new file mode 100644 index 00000000..ccdf3cd2 --- /dev/null +++ b/Stripe/StripeiOSTests/STPShippingMethodsViewControllerLocalizationTests.swift @@ -0,0 +1,80 @@ +// +// STPShippingMethodsViewControllerLocalizationTests.swift +// StripeiOS Tests +// +// Created by Ben Guo on 11/3/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPShippingMethodsViewControllerLocalizationTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func performSnapshotTest(forLanguage language: String?) { + STPLocalizationUtils.overrideLanguage(to: language) + + let method1 = PKShippingMethod() + method1.label = "UPS Ground" + method1.detail = "Arrives in 3-5 days" + method1.amount = NSDecimalNumber(string: "0.00") + method1.identifier = "ups_ground" + let method2 = PKShippingMethod() + method2.label = "FedEx" + method2.detail = "Arrives tomorrow" + method2.amount = NSDecimalNumber(string: "5.99") + method2.identifier = "fedex" + + let shippingVC = STPShippingMethodsViewController( + shippingMethods: [method1, method2], + selectedShippingMethod: method1, + currency: "usd", + theme: STPTheme.defaultTheme + ) + let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: shippingVC)! + STPSnapshotVerifyView(viewToTest, identifier: nil) + STPLocalizationUtils.overrideLanguage(to: nil) + } + + func testGerman() { + performSnapshotTest(forLanguage: "de") + } + + func testEnglish() { + performSnapshotTest(forLanguage: "en") + } + + func testSpanish() { + performSnapshotTest(forLanguage: "es") + } + + func testFrench() { + performSnapshotTest(forLanguage: "fr") + } + + func testItalian() { + performSnapshotTest(forLanguage: "it") + } + + func testJapanese() { + performSnapshotTest(forLanguage: "ja") + } + + func testDutch() { + performSnapshotTest(forLanguage: "nl") + } + + func testChinese() { + performSnapshotTest(forLanguage: "zh-Hans") + } +} diff --git a/Stripe/StripeiOSTests/STPSourceCardDetailsTest.swift b/Stripe/StripeiOSTests/STPSourceCardDetailsTest.swift new file mode 100644 index 00000000..5bddcdc6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceCardDetailsTest.swift @@ -0,0 +1,116 @@ +// +// STPSourceCardDetailsTest.swift +// StripeiOS Tests +// +// Created by Joey Dong on 6/21/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPSourceCardDetailsTest: XCTestCase { + // MARK: - STPSourceCard3DSecureStatus Tests + func testThreeDSecureStatusFromString() { + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "required"), .required) + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "REQUIRED"), .required) + + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "optional"), .optional) + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "OPTIONAL"), .optional) + + XCTAssertEqual( + STPSourceCardDetails.threeDSecureStatus(from: "not_supported"), + .notSupported + ) + XCTAssertEqual( + STPSourceCardDetails.threeDSecureStatus(from: "NOT_SUPPORTED"), + .notSupported + ) + + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "recommended"), .recommended) + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "RECOMMENDED"), .recommended) + + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "unknown"), .unknown) + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "UNKNOWN"), .unknown) + + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "garbage"), .unknown) + XCTAssertEqual(STPSourceCardDetails.threeDSecureStatus(from: "GARBAGE"), .unknown) + } + + func testStringFromThreeDSecureStatus() { + let values: [STPSourceCard3DSecureStatus] = [ + .required, + .optional, + .notSupported, + .recommended, + .unknown, + ] + + for threeDSecureStatus in values { + let string = STPSourceCardDetails.string(fromThreeDSecureStatus: threeDSecureStatus) + + switch threeDSecureStatus { + case .required: + XCTAssertEqual(string, "required") + case .optional: + XCTAssertEqual(string, "optional") + case .notSupported: + XCTAssertEqual(string, "not_supported") + case .recommended: + XCTAssertEqual(string, "recommended") + case .unknown: + XCTAssertNil(string) + default: + break + } + } + } + + // MARK: - Description Tests + func testDescription() { + let cardDetails = STPSourceCardDetails.decodedObject( + fromAPIResponse: STPTestUtils.jsonNamed("CardSource")!["card"] as? [AnyHashable: Any] + ) + XCTAssert(cardDetails?.description != nil) + } + + // MARK: - STPAPIResponseDecodable Tests + func testDecodedObjectFromAPIResponseRequiredFields() { + let requiredFields: [String]? = [] + + for field in requiredFields ?? [] { + var response = STPTestUtils.jsonNamed("CardSource")?["card"] as? [AnyHashable: Any] + response?.removeValue(forKey: field) + + XCTAssertNil(STPSourceCardDetails.decodedObject(fromAPIResponse: response)) + } + + XCTAssert( + (STPSourceCardDetails.decodedObject( + fromAPIResponse: STPTestUtils.jsonNamed("CardSource")!["card"] + as? [AnyHashable: Any] + ) + != nil) + ) + } + + func testDecodedObjectFromAPIResponseMapping() { + let response = STPTestUtils.jsonNamed("CardSource")?["card"] as? [AnyHashable: Any] + let cardDetails = STPSourceCardDetails.decodedObject(fromAPIResponse: response)! + + XCTAssertEqual(cardDetails.brand, .visa) + XCTAssertEqual(cardDetails.country, "US") + XCTAssertEqual(cardDetails.expMonth, UInt(12)) + XCTAssertEqual(cardDetails.expYear, UInt(2034)) + XCTAssertEqual(cardDetails.funding, .debit) + XCTAssertEqual(cardDetails.last4, "5556") + XCTAssertEqual(cardDetails.threeDSecure, .notSupported) + + XCTAssertEqual(cardDetails.allResponseFields as NSDictionary, response! as NSDictionary) + } +} diff --git a/Stripe/StripeiOSTests/STPSourceFunctionalTest.m b/Stripe/StripeiOSTests/STPSourceFunctionalTest.m new file mode 100644 index 00000000..5db95c11 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceFunctionalTest.m @@ -0,0 +1,652 @@ +// +// STPSourceFunctionalTest.m +// Stripe +// +// Created by Ben Guo on 1/23/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; +@import StripeCoreTestUtils; +@import StripeCore; + +#import "STPTestingAPIClient.h" + +@interface STPSourceFunctionalTest : XCTestCase +@end + +@interface STPAPIClient (WritableURL) +@property (nonatomic, readwrite) NSURL *apiURL; +@end + +@implementation STPSourceFunctionalTest + +- (void)testCreateSource_bancontact { + STPSourceParams *params = [STPSourceParams bancontactParamsWithAmount:1099 + name:@"Jenny Rosen" + returnURL:@"https://shop.example.com/crtABC" + statementDescriptor:@"ORDER AT123"]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeBancontact); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_card { + STPCardParams *card = [[STPCardParams alloc] init]; + card.number = @"4242 4242 4242 4242"; + card.expMonth = 6; + card.expYear = 2024; + card.currency = @"usd"; + card.name = @"Jenny Rosen"; + card.address.line1 = @"123 Fake Street"; + card.address.line2 = @"Apartment 4"; + card.address.city = @"New York"; + card.address.state = @"NY"; + card.address.country = @"USA"; + card.address.postalCode = @"10002"; + STPSourceParams *params = [STPSourceParams cardParamsWithCard:card]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeCard); + XCTAssertEqualObjects(source.cardDetails.last4, @"4242"); + XCTAssertEqual(source.cardDetails.expMonth, card.expMonth); + XCTAssertEqual(source.cardDetails.expYear, card.expYear); + XCTAssertEqualObjects(source.owner.name, card.name); + STPAddress *address = source.owner.address; + XCTAssertEqualObjects(address.line1, card.address.line1); + XCTAssertEqualObjects(address.line2, card.address.line2); + XCTAssertEqualObjects(address.city, card.address.city); + XCTAssertEqualObjects(address.state, card.address.state); + XCTAssertEqualObjects(address.country, card.address.country); + XCTAssertEqualObjects(address.postalCode, card.address.postalCode); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_giropay { + STPSourceParams *params = [STPSourceParams giropayParamsWithAmount:1099 + name:@"Jenny Rosen" + returnURL:@"https://shop.example.com/crtABC" + statementDescriptor:@"ORDER AT123"]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeGiropay); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_ideal { + STPSourceParams *params = [STPSourceParams idealParamsWithAmount:1099 + name:@"Jenny Rosen" + returnURL:@"https://shop.example.com/crtABC" + statementDescriptor:@"ORDER AT123" + bank:@"ing"]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeiDEAL); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); + XCTAssertEqualObjects(source.details[@"bank"], @"ing"); + XCTAssertEqualObjects(source.details[@"statement_descriptor"], @"ORDER AT123"); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_ideal_missingOptionalFields { + STPSourceParams *params = [STPSourceParams idealParamsWithAmount:1099 + name:nil + returnURL:@"https://shop.example.com/crtABC" + statementDescriptor:nil + bank:nil]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeiDEAL); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertNil(source.owner.name); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); + XCTAssertNil(source.details[@"bank"]); + XCTAssertNil(source.details[@"statement_descriptor"]); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_ideal_emptyOptionalFields { + STPSourceParams *params = [STPSourceParams idealParamsWithAmount:1099 + name:@"" + returnURL:@"https://shop.example.com/crtABC" + statementDescriptor:@"" + bank:@""]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeiDEAL); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertNil(source.owner.name); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); + XCTAssertNil(source.details[@"bank"]); + XCTAssertNil(source.details[@"statement_descriptor"]); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_sepaDebit { + STPSourceParams *params = [STPSourceParams sepaDebitParamsWithName:@"Jenny Rosen" + iban:@"DE89370400440532013000" + addressLine1:@"Nollendorfstraße 27" + city:@"Berlin" + postalCode:@"10777" + country:@"DE"]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeSEPADebit); + XCTAssertNil(source.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertEqualObjects(source.owner.address.city, @"Berlin"); + XCTAssertEqualObjects(source.owner.address.line1, @"Nollendorfstraße 27"); + XCTAssertEqualObjects(source.owner.address.country, @"DE"); + XCTAssertEqualObjects(source.sepaDebitDetails.country, @"DE"); + XCTAssertEqualObjects(source.sepaDebitDetails.last4, @"3000"); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_sepaDebit_NoAddress { + STPSourceParams *params = [STPSourceParams sepaDebitParamsWithName:@"Jenny Rosen" + iban:@"DE89370400440532013000" + addressLine1:nil + city:nil + postalCode:nil + country:nil]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeSEPADebit); + XCTAssertNil(source.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertNil(source.owner.address.city); + XCTAssertNil(source.owner.address.line1); + XCTAssertNil(source.owner.address.country); + XCTAssertEqualObjects(source.sepaDebitDetails.country, @"DE"); // German IBAN so sepa tells us country here even though we didnt pass it up as owner info + XCTAssertEqualObjects(source.sepaDebitDetails.last4, @"3000"); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_sofort { + STPSourceParams *params = [STPSourceParams sofortParamsWithAmount:1099 + returnURL:@"https://shop.example.com/crtABC" + country:@"DE" + statementDescriptor:@"ORDER AT11990"]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeSofort); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + XCTAssertEqualObjects(source.details[@"country"], @"DE"); + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_threeDSecure { + STPCardParams *card = [[STPCardParams alloc] init]; + card.number = @"4000000000003063"; + card.expMonth = 6; + card.expYear = 2024; + card.currency = @"usd"; + card.address.line1 = @"123 Fake Street"; + card.address.line2 = @"Apartment 4"; + card.address.city = @"New York"; + card.address.state = @"NY"; + card.address.country = @"USA"; + card.address.postalCode = @"10002"; + STPSourceParams *cardParams = [STPSourceParams cardParamsWithCard:card]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *cardExp = [self expectationWithDescription:@"Card Source creation"]; + XCTestExpectation *threeDSExp = [self expectationWithDescription:@"3DS Source creation"]; + [client createSourceWithParams:cardParams completion:^(STPSource *source1, NSError *error1) { + XCTAssertNil(error1); + XCTAssertNotNil(source1); + XCTAssertEqual(source1.cardDetails.threeDSecure, STPSourceCard3DSecureStatusRequired); + [cardExp fulfill]; + + if (source1.stripeID == nil) { + XCTFail(@"stripeID of the Card Source is required to create a 3DS source"); + [threeDSExp fulfill]; + return; + } + + STPSourceParams *params = [STPSourceParams threeDSecureParamsWithAmount:1099 + currency:@"eur" + returnURL:@"https://shop.example.com/crtABC" + card:source1.stripeID]; + params.metadata = @{ @"foo": @"bar" }; + [client createSourceWithParams:params completion:^(STPSource *source2, NSError *error2) { + XCTAssertNil(error2); + XCTAssertNotNil(source2); + XCTAssertEqual(source2.type, STPSourceTypeThreeDSecure); + XCTAssertEqualObjects(source2.amount, params.amount); + XCTAssertEqualObjects(source2.currency, params.currency); + XCTAssertEqual(source2.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source2.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source2.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source2.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + [threeDSExp fulfill]; + }]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)skip_testCreateSourceVisaCheckout { + // The SDK does not have a means of generating Visa Checkout params for testing. Supply your own + // callId, and the correct publishable key, and you can run this test case + // manually after removing the `skip_` prefix. It'll log the source's stripeID, and that + // can be verified in dashboard. + STPSourceParams *params = [STPSourceParams visaCheckoutParamsWithCallId:@""]; + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_"]; + client.apiURL = [NSURL URLWithString:@"https://api.stripe.com/v1"]; + + XCTestExpectation *sourceExp = [self expectationWithDescription:@"VCO source created"]; + [client createSourceWithParams:params completion:^(STPSource * _Nullable source, NSError * _Nullable error) { + [sourceExp fulfill]; + + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeCard); + XCTAssertEqual(source.flow, STPSourceFlowNone); + XCTAssertEqual(source.status, STPSourceStatusChargeable); + XCTAssertEqual(source.usage, STPSourceUsageReusable); + XCTAssertTrue([source.stripeID hasPrefix:@"src_"]); + NSLog(@"Created a VCO source %@", source.stripeID); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)skip_testCreateSourceMasterpass { + // The SDK does not have a means of generating Masterpass params for testing. Supply your own + // cartId & transactionId, and the correct publishable key, and you can run this test case + // manually after removing the `skip_` prefix. It'll log the source's stripeID, and that + // can be verified in dashboard. + STPSourceParams *params = [STPSourceParams masterpassParamsWithCartId:@"" transactionId:@""]; + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_"]; + client.apiURL = [NSURL URLWithString:@"https://api.stripe.com/v1"]; + + XCTestExpectation *sourceExp = [self expectationWithDescription:@"Masterpass source created"]; + [client createSourceWithParams:params completion:^(STPSource * _Nullable source, NSError * _Nullable error) { + [sourceExp fulfill]; + + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeCard); + XCTAssertEqual(source.flow, STPSourceFlowNone); + XCTAssertEqual(source.status, STPSourceStatusChargeable); + XCTAssertEqual(source.usage, STPSourceUsageSingleUse); + XCTAssertTrue([source.stripeID hasPrefix:@"src_"]); + NSLog(@"Created a Masterpass source %@", source.stripeID); + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_alipay { + STPSourceParams *params = [STPSourceParams alipayParamsWithAmount:1099 + currency:@"usd" + returnURL:@"https://shop.example.com/crtABC"]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Alipay Source creation"]; + + params.metadata = @{ @"foo": @"bar" }; + [client createSourceWithParams:params completion:^(STPSource *source, NSError *error2) { + XCTAssertNil(error2); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeAlipay); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_p24 { + STPSourceParams *params = [STPSourceParams p24ParamsWithAmount:1099 + currency:@"eur" + email:@"user@example.com" + name:@"Jenny Rosen" + returnURL:@"https://shop.example.com/crtABC"]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"P24 Source creation"]; + + params.metadata = @{ @"foo": @"bar" }; + [client createSourceWithParams:params completion:^(STPSource *source, NSError *error2) { + XCTAssertNil(error2); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeP24); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.email, params.owner[@"email"]); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + [expectation fulfill]; + }]; + + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testRetrieveSource_sofort { + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_test_vOo1umqsYxSrP5UXfOeL3ecm"]; + STPSourceParams *params = [STPSourceParams new]; + params.type = STPSourceTypeSofort; + params.amount = @1099; + params.currency = @"eur"; + params.redirect = @{@"return_url": @"https://shop.example.com/crtA6B28E1"}; + params.metadata = @{@"foo": @"bar"}; + params.additionalAPIParameters = @{ @"sofort": @{ @"country": @"DE" } }; + XCTestExpectation *createExp = [self expectationWithDescription:@"Source creation"]; + XCTestExpectation *retrieveExp = [self expectationWithDescription:@"Source retrieval"]; + [client createSourceWithParams:params completion:^(STPSource *source1, NSError *error1) { + XCTAssertNil(error1); + XCTAssertNotNil(source1); + [createExp fulfill]; + [client retrieveSourceWithId:source1.stripeID + clientSecret:source1.clientSecret + completion:^(STPSource *source2, NSError *error2) { + XCTAssertNil(error2); + XCTAssertNotNil(source2); + XCTAssertEqualObjects(source1, source2); + [retrieveExp fulfill]; + }]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_eps { + STPSourceParams *params = [STPSourceParams epsParamsWithAmount:1099 + name:@"Jenny Rosen" + returnURL:@"https://shop.example.com/crtABC" + statementDescriptor:@"ORDER AT123"]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeEPS); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + XCTAssertEqualObjects(source.allResponseFields[@"statement_descriptor"], @"ORDER AT123"); + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_eps_no_statement_descriptor { + STPSourceParams *params = [STPSourceParams epsParamsWithAmount:1099 + name:@"Jenny Rosen" + returnURL:@"https://shop.example.com/crtABC" + statementDescriptor:nil]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeEPS); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqualObjects(source.currency, params.currency); + XCTAssertEqualObjects(source.owner.name, params.owner[@"name"]); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + XCTAssertNil(source.allResponseFields[@"statement_descriptor"]); + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_multibanco { + STPSourceParams *params = [STPSourceParams multibancoParamsWithAmount:1099 + returnURL:@"https://shop.example.com/crtABC" + email:@"user@example.com"]; + params.metadata = @{@"foo": @"bar"}; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingDefaultPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeMultibanco); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/crtABC?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata, @"Metadata is not returned."); +#pragma clang diagnostic pop + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_klarna { + NSArray *lineItems = @[[[STPKlarnaLineItem alloc] initWithItemType:STPKlarnaLineItemTypeSKU itemDescription:@"Test Item" quantity:@(2) totalAmount:@(500)], + [[STPKlarnaLineItem alloc] initWithItemType:STPKlarnaLineItemTypeTax itemDescription:@"Tax" quantity:@(1) totalAmount:@(100)]]; + STPAddress *address = [[STPAddress alloc] init]; + address.line1 = @"29 Arlington Avenue"; + address.email = @"test@example.com"; + address.city = @"London"; + address.postalCode = @"N1 7BE"; + address.country = @"GB"; + address.phone = @"02012267709"; + STPDateOfBirth *dob = [[STPDateOfBirth alloc] init]; + dob.day = 11; + dob.month = 3; + dob.year = 1952; + STPSourceParams *params = [STPSourceParams klarnaParamsWithReturnURL:@"https://shop.example.com/return" currency:@"GBP" purchaseCountry:@"GB" items:lineItems customPaymentMethods:@[@(STPKlarnaPaymentMethodsNone)] billingAddress:address billingFirstName:@"Arthur" billingLastName:@"Dent" billingDOB:dob]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:STPTestingGBPublishableKey]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeKlarna); + XCTAssertEqualObjects(source.amount, @(600)); + XCTAssertEqualObjects(source.owner.address.line1, address.line1); + XCTAssertEqualObjects(source.klarnaDetails.purchaseCountry, @"GB"); + XCTAssertEqual(source.redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(source.redirect.returnURL, [NSURL URLWithString:@"https://shop.example.com/return?redirect_merchant_name=xctest"]); + XCTAssertNotNil(source.redirect.url); + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +- (void)testCreateSource_wechatPay { + STPSourceParams *params = [STPSourceParams wechatPayParamsWithAmount:1010 + currency:@"usd" + appId:@"wxa0df51ec63e578ce" + statementDescriptor:nil]; + + STPAPIClient *client = [[STPAPIClient alloc] initWithPublishableKey:@"pk_live_L4KL0pF017Jgv9hBaWzk4xoB"]; + XCTestExpectation *expectation = [self expectationWithDescription:@"Source creation"]; + [client createSourceWithParams:params completion:^(STPSource *source, NSError * error) { + XCTAssertNil(error); + XCTAssertNotNil(source); + XCTAssertEqual(source.type, STPSourceTypeWeChatPay); + XCTAssertEqual(source.status, STPSourceStatusPending); + XCTAssertEqualObjects(source.amount, params.amount); + XCTAssertNil(source.redirect); + + STPSourceWeChatPayDetails *wechat = source.weChatPayDetails; + XCTAssertNotNil(wechat); + XCTAssertNotNil(wechat.weChatAppURL); + + [expectation fulfill]; + }]; + [self waitForExpectationsWithTimeout:TestConstants.STPTestingNetworkRequestTimeout handler:nil]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPSourceOwnerTest.m b/Stripe/StripeiOSTests/STPSourceOwnerTest.m new file mode 100644 index 00000000..0c8f0ab4 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceOwnerTest.m @@ -0,0 +1,63 @@ +// +// STPSourceOwnerTest.m +// Stripe +// +// Created by Joey Dong on 6/23/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPSourceOwnerTest : XCTestCase + +@end + +@implementation STPSourceOwnerTest + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:STPTestJSONSource3DS][@"owner"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPSourceOwner decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPSourceOwner decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONSource3DS][@"owner"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONSource3DS][@"owner"]; + STPSourceOwner *owner = [STPSourceOwner decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(owner.address.city, @"Pittsburgh"); + XCTAssertEqualObjects(owner.address.country, @"US"); + XCTAssertEqualObjects(owner.address.line1, @"123 Fake St"); + XCTAssertEqualObjects(owner.address.line2, @"Apt 1"); + XCTAssertEqualObjects(owner.address.postalCode, @"19219"); + XCTAssertEqualObjects(owner.address.state, @"PA"); + XCTAssertEqualObjects(owner.email, @"jenny.rosen@example.com"); + XCTAssertEqualObjects(owner.name, @"Jenny Rosen"); + XCTAssertEqualObjects(owner.phone, @"555-867-5309"); + XCTAssertEqualObjects(owner.verifiedAddress.city, @"Pittsburgh"); + XCTAssertEqualObjects(owner.verifiedAddress.country, @"US"); + XCTAssertEqualObjects(owner.verifiedAddress.line1, @"123 Fake St"); + XCTAssertEqualObjects(owner.verifiedAddress.line2, @"Apt 1"); + XCTAssertEqualObjects(owner.verifiedAddress.postalCode, @"19219"); + XCTAssertEqualObjects(owner.verifiedAddress.state, @"PA"); + XCTAssertEqualObjects(owner.verifiedEmail, @"jenny.rosen@example.com"); + XCTAssertEqualObjects(owner.verifiedName, @"Jenny Rosen"); + XCTAssertEqualObjects(owner.verifiedPhone, @"555-867-5309"); + + XCTAssertNotEqual(owner.allResponseFields, response); + XCTAssertEqualObjects(owner.allResponseFields, response); +} + +@end diff --git a/Stripe/StripeiOSTests/STPSourceParamsTest.swift b/Stripe/StripeiOSTests/STPSourceParamsTest.swift new file mode 100644 index 00000000..0534c5b8 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceParamsTest.swift @@ -0,0 +1,317 @@ +// +// STPSourceParamsTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 1/25/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPSourceParamsTest: XCTestCase { + // MARK: - + func testInit() { + let sourceParams = STPSourceParams() + XCTAssertEqual(sourceParams.rawTypeString, "") + XCTAssertEqual(sourceParams.flow, .unknown) + XCTAssertEqual(sourceParams.usage, .unknown) + XCTAssertEqual(sourceParams.additionalAPIParameters as NSDictionary, [:] as NSDictionary) + } + + func testType() { + let sourceParams = STPSourceParams() + XCTAssertEqual(sourceParams.type, .unknown) + + sourceParams.rawTypeString = "bancontact" + XCTAssertEqual(sourceParams.type, .bancontact) + + sourceParams.rawTypeString = "card" + XCTAssertEqual(sourceParams.type, .card) + + sourceParams.rawTypeString = "giropay" + XCTAssertEqual(sourceParams.type, .giropay) + + sourceParams.rawTypeString = "ideal" + XCTAssertEqual(sourceParams.type, .iDEAL) + + sourceParams.rawTypeString = "sepa_debit" + XCTAssertEqual(sourceParams.type, .SEPADebit) + + sourceParams.rawTypeString = "sofort" + XCTAssertEqual(sourceParams.type, .sofort) + + sourceParams.rawTypeString = "three_d_secure" + XCTAssertEqual(sourceParams.type, .threeDSecure) + + sourceParams.rawTypeString = "alipay" + XCTAssertEqual(sourceParams.type, .alipay) + + sourceParams.rawTypeString = "p24" + XCTAssertEqual(sourceParams.type, .P24) + + sourceParams.rawTypeString = "eps" + XCTAssertEqual(sourceParams.type, .EPS) + + sourceParams.rawTypeString = "multibanco" + XCTAssertEqual(sourceParams.type, .multibanco) + + sourceParams.rawTypeString = "klarna" + XCTAssertEqual(sourceParams.type, .klarna) + + sourceParams.rawTypeString = "unknown" + XCTAssertEqual(sourceParams.type, .unknown) + + sourceParams.rawTypeString = "garbage" + XCTAssertEqual(sourceParams.type, .unknown) + } + + func testSetType() { + let sourceParams = STPSourceParams() + XCTAssertEqual(sourceParams.type, .unknown) + + sourceParams.type = .bancontact + XCTAssertEqual(sourceParams.rawTypeString, "bancontact") + + sourceParams.type = .card + XCTAssertEqual(sourceParams.rawTypeString, "card") + + sourceParams.type = .giropay + XCTAssertEqual(sourceParams.rawTypeString, "giropay") + + sourceParams.type = .iDEAL + XCTAssertEqual(sourceParams.rawTypeString, "ideal") + + sourceParams.type = .SEPADebit + XCTAssertEqual(sourceParams.rawTypeString, "sepa_debit") + + sourceParams.type = .sofort + XCTAssertEqual(sourceParams.rawTypeString, "sofort") + + sourceParams.type = .threeDSecure + XCTAssertEqual(sourceParams.rawTypeString, "three_d_secure") + + sourceParams.type = .alipay + XCTAssertEqual(sourceParams.rawTypeString, "alipay") + + sourceParams.type = .P24 + XCTAssertEqual(sourceParams.rawTypeString, "p24") + + sourceParams.type = .EPS + XCTAssertEqual(sourceParams.rawTypeString, "eps") + + sourceParams.type = .multibanco + XCTAssertEqual(sourceParams.rawTypeString, "multibanco") + + sourceParams.type = .klarna + XCTAssertEqual(sourceParams.rawTypeString, "klarna") + + sourceParams.type = .unknown + XCTAssertNil(sourceParams.rawTypeString) + } + + func testSetTypePreserveUnknownRawTypeString() { + let sourceParams = STPSourceParams() + sourceParams.rawTypeString = "money" + sourceParams.type = .unknown + XCTAssertEqual(sourceParams.rawTypeString, "money") + } + + func testRawTypeString() { + let sourceParams = STPSourceParams() + + // Check defaults to unknown + XCTAssertEqual(sourceParams.type, .unknown) + + // Check changing type sets rawTypeString + sourceParams.type = .card + XCTAssertEqual(sourceParams.rawTypeString, STPSource.string(from: .card)) + + // Check changing to unknown raw string sets type to unknown + sourceParams.rawTypeString = "new_source_type" + XCTAssertEqual(sourceParams.type, .unknown) + + // Check once unknown that setting type to unknown doesnt clobber string + sourceParams.type = .unknown + XCTAssertEqual(sourceParams.rawTypeString, "new_source_type") + + // Check setting string to known type sets type correctly + sourceParams.rawTypeString = STPSource.string(from: .iDEAL) + XCTAssertEqual(sourceParams.type, .iDEAL) + } + + func testFlowString() { + let sourceParams = STPSourceParams() + XCTAssertNil(sourceParams.flowString()) + + sourceParams.flow = .redirect + XCTAssertEqual(sourceParams.flowString(), "redirect") + + sourceParams.flow = .receiver + XCTAssertEqual(sourceParams.flowString(), "receiver") + + sourceParams.flow = .codeVerification + XCTAssertEqual(sourceParams.flowString(), "code_verification") + + sourceParams.flow = .none + XCTAssertEqual(sourceParams.flowString(), "none") + } + + // MARK: - Constructors Tests + func testCardParamsWithCard() { + let card = STPCardParams() + card.number = "4242 4242 4242 4242" + card.cvc = "123" + card.expMonth = 6 + card.expYear = 2024 + card.currency = "usd" + card.name = "Jenny Rosen" + card.address.line1 = "123 Fake Street" + card.address.line2 = "Apartment 4" + card.address.city = "New York" + card.address.state = "NY" + card.address.country = "USA" + card.address.postalCode = "10002" + + let source = STPSourceParams.cardParams(withCard: card) + let sourceCard = source.additionalAPIParameters["card"] as! [String: AnyHashable] + XCTAssertEqual(sourceCard["number"], card.number) + XCTAssertEqual(sourceCard["cvc"], card.cvc) + XCTAssertEqual(sourceCard["exp_month"], NSNumber(value: card.expMonth)) + XCTAssertEqual(sourceCard["exp_year"], NSNumber(value: card.expYear)) + XCTAssertEqual(source.owner!["name"] as? String, card.name) + let sourceAddress = source.owner!["address"] as! [AnyHashable: Any] + XCTAssertEqual(sourceAddress["line1"] as? String, card.address.line1) + XCTAssertEqual(sourceAddress["line2"] as? String, card.address.line2) + XCTAssertEqual(sourceAddress["city"] as? String, card.address.city) + XCTAssertEqual(sourceAddress["state"] as? String, card.address.state) + XCTAssertEqual(sourceAddress["postal_code"] as? String, card.address.postalCode) + XCTAssertEqual(sourceAddress["country"] as? String, card.address.country) + } + + func testParamsWithVisaCheckout() { + let params = STPSourceParams.visaCheckoutParams(withCallId: "12345678") + + XCTAssertEqual(params.type, .card) + let sourceCard = params.additionalAPIParameters["card"] as? [AnyHashable: Any] + XCTAssertNotNil(sourceCard) + let sourceVisaCheckout = sourceCard!["visa_checkout"] as? [AnyHashable: Any] + XCTAssertNotNil(sourceVisaCheckout) + XCTAssertEqual(sourceVisaCheckout!["callid"] as! String, "12345678") + } + + func testParamsWithMasterPass() { + let params = STPSourceParams.masterpassParams( + withCartId: "12345678", + transactionId: "87654321" + ) + + XCTAssertEqual(params.type, .card) + let sourceCard = params.additionalAPIParameters["card"] as? [AnyHashable: Any] + XCTAssertNotNil(sourceCard) + let sourceMasterpass = sourceCard!["masterpass"] as? [AnyHashable: Any] + XCTAssertNotNil(sourceMasterpass) + XCTAssertEqual(sourceMasterpass!["cart_id"] as! String, "12345678") + XCTAssertEqual(sourceMasterpass!["transaction_id"] as! String, "87654321") + } + + func testKlarnaParams() { + let params = STPSourceParams.klarnaParams( + withReturnURL: "return_url", + currency: "USD", + purchaseCountry: "US", + items: [], + customPaymentMethods: [.none], + billingAddress: nil, + billingFirstName: nil, + billingLastName: nil, + billingDOB: nil + ) + + XCTAssertNotNil(params) + } + + // MARK: - Redirect Dictionary Tests + func redirectMerchantNameQueryItemValue(fromURLString urlString: String?) -> String? { + let components = NSURLComponents(string: urlString ?? "") + for item in components!.queryItems! { + if item.name == "redirect_merchant_name" { + return item.value + } + } + return nil + } + + func testRedirectMerchantNameURL() { + var sourceParams = STPSourceParams.sofortParams( + withAmount: 1000, + returnURL: "test://foo?value=baz", + country: "DE", + statementDescriptor: nil + ) + + var params = STPFormEncoder.dictionary(forObject: sourceParams) + // Should be nil because we have no app name in tests + XCTAssertNil( + redirectMerchantNameQueryItemValue( + fromURLString: (params["redirect"] as! [String: AnyHashable])["return_url"] + as? String + ) + ) + + sourceParams.redirectMerchantName = "bar" + params = STPFormEncoder.dictionary(forObject: sourceParams) + XCTAssertEqual( + redirectMerchantNameQueryItemValue( + fromURLString: (params["redirect"] as! [String: AnyHashable])["return_url"] + as? String + ), + "bar" + ) + + sourceParams = STPSourceParams.sofortParams( + withAmount: 1000, + returnURL: "test://foo?redirect_merchant_name=Manual%20Custom%20Name", + country: "DE", + statementDescriptor: nil + ) + sourceParams.redirectMerchantName = "bar" + params = STPFormEncoder.dictionary(forObject: sourceParams) + // Don't override names set by the user directly in the url + XCTAssertEqual( + redirectMerchantNameQueryItemValue( + fromURLString: (params["redirect"] as! [String: AnyHashable])["return_url"] + as? String + ), + "Manual Custom Name" + ) + + } + + // MARK: - STPFormEncodable Tests + func testRootObjectName() { + XCTAssertNil(STPSourceParams.rootObjectName()) + } + + func testPropertyNamesToFormFieldNamesMapping() { + let sourceParams = STPSourceParams() + + let mapping = STPSourceParams.propertyNamesToFormFieldNamesMapping() + + for propertyName in mapping.keys { + XCTAssertFalse(propertyName.contains(":")) + XCTAssert(sourceParams.responds(to: NSSelectorFromString(propertyName))) + } + + for formFieldName in mapping.values { + XCTAssert(formFieldName.count > 0) + } + + XCTAssertEqual(mapping.values.count, Set(mapping.values).count) + } +} diff --git a/Stripe/StripeiOSTests/STPSourceReceiverTest.m b/Stripe/StripeiOSTests/STPSourceReceiverTest.m new file mode 100644 index 00000000..b0f9dec5 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceReceiverTest.m @@ -0,0 +1,58 @@ +// +// STPSourceReceiverTest.m +// Stripe +// +// Created by Joey Dong on 6/26/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +#import "STPFixtures.h" +#import "STPTestUtils.h" + +@interface STPSourceReceiverTest : XCTestCase + +@end + +@implementation STPSourceReceiverTest + +#pragma mark - Description Tests + +- (void)testDescription { + STPSourceReceiver *receiver = [STPSourceReceiver decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONSource3DS][@"receiver"]]; + XCTAssert(receiver.description); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[ + @"address", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:STPTestJSONSource3DS][@"receiver"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPSourceReceiver decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPSourceReceiver decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:STPTestJSONSource3DS][@"receiver"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONSource3DS][@"receiver"]; + STPSourceReceiver *receiver = [STPSourceReceiver decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(receiver.address, @"test_1MBhWS3uv4ynCfQXF3xQjJkzFPukr4K56N"); + XCTAssertEqualObjects(receiver.amountCharged, @(300)); + XCTAssertEqualObjects(receiver.amountReceived, @(200)); + XCTAssertEqualObjects(receiver.amountReturned, @(100)); + + XCTAssertNotEqual(receiver.allResponseFields, response); + XCTAssertEqualObjects(receiver.allResponseFields, response); +} + +@end diff --git a/Stripe/StripeiOSTests/STPSourceRedirectTest.m b/Stripe/StripeiOSTests/STPSourceRedirectTest.m new file mode 100644 index 00000000..a323e94e --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceRedirectTest.m @@ -0,0 +1,119 @@ +// +// STPSourceRedirectTest.m +// Stripe +// +// Created by Joey Dong on 6/21/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +#import "STPTestUtils.h" + +@interface STPSourceRedirect () + ++ (STPSourceRedirectStatus)statusFromString:(NSString *)string; ++ (NSString *)stringFromStatus:(STPSourceRedirectStatus)status; + +@end + +@interface STPSourceRedirectTest : XCTestCase + +@end + +@implementation STPSourceRedirectTest + +#pragma mark - STPSourceRedirectStatus Tests + +- (void)testStatusFromString { + XCTAssertEqual([STPSourceRedirect statusFromString:@"pending"], STPSourceRedirectStatusPending); + XCTAssertEqual([STPSourceRedirect statusFromString:@"PENDING"], STPSourceRedirectStatusPending); + + XCTAssertEqual([STPSourceRedirect statusFromString:@"succeeded"], STPSourceRedirectStatusSucceeded); + XCTAssertEqual([STPSourceRedirect statusFromString:@"SUCCEEDED"], STPSourceRedirectStatusSucceeded); + + XCTAssertEqual([STPSourceRedirect statusFromString:@"failed"], STPSourceRedirectStatusFailed); + XCTAssertEqual([STPSourceRedirect statusFromString:@"FAILED"], STPSourceRedirectStatusFailed); + + XCTAssertEqual([STPSourceRedirect statusFromString:@"unknown"], STPSourceRedirectStatusUnknown); + XCTAssertEqual([STPSourceRedirect statusFromString:@"UNKNOWN"], STPSourceRedirectStatusUnknown); + + XCTAssertEqual([STPSourceRedirect statusFromString:@"not_required"], STPSourceRedirectStatusNotRequired); + XCTAssertEqual([STPSourceRedirect statusFromString:@"NOT_REQUIRED"], STPSourceRedirectStatusNotRequired); + + XCTAssertEqual([STPSourceRedirect statusFromString:@"garbage"], STPSourceRedirectStatusUnknown); + XCTAssertEqual([STPSourceRedirect statusFromString:@"GARBAGE"], STPSourceRedirectStatusUnknown); +} + +- (void)testStringFromStatus { + NSArray *values = @[ + @(STPSourceRedirectStatusPending), + @(STPSourceRedirectStatusSucceeded), + @(STPSourceRedirectStatusFailed), + @(STPSourceRedirectStatusUnknown), + ]; + + for (NSNumber *statusNumber in values) { + STPSourceRedirectStatus status = (STPSourceRedirectStatus)[statusNumber integerValue]; + NSString *string = [STPSourceRedirect stringFromStatus:status]; + + switch (status) { + case STPSourceRedirectStatusPending: + XCTAssertEqualObjects(string, @"pending"); + break; + case STPSourceRedirectStatusSucceeded: + XCTAssertEqualObjects(string, @"succeeded"); + break; + case STPSourceRedirectStatusFailed: + XCTAssertEqualObjects(string, @"failed"); + break; + case STPSourceRedirectStatusNotRequired: + XCTAssertEqualObjects(string, @"not_required"); + break; + case STPSourceRedirectStatusUnknown: + XCTAssertNil(string); + break; + } + } +} + +#pragma mark - Description Tests + +- (void)testDescription { + STPSourceRedirect *redirect = [STPSourceRedirect decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"3DSSource"][@"redirect"]]; + XCTAssert(redirect.description); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[ + @"return_url", + @"status", + @"url", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"3DSSource"][@"redirect"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPSourceRedirect decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPSourceRedirect decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"3DSSource"][@"redirect"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:@"3DSSource"][@"redirect"]; + STPSourceRedirect *redirect = [STPSourceRedirect decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(redirect.returnURL, [NSURL URLWithString:@"exampleappschema://stripe_callback"]); + XCTAssertEqual(redirect.status, STPSourceRedirectStatusPending); + XCTAssertEqualObjects(redirect.url, [NSURL URLWithString:@"https://hooks.stripe.com/redirect/authenticate/src_19YlvWAHEMiOZZp1QQlOD79v?client_secret=src_client_secret_kBwCSm6Xz5MQETiJ43hUH8qv"]); + + XCTAssertNotEqual(redirect.allResponseFields, response); + XCTAssertEqualObjects(redirect.allResponseFields, response); +} + +@end diff --git a/Stripe/StripeiOSTests/STPSourceSEPADebitDetailsTest.m b/Stripe/StripeiOSTests/STPSourceSEPADebitDetailsTest.m new file mode 100644 index 00000000..f7b906c7 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceSEPADebitDetailsTest.m @@ -0,0 +1,56 @@ +// +// STPSourceSEPADebitDetails.m +// Stripe +// +// Created by Joey Dong on 6/26/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + +#import "STPTestUtils.h" + +@interface STPSourceSEPADebitDetailsTest : XCTestCase + +@end + +@implementation STPSourceSEPADebitDetailsTest + +#pragma mark - Description Tests + +- (void)testDescription { + STPSourceSEPADebitDetails *sepaDebitDetails = [STPSourceSEPADebitDetails decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"SEPADebitSource"][@"sepa_debit"]]; + XCTAssert(sepaDebitDetails.description); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"SEPADebitSource"][@"sepa_debit"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPSourceSEPADebitDetails decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPSourceSEPADebitDetails decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"SEPADebitSource"][@"sepa_debit"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:@"SEPADebitSource"][@"sepa_debit"]; + STPSourceSEPADebitDetails *sepaDebitDetails = [STPSourceSEPADebitDetails decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(sepaDebitDetails.bankCode, @"37040044"); + XCTAssertEqualObjects(sepaDebitDetails.country, @"DE"); + XCTAssertEqualObjects(sepaDebitDetails.fingerprint, @"NxdSyRegc9PsMkWy"); + XCTAssertEqualObjects(sepaDebitDetails.last4, @"3001"); + XCTAssertEqualObjects(sepaDebitDetails.mandateReference, @"NXDSYREGC9PSMKWY"); + XCTAssertEqualObjects(sepaDebitDetails.mandateURL, [NSURL URLWithString:@"https://hooks.stripe.com/adapter/sepa_debit/file/src_18HgGjHNCLa1Vra6Y9TIP6tU/src_client_secret_XcBmS94nTg5o0xc9MSliSlDW"]); + + XCTAssertNotEqual(sepaDebitDetails.allResponseFields, response); + XCTAssertEqualObjects(sepaDebitDetails.allResponseFields, response); +} + +@end diff --git a/Stripe/StripeiOSTests/STPSourceTest.m b/Stripe/StripeiOSTests/STPSourceTest.m new file mode 100644 index 00000000..a462bee6 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceTest.m @@ -0,0 +1,573 @@ +// +// STPSourceTest.m +// Stripe +// +// Created by Ben Guo on 1/24/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +#import "STPFixtures.h" +#import "STPTestUtils.h" + + + +@interface STPSource () + ++ (STPSourceFlow)flowFromString:(NSString *)string; + ++ (STPSourceStatus)statusFromString:(NSString *)string; ++ (NSString *)stringFromStatus:(STPSourceStatus)status; + ++ (STPSourceUsage)usageFromString:(NSString *)string; + +@end + +@interface STPSourceTest : XCTestCase + +@end + +@implementation STPSourceTest + +#pragma mark - STPSourceType Tests + +- (void)testTypeFromString { + XCTAssertEqual([STPSource typeFromString:@"bancontact"], STPSourceTypeBancontact); + XCTAssertEqual([STPSource typeFromString:@"BANCONTACT"], STPSourceTypeBancontact); + + XCTAssertEqual([STPSource typeFromString:@"card"], STPSourceTypeCard); + XCTAssertEqual([STPSource typeFromString:@"CARD"], STPSourceTypeCard); + + XCTAssertEqual([STPSource typeFromString:@"giropay"], STPSourceTypeGiropay); + XCTAssertEqual([STPSource typeFromString:@"GIROPAY"], STPSourceTypeGiropay); + + XCTAssertEqual([STPSource typeFromString:@"ideal"], STPSourceTypeiDEAL); + XCTAssertEqual([STPSource typeFromString:@"IDEAL"], STPSourceTypeiDEAL); + + XCTAssertEqual([STPSource typeFromString:@"sepa_debit"], STPSourceTypeSEPADebit); + XCTAssertEqual([STPSource typeFromString:@"SEPA_DEBIT"], STPSourceTypeSEPADebit); + + XCTAssertEqual([STPSource typeFromString:@"sofort"], STPSourceTypeSofort); + XCTAssertEqual([STPSource typeFromString:@"Sofort"], STPSourceTypeSofort); + + XCTAssertEqual([STPSource typeFromString:@"three_d_secure"], STPSourceTypeThreeDSecure); + XCTAssertEqual([STPSource typeFromString:@"THREE_D_SECURE"], STPSourceTypeThreeDSecure); + + XCTAssertEqual([STPSource typeFromString:@"alipay"], STPSourceTypeAlipay); + XCTAssertEqual([STPSource typeFromString:@"ALIPAY"], STPSourceTypeAlipay); + + XCTAssertEqual([STPSource typeFromString:@"p24"], STPSourceTypeP24); + XCTAssertEqual([STPSource typeFromString:@"P24"], STPSourceTypeP24); + + XCTAssertEqual([STPSource typeFromString:@"eps"], STPSourceTypeEPS); + XCTAssertEqual([STPSource typeFromString:@"EPS"], STPSourceTypeEPS); + + XCTAssertEqual([STPSource typeFromString:@"multibanco"], STPSourceTypeMultibanco); + XCTAssertEqual([STPSource typeFromString:@"MULTIBANCO"], STPSourceTypeMultibanco); + + XCTAssertEqual([STPSource typeFromString:@"unknown"], STPSourceTypeUnknown); + XCTAssertEqual([STPSource typeFromString:@"UNKNOWN"], STPSourceTypeUnknown); + + XCTAssertEqual([STPSource typeFromString:@"garbage"], STPSourceTypeUnknown); + XCTAssertEqual([STPSource typeFromString:@"GARBAGE"], STPSourceTypeUnknown); +} + +- (void)testStringFromType { + NSArray *values = @[ + @(STPSourceTypeBancontact), + @(STPSourceTypeCard), + @(STPSourceTypeGiropay), + @(STPSourceTypeiDEAL), + @(STPSourceTypeSEPADebit), + @(STPSourceTypeSofort), + @(STPSourceTypeThreeDSecure), + @(STPSourceTypeAlipay), + @(STPSourceTypeP24), + @(STPSourceTypeEPS), + @(STPSourceTypeMultibanco), + @(STPSourceTypeUnknown), + ]; + + for (NSNumber *typeNumber in values) { + STPSourceType type = (STPSourceType)[typeNumber integerValue]; + NSString *string = [STPSource stringFromType:type]; + + switch (type) { + case STPSourceTypeBancontact: + XCTAssertEqualObjects(string, @"bancontact"); + break; + case STPSourceTypeCard: + XCTAssertEqualObjects(string, @"card"); + break; + case STPSourceTypeGiropay: + XCTAssertEqualObjects(string, @"giropay"); + break; + case STPSourceTypeiDEAL: + XCTAssertEqualObjects(string, @"ideal"); + break; + case STPSourceTypeSEPADebit: + XCTAssertEqualObjects(string, @"sepa_debit"); + break; + case STPSourceTypeSofort: + XCTAssertEqualObjects(string, @"sofort"); + break; + case STPSourceTypeThreeDSecure: + XCTAssertEqualObjects(string, @"three_d_secure"); + break; + case STPSourceTypeAlipay: + XCTAssertEqualObjects(string, @"alipay"); + break; + case STPSourceTypeP24: + XCTAssertEqualObjects(string, @"p24"); + break; + case STPSourceTypeEPS: + XCTAssertEqualObjects(string, @"eps"); + break; + case STPSourceTypeMultibanco: + XCTAssertEqualObjects(string, @"multibanco"); + break; + case STPSourceTypeWeChatPay: + XCTAssertEqualObjects(string, @"wechat"); + break; + case STPSourceTypeKlarna: + XCTAssertEqualObjects(string, @"klarna"); + break; + case STPSourceTypeUnknown: + XCTAssertNil(string); + break; + } + } +} + +#pragma mark - STPSourceFlow Tests + +- (void)testFlowFromString { + XCTAssertEqual([STPSource flowFromString:@"redirect"], STPSourceFlowRedirect); + XCTAssertEqual([STPSource flowFromString:@"REDIRECT"], STPSourceFlowRedirect); + + XCTAssertEqual([STPSource flowFromString:@"receiver"], STPSourceFlowReceiver); + XCTAssertEqual([STPSource flowFromString:@"RECEIVER"], STPSourceFlowReceiver); + + XCTAssertEqual([STPSource flowFromString:@"code_verification"], STPSourceFlowCodeVerification); + XCTAssertEqual([STPSource flowFromString:@"CODE_VERIFICATION"], STPSourceFlowCodeVerification); + + XCTAssertEqual([STPSource flowFromString:@"none"], STPSourceFlowNone); + XCTAssertEqual([STPSource flowFromString:@"NONE"], STPSourceFlowNone); + + XCTAssertEqual([STPSource flowFromString:@"garbage"], STPSourceFlowUnknown); + XCTAssertEqual([STPSource flowFromString:@"GARBAGE"], STPSourceFlowUnknown); +} + +- (void)testStringFromFlow { + NSArray *values = @[ + @(STPSourceFlowRedirect), + @(STPSourceFlowReceiver), + @(STPSourceFlowCodeVerification), + @(STPSourceFlowNone), + @(STPSourceFlowUnknown), + ]; + + for (NSNumber *flowNumber in values) { + STPSourceFlow flow = (STPSourceFlow)[flowNumber integerValue]; + NSString *string = [STPSource stringFromFlow:flow]; + + switch (flow) { + case STPSourceFlowRedirect: + XCTAssertEqualObjects(string, @"redirect"); + break; + case STPSourceFlowReceiver: + XCTAssertEqualObjects(string, @"receiver"); + break; + case STPSourceFlowCodeVerification: + XCTAssertEqualObjects(string, @"code_verification"); + break; + case STPSourceFlowNone: + XCTAssertEqualObjects(string, @"none"); + break; + case STPSourceFlowUnknown: + XCTAssertNil(string); + break; + } + } +} + +#pragma mark - STPSourceStatus Tests + +- (void)testStatusFromString { + XCTAssertEqual([STPSource statusFromString:@"pending"], STPSourceStatusPending); + XCTAssertEqual([STPSource statusFromString:@"PENDING"], STPSourceStatusPending); + + XCTAssertEqual([STPSource statusFromString:@"chargeable"], STPSourceStatusChargeable); + XCTAssertEqual([STPSource statusFromString:@"CHARGEABLE"], STPSourceStatusChargeable); + + XCTAssertEqual([STPSource statusFromString:@"consumed"], STPSourceStatusConsumed); + XCTAssertEqual([STPSource statusFromString:@"CONSUMED"], STPSourceStatusConsumed); + + XCTAssertEqual([STPSource statusFromString:@"canceled"], STPSourceStatusCanceled); + XCTAssertEqual([STPSource statusFromString:@"CANCELED"], STPSourceStatusCanceled); + + XCTAssertEqual([STPSource statusFromString:@"failed"], STPSourceStatusFailed); + XCTAssertEqual([STPSource statusFromString:@"FAILED"], STPSourceStatusFailed); + + XCTAssertEqual([STPSource statusFromString:@"garbage"], STPSourceStatusUnknown); + XCTAssertEqual([STPSource statusFromString:@"GARBAGE"], STPSourceStatusUnknown); +} + +- (void)testStringFromStatus { + NSArray *values = @[ + @(STPSourceStatusPending), + @(STPSourceStatusChargeable), + @(STPSourceStatusConsumed), + @(STPSourceStatusCanceled), + @(STPSourceStatusFailed), + @(STPSourceStatusUnknown), + ]; + + for (NSNumber *statusNumber in values) { + STPSourceStatus status = (STPSourceStatus)[statusNumber integerValue]; + NSString *string = [STPSource stringFromStatus:status]; + + switch (status) { + case STPSourceStatusPending: + XCTAssertEqualObjects(string, @"pending"); + break; + case STPSourceStatusChargeable: + XCTAssertEqualObjects(string, @"chargeable"); + break; + case STPSourceStatusConsumed: + XCTAssertEqualObjects(string, @"consumed"); + break; + case STPSourceStatusCanceled: + XCTAssertEqualObjects(string, @"canceled"); + break; + case STPSourceStatusFailed: + XCTAssertEqualObjects(string, @"failed"); + break; + case STPSourceStatusUnknown: + XCTAssertNil(string); + break; + } + } +} + +#pragma mark - STPSourceUsage Tests + +- (void)testUsageFromString { + XCTAssertEqual([STPSource usageFromString:@"reusable"], STPSourceUsageReusable); + XCTAssertEqual([STPSource usageFromString:@"REUSABLE"], STPSourceUsageReusable); + + XCTAssertEqual([STPSource usageFromString:@"single_use"], STPSourceUsageSingleUse); + XCTAssertEqual([STPSource usageFromString:@"SINGLE_USE"], STPSourceUsageSingleUse); + + XCTAssertEqual([STPSource usageFromString:@"garbage"], STPSourceUsageUnknown); + XCTAssertEqual([STPSource usageFromString:@"GARBAGE"], STPSourceUsageUnknown); +} + +- (void)testStringFromUsage { + NSArray *values = @[ + @(STPSourceUsageReusable), + @(STPSourceUsageSingleUse), + @(STPSourceUsageUnknown), + ]; + + for (NSNumber *usageNumber in values) { + STPSourceUsage usage = (STPSourceUsage)[usageNumber integerValue]; + NSString *string = [STPSource stringFromUsage:usage]; + + switch (usage) { + case STPSourceUsageReusable: + XCTAssertEqualObjects(string, @"reusable"); + break; + case STPSourceUsageSingleUse: + XCTAssertEqualObjects(string, @"single_use"); + break; + case STPSourceUsageUnknown: + XCTAssertNil(string); + break; + } + } +} + +#pragma mark - Equality Tests + +- (void)testSourceEquals { + STPSource *source1 = [STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"AlipaySource"]]; + STPSource *source2 = [STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"AlipaySource"]]; + + XCTAssertNotEqual(source1, source2); + + XCTAssertEqualObjects(source1, source1); + XCTAssertEqualObjects(source1, source2); + + XCTAssertEqual(source1.hash, source1.hash); + XCTAssertEqual(source1.hash, source2.hash); +} + +#pragma mark - Description Tests + +- (void)testDescription { + STPSource *source = [STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"AlipaySource"]]; + XCTAssert(source.description); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[ + @"id", + @"livemode", + @"status", + @"type", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"AlipaySource"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPSource decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPSource decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"AlipaySource"]]); +} + +- (void)testDecodingSource_3ds { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONSource3DS]; + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(source.stripeID, @"src_456"); + XCTAssertEqualObjects(source.amount, @1099); + XCTAssertEqualObjects(source.clientSecret, @"src_client_secret_456"); + XCTAssertEqualWithAccuracy([source.created timeIntervalSince1970], 1483663790.0, 1.0); + XCTAssertEqualObjects(source.currency, @"eur"); + XCTAssertEqual(source.flow, STPSourceFlowRedirect); + XCTAssertEqual(source.livemode, NO); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata); +#pragma clang diagnostic pop + XCTAssert(source.owner); // STPSourceOwnerTest + XCTAssert(source.receiver); // STPSourceReceiverTest + XCTAssert(source.redirect); // STPSourceRedirectTest + XCTAssertEqual(source.status, STPSourceStatusPending); + XCTAssertEqual(source.type, STPSourceTypeThreeDSecure); + XCTAssertEqual(source.usage, STPSourceUsageSingleUse); + XCTAssertNil(source.verification); + NSMutableDictionary *threedsecure = [response[@"three_d_secure"] mutableCopy]; + [threedsecure removeObjectForKey:@"customer"]; // should be nil + XCTAssertEqualObjects(source.details, threedsecure); + XCTAssertNil(source.cardDetails); // STPSourceCardDetailsTest + XCTAssertNil(source.sepaDebitDetails); // STPSourceSEPADebitDetailsTest + XCTAssertNotEqual(source.allResponseFields, response); // Verify is copy +} + +- (void)testDecodingSource_alipay { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONSourceAlipay]; + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(source.stripeID, @"src_123"); + XCTAssertEqualObjects(source.amount, @1099); + XCTAssertEqualObjects(source.clientSecret, @"src_client_secret_123"); + XCTAssertEqualWithAccuracy([source.created timeIntervalSince1970], 1445277809.0, 1.0); + XCTAssertEqualObjects(source.currency, @"usd"); + XCTAssertEqual(source.flow, STPSourceFlowRedirect); + XCTAssertEqual(source.livemode, YES); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata); +#pragma clang diagnostic pop + XCTAssert(source.owner); // STPSourceOwnerTest + XCTAssertNil(source.receiver); // STPSourceReceiverTest + XCTAssert(source.redirect); // STPSourceRedirectTest + XCTAssertEqual(source.status, STPSourceStatusPending); + XCTAssertEqual(source.type, STPSourceTypeAlipay); + XCTAssertEqual(source.usage, STPSourceUsageSingleUse); + XCTAssertNil(source.verification); + NSMutableDictionary *alipayResponse = [response[@"alipay"] mutableCopy]; + [alipayResponse removeObjectForKey:@"native_url"]; // should be nil + [alipayResponse removeObjectForKey:@"statement_descriptor"]; // should be nil + XCTAssertEqualObjects(source.details, alipayResponse); + XCTAssertNil(source.cardDetails); // STPSourceCardDetailsTest + XCTAssertNil(source.sepaDebitDetails); // STPSourceSEPADebitDetailsTest + XCTAssertNotEqual(source.allResponseFields, response); // Verify is copy +} + +- (void)testDecodingSource_card { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONSourceCard]; + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(source.stripeID, @"src_123"); + XCTAssertNil(source.amount); + XCTAssertEqualObjects(source.clientSecret, @"src_client_secret_123"); + XCTAssertEqualWithAccuracy([source.created timeIntervalSince1970], 1483575790.0, 1.0); + XCTAssertNil(source.currency); + XCTAssertEqual(source.flow, STPSourceFlowNone); + XCTAssertEqual(source.livemode, NO); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata); +#pragma clang diagnostic pop + XCTAssert(source.owner); // STPSourceOwnerTest + XCTAssertNil(source.receiver); // STPSourceReceiverTest + XCTAssertNil(source.redirect); // STPSourceRedirectTest + XCTAssertEqual(source.status, STPSourceStatusChargeable); + XCTAssertEqual(source.type, STPSourceTypeCard); + XCTAssertEqual(source.usage, STPSourceUsageReusable); + XCTAssertNil(source.verification); + XCTAssertEqualObjects(source.details, response[@"card"]); + XCTAssert(source.cardDetails); // STPSourceCardDetailsTest + XCTAssertNil(source.sepaDebitDetails); // STPSourceSEPADebitDetailsTest + XCTAssertNotEqual(source.allResponseFields, response); // Verify is copy +} + +- (void)testDecodingSource_ideal { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONSourceiDEAL]; + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(source.stripeID, @"src_123"); + XCTAssertEqualObjects(source.amount, @1099); + XCTAssertEqualObjects(source.clientSecret, @"src_client_secret_123"); + XCTAssertEqualWithAccuracy([source.created timeIntervalSince1970], 1445277809.0, 1.0); + XCTAssertEqualObjects(source.currency, @"eur"); + XCTAssertEqual(source.flow, STPSourceFlowRedirect); + XCTAssertEqual(source.livemode, YES); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata); +#pragma clang diagnostic pop + XCTAssert(source.owner); // STPSourceOwnerTest + XCTAssertNil(source.receiver); // STPSourceReceiverTest + XCTAssert(source.redirect); // STPSourceRedirectTest + XCTAssertEqual(source.status, STPSourceStatusPending); + XCTAssertEqual(source.type, STPSourceTypeiDEAL); + XCTAssertEqual(source.usage, STPSourceUsageSingleUse); + XCTAssertNil(source.verification); + XCTAssertEqualObjects(source.details, response[@"ideal"]); + XCTAssertNil(source.cardDetails); // STPSourceCardDetailsTest + XCTAssertNil(source.sepaDebitDetails); // STPSourceSEPADebitDetailsTest + XCTAssertNotEqual(source.allResponseFields, response); // Verify is copy +} + +- (void)testDecodingSource_sepa_debit { + NSDictionary *response = [STPTestUtils jsonNamed:STPTestJSONSourceSEPADebit]; + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + XCTAssertEqualObjects(source.stripeID, @"src_18HgGjHNCLa1Vra6Y9TIP6tU"); + XCTAssertNil(source.amount); + XCTAssertEqualObjects(source.clientSecret, @"src_client_secret_XcBmS94nTg5o0xc9MSliSlDW"); + XCTAssertEqualWithAccuracy([source.created timeIntervalSince1970], 1464803577.0, 1.0); + XCTAssertEqualObjects(source.currency, @"eur"); + XCTAssertEqual(source.flow, STPSourceFlowNone); + XCTAssertEqual(source.livemode, NO); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wdeprecated" + XCTAssertNil(source.metadata); +#pragma clang diagnostic pop + XCTAssertEqualObjects(source.owner.name, @"Jenny Rosen"); + XCTAssert(source.owner); // STPSourceOwnerTest + XCTAssertNil(source.receiver); // STPSourceReceiverTest + XCTAssertNil(source.redirect); // STPSourceRedirectTest + XCTAssertEqual(source.status, STPSourceStatusChargeable); + XCTAssertEqual(source.type, STPSourceTypeSEPADebit); + XCTAssertEqual(source.usage, STPSourceUsageReusable); + XCTAssertEqualObjects(source.verification.attemptsRemaining, @5); + XCTAssertEqual(source.verification.status, STPSourceVerificationStatusPending); + XCTAssertEqualObjects(source.details, response[@"sepa_debit"]); + XCTAssertNil(source.cardDetails); // STPSourceCardDetailsTest + XCTAssert(source.sepaDebitDetails); // STPSourceSEPADebitDetailsTest + XCTAssertNotEqual(source.allResponseFields, response); // Verify is copy +} + +#pragma mark - STPPaymentOption Tests + +- (NSArray *)possibleAPIResponses { + return @[[STPTestUtils jsonNamed:STPTestJSONSourceCard], + [STPTestUtils jsonNamed:STPTestJSONSource3DS], + [STPTestUtils jsonNamed:STPTestJSONSourceAlipay], + [STPTestUtils jsonNamed:STPTestJSONSourceBancontact], + [STPTestUtils jsonNamed:STPTestJSONSourceEPS], + [STPTestUtils jsonNamed:STPTestJSONSourceGiropay], + [STPTestUtils jsonNamed:STPTestJSONSourceiDEAL], + [STPTestUtils jsonNamed:STPTestJSONSourceMultibanco], + [STPTestUtils jsonNamed:STPTestJSONSourceP24], + [STPTestUtils jsonNamed:STPTestJSONSourceSEPADebit], + [STPTestUtils jsonNamed:STPTestJSONSourceSofort]]; +} + +- (void)testPaymentOptionImage { + for (NSDictionary *response in [self possibleAPIResponses]) { + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + + switch (source.type) { + case STPSourceTypeCard: + STPAssertEqualImages(source.image, [STPImageLibrary brandImageForCardBrand:source.cardDetails.brand]); + break; + default: + STPAssertEqualImages(source.image, [STPImageLibrary brandImageForCardBrand:STPCardBrandUnknown]); + break; + } + } +} + +- (void)testPaymentOptionTemplateImage { + for (NSDictionary *response in [self possibleAPIResponses]) { + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + + switch (source.type) { + case STPSourceTypeCard: + STPAssertEqualImages(source.templateImage, [STPImageLibrary templatedBrandImageForCardBrand:source.cardDetails.brand]); + break; + default: + STPAssertEqualImages(source.templateImage, [STPImageLibrary templatedBrandImageForCardBrand:STPCardBrandUnknown]); + break; + } + } +} + +- (void)testPaymentOptionLabel { + for (NSDictionary *response in [self possibleAPIResponses]) { + STPSource *source = [STPSource decodedObjectFromAPIResponse:response]; + + switch (source.type) { + case STPSourceTypeBancontact: + XCTAssertEqualObjects(source.label, @"Bancontact"); + break; + case STPSourceTypeCard: + XCTAssertEqualObjects(source.label, @"Visa 5556"); + break; + case STPSourceTypeGiropay: + XCTAssertEqualObjects(source.label, @"giropay"); + break; + case STPSourceTypeiDEAL: + XCTAssertEqualObjects(source.label, @"iDEAL"); + break; + case STPSourceTypeSEPADebit: + XCTAssertEqualObjects(source.label, @"SEPA Debit"); + break; + case STPSourceTypeSofort: + XCTAssertEqualObjects(source.label, @"Sofort"); + break; + case STPSourceTypeThreeDSecure: + XCTAssertEqualObjects(source.label, @"3D Secure"); + break; + case STPSourceTypeAlipay: + XCTAssertEqualObjects(source.label, @"Alipay"); + break; + case STPSourceTypeP24: + XCTAssertEqualObjects(source.label, @"Przelewy24"); + break; + case STPSourceTypeEPS: + XCTAssertEqualObjects(source.label, @"EPS"); + break; + case STPSourceTypeMultibanco: + XCTAssertEqualObjects(source.label, @"Multibanco"); + break; + case STPSourceTypeWeChatPay: + XCTAssertEqualObjects(source.label, @"WeChat Pay"); + case STPSourceTypeKlarna: + XCTAssertEqualObjects(source.label, @"Klarna"); + case STPSourceTypeUnknown: + XCTAssertEqualObjects(source.label, [STPCard stringFromBrand:STPCardBrandUnknown]); + break; + } + + } +} + +@end diff --git a/Stripe/StripeiOSTests/STPSourceVerificationTest.m b/Stripe/StripeiOSTests/STPSourceVerificationTest.m new file mode 100644 index 00000000..7e47c22e --- /dev/null +++ b/Stripe/StripeiOSTests/STPSourceVerificationTest.m @@ -0,0 +1,110 @@ +// +// STPSourceVerificationTest.m +// Stripe +// +// Created by Joey Dong on 6/21/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +@import XCTest; + + +#import "STPTestUtils.h" + +@interface STPSourceVerification () + ++ (STPSourceVerificationStatus)statusFromString:(NSString *)string; ++ (NSString *)stringFromStatus:(STPSourceVerificationStatus)status; + +@end + +@interface STPSourceVerificationTest : XCTestCase + +@end + +@implementation STPSourceVerificationTest + +#pragma mark - STPSourceVerificationStatus Tests + +- (void)testStatusFromString { + XCTAssertEqual([STPSourceVerification statusFromString:@"pending"], STPSourceVerificationStatusPending); + XCTAssertEqual([STPSourceVerification statusFromString:@"pending"], STPSourceVerificationStatusPending); + + XCTAssertEqual([STPSourceVerification statusFromString:@"succeeded"], STPSourceVerificationStatusSucceeded); + XCTAssertEqual([STPSourceVerification statusFromString:@"SUCCEEDED"], STPSourceVerificationStatusSucceeded); + + XCTAssertEqual([STPSourceVerification statusFromString:@"failed"], STPSourceVerificationStatusFailed); + XCTAssertEqual([STPSourceVerification statusFromString:@"FAILED"], STPSourceVerificationStatusFailed); + + XCTAssertEqual([STPSourceVerification statusFromString:@"unknown"], STPSourceVerificationStatusUnknown); + XCTAssertEqual([STPSourceVerification statusFromString:@"UNKNOWN"], STPSourceVerificationStatusUnknown); + + XCTAssertEqual([STPSourceVerification statusFromString:@"garbage"], STPSourceVerificationStatusUnknown); + XCTAssertEqual([STPSourceVerification statusFromString:@"GARBAGE"], STPSourceVerificationStatusUnknown); +} + +- (void)testStringFromStatus { + NSArray *values = @[ + @(STPSourceVerificationStatusPending), + @(STPSourceVerificationStatusSucceeded), + @(STPSourceVerificationStatusFailed), + @(STPSourceVerificationStatusUnknown), + ]; + + for (NSNumber *statusNumber in values) { + STPSourceVerificationStatus status = (STPSourceVerificationStatus)[statusNumber integerValue]; + NSString *string = [STPSourceVerification stringFromStatus:status]; + + switch (status) { + case STPSourceVerificationStatusPending: + XCTAssertEqualObjects(string, @"pending"); + break; + case STPSourceVerificationStatusSucceeded: + XCTAssertEqualObjects(string, @"succeeded"); + break; + case STPSourceVerificationStatusFailed: + XCTAssertEqualObjects(string, @"failed"); + break; + case STPSourceVerificationStatusUnknown: + XCTAssertNil(string); + break; + } + } +} + +#pragma mark - Description Tests + +- (void)testDescription { + STPSourceVerification *verification = [STPSourceVerification decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"SEPADebitSource"][@"verification"]]; + XCTAssert(verification.description); +} + +#pragma mark - STPAPIResponseDecodable Tests + +- (void)testDecodedObjectFromAPIResponseRequiredFields { + NSArray *requiredFields = @[ + @"status", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STPTestUtils jsonNamed:@"SEPADebitSource"][@"verification"] mutableCopy]; + [response removeObjectForKey:field]; + + XCTAssertNil([STPSourceVerification decodedObjectFromAPIResponse:response]); + } + + XCTAssert([STPSourceVerification decodedObjectFromAPIResponse:[STPTestUtils jsonNamed:@"SEPADebitSource"][@"verification"]]); +} + +- (void)testDecodedObjectFromAPIResponseMapping { + NSDictionary *response = [STPTestUtils jsonNamed:@"SEPADebitSource"][@"verification"]; + STPSourceVerification *verification = [STPSourceVerification decodedObjectFromAPIResponse:response]; + + XCTAssertEqualObjects(verification.attemptsRemaining, @5); + XCTAssertEqual(verification.status, STPSourceVerificationStatusPending); + + XCTAssertNotEqual(verification.allResponseFields, response); + XCTAssertEqualObjects(verification.allResponseFields, response); +} + +@end diff --git a/Stripe/StripeiOSTests/STPStackViewWithSeparatorSnapshotTests.swift b/Stripe/StripeiOSTests/STPStackViewWithSeparatorSnapshotTests.swift new file mode 100644 index 00000000..da3aa600 --- /dev/null +++ b/Stripe/StripeiOSTests/STPStackViewWithSeparatorSnapshotTests.swift @@ -0,0 +1,218 @@ +// +// STPStackViewWithSeparatorSnapshotTests.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 10/23/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +@_spi(STP) import StripeUICore + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPStackViewWithSeparatorSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // recordMode = true + } + + func embedInRenderableView(_ stackView: StackViewWithSeparator) -> UIView { + let containingView = UIView(frame: CGRect(x: 0, y: 0, width: 200, height: 400)) + containingView.addSubview(stackView) + stackView.translatesAutoresizingMaskIntoConstraints = false + NSLayoutConstraint.activate([ + stackView.leadingAnchor.constraint(equalTo: containingView.leadingAnchor), + containingView.trailingAnchor.constraint(equalTo: stackView.trailingAnchor), + stackView.topAnchor.constraint(equalTo: containingView.topAnchor), + containingView.bottomAnchor.constraint(equalTo: stackView.bottomAnchor), + ]) + containingView.frame.size = containingView.systemLayoutSizeFitting( + UIView.layoutFittingCompressedSize + ) + return containingView + } + + func testHorizontal() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .horizontal + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testVertical() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .vertical + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testSingleArrangedSubviewHorizontal() { + let label1 = UILabel() + label1.text = "Label 1" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1]) + stackView.axis = .horizontal + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testSingleArrangedSubviewVertical() { + let label1 = UILabel() + label1.text = "Label 1" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1]) + stackView.axis = .vertical + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testCustomColorHorizontal() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .horizontal + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .red + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testCustomColorVertical() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .vertical + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .red + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testDisabledColor() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .horizontal + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + stackView.drawBorder = true + stackView.isUserInteractionEnabled = false + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testCustomBackgroundColor() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .horizontal + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + stackView.drawBorder = true + stackView.customBackgroundColor = .green + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testCustomDisabledColor() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .horizontal + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + stackView.customBackgroundDisabledColor = .green + stackView.drawBorder = true + stackView.isUserInteractionEnabled = false + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testPartialSeparatorHorizontal() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .horizontal + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + stackView.separatorStyle = .partial + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + + func testPartialSeparatorVertical() { + let label1 = UILabel() + label1.text = "Label 1" + let label2 = UILabel() + label2.text = "Label 2" + let label3 = UILabel() + label3.text = "Label 3" + let stackView = StackViewWithSeparator(arrangedSubviews: [label1, label2, label3]) + stackView.axis = .vertical + stackView.distribution = .fillEqually + stackView.spacing = 1 + stackView.separatorColor = .lightGray + stackView.separatorStyle = .partial + + STPSnapshotVerifyView(embedInRenderableView(stackView)) + } + +} diff --git a/Stripe/StripeiOSTests/STPStringUtilsTest.m b/Stripe/StripeiOSTests/STPStringUtilsTest.m new file mode 100644 index 00000000..1cd29577 --- /dev/null +++ b/Stripe/StripeiOSTests/STPStringUtilsTest.m @@ -0,0 +1,35 @@ +// +// STPStringUtilsTest.m +// Stripe +// +// Created by Brian Dorfman on 9/8/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import + + +@interface STPStringUtilsTest : XCTestCase + +@end + +@implementation STPStringUtilsTest + +- (void)testExpirationDateStrings { + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"12/1995"], @"12/95"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"12 / 1995"], @"12 / 95"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"12 /1995"], @"12 /95"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"1295"], @"1295"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"12/95"], @"12/95"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"08/2001"], @"08/01"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@" 08/a 2001"], @" 08/a 2001"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"20/2022"], @"20/22"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"20/202222"], @"20/22"); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@""], @""); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@" "], @" "); + XCTAssertEqualObjects([STPStringUtils expirationDateStringFromString:@"12/"], @"12/"); +} + + + +@end diff --git a/Stripe/StripeiOSTests/STPStringUtilsTest.swift b/Stripe/StripeiOSTests/STPStringUtilsTest.swift new file mode 100644 index 00000000..d2737433 --- /dev/null +++ b/Stripe/StripeiOSTests/STPStringUtilsTest.swift @@ -0,0 +1,95 @@ +// +// STPStringUtilsTest.swift +// StripeiOS Tests +// +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation + +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPStringUtilsSwiftTest: XCTestCase { + func testParseRangeSingleTagSuccess1() { + let exp = self.expectation(description: "Parsed") + STPStringUtils.parseRange( + from: "Test string", + withTag: "b" + ) { string, range in + XCTAssertTrue(NSEqualRanges(range, NSRange(location: 5, length: 6))) + XCTAssertEqual(string, "Test string") + exp.fulfill() + } + waitForExpectations(timeout: 1) + } + + func testParseRangeSingleTagSuccess2() { + let exp = self.expectation(description: "Parsed") + STPStringUtils.parseRange( + from: "Test string", + withTag: "b" + ) { string, range in + XCTAssertTrue(NSEqualRanges(range, NSRange(location: 8, length: 10))) + XCTAssertEqual(string, "Test string") + exp.fulfill() + } + waitForExpectations(timeout: 1) + } + + func testParseRangeSingleTagFailure1() { + let exp = self.expectation(description: "Parsed") + STPStringUtils.parseRange( + from: "Test string", + withTag: "a" + ) { string, range in + XCTAssertEqual(range.location, NSNotFound) + XCTAssertEqual(string, "Test string") + exp.fulfill() + } + waitForExpectations(timeout: 1) + } + + func testParseRangeSingleTagFailure2() { + let exp = self.expectation(description: "Parsed") + STPStringUtils.parseRange( + from: "Test string", + withTag: "b" + ) { string, range in + XCTAssertEqual(range.location, NSNotFound) + XCTAssertEqual(string, "Test string") + exp.fulfill() + } + waitForExpectations(timeout: 1) + } + + func testParseRangeMultiTag1() { + let exp = self.expectation(description: "Parsed") + STPStringUtils.parseRanges( + from: "Test string", + withTags: Set(["a", "b", "c"]) + ) { string, tagMap in + XCTAssertTrue(NSEqualRanges(tagMap["a"]!.rangeValue, NSRange(location: 0, length: 4))) + XCTAssertTrue(NSEqualRanges(tagMap["b"]!.rangeValue, NSRange(location: 5, length: 6))) + XCTAssertEqual(tagMap["c"]!.rangeValue.location, NSNotFound) + XCTAssertEqual(string, "Test string") + exp.fulfill() + } + waitForExpectations(timeout: 1) + } + + func testParseRangeMultiTag2() { + let exp = self.expectation(description: "Parsed") + STPStringUtils.parseRanges(from: "Test string", withTags: Set(["a", "b", "c"])) { + string, + tagMap in + XCTAssertEqual(tagMap["a"]!.rangeValue.location, NSNotFound) + XCTAssertEqual(tagMap["b"]!.rangeValue.location, NSNotFound) + XCTAssertEqual(tagMap["c"]!.rangeValue.location, NSNotFound) + XCTAssertEqual(string, "Test string") + exp.fulfill() + } + waitForExpectations(timeout: 1) + } +} diff --git a/Stripe/StripeiOSTests/STPSwiftFixtures.swift b/Stripe/StripeiOSTests/STPSwiftFixtures.swift new file mode 100644 index 00000000..4771c080 --- /dev/null +++ b/Stripe/StripeiOSTests/STPSwiftFixtures.swift @@ -0,0 +1,87 @@ +// +// STPSwiftFixtures.swift +// StripeiOS Tests +// +// Created by David Estes on 10/2/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import Foundation + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class MockEphemeralKeyProvider: NSObject, STPCustomerEphemeralKeyProvider { + func createCustomerKey( + withAPIVersion apiVersion: String, + completion: @escaping STPJSONResponseCompletionBlock + ) { + completion(STPFixtures.ephemeralKey().allResponseFields, nil) + } +} + +@objcMembers +@objc class Testing_StaticCustomerContext_Objc: Testing_StaticCustomerContext { + +} + +@objcMembers +class Testing_StaticCustomerContext: STPCustomerContext { + var customer: STPCustomer + var paymentMethods: [STPPaymentMethod] + convenience init() { + let customer = STPFixtures.customerWithSingleCardTokenSource() + let paymentMethods = [STPFixtures.paymentMethod()].compactMap { $0 } + self.init( + customer: customer, + paymentMethods: paymentMethods + ) + } + init( + customer: STPCustomer, + paymentMethods: [STPPaymentMethod] + ) { + self.customer = customer + self.paymentMethods = paymentMethods + super.init( + keyManager: STPEphemeralKeyManager( + keyProvider: MockEphemeralKeyProvider(), + apiVersion: "1", + performsEagerFetching: false + ), + apiClient: STPAPIClient.shared + ) + } + + override func retrieveCustomer(_ completion: STPCustomerCompletionBlock?) { + if let completion = completion { + completion(customer, nil) + } + } + + override func listPaymentMethodsForCustomer(completion: STPPaymentMethodsCompletionBlock?) { + if let completion = completion { + completion(paymentMethods, nil) + } + } + + var didAttach = false + override func attachPaymentMethod( + toCustomer paymentMethod: STPPaymentMethod, + completion: STPErrorBlock? + ) { + didAttach = true + if let completion = completion { + completion(nil) + } + } + + override func retrieveLastSelectedPaymentMethodIDForCustomer( + completion: @escaping (String?, Error?) -> Void + ) { + completion(nil, nil) + } +} diff --git a/Stripe/StripeiOSTests/STPTestAPIClient+Swift.swift b/Stripe/StripeiOSTests/STPTestAPIClient+Swift.swift new file mode 100644 index 00000000..7b437aba --- /dev/null +++ b/Stripe/StripeiOSTests/STPTestAPIClient+Swift.swift @@ -0,0 +1,25 @@ +// +// STPTestAPIClient+Swift.swift +// StripeiOS Tests +// +// Created by Cameron Sabol on 4/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation + +/// Just redeclares constants in STPTestingAPIClient.h for ease of use in Swift tests + +let STPTestingDefaultPublishableKey = "pk_test_ErsyMEOTudSjQR8hh0VrQr5X008sBXGOu6" +// Test account in Australia +let STPTestingAUPublishableKey = "pk_test_GNmlCJ6AFgWXm4mJYiyWSOWN00KIIiri7F" +// Test account in Mexico +let STPTestingMEXPublishableKey = + "pk_test_51GvAY5HNG4o8pO5lDEegY72rkF1TMiMyuTxSFJsmsH7U0KjTwmEf2VuXHVHecil64QA8za8Um2uSsFsfrG0BkzFo00sb1uhblF" +// Test account in SG +let STPTestingSGPublishableKey = + "pk_test_51H7oXMAOnZToJom1hqiSvNGsUVTrG1SaXRSBon9xcEp0yDFAxEh5biA4n0ty6paEsD5Mo5ps1b7Taj9WAHQzjup800m8A8Nc3u" +let STPTestingINPublishableKey = + "pk_test_51H7wmsBte6TMTRd4gph9Wm7gnQOKJwdVTCj30AhtB8MhWtlYj6v9xDn1vdCtKYGAE7cybr6fQdbQQtgvzBihE9cl00tOnrTpL9" + +let STPTestingNetworkRequestTimeout: TimeInterval = 8 diff --git a/Stripe/StripeiOSTests/STPTestUtils.h b/Stripe/StripeiOSTests/STPTestUtils.h new file mode 100644 index 00000000..66440c27 --- /dev/null +++ b/Stripe/StripeiOSTests/STPTestUtils.h @@ -0,0 +1,49 @@ +// +// STPTestUtils.h +// Stripe +// +// Created by Ben Guo on 7/14/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import +#import + +@interface STPTestUtils : NSObject + ++ (NSDictionary *)jsonNamed:(NSString *)name; + +/** + Using runtime inspection, what are all the property names for this object? + + @param object the object to introspect + @return list of property names, usable with `valueForKey:` + */ ++ (NSArray *)propertyNamesOf:(NSObject *)object; + +@end + + +/** + Custom assertion function to compare to UIImage instances. + + On iOS 9, `XCTAssertEqualObjects` incorrectly fails when provided with identical images. + + This just calls `XCTAssertEqualObjects` with the `UIImagePNGRepresentation` of each + image. Can be removed when we drop support for iOS 9. + + @param image1 First UIImage to compare + @param image2 Second UIImage to compare + */ +NS_INLINE void STPAssertEqualImages(UIImage *image1, UIImage *image2) { + XCTAssertEqualObjects(UIImagePNGRepresentation(image1), UIImagePNGRepresentation(image2)); +}; + +/** + Calls FBSnapshotVerifyView with a default 2% per-pixel color differentiation, as M1 and Intel machines render shadows differently. + @param view The view to snapshot. + @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. + */ +#define STPSnapshotVerifyView(view__, identifier__) \ +FBSnapshotVerifyViewWithPixelOptions(view__, identifier__, FBSnapshotTestCaseDefaultSuffixes(), 0.02, 0) + diff --git a/Stripe/StripeiOSTests/STPTestUtils.m b/Stripe/StripeiOSTests/STPTestUtils.m new file mode 100644 index 00000000..ce0bff74 --- /dev/null +++ b/Stripe/StripeiOSTests/STPTestUtils.m @@ -0,0 +1,72 @@ +// +// STPTestUtils.m +// Stripe +// +// Created by Ben Guo on 7/14/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import "STPTestUtils.h" + +@import ObjectiveC.runtime; + +@implementation STPTestUtils + ++ (NSDictionary *)jsonNamed:(NSString *)name { + NSData *data = [self dataFromJSONFile:name]; + if (data != nil) { + return [NSJSONSerialization JSONObjectWithData:data options:(NSJSONReadingOptions)kNilOptions error:nil]; + } + return nil; +} + ++ (NSArray *)propertyNamesOf:(NSObject *)object { + uint propertyCount; + objc_property_t *propertyList = class_copyPropertyList([object class], &propertyCount); + NSMutableArray *propertyNames = [NSMutableArray arrayWithCapacity:propertyCount]; + + for (uint i = 0; i < propertyCount; i++) { + objc_property_t property = propertyList[i]; + NSString *propertyName = [NSString stringWithUTF8String:property_getName(property)]; + [propertyNames addObject:propertyName]; + } + free(propertyList); + return propertyNames; +} + +#pragma mark - + ++ (NSBundle *)testBundle { + return [NSBundle bundleForClass:[STPTestUtils class]]; +} + ++ (NSData *)dataFromJSONFile:(NSString *)name { + NSBundle *bundle = [self testBundle]; + NSString *path = [bundle pathForResource:name ofType:@"json"]; + + if (!path) { + // Missing JSON file + return nil; + } + + NSError *error = nil; + NSString *jsonString = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:&error]; + + if (!jsonString) { + // File read error + return nil; + } + + // Strip all lines that begin with `//` + NSMutableArray *jsonLines = [[NSMutableArray alloc] init]; + + for (NSString *line in [jsonString componentsSeparatedByString:@"\n"]) { + if (![line hasPrefix:@"//"]) { + [jsonLines addObject:line]; + } + } + + return [[jsonLines componentsJoinedByString:@"\n"] dataUsingEncoding:NSUTF8StringEncoding]; +} + +@end diff --git a/Stripe/StripeiOSTests/STPTestingAPIClient.h b/Stripe/StripeiOSTests/STPTestingAPIClient.h new file mode 100644 index 00000000..ffc7727a --- /dev/null +++ b/Stripe/StripeiOSTests/STPTestingAPIClient.h @@ -0,0 +1,69 @@ +// +// STPTestingAPIClient.h +// StripeiOS +// +// Created by Cameron Sabol on 2/20/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import + +@import StripeCore; +@class STPEphemeralKey; + +NS_ASSUME_NONNULL_BEGIN + +/** + Test account info: + Account: acct_1G6m1pFY0qyl6XeW + Dashboard login/pw: fetch mobile-payments-sdk-ci + */ +static NSString * const STPTestingDefaultPublishableKey = @"pk_test_ErsyMEOTudSjQR8hh0VrQr5X008sBXGOu6"; +// Test account in Australia +static NSString * const STPTestingAUPublishableKey = @"pk_test_GNmlCJ6AFgWXm4mJYiyWSOWN00KIIiri7F"; +// Test account in Mexico +static NSString * const STPTestingMEXPublishableKey = @"pk_test_51GvAY5HNG4o8pO5lDEegY72rkF1TMiMyuTxSFJsmsH7U0KjTwmEf2VuXHVHecil64QA8za8Um2uSsFsfrG0BkzFo00sb1uhblF"; +// Test account in SG +static NSString * const STPTestingSGPublishableKey = @"pk_test_51H7oXMAOnZToJom1hqiSvNGsUVTrG1SaXRSBon9xcEp0yDFAxEh5biA4n0ty6paEsD5Mo5ps1b7Taj9WAHQzjup800m8A8Nc3u"; +// Test account in Belgium +static NSString * const STPTestingBEPublishableKey = @"pk_test_51HZi0VArGMi59tL4sIXUjwXbMiM5uSHVfsKjNXcepJ80C5niX4bCm5rJ3CeDI1vjZ5Mz55Phsmw9QqjoZTsBFoWh009RQaGx0R"; +static NSString * const STPTestingINPublishableKey = @"pk_test_51H7wmsBte6TMTRd4gph9Wm7gnQOKJwdVTCj30AhtB8MhWtlYj6v9xDn1vdCtKYGAE7cybr6fQdbQQtgvzBihE9cl00tOnrTpL9"; +// Test account in Brazil +static NSString * const STPTestingBRPublishableKey = @"pk_test_51JYFFjJQVROkWvqT6Hy9pW7uPb6UzxT3aACZ0W3olY8KunzDE9mm6OxE5W2EHcdZk7LxN6xk9zumFbZL8zvNwixR0056FVxQmt"; +// Test account in Great Britain +static NSString * const STPTestingGBPublishableKey = @"pk_test_51KmkHbGoesj9fw9QAZJlz1qY4dns8nFmLKc7rXiWKAIj8QU7NPFPwSY1h8mqRaFRKQ9njs9pVJoo2jhN6ZKSDA4h00mjcbGF7b"; + +@interface STPTestingAPIClient : NSObject + ++ (instancetype)sharedClient; + +// Set this to the Stripe SDK session for SWHTTPRecorder recording to work correctly +@property (nonatomic, readwrite) NSURLSessionConfiguration *sessionConfig; + +- (void)createPaymentIntentWithParams:(nullable NSDictionary *)params + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion; + +- (void)createPaymentIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account // nil for default or "au" for Australia test account or "mex" for Mexico test account + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion; + +- (void)createPaymentIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account // nil for default or "au" for Australia test account or "mex" for Mexico test account + apiVersion:(nullable NSString *)apiVersion // nil for default or pass with beta headers + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion; + +- (void)createSetupIntentWithParams:(nullable NSDictionary *)params + completion:(void (^)(NSString *_Nullable, NSError * _Nullable))completion; + +- (void)createSetupIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account // nil for default or "au" for Australia test account or "mex" for Mexico test account + completion:(void (^)(NSString *_Nullable, NSError * _Nullable))completion; + +- (void)createSetupIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account // nil for default or "au" for Australia test account or "mex" for Mexico test account + apiVersion:(nullable NSString *)apiVersion // nil for default or pass with beta headers + completion:(void (^)(NSString *_Nullable, NSError * _Nullable))completion; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe/StripeiOSTests/STPTestingAPIClient.m b/Stripe/StripeiOSTests/STPTestingAPIClient.m new file mode 100644 index 00000000..ce44df9e --- /dev/null +++ b/Stripe/StripeiOSTests/STPTestingAPIClient.m @@ -0,0 +1,168 @@ +// +// STPTestingAPIClient.m +// StripeiOS +// +// Created by Cameron Sabol on 2/20/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#import "STPTestingAPIClient.h" +@import Stripe; +@import StripeCore; + +static NSString * const STPTestingBackendURL = @"https://stp-mobile-ci-test-backend-e1b3.stripedemos.com/"; + +NS_ASSUME_NONNULL_BEGIN + +@implementation STPTestingAPIClient + ++ (instancetype)sharedClient { + static dispatch_once_t onceToken; + static STPTestingAPIClient *sharedClient = nil; + dispatch_once(&onceToken, ^{ + sharedClient = [[STPTestingAPIClient alloc] init]; + }); + + return sharedClient; +} + +- (instancetype)init { + self = [super init]; + self.sessionConfig = [[NSURLSession sharedSession] configuration]; + return self; +} + +- (void)createPaymentIntentWithParams:(nullable NSDictionary *)params + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion { + [self createPaymentIntentWithParams:params + account:nil + completion:completion]; +} + +- (void)createPaymentIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion { + [self createPaymentIntentWithParams:params + account:account + apiVersion:nil + completion:completion]; +} + +- (void)createPaymentIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account + apiVersion:(nullable NSString *)apiVersion + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion { + NSURLSession *session = [NSURLSession sessionWithConfiguration:self.sessionConfig]; + NSURL *url = [NSURL URLWithString:[STPTestingBackendURL stringByAppendingString:@"create_payment_intent"]]; + + NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:url]; + request.HTTPMethod = @"POST"; + [request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"]; + + NSData *postData = [NSJSONSerialization dataWithJSONObject:@{@"account" : account ?: @"", + @"create_params": params ?: @{}, + @"version": apiVersion ?: STPAPIClient.apiVersion, + } options:0 error:NULL]; + + NSURLSessionUploadTask *uploadTask = [session uploadTaskWithRequest:request + fromData:postData + completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) { + NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *)response; + if (error) { + completion(nil, error); + } else if (data == nil || httpResponse.statusCode != 200) { + dispatch_async(dispatch_get_main_queue(), ^{ + NSDictionary *userInfo = @{ + STPError.errorMessageKey: [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding], + }; + NSError *apiError = [NSError errorWithDomain:STPError.stripeDomain code:STPAPIError userInfo:userInfo]; + NSLog(@"%@", apiError); + completion(nil, apiError); + }); + } else { + NSError *jsonError = nil; + id json = [NSJSONSerialization JSONObjectWithData:data options:0 error:&jsonError]; + + if (json && + [json isKindOfClass:[NSDictionary class]] && + [json[@"secret"] isKindOfClass:[NSString class]]) { + dispatch_async(dispatch_get_main_queue(), ^{ + completion(json[@"secret"], nil); + }); + } else { + dispatch_async(dispatch_get_main_queue(), ^{ + completion(nil, jsonError); + }); + } + } + }]; + + [uploadTask resume]; +} + +- (void)createSetupIntentWithParams:(nullable NSDictionary *)params + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion { + [self createSetupIntentWithParams:params + account:nil + completion:completion]; +} + +- (void)createSetupIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account + completion:(void (^)(NSString * _Nullable, NSError * _Nullable))completion { + [self createSetupIntentWithParams:params + account:account + apiVersion:nil + completion:completion]; +} + +- (void)createSetupIntentWithParams:(nullable NSDictionary *)params + account:(nullable NSString *)account + apiVersion:(nullable NSString *)apiVersion + completion:(void (^)(NSString *_Nullable, NSError * _Nullable))completion { + NSURLSession *session = [NSURLSession sessionWithConfiguration:self.sessionConfig]; + NSURL *url = [NSURL URLWithString:[STPTestingBackendURL stringByAppendingString:@"create_setup_intent"]]; + + NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:url]; + request.HTTPMethod = @"POST"; + [request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"]; + + NSData *postData = [NSJSONSerialization dataWithJSONObject:@{@"account" : account ?: @"", + @"create_params": params ?: @{}, + @"version": apiVersion ?: STPAPIClient.apiVersion, + } options:0 error:NULL]; + + NSURLSessionUploadTask *uploadTask = [session uploadTaskWithRequest:request + fromData:postData + completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) { + NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *)response; + + if (error || data == nil || httpResponse.statusCode != 200) { + dispatch_async(dispatch_get_main_queue(), ^{ + completion(nil, error); + }); + } else { + NSError *jsonError = nil; + id json = [NSJSONSerialization JSONObjectWithData:data options:0 error:&jsonError]; + + if (json && + [json isKindOfClass:[NSDictionary class]] && + [json[@"secret"] isKindOfClass:[NSString class]]) { + dispatch_async(dispatch_get_main_queue(), ^{ + completion(json[@"secret"], nil); + }); + } else { + dispatch_async(dispatch_get_main_queue(), ^{ + completion(nil, jsonError); + }); + } + } + }]; + + [uploadTask resume]; +} + +@end + + +NS_ASSUME_NONNULL_END diff --git a/Stripe/StripeiOSTests/STPTextFieldDelegateProxyTests.swift b/Stripe/StripeiOSTests/STPTextFieldDelegateProxyTests.swift new file mode 100644 index 00000000..31c53a7c --- /dev/null +++ b/Stripe/StripeiOSTests/STPTextFieldDelegateProxyTests.swift @@ -0,0 +1,37 @@ +// +// STPTextFieldDelegateProxyTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 11/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPTextFieldDelegateProxyTests: XCTestCase { + + func testProxyShouldDeleteLeadingWhitespace() { + let textField = STPFormTextField() + textField.autoFormattingBehavior = .cardNumbers + textField.text = " " // space + + let sut = STPTextFieldDelegateProxy() + + let result = sut.textField( + textField, + shouldChangeCharactersIn: NSRange(location: 0, length: 1), + replacementString: "" + ) + + // Proxy should handle the deletion and return `false`. + XCTAssertFalse(result) + XCTAssertEqual(textField.text, "") + } + +} diff --git a/Stripe/StripeiOSTests/STPThreeDSButtonCustomizationTest.swift b/Stripe/StripeiOSTests/STPThreeDSButtonCustomizationTest.swift new file mode 100644 index 00000000..85a2e8fa --- /dev/null +++ b/Stripe/StripeiOSTests/STPThreeDSButtonCustomizationTest.swift @@ -0,0 +1,40 @@ +// +// STPThreeDSButtonCustomizationTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/17/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPThreeDSButtonCustomizationTest: XCTestCase { + func testPropertiesAreForwarded() { + let customization = STPThreeDSButtonCustomization.defaultSettings(for: .next) + customization.backgroundColor = UIColor.red + customization.cornerRadius = -1 + customization.titleStyle = .lowercase + customization.font = UIFont.italicSystemFont(ofSize: 1) + customization.textColor = UIColor.blue + + let stdsCustomization = customization.buttonCustomization + XCTAssertEqual(UIColor.red, stdsCustomization.backgroundColor) + XCTAssertEqual(stdsCustomization.backgroundColor, customization.backgroundColor) + + XCTAssertEqual(-1, stdsCustomization.cornerRadius, accuracy: 0.1) + XCTAssertEqual(stdsCustomization.cornerRadius, customization.cornerRadius, accuracy: 0.1) + + XCTAssertEqual(.lowercase, stdsCustomization.titleStyle) + XCTAssertEqual(stdsCustomization.titleStyle.rawValue, customization.titleStyle.rawValue) + + XCTAssertEqual(UIFont.italicSystemFont(ofSize: 1), stdsCustomization.font) + XCTAssertEqual(stdsCustomization.font, customization.font) + + XCTAssertEqual(UIColor.blue, stdsCustomization.textColor) + XCTAssertEqual(stdsCustomization.textColor, customization.textColor) + } +} diff --git a/Stripe/StripeiOSTests/STPThreeDSFooterCustomizationTest.swift b/Stripe/StripeiOSTests/STPThreeDSFooterCustomizationTest.swift new file mode 100644 index 00000000..81447030 --- /dev/null +++ b/Stripe/StripeiOSTests/STPThreeDSFooterCustomizationTest.swift @@ -0,0 +1,45 @@ +// +// STPThreeDSFooterCustomizationTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/17/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPThreeDSFooterCustomizationTest: XCTestCase { + func testPropertiesAreForwarded() { + let customization = STPThreeDSFooterCustomization.defaultSettings() + customization.backgroundColor = UIColor.red + customization.chevronColor = UIColor.blue + customization.headingTextColor = UIColor.green + customization.headingFont = UIFont.systemFont(ofSize: 1) + customization.font = UIFont.systemFont(ofSize: 2) + customization.textColor = UIColor.magenta + + let stdsCustomization = customization.footerCustomization + + XCTAssertEqual(UIColor.red, stdsCustomization.backgroundColor) + XCTAssertEqual(stdsCustomization.backgroundColor, customization.backgroundColor) + + XCTAssertEqual(UIColor.blue, stdsCustomization.chevronColor) + XCTAssertEqual(stdsCustomization.chevronColor, customization.chevronColor) + + XCTAssertEqual(UIColor.green, stdsCustomization.headingTextColor) + XCTAssertEqual(stdsCustomization.headingTextColor, customization.headingTextColor) + + XCTAssertEqual(UIFont.systemFont(ofSize: 1), stdsCustomization.headingFont) + XCTAssertEqual(stdsCustomization.headingFont, customization.headingFont) + + XCTAssertEqual(UIFont.systemFont(ofSize: 2), stdsCustomization.font) + XCTAssertEqual(stdsCustomization.font, customization.font) + + XCTAssertEqual(UIColor.magenta, stdsCustomization.textColor) + XCTAssertEqual(stdsCustomization.textColor, customization.textColor) + } +} diff --git a/Stripe/StripeiOSTests/STPThreeDSLabelCustomizationTest.swift b/Stripe/StripeiOSTests/STPThreeDSLabelCustomizationTest.swift new file mode 100644 index 00000000..bd546dfc --- /dev/null +++ b/Stripe/StripeiOSTests/STPThreeDSLabelCustomizationTest.swift @@ -0,0 +1,37 @@ +// +// STPThreeDSLabelCustomizationTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/17/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPThreeDSLabelCustomizationTest: XCTestCase { + func testPropertiesAreForwarded() { + let customization = STPThreeDSLabelCustomization.defaultSettings() + customization.headingFont = UIFont.systemFont(ofSize: 1) + customization.headingTextColor = UIColor.red + customization.font = UIFont.systemFont(ofSize: 2) + customization.textColor = UIColor.blue + + let stdsCustomization = customization.labelCustomization + + XCTAssertEqual(UIFont.systemFont(ofSize: 1), stdsCustomization.headingFont) + XCTAssertEqual(stdsCustomization.headingFont, customization.headingFont) + + XCTAssertEqual(UIColor.red, stdsCustomization.headingTextColor) + XCTAssertEqual(stdsCustomization.headingTextColor, customization.headingTextColor) + + XCTAssertEqual(UIFont.systemFont(ofSize: 2), stdsCustomization.font) + XCTAssertEqual(stdsCustomization.font, customization.font) + + XCTAssertEqual(UIColor.blue, stdsCustomization.textColor) + XCTAssertEqual(stdsCustomization.textColor, customization.textColor) + } +} diff --git a/Stripe/StripeiOSTests/STPThreeDSNavigationBarCustomizationTest.swift b/Stripe/StripeiOSTests/STPThreeDSNavigationBarCustomizationTest.swift new file mode 100644 index 00000000..a2dcdf5b --- /dev/null +++ b/Stripe/StripeiOSTests/STPThreeDSNavigationBarCustomizationTest.swift @@ -0,0 +1,48 @@ +// +// STPThreeDSNavigationBarCustomizationTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/17/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPThreeDSNavigationBarCustomizationTest: XCTestCase { + func testPropertiesAreForwarded() { + let customization = STPThreeDSNavigationBarCustomization.defaultSettings() + customization.font = UIFont.italicSystemFont(ofSize: 1) + customization.textColor = UIColor.blue + customization.barTintColor = UIColor.red + customization.barStyle = UIBarStyle.blackOpaque + customization.translucent = false + customization.headerText = "foo" + customization.buttonText = "bar" + + let stdsCustomization = customization.navigationBarCustomization + XCTAssertEqual(UIFont.italicSystemFont(ofSize: 1), stdsCustomization.font) + XCTAssertEqual(stdsCustomization.font, customization.font) + + XCTAssertEqual(UIColor.blue, stdsCustomization.textColor) + XCTAssertEqual(stdsCustomization.textColor, customization.textColor) + + XCTAssertEqual(UIColor.red, stdsCustomization.barTintColor) + XCTAssertEqual(stdsCustomization.barTintColor, customization.barTintColor) + + XCTAssertEqual(UIBarStyle.blackOpaque, stdsCustomization.barStyle) + XCTAssertEqual(stdsCustomization.barStyle, customization.barStyle) + + XCTAssertEqual(false, stdsCustomization.translucent) + XCTAssertEqual(stdsCustomization.translucent, customization.translucent) + + XCTAssertEqual("foo", stdsCustomization.headerText) + XCTAssertEqual(stdsCustomization.headerText, customization.headerText) + + XCTAssertEqual("bar", stdsCustomization.buttonText) + XCTAssertEqual(stdsCustomization.buttonText, customization.buttonText) + } +} diff --git a/Stripe/StripeiOSTests/STPThreeDSSelectionCustomizationTest.swift b/Stripe/StripeiOSTests/STPThreeDSSelectionCustomizationTest.swift new file mode 100644 index 00000000..344f04e1 --- /dev/null +++ b/Stripe/StripeiOSTests/STPThreeDSSelectionCustomizationTest.swift @@ -0,0 +1,42 @@ +// +// STPThreeDSSelectionCustomizationTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/18/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPThreeDSSelectionCustomizationTest: XCTestCase { + func testPropertiesAreForwarded() { + let customization = STPThreeDSSelectionCustomization.defaultSettings() + customization.primarySelectedColor = UIColor.red + customization.secondarySelectedColor = UIColor.blue + customization.unselectedBorderColor = UIColor.brown + customization.unselectedBackgroundColor = UIColor.cyan + + let stdsCustomization = customization.selectionCustomization + XCTAssertEqual(UIColor.red, stdsCustomization.primarySelectedColor) + XCTAssertEqual(stdsCustomization.primarySelectedColor, customization.primarySelectedColor) + + XCTAssertEqual(UIColor.blue, stdsCustomization.secondarySelectedColor) + XCTAssertEqual( + stdsCustomization.secondarySelectedColor, + customization.secondarySelectedColor + ) + + XCTAssertEqual(UIColor.brown, stdsCustomization.unselectedBorderColor) + XCTAssertEqual(stdsCustomization.unselectedBorderColor, customization.unselectedBorderColor) + + XCTAssertEqual(UIColor.cyan, stdsCustomization.unselectedBackgroundColor) + XCTAssertEqual( + stdsCustomization.unselectedBackgroundColor, + customization.unselectedBackgroundColor + ) + } +} diff --git a/Stripe/StripeiOSTests/STPThreeDSTextFieldCustomizationTest.swift b/Stripe/StripeiOSTests/STPThreeDSTextFieldCustomizationTest.swift new file mode 100644 index 00000000..2bc87ac9 --- /dev/null +++ b/Stripe/StripeiOSTests/STPThreeDSTextFieldCustomizationTest.swift @@ -0,0 +1,48 @@ +// +// STPThreeDSTextFieldCustomizationTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/18/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPThreeDSTextFieldCustomizationTest: XCTestCase { + func testPropertiesAreForwarded() { + let customization = STPThreeDSTextFieldCustomization.defaultSettings() + customization.font = UIFont.italicSystemFont(ofSize: 1) + customization.textColor = UIColor.blue + customization.borderWidth = -1 + customization.borderColor = UIColor.red + customization.cornerRadius = -8 + customization.keyboardAppearance = .alert + customization.placeholderTextColor = UIColor.green + + let stdsCustomization = customization.textFieldCustomization + XCTAssertEqual(UIFont.italicSystemFont(ofSize: 1), stdsCustomization.font) + XCTAssertEqual(stdsCustomization.font, customization.font) + + XCTAssertEqual(UIColor.blue, stdsCustomization.textColor) + XCTAssertEqual(stdsCustomization.textColor, customization.textColor) + + XCTAssertEqual(-1, stdsCustomization.borderWidth, accuracy: 0.1) + XCTAssertEqual(stdsCustomization.borderWidth, customization.borderWidth, accuracy: 0.1) + + XCTAssertEqual(UIColor.red, stdsCustomization.borderColor) + XCTAssertEqual(stdsCustomization.borderColor, customization.borderColor) + + XCTAssertEqual(-8, stdsCustomization.cornerRadius, accuracy: 0.1) + XCTAssertEqual(stdsCustomization.cornerRadius, customization.cornerRadius, accuracy: 0.1) + + XCTAssertEqual(UIKeyboardAppearance.alert, stdsCustomization.keyboardAppearance) + XCTAssertEqual(stdsCustomization.keyboardAppearance, customization.keyboardAppearance) + + XCTAssertEqual(UIColor.green, stdsCustomization.placeholderTextColor) + XCTAssertEqual(stdsCustomization.placeholderTextColor, customization.placeholderTextColor) + } +} diff --git a/Stripe/StripeiOSTests/STPThreeDSUICustomizationTest.swift b/Stripe/StripeiOSTests/STPThreeDSUICustomizationTest.swift new file mode 100644 index 00000000..392c8422 --- /dev/null +++ b/Stripe/StripeiOSTests/STPThreeDSUICustomizationTest.swift @@ -0,0 +1,137 @@ +// +// STPThreeDSUICustomizationTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 6/17/19. +// Copyright © 2019 Stripe, Inc. All rights reserved. +// + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class STPThreeDSUICustomizationTest: XCTestCase { + func testPropertiesPassedThrough() { + let customization = STPThreeDSUICustomization.defaultSettings() + + // Maintains button customization objects + customization.buttonCustomization(for: .next).backgroundColor = UIColor.cyan + customization.buttonCustomization(for: .resend).backgroundColor = UIColor.cyan + customization.buttonCustomization(for: .submit).backgroundColor = UIColor.cyan + customization.buttonCustomization(for: .continue).backgroundColor = UIColor.cyan + customization.buttonCustomization(for: .cancel).backgroundColor = UIColor.cyan + XCTAssertEqual( + customization.uiCustomization.buttonCustomization(for: .next).backgroundColor, + UIColor.cyan + ) + XCTAssertEqual( + customization.uiCustomization.buttonCustomization(for: .resend).backgroundColor, + UIColor.cyan + ) + XCTAssertEqual( + customization.uiCustomization.buttonCustomization(for: .submit).backgroundColor, + UIColor.cyan + ) + XCTAssertEqual( + customization.uiCustomization.buttonCustomization(for: .continue).backgroundColor, + UIColor.cyan + ) + XCTAssertEqual( + customization.uiCustomization.buttonCustomization(for: .cancel).backgroundColor, + UIColor.cyan + ) + + let buttonCustomization = STPThreeDSButtonCustomization.defaultSettings(for: .next) + customization.setButtonCustomization(buttonCustomization, for: .next) + XCTAssertEqual( + customization.uiCustomization.buttonCustomization(for: .next), + buttonCustomization.buttonCustomization + ) + + // Footer + customization.footerCustomization.backgroundColor = UIColor.cyan + XCTAssertEqual( + customization.uiCustomization.footerCustomization.backgroundColor, + UIColor.cyan + ) + + let footerCustomization = STPThreeDSFooterCustomization.defaultSettings() + customization.footerCustomization = footerCustomization + XCTAssertEqual( + customization.uiCustomization.footerCustomization, + footerCustomization.footerCustomization + ) + + // Label + customization.labelCustomization.textColor = UIColor.cyan + XCTAssertEqual(customization.uiCustomization.labelCustomization.textColor, UIColor.cyan) + + let labelCustomization = STPThreeDSLabelCustomization.defaultSettings() + customization.labelCustomization = labelCustomization + XCTAssertEqual( + customization.uiCustomization.labelCustomization, + labelCustomization.labelCustomization + ) + + // Navigation Bar + customization.navigationBarCustomization.textColor = UIColor.cyan + XCTAssertEqual( + customization.uiCustomization.navigationBarCustomization.textColor, + UIColor.cyan + ) + + let navigationBar = STPThreeDSNavigationBarCustomization.defaultSettings() + customization.navigationBarCustomization = navigationBar + XCTAssertEqual( + customization.uiCustomization.navigationBarCustomization, + navigationBar.navigationBarCustomization + ) + + // Selection + customization.selectionCustomization.primarySelectedColor = UIColor.cyan + XCTAssertEqual( + customization.uiCustomization.selectionCustomization.primarySelectedColor, + UIColor.cyan + ) + + let selection = STPThreeDSSelectionCustomization.defaultSettings() + customization.selectionCustomization = selection + XCTAssertEqual( + customization.uiCustomization.selectionCustomization, + selection.selectionCustomization + ) + + // Text Field + customization.textFieldCustomization.textColor = UIColor.cyan + XCTAssertEqual(customization.uiCustomization.textFieldCustomization.textColor, UIColor.cyan) + + let textField = STPThreeDSTextFieldCustomization.defaultSettings() + customization.textFieldCustomization = textField + XCTAssertEqual( + customization.uiCustomization.textFieldCustomization, + textField.textFieldCustomization + ) + + // Other + customization.backgroundColor = UIColor.red + customization.activityIndicatorViewStyle = UIActivityIndicatorView.Style.whiteLarge + customization.blurStyle = UIBlurEffect.Style.dark + + XCTAssertEqual(UIColor.red, customization.backgroundColor) + XCTAssertEqual(customization.backgroundColor, customization.uiCustomization.backgroundColor) + + XCTAssertEqual( + UIActivityIndicatorView.Style.whiteLarge, + customization.activityIndicatorViewStyle + ) + XCTAssertEqual( + customization.activityIndicatorViewStyle, + customization.uiCustomization.activityIndicatorViewStyle + ) + + XCTAssertEqual(UIBlurEffect.Style.dark, customization.blurStyle) + XCTAssertEqual(customization.blurStyle, customization.uiCustomization.blurStyle) + } +} diff --git a/Stripe/StripeiOSTests/STPTokenTest.m b/Stripe/StripeiOSTests/STPTokenTest.m new file mode 100644 index 00000000..802aed9b --- /dev/null +++ b/Stripe/StripeiOSTests/STPTokenTest.m @@ -0,0 +1,58 @@ +// +// STPTokenTest.m +// Stripe +// +// Created by Saikat Chakrabarti on 11/9/12. +// +// + +@import XCTest; + + + +@interface STPTokenTest : XCTestCase +@end + +@implementation STPTokenTest + +- (NSDictionary *)buildTestTokenResponse { + NSDictionary *cardDict = @{ + @"id": @"card_123", + @"exp_month": @"12", + @"exp_year": @"2013", + @"name": @"Smerlock Smolmes", + @"address_line1": @"221A Baker Street", + @"address_city": @"New York", + @"address_state": @"NY", + @"address_zip": @"12345", + @"address_country": @"US", + @"last4": @"1234", + @"brand": @"Visa", + @"fingerprint": @"Fingolfin", + @"country": @"JP", + }; + + NSDictionary *tokenDict = @{ @"id": @"id_for_token", @"object": @"token", @"livemode": @NO, @"created": @1353025450.0, @"used": @NO, @"card": cardDict, @"type": @"card" }; + return tokenDict; +} + +- (void)testCreatingTokenWithAttributeDictionarySetsAttributes { + STPToken *token = [STPToken decodedObjectFromAPIResponse:[self buildTestTokenResponse]]; + XCTAssertEqualObjects([token tokenId], @"id_for_token", @"Generated token has the correct id"); + XCTAssertEqual([token livemode], NO, @"Generated token has the correct livemode"); + XCTAssertEqual([token type], STPTokenTypeCard, @"Generated token has incorrect type"); + + XCTAssertEqualWithAccuracy([[token created] timeIntervalSince1970], 1353025450.0, 1.0, @"Generated token has the correct created time"); +} + +- (void)testCreatingTokenSetsAdditionalResponseFields { + NSMutableDictionary *tokenResponse = [[self buildTestTokenResponse] mutableCopy]; + tokenResponse[@"foo"] = @"bar"; + STPToken *token = [STPToken decodedObjectFromAPIResponse:tokenResponse]; + NSDictionary *allResponseFields = token.allResponseFields; + XCTAssertEqualObjects(allResponseFields[@"foo"], @"bar"); + XCTAssertEqualObjects(allResponseFields[@"livemode"], @NO); + XCTAssertNil(allResponseFields[@"baz"]); +} + +@end diff --git a/Stripe/StripeiOSTests/STPUIVCStripeParentViewControllerTests.m b/Stripe/StripeiOSTests/STPUIVCStripeParentViewControllerTests.m new file mode 100644 index 00000000..d2d2951b --- /dev/null +++ b/Stripe/StripeiOSTests/STPUIVCStripeParentViewControllerTests.m @@ -0,0 +1,46 @@ +// +// STPUIVCStripeParentViewControllerTests.m +// Stripe +// +// Created by Jack Flintermann on 1/19/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import +#import + + +@interface TestViewController : UIViewController +@end + +@implementation TestViewController +@end + +@interface STPUIVCStripeParentViewControllerTests : XCTestCase +@end + +@implementation STPUIVCStripeParentViewControllerTests + +- (void)testNilParent { + UIViewController *vc = [UIViewController new]; + XCTAssertNil([vc stp_parentViewControllerOfClass:[UIViewController class]]); +} + +- (void)testNavigationController { + UIViewController *vc = [UIViewController new]; + UINavigationController *nav = [[UINavigationController alloc] initWithRootViewController:vc]; + UINavigationController *parent = (UINavigationController *)[vc stp_parentViewControllerOfClass:[UINavigationController class]]; + XCTAssertEqual(nav, parent); +} + +- (void)testDeepHeirarchy { + UIViewController *topLevel = [TestViewController new]; + UIViewController *vc = [UIViewController new]; + UINavigationController *nav = [[UINavigationController alloc] initWithRootViewController:vc]; + [topLevel addChildViewController:nav]; + [nav didMoveToParentViewController:topLevel]; + TestViewController *parent = (TestViewController *)[vc stp_parentViewControllerOfClass:[TestViewController class]]; + XCTAssertEqual(topLevel, parent); +} + +@end diff --git a/Stripe/StripeiOSTests/SWHttpTrafficRecorder.h b/Stripe/StripeiOSTests/SWHttpTrafficRecorder.h new file mode 100755 index 00000000..c6959ce8 --- /dev/null +++ b/Stripe/StripeiOSTests/SWHttpTrafficRecorder.h @@ -0,0 +1,200 @@ +/*********************************************************************************** + * Copyright 2015 Capital One Services, LLC + * SPDX-License-Identifier: Apache-2.0 + * SPDX-Copyright: Copyright (c) Capital One Services, LLC + + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + + * http://www.apache.org/licenses/LICENSE-2.0 + + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + ***********************************************************************************/ + + +//////////////////////////////////////////////////////////////////////////////// + +// Created by Jinlian (Sunny) Wang on 8/23/15. + +#import + +//! Project version number for SWHttpTrafficRecorder. +FOUNDATION_EXPORT double SWHttpTrafficRecorderVersionNumber; + +//! Project version string for SWHttpTrafficRecorder. +FOUNDATION_EXPORT const unsigned char SWHttpTrafficRecorderVersionString[]; + +/** + * Recording formats that is supported by SWHttpTrafficRecorder. + */ +typedef NS_ENUM(NSInteger, SWHTTPTrafficRecordingFormat) { + /* Custom format when the recorder records a request through an optional createFileInCustomFormatBlock block. */ + SWHTTPTrafficRecordingFormatCustom = -1, + + /* For BodyOnly format, the recorder creates a recorded file for each request using only its response body. If it is a JSON response, the file uses .json extension. Otherwise, .txt extenion is used. */ + SWHTTPTrafficRecordingFormatBodyOnly = 1, + + /* For Mocktail format, the recorder creates a recorded file for each request and its response in format that is defined by Mocktail framework at https://github.com/puls/objc-mocktail. The file uses .tail extension. */ + SWHTTPTrafficRecordingFormatMocktail = 2, + + /* For HTTPMessage format, the recorder creates a recorded file for each request and its response in format can be deserialized into a CFHTTPMessageRef raw HTTP Message. 'curl -is' also outputs in this format. */ + SWHTTPTrafficRecordingFormatHTTPMessage = 3 +}; + +/** + * Error codes for SWHttpTrafficRecorder. + */ +typedef NS_ENUM(NSInteger, SWHttpTrafficRecorderError) { + /** The specified path does not exist and cannot be created */ + SWHttpTrafficRecorderErrorPathFailedToCreate = 1, + /** The specified path was not writable */ + SWHttpTrafficRecorderErrorPathNotWritable +}; + +/** + * Recording progress that is reported by SWHttpTrafficRecorder at each phase of recording a request and its response. + */ +typedef NS_ENUM(NSInteger, SWHTTPTrafficRecordingProgressKind) { + /* A HTTP Request is received by the recorder. */ + SWHTTPTrafficRecordingProgressReceived = 1, + + /* A HTTP Request is skipped by the recorder for recording. */ + SWHTTPTrafficRecordingProgressSkipped = 2, + + /* The recorder starts downloading the response for a request. */ + SWHTTPTrafficRecordingProgressStarted = 3, + + /* The recorder finishes downloading the response for a request. */ + SWHTTPTrafficRecordingProgressLoaded = 4, + + /* The recorder finishes recording the response for a request. */ + SWHTTPTrafficRecordingProgressRecorded = 5, + + /* The recorder fails to download the response for a request for whatever reason. */ + SWHTTPTrafficRecordingProgressFailedToLoad = 6, + + /* The recorder fails to record the response for a request for whatever reason. */ + SWHTTPTrafficRecordingProgressFailedToRecord = 7 +}; + +/* The key in a recording progress info dictionary whose value indicates the current NSURLRequest that is being recorded.*/ +FOUNDATION_EXPORT NSString * const SWHTTPTrafficRecordingProgressRequestKey; + +/* The key in a recording progress info dictionary whose value indicates the current NSHTTPURLResponse that is being recorded.*/ +FOUNDATION_EXPORT NSString * const SWHTTPTrafficRecordingProgressResponseKey; + +/* The key in a recording progress info dictionary whose value indicates the current NSData response body that is being recorded.*/ +FOUNDATION_EXPORT NSString * const SWHTTPTrafficRecordingProgressBodyDataKey; + +/* The key in a recording progress info dictionary whose value indicates the current file path that is used for the recorded file.*/ +FOUNDATION_EXPORT NSString * const SWHTTPTrafficRecordingProgressFilePathKey; + +/* The key in a recording progress info dictionary whose value indicates the current recording format.*/ +FOUNDATION_EXPORT NSString * const SWHTTPTrafficRecordingProgressFileFormatKey; + +/* The key in a recording progress info dictionary whose value indicates the NSErrror which fails the recording.*/ +FOUNDATION_EXPORT NSString * const SWHTTPTrafficRecordingProgressErrorKey; + +/* The error domain for SWHttpTrafficRecorder. */ +FOUNDATION_EXPORT NSString * const SWHttpTrafficRecorderErrorDomain; + +/** An optional delegate SWHttpTrafficRecorder uses to report its recording progress. + */ +@protocol SWHttpTrafficRecordingProgressDelegate + +/** + * Delegate method to be called by the recorder to update its current progress. + * @param currentProgress The current progress of the recording. + * @param info A recording progress info dictionary where its values (including a request object, its response, response body data, file path, recording format and NSError) can be retrieved through different keys. The available values depend on the current progress. + */ +- (void)updateRecordingProgress:(SWHTTPTrafficRecordingProgressKind)currentProgress userInfo:(NSDictionary *)info; + +@end + +/** + * An SWHttpTrafficRecorder lets you intercepts the http requests made by an application and records their responses in a specified format. There are three built-in formats supported: ResponseBodyOnly, Mocktail and HTTPMessage. These formats are widely used by various mocking/stubbing frameworks such as Mocktail(https://github.com/puls/objc-mocktail), OHHTTPStubs(https://github.com/AliSoftware/OHHTTPStubs/tree/master/OHHTTPStubs), Nocilla(https://github.com/luisobo/Nocilla), etc. You can also use it to monitor the traffic for debugging purpose. + */ +@interface SWHttpTrafficRecorder : NSObject + +/** + * Returns the shared recorder object. + */ ++ (instancetype)sharedRecorder; + +/** + * Method to start recording using default path. + */ +- (BOOL)startRecording; + +/** + * Method to start recording and saves recorded files at a specified location. + * @param path The path where recorded files are saved. + * @param error An out value that returns any error encountered while accessing the recordingPath. Returns an NSError object if any error; otherwise returns nil. + */ +- (BOOL)startRecordingAtPath:(NSString *)recordingPath error:(NSError **) error; + +/** + * Method to start recording and saves recorded files at a specified location using given session configuration. + * @param recordingPath The path where recorded files are saved. + * @param sessionConfig The NSURLSessionConfiguration which will be modified. + * @param error An out value that returns any error encountered while accessing the recordingPath. Returns an NSError object if any error; otherwise returns nil. + */ +- (BOOL)startRecordingAtPath:(NSString *)recordingPath forSessionConfiguration:(NSURLSessionConfiguration *)sessionConfig error:(NSError **) error; + +/** + * Method to stop recording. + */ +- (void)stopRecording; + +/** + * A Boolean value which indicates whether the recording is recording traffic. + */ +@property(nonatomic, readonly, assign) BOOL isRecording; + +/** + * A Enum value which indicates the format the recording is using to record traffic. + */ +@property(nonatomic, assign) SWHTTPTrafficRecordingFormat recordingFormat; + +/** + * A Dictionary containing Regex/Token pairs for replacement in response data + */ +@property(nonatomic, assign) NSMutableDictionary *replacementDict; + +/** + * The delegate where the recording progress are reported. + */ +@property(nonatomic, assign) id progressDelegate; + +/** + * The optional block (if provided) to be applied to every request to determine whether the request shall be recorded by the recorder. It takes a NSURLRequest as parameter and returns a Boolean value that indicates whether the request shall be recorded. + */ +@property(nonatomic, copy) BOOL(^recordingTestBlock)(NSURLRequest *request); + +/** + * The optional block (if provided) to be applied to every request to determine whether the response body shall be base64 encodes before recording. It takes a NSURLRequest as parameter and returns a Boolean value that indicates whether the response body shall be base64 encoded. + */ +@property(nonatomic, copy) BOOL(^base64TestBlock)(NSURLRequest *request, NSURLResponse *response); + +/** + * The optional block (if provided) to be applied to every request to determine what file name is to be used while creating the recorded file. It takes a NSURLRequest and a default name that is generated by the recorder as parameters and returns a NSString value which is used as filename while creating the recorded file. + */ +@property(nonatomic, copy) NSString*(^fileNamingBlock)(NSURLRequest *request, NSURLResponse *response, NSString *defaultName); + +/** + * The optional block (if provided) to be applied to every request to determine what regular expression is to be used while creating a recorded file of Mocktail format. It takes a NSURLRequest and a default regular expression pattern that is generated by the recorder as parameters and returns a NSString value which is used as the regular expression pattern while creating the recorded file. + */ +@property(nonatomic, copy) NSString*(^urlRegexPatternBlock)(NSURLRequest *request, NSString *defaultPattern); + +/** + * The optional block (if provided) to be applied to every request to create the recorded file when the recording format is custom. It takes a NSURLRequest, its response, a body data and a filePath as parameters and be expected to create the recorded file at the filePath. + */ +@property(nonatomic, copy) NSString*(^createFileInCustomFormatBlock)(NSURLRequest *request, NSURLResponse *response, NSData *bodyData, NSString *filePath); + +@end diff --git a/Stripe/StripeiOSTests/SWHttpTrafficRecorder.m b/Stripe/StripeiOSTests/SWHttpTrafficRecorder.m new file mode 100755 index 00000000..5d1f7df1 --- /dev/null +++ b/Stripe/StripeiOSTests/SWHttpTrafficRecorder.m @@ -0,0 +1,513 @@ +/*********************************************************************************** + * Copyright 2015 Capital One Services, LLC + * SPDX-License-Identifier: Apache-2.0 + * SPDX-Copyright: Copyright (c) Capital One Services, LLC + + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + + * http://www.apache.org/licenses/LICENSE-2.0 + + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + ***********************************************************************************/ + + +//////////////////////////////////////////////////////////////////////////////// + +// Created by Jinlian (Sunny) Wang on 8/23/15. + +#import "SWHttpTrafficRecorder.h" + +NSString * const SWHTTPTrafficRecordingProgressRequestKey = @"REQUEST_KEY"; +NSString * const SWHTTPTrafficRecordingProgressResponseKey = @"RESPONSE_KEY"; +NSString * const SWHTTPTrafficRecordingProgressBodyDataKey = @"BODY_DATA_KEY"; +NSString * const SWHTTPTrafficRecordingProgressFilePathKey = @"FILE_PATH_KEY"; +NSString * const SWHTTPTrafficRecordingProgressFileFormatKey= @"FILE_FORMAT_KEY"; +NSString * const SWHTTPTrafficRecordingProgressErrorKey = @"ERROR_KEY"; + +NSString * const SWHttpTrafficRecorderErrorDomain = @"RECORDER_ERROR_DOMAIN"; + +@interface SWHttpTrafficRecorder() +@property(nonatomic, assign, readwrite) BOOL isRecording; +@property(nonatomic, strong) NSString *recordingPath; +@property(nonatomic, assign) int fileNo; +@property(nonatomic, strong) NSOperationQueue *fileCreationQueue; +@property(nonatomic, strong) NSURLSessionConfiguration *sessionConfig; +@property(nonatomic, assign) NSUInteger runTimeStamp; +@property(nonatomic, strong) NSDictionary *fileExtensionMapping; +@end + +@interface SWRecordingProtocol : NSURLProtocol @end + +@implementation SWHttpTrafficRecorder + ++ (instancetype)sharedRecorder +{ + static SWHttpTrafficRecorder *shared = nil; + static dispatch_once_t onceToken; + dispatch_once(&onceToken, ^{ + shared = self.new; + shared.isRecording = NO; + shared.fileNo = 0; + shared.fileCreationQueue = [[NSOperationQueue alloc] init]; + shared.runTimeStamp = 0; + shared.recordingFormat = SWHTTPTrafficRecordingFormatMocktail; + }); + return shared; +} + +- (BOOL)startRecording{ + return [self startRecordingAtPath:nil forSessionConfiguration:nil error:nil]; +} + +- (BOOL)startRecordingAtPath:(NSString *)recordingPath error:(NSError **) error { + return [self startRecordingAtPath:recordingPath forSessionConfiguration:nil error:error]; +} + +- (BOOL)startRecordingAtPath:(NSString *)recordingPath forSessionConfiguration:(NSURLSessionConfiguration *)sessionConfig error:(NSError **) error { + if(!self.isRecording){ + if(recordingPath){ + self.recordingPath = recordingPath; + NSFileManager *fileManager = [NSFileManager defaultManager]; + if(![fileManager fileExistsAtPath:recordingPath]){ + NSError *bError = nil; + if(![fileManager createDirectoryAtPath:recordingPath withIntermediateDirectories:YES attributes:nil error:&bError]){ + if(error){ + *error = [NSError errorWithDomain:SWHttpTrafficRecorderErrorDomain code:SWHttpTrafficRecorderErrorPathFailedToCreate userInfo:@{NSLocalizedDescriptionKey: [NSString stringWithFormat:@"Path '%@' does not exist and error while creating it.", recordingPath], NSUnderlyingErrorKey: bError}]; + } + return NO; + } + } else if(![fileManager isWritableFileAtPath:recordingPath]){ + if (error){ + *error = [NSError errorWithDomain:SWHttpTrafficRecorderErrorDomain code:SWHttpTrafficRecorderErrorPathNotWritable userInfo:@{NSLocalizedDescriptionKey: [NSString stringWithFormat:@"Path '%@' is not writable.", recordingPath]}]; + } + return NO; + } + } else { + self.recordingPath = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)[0]; + } + + self.fileNo = 0; + self.runTimeStamp = (NSUInteger)[NSDate timeIntervalSinceReferenceDate]; + } + if(sessionConfig){ + self.sessionConfig = sessionConfig; + NSMutableOrderedSet *mutableProtocols = [[NSMutableOrderedSet alloc] initWithArray:sessionConfig.protocolClasses]; + [mutableProtocols insertObject:[SWRecordingProtocol class] atIndex:0]; + sessionConfig.protocolClasses = [mutableProtocols array]; + } + else { + [NSURLProtocol registerClass:[SWRecordingProtocol class]]; + } + + self.isRecording = YES; + + return YES; +} + +- (void)stopRecording{ + if(self.isRecording){ + if(self.sessionConfig) { + NSMutableArray *mutableProtocols = [[NSMutableArray alloc] initWithArray:self.sessionConfig.protocolClasses]; + [mutableProtocols removeObject:[SWRecordingProtocol class]]; + self.sessionConfig.protocolClasses = mutableProtocols; + self.sessionConfig = nil; + } + else { + [NSURLProtocol unregisterClass:[SWRecordingProtocol class]]; + } + } + self.isRecording = NO; +} + +- (int)increaseFileNo{ + @synchronized(self) { + return self.fileNo++; + } +} + +- (NSDictionary *)fileExtensionMapping{ + if(!_fileExtensionMapping){ + _fileExtensionMapping = @{ + @"application/json": @"json", @"image/png": @"png", @"image/jpeg" : @"jpg", + @"image/gif": @"gif", @"image/bmp": @"bmp", @"text/plain": @"txt", + @"text/css": @"css", @"text/html": @"html", @"application/javascript": @"js", + @"text/javascript": @"js", @"application/xml": @"xml", @"text/xml": @"xml", + @"image/tiff": @"tiff", @"image/x-tiff": @"tiff" + }; + } + return _fileExtensionMapping; +} + +@end + + +//////////////////////////////////////////////////////////////////////////////// +#pragma mark - Private Protocol Class + + +static NSString * const SWRecordingLProtocolHandledKey = @"SWRecordingLProtocolHandledKey"; + +@interface SWRecordingProtocol () + +@property (nonatomic, strong) NSURLSessionDataTask *dataTask; +@property (nonatomic, strong) NSMutableData *mutableData; +@property (nonatomic, strong) NSURLResponse *response; +@property (nonatomic, strong) NSURLSession *session; + +@end + + +@implementation SWRecordingProtocol + +#pragma mark - NSURLProtocol overrides + ++ (BOOL)canInitWithRequest:(NSURLRequest *)request { + BOOL isHTTP = [request.URL.scheme isEqualToString:@"https"] || [request.URL.scheme isEqualToString:@"http"]; + if ([NSURLProtocol propertyForKey:SWRecordingLProtocolHandledKey inRequest:request] || !isHTTP) { + return NO; + } + + [self updateRecorderProgressDelegate:SWHTTPTrafficRecordingProgressReceived userInfo:@{SWHTTPTrafficRecordingProgressRequestKey: request}]; + + BOOL(^testBlock)(NSURLRequest *request) = [SWHttpTrafficRecorder sharedRecorder].recordingTestBlock; + BOOL canInit = YES; + if(testBlock){ + canInit = testBlock(request); + } + if(!canInit){ + [self updateRecorderProgressDelegate:SWHTTPTrafficRecordingProgressSkipped userInfo:@{SWHTTPTrafficRecordingProgressRequestKey: request}]; + } + return canInit; +} + ++ (NSURLRequest *) canonicalRequestForRequest:(NSURLRequest *)request { + return request; +} + +- (void) startLoading { + NSMutableURLRequest *newRequest = [self.request mutableCopy]; + [NSURLProtocol setProperty:@YES forKey:SWRecordingLProtocolHandledKey inRequest:newRequest]; + + [self.class updateRecorderProgressDelegate:SWHTTPTrafficRecordingProgressStarted userInfo:@{SWHTTPTrafficRecordingProgressRequestKey: self.request}]; + + self.session = [NSURLSession sessionWithConfiguration:SWHttpTrafficRecorder.sharedRecorder.sessionConfig + delegate:self + delegateQueue:nil]; + self.dataTask = [self.session dataTaskWithRequest:newRequest]; + [self.dataTask resume]; +} + +- (void) stopLoading { + [self.dataTask cancel]; + self.mutableData = nil; +} + +#pragma mark - NSURLConnectionDelegate + +- (void)URLSession:(NSURLSession *)session + dataTask:(NSURLSessionDataTask *)dataTask +didReceiveResponse:(NSURLResponse *)response + completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler { + [self.client URLProtocol:self didReceiveResponse:response cacheStoragePolicy:NSURLCacheStorageNotAllowed]; + + self.response = response; + self.mutableData = [[NSMutableData alloc] init]; + completionHandler(NSURLSessionResponseAllow); +} + +- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data { + [self.client URLProtocol:self didLoadData:data]; + + [self.mutableData appendData:data]; +} + +- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error { + NSData *data = [self.mutableData copy]; + + if (error != nil) { + [self.client URLProtocol:self didFailWithError:error]; + + [self.class updateRecorderProgressDelegate:SWHTTPTrafficRecordingProgressFailedToLoad + userInfo:@{SWHTTPTrafficRecordingProgressRequestKey: self.request, + SWHTTPTrafficRecordingProgressErrorKey: error + }]; + + return; + } + + [self.client URLProtocolDidFinishLoading:self]; + + NSHTTPURLResponse *response = (NSHTTPURLResponse *)self.response; + NSURLRequest *request = (NSURLRequest*)task.currentRequest; + + [self.class updateRecorderProgressDelegate:SWHTTPTrafficRecordingProgressLoaded + userInfo:@{SWHTTPTrafficRecordingProgressRequestKey: request, + SWHTTPTrafficRecordingProgressResponseKey: response, + SWHTTPTrafficRecordingProgressBodyDataKey: data + }]; + + NSString *path = [self getFilePath:request response:response]; + SWHTTPTrafficRecordingFormat format = [SWHttpTrafficRecorder sharedRecorder].recordingFormat; + if(format == SWHTTPTrafficRecordingFormatBodyOnly){ + [self createBodyOnlyFileWithRequest:request response:response data:data atFilePath:path]; + } else if(format == SWHTTPTrafficRecordingFormatMocktail){ + [self createMocktailFileWithRequest:request response:response data:data atFilePath:path]; + } else if(format == SWHTTPTrafficRecordingFormatHTTPMessage){ + [self createHTTPMessageFileWithRequest:request response:response data:data atFilePath:path]; + } else if(format == SWHTTPTrafficRecordingFormatCustom && [SWHttpTrafficRecorder sharedRecorder].createFileInCustomFormatBlock != nil){ + [SWHttpTrafficRecorder sharedRecorder].createFileInCustomFormatBlock(request, response, data, path); + } else { + NSLog(@"File format: %ld is not supported.", (long)format); + } +} + +- (void)URLSession:(NSURLSession *)session + task:(NSURLSessionTask *)task +willPerformHTTPRedirection:(NSHTTPURLResponse *)response + newRequest:(NSURLRequest *)request + completionHandler:(void (^)(NSURLRequest * _Nullable))completionHandler { + if (response != nil) { + [[self client] URLProtocol:self wasRedirectedToRequest:request redirectResponse:response]; + } + completionHandler(request); +} + + +#pragma mark - File Creation Utility Methods + +-(NSString *)getFileName:(NSURLRequest *)request response:(NSHTTPURLResponse *)response{ + NSString *fileName = [request.URL lastPathComponent]; + + if(!fileName || [self isNotValidFileName: fileName]){ + fileName = @"Mocktail"; + } + + fileName = [NSString stringWithFormat:@"%@_%lu_%d", fileName, (unsigned long)[SWHttpTrafficRecorder sharedRecorder].runTimeStamp, [[SWHttpTrafficRecorder sharedRecorder] increaseFileNo]]; + + fileName = [fileName stringByAppendingPathExtension:[self getFileExtension:request response:response]]; + + NSString *(^fileNamingBlock)(NSURLRequest *request, NSURLResponse *response, NSString *defaultName) = [SWHttpTrafficRecorder sharedRecorder].fileNamingBlock; + + if(fileNamingBlock){ + fileName = fileNamingBlock(request, response, fileName); + } + return fileName; +} + +-(BOOL)isNotValidFileName:(NSString*) fileName{ + return NO; +} + +-(NSString *)getFilePath:(NSURLRequest *)request response:(NSHTTPURLResponse *)response{ + NSString *recordingPath = [SWHttpTrafficRecorder sharedRecorder].recordingPath; + NSString *filePath = [recordingPath stringByAppendingPathComponent:[self getFileName:request response:response]]; + + return filePath; +} + +-(NSString *)getFileExtension:(NSURLRequest *)request response:(NSHTTPURLResponse *)response{ + SWHTTPTrafficRecordingFormat format = [SWHttpTrafficRecorder sharedRecorder].recordingFormat; + if(format == SWHTTPTrafficRecordingFormatBodyOnly){ + /* Based on http://blog.ablepear.com/2010/08/how-to-get-file-extension-for-mime-type.html, we may be able to get the file extension from mime type. Use a fixed mapping for simpilicity for now unless there is a need later on */ + return [SWHttpTrafficRecorder sharedRecorder].fileExtensionMapping[response.MIMEType] ?: @"unknown"; + } else if(format == SWHTTPTrafficRecordingFormatMocktail){ + return @"tail"; + } else if(format == SWHTTPTrafficRecordingFormatHTTPMessage){ + return @"response"; + } + + return @"unknown"; +} + +-(BOOL)toBase64Body:(NSURLRequest *)request andResponse:(NSHTTPURLResponse *)response{ + if([SWHttpTrafficRecorder sharedRecorder].base64TestBlock){ + return [SWHttpTrafficRecorder sharedRecorder].base64TestBlock(request, response); + } + return [response.MIMEType hasPrefix:@"image"]; +} + +-(NSData *)doBase64:(NSData *)bodyData request: (NSURLRequest*)request response:(NSHTTPURLResponse*)response{ + BOOL toBase64 = [self toBase64Body:request andResponse:response]; + if(toBase64 && bodyData){ + return [bodyData base64EncodedDataWithOptions:0]; + } else { + return bodyData; + } +} + +-(NSData *)doJSONPrettyPrint:(NSData *)bodyData request: (NSURLRequest*)request response:(NSHTTPURLResponse*)response{ + if([response.MIMEType isEqualToString:@"application/json"] && bodyData) + { + NSError *error; + id json = [NSJSONSerialization JSONObjectWithData:bodyData options:0 error:&error]; + if(json && !error){ + bodyData = [NSJSONSerialization dataWithJSONObject:json options:NSJSONWritingPrettyPrinted error:&error]; + if(error){ + NSLog(@"Somehow the content is not a json though the mime type is json: %@", error); + } + } else { + NSLog(@"Somehow the content is not a json though the mime type is json: %@", error); + } + } + return bodyData; +} + +-(void)createFileAt:(NSString *)filePath usingData:(NSData *)data completionHandler:(void(^)(BOOL created))completionHandler{ + __block BOOL created = NO; + NSBlockOperation* creationOp = [NSBlockOperation blockOperationWithBlock: ^{ + created = [NSFileManager.defaultManager createFileAtPath:filePath contents:data attributes:[NSDictionary dictionaryWithObject:NSFileProtectionComplete forKey:NSFileProtectionKey]]; + }]; + creationOp.completionBlock = ^{ + completionHandler(created); + }; + [[SWHttpTrafficRecorder sharedRecorder].fileCreationQueue addOperation:creationOp]; +} + +#pragma mark - BodyOnly File Creation + +-(void)createBodyOnlyFileWithRequest:(NSURLRequest*)request response:(NSHTTPURLResponse*)response data:(NSData*)data atFilePath:(NSString *)filePath +{ + data = [self doJSONPrettyPrint:data request:request response:response]; + + NSDictionary *userInfo = @{SWHTTPTrafficRecordingProgressRequestKey: request, + SWHTTPTrafficRecordingProgressResponseKey: response, + SWHTTPTrafficRecordingProgressBodyDataKey: data, + SWHTTPTrafficRecordingProgressFileFormatKey: @(SWHTTPTrafficRecordingFormatBodyOnly), + SWHTTPTrafficRecordingProgressFilePathKey: filePath + }; + [self createFileAt:filePath usingData:data completionHandler:^(BOOL created) { + [self.class updateRecorderProgressDelegate:(created ? SWHTTPTrafficRecordingProgressRecorded : SWHTTPTrafficRecordingProgressFailedToRecord) userInfo:userInfo]; + }]; +} + +#pragma mark - Mocktail File Creation + +-(void)createMocktailFileWithRequest:(NSURLRequest*)request response:(NSHTTPURLResponse*)response data:(NSData*)data atFilePath:(NSString *)filePath +{ + NSMutableString *tail = NSMutableString.new; + + [tail appendFormat:@"%@\n", request.HTTPMethod]; + [tail appendFormat:@"%@\n", [self getURLRegexPattern:request]]; + [tail appendFormat:@"%ld\n", (long)response.statusCode]; + [tail appendFormat:@"%@%@\n", response.MIMEType, [self toBase64Body:request andResponse:response] ? @";base64": @""]; + NSEnumerator *headerKeys = [response.allHeaderFields keyEnumerator]; + for (NSString *key in headerKeys) { + [tail appendFormat:@"%@: %@\n", key, (NSString*)[response.allHeaderFields objectForKey:key]]; + } + + [tail appendString:@"\n"]; + + data = [self doBase64:data request:request response:response]; + + data = [self doJSONPrettyPrint:data request:request response:response]; + + data = [self replaceRegexWithTokensInData:data]; + + [tail appendFormat:@"%@", data ? [[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding] : @""]; + + NSDictionary *userInfo = @{SWHTTPTrafficRecordingProgressRequestKey: request, + SWHTTPTrafficRecordingProgressResponseKey: response, + SWHTTPTrafficRecordingProgressBodyDataKey: data, + SWHTTPTrafficRecordingProgressFileFormatKey: @(SWHTTPTrafficRecordingFormatMocktail), + SWHTTPTrafficRecordingProgressFilePathKey: filePath + }; + [self createFileAt:filePath usingData:[tail dataUsingEncoding:NSUTF8StringEncoding] completionHandler:^(BOOL created) { + [self.class updateRecorderProgressDelegate:(created ? SWHTTPTrafficRecordingProgressRecorded : SWHTTPTrafficRecordingProgressFailedToRecord) userInfo: userInfo]; + }]; +} + +-(NSData *)replaceRegexWithTokensInData: (NSData *) data { + SWHttpTrafficRecorder *recorder = [SWHttpTrafficRecorder sharedRecorder]; + if(![recorder replacementDict]) { + return data; + } + else { + NSString *dataString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]; + for(NSString *key in [recorder replacementDict]) { + if([[[recorder replacementDict] objectForKey: key] isKindOfClass:[NSRegularExpression class]]) { + dataString = [[[recorder replacementDict] objectForKey:key] stringByReplacingMatchesInString:dataString options:0 range:NSMakeRange(0, [dataString length]) withTemplate:key]; + } + } + data = [dataString dataUsingEncoding:NSUTF8StringEncoding]; + return data; + } +} + +-(NSString *)getURLRegexPattern:(NSURLRequest *)request{ + NSString *urlPattern = request.URL.path; + if(request.URL.query){ + NSArray *queryArray = [request.URL.query componentsSeparatedByString:@"&"]; + NSMutableArray *processedQueryArray = [[NSMutableArray alloc] initWithCapacity:queryArray.count]; + for (NSString *part in queryArray) { + NSRegularExpression *urlRegex = [NSRegularExpression regularExpressionWithPattern:@"(.*)=(.*)" options:NSRegularExpressionCaseInsensitive error:nil]; + NSString *newPart = [urlRegex stringByReplacingMatchesInString:part options:0 range:NSMakeRange(0, part.length) withTemplate:@"$1=.*"]; + [processedQueryArray addObject:newPart]; + } + urlPattern = [NSString stringWithFormat:@"%@\\?%@", request.URL.path, [processedQueryArray componentsJoinedByString:@"&"]]; + } + + NSString *(^urlRegexPatternBlock)(NSURLRequest *request, NSString *defaultPattern) = [SWHttpTrafficRecorder sharedRecorder].urlRegexPatternBlock; + + if(urlRegexPatternBlock){ + urlPattern = urlRegexPatternBlock(request, urlPattern); + } + + urlPattern = [urlPattern stringByAppendingString:@"$"]; + + return urlPattern; +} + +#pragma mark - HTTP Message File Creation + +-(void)createHTTPMessageFileWithRequest:(NSURLRequest*)request response:(NSHTTPURLResponse*)response data:(NSData*)data atFilePath:(NSString *)filePath +{ + NSMutableString *dataString = NSMutableString.new; + + [dataString appendFormat:@"%@\n", [self statusLineFromResponse:response]]; + + NSDictionary *headers = response.allHeaderFields; + for(NSString *key in headers){ + [dataString appendFormat:@"%@: %@\n", key, headers[key]]; + } + + [dataString appendString:@"\n"]; + + NSMutableData *responseData = [NSMutableData dataWithData:[dataString dataUsingEncoding:NSUTF8StringEncoding]]; + [responseData appendData:data]; + + NSDictionary *userInfo = @{SWHTTPTrafficRecordingProgressRequestKey: request, + SWHTTPTrafficRecordingProgressResponseKey: response, + SWHTTPTrafficRecordingProgressBodyDataKey: data, + SWHTTPTrafficRecordingProgressFileFormatKey: @(SWHTTPTrafficRecordingFormatHTTPMessage), + SWHTTPTrafficRecordingProgressFilePathKey: filePath + }; + + [self createFileAt:filePath usingData:responseData completionHandler:^(BOOL created) { + [self.class updateRecorderProgressDelegate:(created ? SWHTTPTrafficRecordingProgressRecorded : SWHTTPTrafficRecordingProgressFailedToRecord) userInfo:userInfo]; + }]; +} + +- (NSString *)statusLineFromResponse:(NSHTTPURLResponse*)response{ + CFHTTPMessageRef message = CFHTTPMessageCreateResponse(kCFAllocatorDefault, [response statusCode], NULL, kCFHTTPVersion1_1); + NSString *statusLine = (__bridge_transfer NSString *)CFHTTPMessageCopyResponseStatusLine(message); + CFRelease(message); + return statusLine; +} + +#pragma mark - Recording Progress + ++ (void)updateRecorderProgressDelegate:(SWHTTPTrafficRecordingProgressKind)progress userInfo:(NSDictionary *)info{ + SWHttpTrafficRecorder *recorder = [SWHttpTrafficRecorder sharedRecorder]; + if(recorder.progressDelegate && [recorder.progressDelegate respondsToSelector:@selector(updateRecordingProgress:userInfo:)]){ + [recorder.progressDelegate updateRecordingProgress:progress userInfo:info]; + } +} + +@end diff --git a/Stripe/StripeiOSTests/ServerErrorMapperTest.swift b/Stripe/StripeiOSTests/ServerErrorMapperTest.swift new file mode 100644 index 00000000..11471eb8 --- /dev/null +++ b/Stripe/StripeiOSTests/ServerErrorMapperTest.swift @@ -0,0 +1,94 @@ +// +// ServerErrorMapperTest.swift +// StripeiOS Tests +// +// Created by Nick Porter on 9/13/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class ServerErrorMapperTest: XCTestCase { + + func testFromMissingPublishableKey() { + let serverErrorMessage = + "You did not provide an API key. You need to provide your API key in the Authorization header, using Bearer auth (e.g. \'Authorization: Bearer YOUR_SECRET_KEY\'). See https://stripe.com/docs/api#authentication for details, or we can help at https://support.stripe.com/." + + XCTAssertTrue( + ServerErrorMapper.mobileErrorMessage( + from: serverErrorMessage, + httpResponse: HTTPURLResponse() + )!.hasPrefix("No valid API key provided. Set `STPAPIClient.shared()") + ) + } + + func testFromInvalidPublishableKey() { + let serverErrorMessage = "Invalid API Key provided: pk_test" + + XCTAssertTrue( + ServerErrorMapper.mobileErrorMessage( + from: serverErrorMessage, + httpResponse: HTTPURLResponse() + )!.hasPrefix("No valid API key provided. Set `STPAPIClient.shared()") + ) + } + + func testFromInvalidCustEphKey() { + let serverErrorMessage = "Invalid API Key provided: badCustEphKey" + + let url = URL(string: "https://api.stripe.com/v1/payment_methods?customer=")! + let httpResponse = HTTPURLResponse( + url: url, + statusCode: 401, + httpVersion: nil, + headerFields: nil + ) + XCTAssertTrue( + ServerErrorMapper.mobileErrorMessage( + from: serverErrorMessage, + httpResponse: httpResponse + )!.hasPrefix("Invalid customer ephemeral key secret.") + ) + } + + func testFromNoSuchPaymentIntent() { + let serverErrorMessage = "No such payment_intent: pi_123" + + XCTAssertTrue( + ServerErrorMapper.mobileErrorMessage( + from: serverErrorMessage, + httpResponse: HTTPURLResponse() + )!.hasPrefix("No matching PaymentIntent could") + ) + } + + func testFromNoSuchSetupIntent() { + let serverErrorMessage = "No such setup_intent: si_123" + + XCTAssertTrue( + ServerErrorMapper.mobileErrorMessage( + from: serverErrorMessage, + httpResponse: HTTPURLResponse() + )!.hasPrefix("No matching SetupIntent could") + ) + } + + func testFromUnknownErrorMessage() { + let serverErrorMessage = + "This error message is not known to ServerErrorMapper, should return nil." + + XCTAssertNil( + ServerErrorMapper.mobileErrorMessage( + from: serverErrorMessage, + httpResponse: HTTPURLResponse() + ) + ) + } + +} diff --git a/Stripe/StripeiOSTests/StripeErrorTest.swift b/Stripe/StripeiOSTests/StripeErrorTest.swift new file mode 100644 index 00000000..c9c9750f --- /dev/null +++ b/Stripe/StripeiOSTests/StripeErrorTest.swift @@ -0,0 +1,254 @@ +// +// StripeErrorTest.swift +// StripeiOS Tests +// +// Created by Ben Guo on 4/14/17. +// Copyright © 2017 Stripe, Inc. All rights reserved. +// + +import Foundation +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI + +class StripeErrorTest: XCTestCase { + func testEmptyResponse() { + let response: [AnyHashable: Any] = [:] + let error = NSError.stp_error(fromStripeResponse: response) + XCTAssertNil(error) + } + + func testResponseWithUnknownTypeAndNoMessage() { + let response = [ + "error": [ + "type": "foo", + "code": "error_code", + ], + ] + let error = NSError.stp_error(fromStripeResponse: response)! + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual(error.code, STPErrorCode.apiError.rawValue) + XCTAssertEqual( + error.userInfo[NSLocalizedDescriptionKey] as! String, + NSError.stp_unexpectedErrorMessage() + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorTypeKey] as! String, + response["error"]!["type"]! + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorCodeKey] as! String, + response["error"]!["code"]! + ) + XCTAssertTrue( + (error.userInfo[STPError.errorMessageKey]! as! String).hasPrefix( + "Could not interpret the error response" + ) + ) + } + + func testAPIError() { + let response = [ + "error": [ + "type": "api_error", + "message": "some message", + ], + ] + let error = NSError.stp_error(fromStripeResponse: response)! + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual(error.code, STPErrorCode.apiError.rawValue) + XCTAssertEqual( + error.userInfo[NSLocalizedDescriptionKey] as! String?, + NSError.stp_unexpectedErrorMessage() + ) + XCTAssertEqual( + error.userInfo[STPError.errorMessageKey] as! String?, + response["error"]!["message"] + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorTypeKey] as! String?, + response["error"]!["type"] + ) + } + + func testInvalidRequestErrorMissingParameter() { + let response = [ + "error": [ + "type": "invalid_request_error", + "message": "The payment method `card` requires the parameter: card[exp_year].", + "param": "card[exp_year]", + ], + ] + let error = NSError.stp_error(fromStripeResponse: response)! + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual(error.code, STPErrorCode.invalidRequestError.rawValue) + XCTAssertEqual( + error.userInfo[NSLocalizedDescriptionKey] as? String, + NSError.stp_unexpectedErrorMessage() + ) + XCTAssertEqual( + error.userInfo[STPError.errorMessageKey] as? String, + response["error"]!["message"] + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorTypeKey] as? String, + response["error"]!["type"] + ) + XCTAssertEqual(error.userInfo[STPError.errorParameterKey] as! String, "card[expYear]") + } + + func testAuthenticationError() { + // Given an `invalid_request_error` response + let response = [ + "error": [ + "type": "invalid_request_error", + "message": "Invalid API Key provided: pk_test_***************************00", + ], + ] + + // with a `401` HTTP status code + let httpResponse = HTTPURLResponse( + url: URL(string: "https://api.stripe.com/v1/payment_intents")!, + statusCode: 401, + httpVersion: "1.1", + headerFields: nil + ) + + let error = NSError.stp_error(fromStripeResponse: response, httpResponse: httpResponse)! + + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual( + error.code, + STPErrorCode.authenticationError.rawValue, + "`error.code` should be equals to `STPErrorCode.authenticationError`" + ) + } + + func testAuthenticationErrorDueToExpiredKey() { + // Given an `invalid_request_error` response due to an expired key + let response = [ + "error": [ + "code": "api_key_expired", + "type": "invalid_request_error", + "message": "Expired API Key provided: pk_test_***************************00", + ], + ] + + // with a `401` HTTP status code + let httpResponse = HTTPURLResponse( + url: URL(string: "https://api.stripe.com/v1/payment_intents")!, + statusCode: 401, + httpVersion: "1.1", + headerFields: nil + ) + + let error = NSError.stp_error(fromStripeResponse: response, httpResponse: httpResponse)! + + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual( + error.code, + STPErrorCode.authenticationError.rawValue, + "`error.code` should be equals to `STPErrorCode.authenticationError`" + ) + } + + func testInvalidRequestErrorIncorrectNumber() { + let response = [ + "error": [ + "type": "invalid_request_error", + "message": "Your card number is incorrect.", + "code": "incorrect_number", + ], + ] + let error = NSError.stp_error(fromStripeResponse: response)! + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual(error.code, STPErrorCode.invalidRequestError.rawValue) + // Error type is not `card_error`, so `NSLocalizedDescription` will be a generic error. + XCTAssertEqual( + error.userInfo[NSLocalizedDescriptionKey] as! String, + NSError.stp_unexpectedErrorMessage() + ) + XCTAssertEqual( + error.userInfo[STPError.cardErrorCodeKey] as! String, + STPError.incorrectNumber + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorTypeKey] as? String, + response["error"]!["type"] + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorCodeKey] as? String, + response["error"]!["code"] + ) + XCTAssertEqual( + error.userInfo[STPError.errorMessageKey] as? String, + response["error"]!["message"] + ) + } + + func testCardErrorIncorrectNumber() { + let response = [ + "error": [ + "type": "card_error", + "message": "Your card number is incorrect.", + "code": "incorrect_number", + ], + ] + let error = NSError.stp_error(fromStripeResponse: response)! + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual(error.code, STPErrorCode.cardError.rawValue) + XCTAssertEqual( + error.userInfo[NSLocalizedDescriptionKey] as! String, + "Your card number is incorrect." + ) + XCTAssertEqual( + error.userInfo[STPError.cardErrorCodeKey] as! String, + STPError.incorrectNumber + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorTypeKey] as? String, + response["error"]!["type"] + ) + XCTAssertEqual( + error.userInfo[STPError.stripeErrorCodeKey] as? String, + response["error"]!["code"] + ) + XCTAssertEqual( + error.userInfo[STPError.errorMessageKey] as? String, + response["error"]!["message"] + ) + } + + func testCardDeclinedError() { + let response = [ + "error": [ + "type": "card_error", + "code": "card_declined", + "decline_code": "insufficient_funds", + ], + ] + guard let error = NSError.stp_error(fromStripeResponse: response) else { + XCTFail() + return + } + XCTAssertEqual(error.domain, STPError.stripeDomain) + XCTAssertEqual(error.code, STPErrorCode.cardError.rawValue) + XCTAssertEqual( + error.userInfo[STPError.cardErrorCodeKey] as? String, + STPCardErrorCode.cardDeclined.rawValue + ) + // Response didn't include a message, so a built in message will be used. + XCTAssertEqual( + error.userInfo[NSLocalizedDescriptionKey] as? String, + NSError.stp_cardErrorDeclinedUserMessage() + ) + XCTAssertEqual( + error.userInfo[STPError.stripeDeclineCodeKey] as? String, + "insufficient_funds" + ) + } +} diff --git a/Stripe/StripeiOSTests/StripeTests-Prefix.pch b/Stripe/StripeiOSTests/StripeTests-Prefix.pch new file mode 100644 index 00000000..0540b07f --- /dev/null +++ b/Stripe/StripeiOSTests/StripeTests-Prefix.pch @@ -0,0 +1,21 @@ +// +// StripeTests-Prefix.pch +// StripeiOS Tests +// +// Created by David Estes on 10/8/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +#ifndef StripeTests_Prefix_pch +#define StripeTests_Prefix_pch + +// Include any system framework and library headers here that should be included in all compilation units. +// You will also need to set the Prefix Header build setting of one or more of your targets to reference this file. + +#import "STPBlocks.h" +@import StripeApplePay; +@import Stripe; +@import StripePayments; +@import StripePaymentsUI; + +#endif /* StripeTests_Prefix_pch */ diff --git a/Stripe/StripeiOSTests/StripeiOS Tests-Bridging-Header.h b/Stripe/StripeiOSTests/StripeiOS Tests-Bridging-Header.h new file mode 100644 index 00000000..7f87b154 --- /dev/null +++ b/Stripe/StripeiOSTests/StripeiOS Tests-Bridging-Header.h @@ -0,0 +1,9 @@ +// +// Use this file to import your target's public headers that you would like to expose to Swift. +// + +#import "SWHttpTrafficRecorder.h" +#import "STPTestingAPIClient.h" +#import "STPTestUtils.h" +#import "STPFixtures.h" +#import "NSLocale+STPSwizzling.h" diff --git a/Stripe/StripeiOSTests/TextFieldElement+CardTest.swift b/Stripe/StripeiOSTests/TextFieldElement+CardTest.swift new file mode 100644 index 00000000..c5278b5e --- /dev/null +++ b/Stripe/StripeiOSTests/TextFieldElement+CardTest.swift @@ -0,0 +1,366 @@ +// +// TextFieldElement+CardTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 2/25/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class TextFieldElementCardTest: XCTestCase { + func testPANValidation() throws { + typealias Error = TextFieldElement.PANConfiguration.Error + let testcases: [String: ElementValidationState] = [ + "": .invalid(error: Error.empty, shouldDisplay: false), + + // Incomplete + "4": .invalid(error: Error.incomplete, shouldDisplay: true), + "424242424242": .invalid(error: Error.incomplete, shouldDisplay: true), + // diners club (14 digit, but 13 digits given) + "3622720627166": .invalid(error: Error.incomplete, shouldDisplay: true), + + // Unknown card brand + "0000000000000000": .invalid(error: Error.invalidBrand, shouldDisplay: true), + "1000000000000000": .invalid(error: Error.invalidBrand, shouldDisplay: true), + "1234567812345678": .invalid(error: Error.invalidBrand, shouldDisplay: true), + "9999999999999995": .invalid(error: Error.invalidBrand, shouldDisplay: true), + "1234123412341234": .invalid(error: Error.invalidBrand, shouldDisplay: true), + "9999999999999999999999": .invalid(error: Error.invalidBrand, shouldDisplay: true), + + // Fails luhn check + "4012888888881889": .invalid(error: Error.invalidLuhn, shouldDisplay: true), + "2223000010089809": .invalid(error: Error.invalidLuhn, shouldDisplay: true), + "3530111333300009": .invalid(error: Error.invalidLuhn, shouldDisplay: true), + // mastercard (prepaid) + "5105105105105109": .invalid(error: Error.invalidLuhn, shouldDisplay: true), + // discover + "6011111111111119": .invalid(error: Error.invalidLuhn, shouldDisplay: true), + // cup + "6200000000000009": .invalid(error: Error.invalidLuhn, shouldDisplay: true), + + // Valid (luhn-passing) PANs + "4012888888881881": .valid, + "2223000010089800": .valid, + "3530111333300000": .valid, + "4242424242424242": .valid, // visa + "4000056655665556": .valid, // visa (debit) + "5555555555554444": .valid, // mastercard + "2223003122003222": .valid, // mastercard (2-series) + "5200828282828210": .valid, // mastercard (debit) + "5105105105105100": .valid, // mastercard (prepaid) + "378282246310005": .valid, // amex + "371449635398431": .valid, // amex + "6011111111111117": .valid, // discover + "6011000990139424": .valid, // discover + "3056930009020004": .valid, // diners club + "36227206271667": .valid, // diners club (14 digit) + "3566002020360505": .valid, // jcb + "6200000000000005": .valid, // cup + + // ⚠️ Don't test variable length PANs here - + // they trigger STPBINRange async calls that pollute other tests + + // Non-US + "4000000760000002": .valid, // br + "4000001240000000": .valid, // ca + "4000004840008001": .valid, // mx + ] + + let configuration = TextFieldElement.PANConfiguration() + for (text, expected) in testcases { + let actual = configuration.simulateValidationState(text) + XCTAssertTrue( + actual == expected, + "Input \"\(text)\": expected \(expected) but got \(actual)" + ) + } + } + + func testBINRangeThatRequiresNetworkCallToValidate() { + // Set a publishable key for the metadata service + STPAPIClient.shared.publishableKey = STPTestingDefaultPublishableKey + var configuration = TextFieldElement.PANConfiguration() + let binController = STPBINController() + configuration.binController = binController + + // Given a 19-digit Union Pay variable length number i.e., requires a network call in order to be know the correct length... + // (I got this number from https://hubble.corp.stripe.com/queries/ek/5612e1d3) + let unionPay19 = "6235510000000000009" // 19-digit, valid luhn Union Pay + // a 16-digit valid luhn Union Pay that should be 19 digits according to its BIN prefix. + let unionPay19_but_16_digits_entered = "6235510000000002" + XCTAssertFalse(binController.hasBINRanges(forPrefix: unionPay19_but_16_digits_entered)) + XCTAssertFalse(binController.hasBINRanges(forPrefix: unionPay19)) + + // ...we should allow a 16 digit number, since we don't know the correct length yet... + XCTAssertEqual( + configuration.simulateValidationState(unionPay19_but_16_digits_entered), + .valid + ) + // ...and we should allow a 19 digit number + XCTAssertEqual( + configuration.simulateValidationState(unionPay19_but_16_digits_entered), + .valid + ) + // ...and we should mark a 15 digit number as incomplete + XCTAssertEqual( + configuration.simulateValidationState( + unionPay19_but_16_digits_entered.stp_safeSubstring(to: 15) + ), + .invalid(error: TextFieldElement.PANConfiguration.Error.incomplete, shouldDisplay: true) + ) + + // ...and load the BIN range. + XCTAssertTrue( + binController.isLoadingCardMetadata(forPrefix: unionPay19_but_16_digits_entered) + ) + + // After we've loaded the bin range... + let e = expectation(description: "Fetch BIN Range") + binController.retrieveBINRanges(forPrefix: unionPay19_but_16_digits_entered) { _ in + e.fulfill() + } + waitForExpectations(timeout: 10, handler: nil) + XCTAssertTrue(binController.hasBINRanges(forPrefix: unionPay19_but_16_digits_entered)) + XCTAssertTrue(binController.hasBINRanges(forPrefix: unionPay19)) + + // ...a 16 digit number should be considered incomplete + XCTAssertEqual( + configuration.simulateValidationState(unionPay19_but_16_digits_entered), + .invalid(error: TextFieldElement.PANConfiguration.Error.incomplete, shouldDisplay: true) + ) + // ...and the 19 digit number should still be valid + XCTAssertEqual( + configuration.simulateValidationState(unionPay19), + .valid + ) + + // Hack to let STPBINRange finish network calls before running another test + let allRetrievalsAreComplete = expectation(description: "Fetch BIN Range") + binController.retrieveBINRanges(forPrefix: unionPay19_but_16_digits_entered) { _ in + allRetrievalsAreComplete.fulfill() + } + waitForExpectations(timeout: 10, handler: nil) + } + + func testBINRangeThatRequiresNetworkCallToValidateWhenCallFails() { + // We set an invalid publishable key so that STPBINRange calls to the API fail + STPAPIClient.shared.publishableKey = "" + var configuration = TextFieldElement.PANConfiguration() + let binController = STPBINController() + configuration.binController = binController + + // Given a 19-digit Union Pay variable length number i.e., requires a network call in order to be know the correct length... + // (I got this number from https://hubble.corp.stripe.com/queries/ek/5612e1d3) + let unionPay19 = "6235510000000000009" // 19-digit, valid luhn Union Pay + // a 16-digit valid luhn Union Pay that should be 19 digits according to its BIN prefix. + let unionPay19_but_16_digits_entered = "6235510000000002" + XCTAssertFalse(binController.hasBINRanges(forPrefix: unionPay19_but_16_digits_entered)) + XCTAssertFalse(binController.hasBINRanges(forPrefix: unionPay19)) + + // ...we should allow a 16 digit number, since we don't know the correct length yet... + XCTAssertEqual( + configuration.simulateValidationState(unionPay19_but_16_digits_entered), + .valid + ) + // ...and we should allow a 19 digit number + XCTAssertEqual( + configuration.simulateValidationState(unionPay19_but_16_digits_entered), + .valid + ) + + // ...and load the BIN range. + XCTAssertTrue( + binController.isLoadingCardMetadata(forPrefix: unionPay19_but_16_digits_entered) + ) + + // After we've unsuccessfully loaded the bin range... + let e = expectation(description: "Fetch BIN Range") + binController.retrieveBINRanges(forPrefix: unionPay19_but_16_digits_entered) { _ in + e.fulfill() + } + waitForExpectations(timeout: 10, handler: nil) + XCTAssertFalse(binController.hasBINRanges(forPrefix: unionPay19_but_16_digits_entered)) + XCTAssertFalse(binController.hasBINRanges(forPrefix: unionPay19)) + + // ...16 and 19 digit numbers should still be allowed, since we still don't know the correct length + XCTAssertEqual(configuration.maxLength(for: unionPay19_but_16_digits_entered), 19) + XCTAssertEqual( + configuration.simulateValidationState(unionPay19_but_16_digits_entered), + .valid + ) + // ...and the 19 digit number should still be valid + XCTAssertEqual(configuration.maxLength(for: unionPay19), 19) + XCTAssertEqual( + configuration.simulateValidationState(unionPay19), + .valid + ) + + // Hack to let STPBINRange finish network calls before running another test + let allRetrievalsAreComplete = expectation(description: "Fetch BIN Range") + binController.retrieveBINRanges(forPrefix: unionPay19_but_16_digits_entered) { _ in + allRetrievalsAreComplete.fulfill() + } + waitForExpectations(timeout: 10, handler: nil) + } + + func testCVCValidation() { + let emptyError = TextFieldElement.Error.empty + let incompleteError = TextFieldElement.Error.incomplete( + localizedDescription: .Localized.your_cards_security_code_is_incomplete + ) + let testcases: [(String, STPCardBrand, ElementValidationState)] = [ + // VISA CVC are 3 digits + ("", .visa, .invalid(error: emptyError, shouldDisplay: false)), + ("1", .visa, .invalid(error: incompleteError, shouldDisplay: true)), + ("12", .visa, .invalid(error: incompleteError, shouldDisplay: true)), + ("123", .visa, .valid), + + // Unknown card brand allows 3 or 4 digits + ("", .unknown, .invalid(error: emptyError, shouldDisplay: false)), + ("1", .unknown, .invalid(error: incompleteError, shouldDisplay: true)), + ("12", .unknown, .invalid(error: incompleteError, shouldDisplay: true)), + ("123", .unknown, .valid), + ("1234", .unknown, .valid), + + // Amex CVV allow 3 or 4 digits + ("", .amex, .invalid(error: emptyError, shouldDisplay: false)), + ("1", .amex, .invalid(error: incompleteError, shouldDisplay: true)), + ("12", .amex, .invalid(error: incompleteError, shouldDisplay: true)), + ("123", .amex, .valid), + ("1234", .amex, .valid), + ] + for (text, brand, expected) in testcases { + let config = TextFieldElement.CVCConfiguration(cardBrandProvider: { + return brand + }) + let actual = config.simulateValidationState(text) + XCTAssertTrue( + actual == expected, + "Input \"\(text), \(brand)\": expected \(expected) but got \(actual)" + ) + } + } + + func testExpiryValidation() throws { + typealias Error = TextFieldElement.ExpiryDateConfiguration.Error + + let calendar = Calendar(identifier: .gregorian) + + let dateFormatter = DateFormatter() + dateFormatter.dateFormat = "MMyy" + dateFormatter.calendar = calendar + + let currentMonth = try XCTUnwrap( + calendar.date(from: calendar.dateComponents([.year, .month], from: Date())) + ) + let lastMonth = try XCTUnwrap(calendar.date(byAdding: .month, value: -1, to: currentMonth)) + let nextMonth = try XCTUnwrap(calendar.date(byAdding: .month, value: 1, to: currentMonth)) + let oneYearFromNow = try XCTUnwrap( + calendar.date(byAdding: .year, value: 1, to: currentMonth) + ) + + let testcases: [String: ElementValidationState] = [ + // Test empty -> incomplete -> complete + "": .invalid(error: TextFieldElement.Error.empty, shouldDisplay: false), + "0": .invalid(error: Error.incomplete, shouldDisplay: true), + "1": .invalid(error: Error.incomplete, shouldDisplay: true), + "12": .invalid(error: Error.incomplete, shouldDisplay: true), + "12/2": .invalid(error: Error.incomplete, shouldDisplay: true), + "12/49": .valid, + // Cards expire at end of last day of printed month. + dateFormatter.string(from: currentMonth): .valid, + dateFormatter.string(from: oneYearFromNow): .valid, + dateFormatter.string(from: nextMonth): .valid, + + // Test invalid months + "00": .invalid(error: Error.invalidMonth, shouldDisplay: true), + "00/9": .invalid(error: Error.invalidMonth, shouldDisplay: true), + "00/99": .invalid(error: Error.invalidMonth, shouldDisplay: true), + "13": .invalid(error: Error.invalidMonth, shouldDisplay: true), + + // Test expired dates + "12/21": .invalid(error: Error.invalid, shouldDisplay: true), + "01/22": .invalid(error: Error.invalid, shouldDisplay: true), + dateFormatter.string(from: lastMonth): .invalid( + error: Error.invalid, + shouldDisplay: true + ), + ] + let configuration = TextFieldElement.ExpiryDateConfiguration() + for (text, expected) in testcases { + let actual = configuration.simulateValidationState(text) + XCTAssertTrue( + actual == expected, + "Input \"\(text)\": expected \(expected) but got \(actual)" + ) + } + } + + func testExpiryDisplayText() { + let configuration = TextFieldElement.ExpiryDateConfiguration() + let textFieldElement = TextFieldElement(configuration: configuration) + + let testcases = [ + "0": "0", + "1": "1", + "2": "02", + "3": "03", + "4": "04", + "5": "05", + "6": "06", + "7": "07", + "8": "08", + "9": "09", + "10": "10", + "102": "10/2", + "1021": "10/21", + ] + + for (text, expected) in testcases { + // Simulate user input + textFieldElement.textFieldView.textField.text = text + textFieldElement.textFieldView.textDidChange() + let actual = textFieldElement.textFieldView.textField.text + XCTAssertTrue( + actual == expected, + "Input \"\(text)\": expected \(expected) but got \(actual!)" + ) + } + } +} + +extension TextFieldElementConfiguration { + // MARK: - Helpers + func simulateValidationState(_ input: String) -> ElementValidationState { + let textFieldElement = TextFieldElement(configuration: self) + textFieldElement.textFieldView.textField.text = input + textFieldElement.textFieldView.textDidChange() + return textFieldElement.validationState + } +} + +extension ElementValidationState: Equatable { + /// - Note: Assumes errors are equal if their localized descriptions are equal + public static func == (lhs: ElementValidationState, rhs: ElementValidationState) -> Bool { + switch (lhs, rhs) { + case (.valid, .valid): + return true + case ( + .invalid(let lhs_error, let lhs_shouldDisplay), + .invalid(let rhs_error, let rhs_shouldDisplay) + ): + return lhs_error.localizedDescription == rhs_error.localizedDescription + && lhs_shouldDisplay == rhs_shouldDisplay + default: + return false + } + } +} diff --git a/Stripe/StripeiOSTests/TextFieldElement+IBANTest.swift b/Stripe/StripeiOSTests/TextFieldElement+IBANTest.swift new file mode 100644 index 00000000..01b65d5c --- /dev/null +++ b/Stripe/StripeiOSTests/TextFieldElement+IBANTest.swift @@ -0,0 +1,140 @@ +// +// TextFieldElement+IBANTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 5/23/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentSheet +@testable@_spi(STP) import StripePaymentsUI +@testable@_spi(STP) import StripeUICore + +class TextFieldElementIBANTest: XCTestCase { + typealias IBANError = TextFieldElement.IBANError + typealias Error = TextFieldElement.Error + + func testValidation() throws { + let testcases: [String: TextFieldElement.ValidationState] = [ + "": .invalid(Error.empty), + "G": .invalid(IBANError.incomplete), + "GB": .invalid(IBANError.incomplete), + "GB1": .invalid(IBANError.incomplete), + "GB12": .invalid(IBANError.incomplete), + + "1": .invalid(IBANError.shouldStartWithCountryCode), + "12": .invalid(IBANError.shouldStartWithCountryCode), + "Z1": .invalid(IBANError.shouldStartWithCountryCode), + "🤦🏻🇺🇸": .invalid(IBANError.shouldStartWithCountryCode), + + "ZZ": .invalid(IBANError.invalidCountryCode(countryCode: "ZZ")), + + "GB82WEST12345698765432🇺🇸": .invalid(IBANError.invalidFormat), + // https://www.iban.com/testibans + "GB94BARC20201530093459": .invalid(IBANError.invalidFormat), + + "GB33BUKB20201555555555": .valid, + "GB94BARC10201530093459": .valid, + "SK6902000000001933504555": .valid, + "BG09STSA93000021741508": .valid, + "FR1420041010050500013M02606": .valid, + "AT611904300234573201": .valid, + "AT861904300235473202": .valid, + ] + + let config = TextFieldElement.IBANConfiguration(defaultValue: nil) + for (text, expected) in testcases { + let actual = config.validate(text: text, isOptional: false) + XCTAssertTrue( + actual == expected, + "Input \"\(text)\": expected \(expected) but got \(actual)" + ) + } + } + + func testValidateCountryCode() { + let testcases: [String: TextFieldElement.ValidationState] = [ + "": .invalid(IBANError.incomplete), + "A": .invalid(IBANError.incomplete), + "D": .invalid(IBANError.incomplete), + + "ū": .invalid(IBANError.shouldStartWithCountryCode), + "1": .invalid(IBANError.shouldStartWithCountryCode), + ".": .invalid(IBANError.shouldStartWithCountryCode), + + "AT": .valid, + "DE": .valid, + ] + for (test, expected) in testcases { + let actual = TextFieldElement.IBANConfiguration.validateCountryCode(test) + XCTAssertTrue(actual == expected) + } + } + + func testTransformToASCIIDigits() { + let testcases: [String: String] = [ + "": "", + "1234": "1234", + "GB82": "161182", + "AAAA": "10101010", + "ZZZZ": "35353535", + ] + for (test, expected) in testcases { + let actual = TextFieldElement.IBANConfiguration.transformToASCIIDigits(test) + XCTAssertTrue(actual == expected) + } + } + + func testMod97() { + let testcases: [String: Int?] = [ + "0": 0, + "97": 0, + "96": 96, + "00001": 1, + "13985713857180375018375081735081735": 15, + "13985713857180375018375081735081720": 0, + ] + for (test, expected) in testcases { + let actual = TextFieldElement.IBANConfiguration.mod97(test) + XCTAssertTrue(actual == expected) + } + + for _ in 0...100 { + let test = Int.random(in: 0...Int.max) + let actual = TextFieldElement.IBANConfiguration.mod97(String(test)) + XCTAssertTrue(actual == test % 97) + } + } +} + +// MARK: - Helpers + +// TODO(mludowise): These should get migrated to a shared StripeUICoreTestUtils target + +extension TextFieldElement.ValidationState: Equatable { + public static func == ( + lhs: TextFieldElement.ValidationState, + rhs: TextFieldElement.ValidationState + ) -> Bool { + switch (lhs, rhs) { + case (.valid, .valid): + return true + case (.invalid(let lhsError), .invalid(let rhsError)): + return lhsError == rhsError + default: + return false + } + } +} + +func == (lhs: TextFieldValidationError, rhs: TextFieldValidationError) -> Bool { + guard String(describing: lhs) == String(describing: rhs) else { + return false + } + return (lhs as NSError).isEqual(rhs as NSError) +} diff --git a/Stripe/StripeiOSTests/UINavigationBar+StripeTest.m b/Stripe/StripeiOSTests/UINavigationBar+StripeTest.m new file mode 100644 index 00000000..c80b5898 --- /dev/null +++ b/Stripe/StripeiOSTests/UINavigationBar+StripeTest.m @@ -0,0 +1,52 @@ +// +// UINavigationBar+StripeTest.m +// Stripe +// +// Created by Brian Dorfman on 12/9/16. +// Copyright © 2016 Stripe, Inc. All rights reserved. +// + +#import +#import +@import Stripe; +#import "STPFixtures.h" +#import "STPMocks.h" + +@interface UINavigationBar_StripeTest : XCTestCase + +@end + +@implementation UINavigationBar_StripeTest + +- (STPPaymentOptionsViewController *)buildPaymentOptionsViewController { + id customerContext = [STPMocks staticCustomerContext]; + STPPaymentConfiguration *config = [STPFixtures paymentConfiguration]; + STPTheme *theme = [STPTheme defaultTheme]; + id delegate = OCMProtocolMock(@protocol(STPPaymentOptionsViewControllerDelegate)); + STPPaymentOptionsViewController *paymentOptionsVC = [[STPPaymentOptionsViewController alloc] initWithConfiguration:config + theme:theme + customerContext:customerContext + delegate:delegate]; + return paymentOptionsVC; +} + +- (void)testVCUsesNavigationBarColor { + STPPaymentOptionsViewController *paymentOptionsVC = [self buildPaymentOptionsViewController]; + STPTheme *navTheme = [STPTheme new]; + navTheme.accentColor = [UIColor purpleColor]; + + UINavigationController *navController = [[UINavigationController alloc] initWithRootViewController:paymentOptionsVC]; + navController.navigationBar.stp_theme = navTheme; + __unused UIView *view = paymentOptionsVC.view; + XCTAssertEqualObjects(paymentOptionsVC.navigationItem.leftBarButtonItem.tintColor, [UIColor purpleColor]); +} + +- (void)testVCDoesNotUseNavigationBarColor { + STPPaymentOptionsViewController *paymentOptionsVC = [self buildPaymentOptionsViewController]; + __unused UINavigationController *navController = [[UINavigationController alloc] initWithRootViewController:paymentOptionsVC]; + __unused UIView *view = paymentOptionsVC.view; + XCTAssertEqualObjects(paymentOptionsVC.navigationItem.leftBarButtonItem.tintColor, [STPTheme defaultTheme].accentColor); +} + + +@end diff --git a/Stripe/StripeiOSTests/UserDefaults+StripeTest.swift b/Stripe/StripeiOSTests/UserDefaults+StripeTest.swift new file mode 100644 index 00000000..4803d5df --- /dev/null +++ b/Stripe/StripeiOSTests/UserDefaults+StripeTest.swift @@ -0,0 +1,33 @@ +// +// UserDefaults+StripeTest.swift +// StripeiOS Tests +// +// Created by Yuki Tokuhiro on 5/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePayments +@testable@_spi(STP) import StripePaymentsUI + +class UserDefaults_StripeTest: XCTestCase { + func testFraudDetectionData() throws { + let fraudDetectionData = FraudDetectionData( + sid: UUID().uuidString, + muid: UUID().uuidString, + guid: UUID().uuidString, + sidCreationDate: Date() + ) + UserDefaults.standard.fraudDetectionData = fraudDetectionData + XCTAssertEqual(UserDefaults.standard.fraudDetectionData, fraudDetectionData) + } + + func testCustomerToLastSelectedPaymentMethod() throws { + let c = [UUID().uuidString: UUID().uuidString] + UserDefaults.standard.customerToLastSelectedPaymentMethod = c + XCTAssertEqual(UserDefaults.standard.customerToLastSelectedPaymentMethod, c) + } +} diff --git a/Stripe/StripeiOSTests/WalletHeaderViewSnapshotTests.swift b/Stripe/StripeiOSTests/WalletHeaderViewSnapshotTests.swift new file mode 100644 index 00000000..394386e9 --- /dev/null +++ b/Stripe/StripeiOSTests/WalletHeaderViewSnapshotTests.swift @@ -0,0 +1,198 @@ +// +// WalletHeaderViewSnapshotTests.swift +// StripeiOS Tests +// +// Created by Ramon Torres on 12/9/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import iOSSnapshotTestCase +import StripeCoreTestUtils +import UIKit + +@testable@_spi(STP) import Stripe +@testable@_spi(STP) import StripeCore +@testable@_spi(STP) import StripePaymentSheet + +class WalletHeaderViewSnapshotTests: FBSnapshotTestCase { + + override func setUp() { + super.setUp() + // self.recordMode = true + } + + func testApplePayButton() { + let headerView = PaymentSheetViewController.WalletHeaderView( + options: .applePay, + delegate: nil + ) + verify(headerView) + } + + func testApplePayButtonWithCustomCta() { + let headerView = PaymentSheetViewController.WalletHeaderView( + options: .applePay, + applePayButtonType: .buy, + delegate: nil + ) + verify(headerView) + } + + func testLinkButton() { + let headerView = PaymentSheetViewController.WalletHeaderView( + options: .link, + delegate: nil + ) + verify(headerView) + } + + // Tests UI elements that adapt their color based on the `PaymentSheet.Appearance` + @available(iOS 13.0, *) + func testAdaptiveElements() { + var darkMode = false + + var appearance = PaymentSheet.Appearance() + appearance.colors.background = UIColor.init(dynamicProvider: { _ in + if darkMode { + return .black + } + + return .white + }) + + appearance.cornerRadius = 0 + let headerView = PaymentSheetViewController.WalletHeaderView( + options: .applePay, + appearance: appearance, + delegate: nil + ) + + verify(headerView, identifier: "Light") + + darkMode = true + headerView.traitCollectionDidChange(nil) + + verify(headerView, identifier: "Dark") + } + + // Tests UI elements that adapt their color based on the `PaymentSheet.Appearance` + @available(iOS 13.0, *) + func testAdaptiveElementsWithCustomApplePayCta() { + var darkMode = false + + var appearance = PaymentSheet.Appearance() + appearance.colors.background = UIColor.init(dynamicProvider: { _ in + if darkMode { + return .black + } + + return .white + }) + + appearance.cornerRadius = 0 + let headerView = PaymentSheetViewController.WalletHeaderView( + options: .applePay, + appearance: appearance, + applePayButtonType: .buy, + delegate: nil + ) + + verify(headerView, identifier: "Light") + + darkMode = true + headerView.traitCollectionDidChange(nil) + + verify(headerView, identifier: "Dark") + } + + func testAllButtons() { + let headerView = PaymentSheetViewController.WalletHeaderView( + options: [.applePay, .link], + delegate: nil + ) + verify(headerView) + + headerView.showsCardPaymentMessage = true + verify(headerView, identifier: "Card only") + } + + func testAllButtonsWithCustomApplePayCta() { + let headerView = PaymentSheetViewController.WalletHeaderView( + options: [.applePay, .link], + applePayButtonType: .buy, + delegate: nil + ) + verify(headerView) + + headerView.showsCardPaymentMessage = true + verify(headerView, identifier: "Card only") + } + + func testCustomFont() throws { + var appearance = PaymentSheet.Appearance.default + appearance.font.base = try XCTUnwrap(UIFont(name: "AmericanTypewriter", size: 12.0)) + + let headerView = PaymentSheetViewController.WalletHeaderView( + options: [.applePay, .link], + appearance: appearance, + delegate: nil + ) + + verify(headerView) + } + + func testCustomFontScales() throws { + var appearance = PaymentSheet.Appearance.default + appearance.font.base = try XCTUnwrap(UIFont(name: "AmericanTypewriter", size: 12.0)) + appearance.font.sizeScaleFactor = 1.25 + + let headerView = PaymentSheetViewController.WalletHeaderView( + options: [.applePay, .link], + appearance: appearance, + delegate: nil + ) + + verify(headerView) + } + + func testCustomCornerRadius() { + var appearance = PaymentSheet.Appearance.default + appearance.cornerRadius = 14.5 + + let headerView = PaymentSheetViewController.WalletHeaderView( + options: [.applePay, .link], + appearance: appearance, + delegate: nil + ) + + verify(headerView) + } + + func verify( + _ view: UIView, + identifier: String? = nil, + file: StaticString = #filePath, + line: UInt = #line + ) { + view.autosizeHeight(width: 300) + STPSnapshotVerifyView(view, identifier: identifier, file: file, line: line) + } +} + +extension WalletHeaderViewSnapshotTests { + fileprivate struct LinkAccountStub: PaymentSheetLinkAccountInfoProtocol { + let email: String + let redactedPhoneNumber: String? + let isRegistered: Bool + let isLoggedIn: Bool + } + + fileprivate func makeLinkAccountStub() -> LinkAccountStub { + return LinkAccountStub( + email: "customer@example.com", + redactedPhoneNumber: "+1********55", + isRegistered: true, + isLoggedIn: true + ) + } +} diff --git a/Stripe3DS2/BuildConfigurations/Project-Debug.xcconfig b/Stripe3DS2/BuildConfigurations/Project-Debug.xcconfig new file mode 100644 index 00000000..039738ae --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Project-Debug.xcconfig @@ -0,0 +1,15 @@ +// +// Project-Debug.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Project-Shared.xcconfig" + +ENABLE_TESTABILITY = YES +GCC_DYNAMIC_NO_PIC = NO +GCC_OPTIMIZATION_LEVEL = 0 +GCC_PREPROCESSOR_DEFINITIONS = DEBUG=1 $(inherited) +MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE +ONLY_ACTIVE_ARCH = YES \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Project-Release.xcconfig b/Stripe3DS2/BuildConfigurations/Project-Release.xcconfig new file mode 100644 index 00000000..e2ec59ca --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Project-Release.xcconfig @@ -0,0 +1,12 @@ +// +// Project-Release.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Project-Shared.xcconfig" + +ENABLE_NS_ASSERTIONS = NO +MTL_ENABLE_DEBUG_INFO = NO +VALIDATE_PRODUCT = YES \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Project-Shared.xcconfig b/Stripe3DS2/BuildConfigurations/Project-Shared.xcconfig new file mode 100644 index 00000000..cca99b2c --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Project-Shared.xcconfig @@ -0,0 +1,63 @@ +// +// Project-Shared.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +ALWAYS_SEARCH_USER_PATHS = NO +CLANG_ANALYZER_NONNULL = YES +CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE +CLANG_CXX_LANGUAGE_STANDARD = gnu++14 +CLANG_CXX_LIBRARY = libc++ +CLANG_ENABLE_MODULES = YES +CLANG_ENABLE_OBJC_ARC = YES +CLANG_ENABLE_OBJC_WEAK = YES +CLANG_WARN__DUPLICATE_METHOD_MATCH = YES +CLANG_WARN_ASSIGN_ENUM = YES +CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES +CLANG_WARN_BOOL_CONVERSION = YES +CLANG_WARN_COMMA = YES +CLANG_WARN_CONSTANT_CONVERSION = YES +CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES +CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR +CLANG_WARN_DOCUMENTATION_COMMENTS = YES +CLANG_WARN_EMPTY_BODY = YES +CLANG_WARN_ENUM_CONVERSION = YES +CLANG_WARN_IMPLICIT_SIGN_CONVERSION = YES +CLANG_WARN_INFINITE_RECURSION = YES +CLANG_WARN_INT_CONVERSION = YES +CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES +CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES +CLANG_WARN_OBJC_INTERFACE_IVARS = YES +CLANG_WARN_OBJC_LITERAL_CONVERSION = YES +CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR +CLANG_WARN_RANGE_LOOP_ANALYSIS = YES +CLANG_WARN_SEMICOLON_BEFORE_METHOD_BODY = YES +CLANG_WARN_STRICT_PROTOTYPES = YES +CLANG_WARN_SUSPICIOUS_IMPLICIT_CONVERSION = YES +CLANG_WARN_SUSPICIOUS_MOVE = YES +CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE +CLANG_WARN_UNREACHABLE_CODE = YES +COPY_PHASE_STRIP = NO +CURRENT_PROJECT_VERSION = 1.0 +DEBUG_INFORMATION_FORMAT = dwarf-with-dsym +ENABLE_STRICT_OBJC_MSGSEND = YES +GCC_C_LANGUAGE_STANDARD = gnu11 +GCC_NO_COMMON_BLOCKS = YES +GCC_TREAT_INCOMPATIBLE_POINTER_TYPE_WARNINGS_AS_ERRORS = YES +GCC_TREAT_WARNINGS_AS_ERRORS = YES +GCC_WARN_64_TO_32_BIT_CONVERSION = YES +GCC_WARN_ABOUT_MISSING_NEWLINE = YES +GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR +GCC_WARN_SIGN_COMPARE = YES +GCC_WARN_UNDECLARED_SELECTOR = YES +GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE +GCC_WARN_UNUSED_FUNCTION = YES +GCC_WARN_UNUSED_VARIABLE = YES +IPHONEOS_DEPLOYMENT_TARGET = 13.0 +MTL_FAST_MATH = YES +SDKROOT = iphoneos +SWIFT_VERSION=5.0 +VERSION_INFO_PREFIX = +VERSIONING_SYSTEM = apple-generic diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2-Debug.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2-Debug.xcconfig new file mode 100644 index 00000000..34a46e79 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2-Debug.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2-Debug.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2-Release.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2-Release.xcconfig new file mode 100644 index 00000000..1885c226 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2-Release.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2-Release.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2-Shared.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2-Shared.xcconfig new file mode 100644 index 00000000..02d8fe27 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2-Shared.xcconfig @@ -0,0 +1,23 @@ +// +// Stripe3DS2-Shared.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +APPLICATION_EXTENSION_API_ONLY = YES +BUILD_LIBRARY_FOR_DISTRIBUTION = YES +CODE_SIGN_STYLE = Automatic +DEFINES_MODULE = YES +DEPLOYMENT_POSTPROCESSING = YES +DYLIB_COMPATIBILITY_VERSION = 1 +DYLIB_CURRENT_VERSION = 1 +DYLIB_INSTALL_NAME_BASE = @rpath +GCC_PREFIX_HEADER = $(SRCROOT)/Stripe3DS2/include/Stripe3DS2-Prefix.pch +INSTALL_PATH = $(LOCAL_LIBRARY_DIR)/Frameworks +IPHONEOS_DEPLOYMENT_TARGET = 13.0 +OTHER_LDFLAGS = +SKIP_INSTALL = YES +STRIP_STYLE = non-global +TARGETED_DEVICE_FAMILY = 1,2 +LD_RUNPATH_SEARCH_PATHS = $(inherited) @executable_path/Frameworks @loader_path/Frameworks diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Debug.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Debug.xcconfig new file mode 100644 index 00000000..bd71fac1 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Debug.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2DemoUI-Debug.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2DemoUI-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Release.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Release.xcconfig new file mode 100644 index 00000000..baf74545 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Release.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2DemoUI-Release.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2DemoUI-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Shared.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Shared.xcconfig new file mode 100644 index 00000000..fb8baac4 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUI-Shared.xcconfig @@ -0,0 +1,13 @@ +// +// Stripe3DS2DemoUI-Shared.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon +CODE_SIGN_STYLE = Automatic +IPHONEOS_DEPLOYMENT_TARGET = 13.0 +TARGETED_DEVICE_FAMILY = 1,2 +DYLIB_INSTALL_NAME_BASE = @rpath +LD_RUNPATH_SEARCH_PATHS = $(inherited) @executable_path/Frameworks @loader_path/Frameworks diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Debug.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Debug.xcconfig new file mode 100644 index 00000000..53c3f9ff --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Debug.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2DemoUITests-Debug.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2DemoUITests-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Release.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Release.xcconfig new file mode 100644 index 00000000..7251a5cc --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Release.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2DemoUITests-Release.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2DemoUITests-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Shared.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Shared.xcconfig new file mode 100644 index 00000000..e520a8af --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2DemoUITests-Shared.xcconfig @@ -0,0 +1,13 @@ +// +// Stripe3DS2DemoUITests-Shared.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +BUNDLE_LOADER = $(TEST_HOST) +CODE_SIGN_STYLE = Automatic +TARGETED_DEVICE_FAMILY = 1,2 +TEST_HOST = $(BUILT_PRODUCTS_DIR)/Stripe3DS2DemoUI.app/Stripe3DS2DemoUI +DYLIB_INSTALL_NAME_BASE = @rpath +LD_RUNPATH_SEARCH_PATHS = $(inherited) @executable_path/Frameworks @loader_path/Frameworks diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Debug.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Debug.xcconfig new file mode 100644 index 00000000..1744b025 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Debug.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2Tests-Debug.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2Tests-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Release.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Release.xcconfig new file mode 100644 index 00000000..f03b2ac4 --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Release.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2Tests-Release.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +#include "Stripe3DS2Tests-Shared.xcconfig" + +//********************************************// +//* Currently no build settings in this file *// +//********************************************// \ No newline at end of file diff --git a/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Shared.xcconfig b/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Shared.xcconfig new file mode 100644 index 00000000..2e597a2e --- /dev/null +++ b/Stripe3DS2/BuildConfigurations/Stripe3DS2Tests-Shared.xcconfig @@ -0,0 +1,12 @@ +// +// Stripe3DS2Tests-Shared.xcconfig +// +// Generated by BuildSettingExtractor on 12/5/22 +// https://buildsettingextractor.com +// + +CODE_SIGN_STYLE = Automatic +OTHER_LDFLAGS = -ObjC +TARGETED_DEVICE_FAMILY = 1,2 +DYLIB_INSTALL_NAME_BASE = @rpath +LD_RUNPATH_SEARCH_PATHS = $(inherited) @executable_path/Frameworks @loader_path/Frameworks diff --git a/Stripe3DS2/Project.swift b/Stripe3DS2/Project.swift new file mode 100644 index 00000000..f9d34aac --- /dev/null +++ b/Stripe3DS2/Project.swift @@ -0,0 +1,126 @@ +import ProjectDescription +import ProjectDescriptionHelpers + +let project = Project( + name: "Stripe3DS2", + options: .options( + automaticSchemesOptions: .disabled, + disableBundleAccessors: true, + disableSynthesizedResourceAccessors: true + ), + packages: [ + .remote( + url: "https://github.com/uber/ios-snapshot-test-case", + requirement: .upToNextMajor(from: "8.0.0") + ), + ], + settings: .settings( + configurations: [ + .debug( + name: "Debug", + xcconfig: "BuildConfigurations/Project-Debug.xcconfig" + ), + .release( + name: "Release", + xcconfig: "BuildConfigurations/Project-Release.xcconfig" + ), + ], + defaultSettings: .none + ), + targets: [ + Target( + name: "Stripe3DS2", + platform: .iOS, + product: .framework, + bundleId: "com.stripe.stripe-3ds2", + infoPlist: "Stripe3DS2/Info.plist", + sources: "Stripe3DS2/**/*.m", + resources: "Stripe3DS2/Resources/**", + headers: .headers( + public: [ + "Stripe3DS2/include/*.h", + ], + project: "Stripe3DS2/*.h" + ), + settings: .stripeTargetSettings( + baseXcconfigFilePath: "BuildConfigurations/Stripe3DS2" + ) + ), + Target( + name: "Stripe3DS2Tests", + platform: .iOS, + product: .unitTests, + bundleId: "com.stripe.Stripe3DS2Tests", + infoPlist: "Stripe3DS2Tests/Info.plist", + sources: "Stripe3DS2Tests/**/*.m", + resources: "Stripe3DS2Tests/JSON/**", + headers: .headers( + project: "Stripe3DS2/**/*.h" + ), + dependencies: [ + .xctest, + .target(name: "Stripe3DS2"), + ], + settings: .stripeTargetSettings( + baseXcconfigFilePath: "BuildConfigurations/Stripe3DS2Tests" + ) + ), + Target( + name: "Stripe3DS2DemoUI", + platform: .iOS, + product: .app, + bundleId: "com.stripe.Stripe3DS2DemoUI", + infoPlist: "Stripe3DS2DemoUI/Info.plist", + sources: "Stripe3DS2DemoUI/Sources/**/*.m", + resources: "Stripe3DS2DemoUI/Resources/**", + headers: .headers( + project: "Stripe3DS2DemoUI/Sources/**/*.h" + ), + dependencies: [ + .target(name: "Stripe3DS2"), + ], + settings: .stripeTargetSettings( + baseXcconfigFilePath: "BuildConfigurations/Stripe3DS2DemoUI" + ) + ), + Target( + name: "Stripe3DS2DemoUITests", + platform: .iOS, + product: .unitTests, + bundleId: "com.stripe.Stripe3DS2DemoUITests", + infoPlist: "Stripe3DS2DemoUITests/Info.plist", + sources: "Stripe3DS2DemoUITests/**/*.m", + dependencies: [ + .xctest, + .target(name: "Stripe3DS2"), + .target(name: "Stripe3DS2DemoUI"), + .package(product: "iOSSnapshotTestCase"), + ], + settings: .stripeTargetSettings( + baseXcconfigFilePath: "BuildConfigurations/Stripe3DS2DemoUITests" + ) + ), + ], + schemes: [ + Scheme( + name: "Stripe3DS2", + buildAction: .buildAction(targets: ["Stripe3DS2"]), + testAction: .targets(["Stripe3DS2Tests"]) + ), + Scheme( + name: "Stripe3DS2DemoUI", + buildAction: .buildAction(targets: ["Stripe3DS2DemoUI"]), + testAction: .targets( + ["Stripe3DS2DemoUITests"], + arguments: Arguments( + environment: [ + "FB_REFERENCE_IMAGE_DIR": + "$(SOURCE_ROOT)/../Tests/ReferenceImages", + ] + ), + expandVariableFromTarget: "Stripe3DS2DemoUITests" + ), + runAction: .runAction(executable: "Stripe3DS2DemoUI") + ), + ] +) diff --git a/Stripe3DS2/Stripe3DS2.xcodeproj/project.pbxproj b/Stripe3DS2/Stripe3DS2.xcodeproj/project.pbxproj new file mode 100644 index 00000000..8e5d362a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2.xcodeproj/project.pbxproj @@ -0,0 +1,1748 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 52; + objects = { + +/* Begin PBXBuildFile section */ + 01706B4660728A5BFAC12840 /* STDSRuntimeException.m in Sources */ = {isa = PBXBuildFile; fileRef = CD33421F13A675DCFE597FFB /* STDSRuntimeException.m */; }; + 05647ABCFBDBE1209D5044AC /* STDSEphemeralKeyPair.m in Sources */ = {isa = PBXBuildFile; fileRef = F9D521B45783D36C8E83F0EA /* STDSEphemeralKeyPair.m */; }; + 0615DD02C0B022AE207DADF0 /* STDSUICustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = 210B22FF4DCB0C6E7C763EAB /* STDSUICustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 081347606D48F64161D95B45 /* mastercard.der in Resources */ = {isa = PBXBuildFile; fileRef = 1104D5855D28E49E104CA004 /* mastercard.der */; }; + 089DAFECA444861762781974 /* NSError+Stripe3DS2.m in Sources */ = {isa = PBXBuildFile; fileRef = 26C20D4B77372845627D6466 /* NSError+Stripe3DS2.m */; }; + 08ECC7E70E7478E76793ED1E /* UIViewController+Stripe3DS2.m in Sources */ = {isa = PBXBuildFile; fileRef = 99451064136843047C7881AD /* UIViewController+Stripe3DS2.m */; }; + 09DFAF7B38EAB76BC66AC9C8 /* STDSChallengeResponseObject.m in Sources */ = {isa = PBXBuildFile; fileRef = 24CE086326AF3DE336BF4F4C /* STDSChallengeResponseObject.m */; }; + 09F0B1945CC12FB1215119FD /* STDSProcessingView.h in Headers */ = {isa = PBXBuildFile; fileRef = 95641ECBE1AE1CC198013405 /* STDSProcessingView.h */; }; + 0A0D73B3004DC5854D2BC639 /* mastercard-logo@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 5862FA7E6300A2169A3CD3D7 /* mastercard-logo@3x.png */; }; + 0A2BC6A9E388B242C46C09D9 /* UIButton+CustomInitialization.h in Headers */ = {isa = PBXBuildFile; fileRef = 4DD55BECC3FB85216FE46218 /* UIButton+CustomInitialization.h */; }; + 0BA0F0CADC4CFF419E1F7EFC /* STDSLabelCustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = 6932DD26C7C16938DFA0E198 /* STDSLabelCustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 0D3D9C95CB14A610EAAAEF6E /* STDSJSONEncoder.m in Sources */ = {isa = PBXBuildFile; fileRef = 0681C043F732148587737233 /* STDSJSONEncoder.m */; }; + 0D4B42EC5019D213BDBC84E7 /* NSDictionary+DecodingHelpers.m in Sources */ = {isa = PBXBuildFile; fileRef = C566D6E444FCA4BCEC65F3D2 /* NSDictionary+DecodingHelpers.m */; }; + 0EEFDE04392FD017EA0A017A /* STDSDeviceInformationParameter.m in Sources */ = {isa = PBXBuildFile; fileRef = 25C7D18868DDADEF4CB6220C /* STDSDeviceInformationParameter.m */; }; + 1134F81C7D015BA5EA52BFD1 /* NSDictionary+DecodingHelpers.h in Headers */ = {isa = PBXBuildFile; fileRef = C78AA1D7AD2BAB52EE58D078 /* NSDictionary+DecodingHelpers.h */; }; + 1156B25EBE0135B627E174D2 /* STDSProgressViewController.m in Sources */ = {isa = PBXBuildFile; fileRef = 84967ED0D9B206B3F5AA4F02 /* STDSProgressViewController.m */; }; + 125427F1322501AA12EAEBE6 /* visa.der in Resources */ = {isa = PBXBuildFile; fileRef = A70071543985284E0D9DAC64 /* visa.der */; }; + 128B64380D6724F016EE8D56 /* STDSDemoViewController.m in Sources */ = {isa = PBXBuildFile; fileRef = 382C583385FAECA91366769E /* STDSDemoViewController.m */; }; + 136255493F3F0E930FBA8606 /* STDSImageLoader.m in Sources */ = {isa = PBXBuildFile; fileRef = A09A89BBE7360F93195641C9 /* STDSImageLoader.m */; }; + 1469551DB874336B68F6C39D /* Stripe3DS2.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = CE4030C50383B488C97F4B56 /* Stripe3DS2.framework */; }; + 149FE0C2910F08910B250BD8 /* STDSException+Internal.h in Headers */ = {isa = PBXBuildFile; fileRef = 461F938CDE7ED6D06D9D2700 /* STDSException+Internal.h */; }; + 14BDBD98F8E15576403255C9 /* STDSChallengeInformationView.m in Sources */ = {isa = PBXBuildFile; fileRef = D24777EE9931075405F370DB /* STDSChallengeInformationView.m */; }; + 15D4A2F07EA477B84CD48B15 /* STDSRuntimeErrorEvent.h in Headers */ = {isa = PBXBuildFile; fileRef = DE7D85369E012B15702D8DF9 /* STDSRuntimeErrorEvent.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 1767509FDC216602A5E43F0E /* STDSErrorMessageTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 278F2C40987F5218BD031E3D /* STDSErrorMessageTest.m */; }; + 1B20B454D0C309613A1FAF68 /* STDSThreeDSProtocolVersion.m in Sources */ = {isa = PBXBuildFile; fileRef = 8AC648332A706809F1BDFD59 /* STDSThreeDSProtocolVersion.m */; }; + 206B93BDE91CFED74A053B87 /* STDSStackView.h in Headers */ = {isa = PBXBuildFile; fileRef = B7AE3B4732D203134FE096FE /* STDSStackView.h */; }; + 2086DFD1FC02783FB106AD35 /* UIColor+DefaultColors.m in Sources */ = {isa = PBXBuildFile; fileRef = 869733153BCA6BE2EB7A452F /* UIColor+DefaultColors.m */; }; + 2224463CE5FB99D4BEE6A479 /* STDSJSONEncoder.h in Headers */ = {isa = PBXBuildFile; fileRef = 06C5207432FB07BD71703BCC /* STDSJSONEncoder.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 24D49F833E1ED28DE557F219 /* STDSException.h in Headers */ = {isa = PBXBuildFile; fileRef = B71A1C110DCC23A9CE929837 /* STDSException.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 24F626CD0F93B7AF59B1D6AA /* STDSThreeDS2Service.m in Sources */ = {isa = PBXBuildFile; fileRef = 77646CEC7EE754F47EDBDA4F /* STDSThreeDS2Service.m */; }; + 27F3EA1BB7E7B56A1A8524BA /* UIColor+ThirteenSupport.h in Headers */ = {isa = PBXBuildFile; fileRef = 9A8B0001705400C661678617 /* UIColor+ThirteenSupport.h */; }; + 284F72D3FD8C06C0E5974086 /* STDSTextFieldCustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = 97163B5E9598CA3A298920B9 /* STDSTextFieldCustomization.m */; }; + 28AD7EC6F6DA2AECD7F61BA7 /* NSLayoutConstraint+LayoutSupport.h in Headers */ = {isa = PBXBuildFile; fileRef = E0949399E96663964091C0EF /* NSLayoutConstraint+LayoutSupport.h */; }; + 2938D9CC41899DC78D01EF9F /* STDSDeviceInformationParameter.h in Headers */ = {isa = PBXBuildFile; fileRef = 172AF5C85EDBD92A1030E361 /* STDSDeviceInformationParameter.h */; }; + 2970D4880EBFAEFABA2E8EBD /* STDSChallengeRequestParameters.m in Sources */ = {isa = PBXBuildFile; fileRef = 60693BC560C4927ACFD2E6C3 /* STDSChallengeRequestParameters.m */; }; + 2AB06EC20B22306F1DDF9F4B /* STDSChallengeResponseImageObject.m in Sources */ = {isa = PBXBuildFile; fileRef = 3A9F828840A3B361842E38DE /* STDSChallengeResponseImageObject.m */; }; + 2B09B7FDF8C4AA551F7EE18E /* STDSDeviceInformationParameterTests.m in Sources */ = {isa = PBXBuildFile; fileRef = FC52C3C844BDA981EE34BAB9 /* STDSDeviceInformationParameterTests.m */; }; + 2B2787A7103C0D0AB5F00B78 /* STDSAuthenticationResponseObject.h in Headers */ = {isa = PBXBuildFile; fileRef = 138E82072FC6572612A8E248 /* STDSAuthenticationResponseObject.h */; }; + 2BBA82878804711E06CBFCCB /* STDSErrorMessage.h in Headers */ = {isa = PBXBuildFile; fileRef = 645D0BB927B7F8A0D37EE04D /* STDSErrorMessage.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 2C5621AABECF6192F92A20CF /* STDSStripe3DS2Error.m in Sources */ = {isa = PBXBuildFile; fileRef = 9F4376BB07FD1EE783249AED /* STDSStripe3DS2Error.m */; }; + 2C6DB6516699B4EFADDF3360 /* STDSRuntimeException.h in Headers */ = {isa = PBXBuildFile; fileRef = B8619CA38E2A5B49DBF8546B /* STDSRuntimeException.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 2CB6DF5CCCBC1EB9ABD03143 /* STDSNotInitializedException.h in Headers */ = {isa = PBXBuildFile; fileRef = 2FCABBF51CBA7F9FEF757B4C /* STDSNotInitializedException.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 2E2022723BC7A4B2DD5ECE76 /* STDSErrorMessage+Internal.m in Sources */ = {isa = PBXBuildFile; fileRef = 2CB591796654445B8434A501 /* STDSErrorMessage+Internal.m */; }; + 30632F7C730BF8D23B607CF7 /* STDSWarningTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 474F39AA3F47D84F8D54E5B7 /* STDSWarningTests.m */; }; + 30B25163BEDEB99CDD101B5F /* NSString+EmptyChecking.h in Headers */ = {isa = PBXBuildFile; fileRef = A8A77DB1C711B8696B2E5FEF /* NSString+EmptyChecking.h */; }; + 3192FCC841152A7E5E6F19D6 /* STDSAlreadyInitializedException.h in Headers */ = {isa = PBXBuildFile; fileRef = 1F1FD11407319293954C6D81 /* STDSAlreadyInitializedException.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 31C636A25BF83316EE7EB57D /* STDSThreeDS2Service.h in Headers */ = {isa = PBXBuildFile; fileRef = 5EAA444FA7C82BDA4AD907BF /* STDSThreeDS2Service.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 326AD1DD7BB8A2A6DA66EC67 /* STDSThreeDSProtocolVersion+Private.h in Headers */ = {isa = PBXBuildFile; fileRef = 47C29E9F3D2A62676B424CD1 /* STDSThreeDSProtocolVersion+Private.h */; }; + 32BE38D04DD76CA4490389C6 /* STDSConfigParametersTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 5F551808E65CDADD9B3A9CB8 /* STDSConfigParametersTests.m */; }; + 3343DA94A5032627843A343C /* STDSEphemeralKeyPair+Testing.h in Headers */ = {isa = PBXBuildFile; fileRef = 52D6E4F76CBA258163E57DF3 /* STDSEphemeralKeyPair+Testing.h */; }; + 335553A547A3CA80B3901CCF /* STDSStripe3DS2Error.h in Headers */ = {isa = PBXBuildFile; fileRef = AE53BAA72835AFA3B35C58D3 /* STDSStripe3DS2Error.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 34256504F0E0E519D586B7CE /* STDSErrorMessage+Internal.h in Headers */ = {isa = PBXBuildFile; fileRef = 5463EB1CC8E85BE3BD12DEFF /* STDSErrorMessage+Internal.h */; }; + 3617B07ABD719F1A5F8302FA /* NSError+Stripe3DS2.h in Headers */ = {isa = PBXBuildFile; fileRef = A6778F8CE36F769DF608F932 /* NSError+Stripe3DS2.h */; }; + 3844F0E21742F43BCB32499A /* STDSDeviceInformationParameter+Private.h in Headers */ = {isa = PBXBuildFile; fileRef = 4869D6E14F01EDB960CB3065 /* STDSDeviceInformationParameter+Private.h */; }; + 390691CA0EAF81418B0C65DB /* UIView+LayoutSupport.m in Sources */ = {isa = PBXBuildFile; fileRef = 8B0F0485B7048DDAA7DA8F9B /* UIView+LayoutSupport.m */; }; + 3909D8028AC485BCCF17B48B /* STDSStackView.m in Sources */ = {isa = PBXBuildFile; fileRef = 1D6269F8B91341C387A911DB /* STDSStackView.m */; }; + 3B430F172A3AD6AA9AFDFC04 /* STDSNavigationBarCustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = E1630D876436E43547FF69CE /* STDSNavigationBarCustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 3C6D16D8E7B5BC865D956A0B /* iOSSnapshotTestCase in Frameworks */ = {isa = PBXBuildFile; productRef = 0118969F6608A92583CC6C98 /* iOSSnapshotTestCase */; }; + 3C88AE37A760B91E93D423CF /* STDSException.m in Sources */ = {isa = PBXBuildFile; fileRef = 6F030410FDABFF833CD8FE46 /* STDSException.m */; }; + 3C8C3BCDDDDEF6B9E150D4E1 /* STDSChallengeResponseSelectionInfo.h in Headers */ = {isa = PBXBuildFile; fileRef = 52EEBC7D74623E1A1D7E3219 /* STDSChallengeResponseSelectionInfo.h */; }; + 3D674B4EE5ABACD65EB21A1E /* discover.der in Resources */ = {isa = PBXBuildFile; fileRef = 04AB48F014A8D5BA56FEF463 /* discover.der */; }; + 3F2624C33291FCE00D596768 /* STDSDeviceInformationManager.m in Sources */ = {isa = PBXBuildFile; fileRef = 9580A82A59EB2B0103EDF47A /* STDSDeviceInformationManager.m */; }; + 3F30A8F418870D176A626F00 /* STDSSpacerView.m in Sources */ = {isa = PBXBuildFile; fileRef = 51E4A57094A671EB14380882 /* STDSSpacerView.m */; }; + 41246FCC0695BABE1489C53E /* UIViewController+Stripe3DS2.h in Headers */ = {isa = PBXBuildFile; fileRef = 8F72413F99F8E50FD0FA085D /* UIViewController+Stripe3DS2.h */; }; + 4142A3AB3E384F91A1E5550B /* STDSExpandableInformationView.m in Sources */ = {isa = PBXBuildFile; fileRef = AE1BBF3A441B1F782B766F61 /* STDSExpandableInformationView.m */; }; + 425849564519D6454DDA3424 /* NSData+JWEHelpers.h in Headers */ = {isa = PBXBuildFile; fileRef = 5B05A609753EE71502409082 /* NSData+JWEHelpers.h */; }; + 435EFC9119D4B42B2C6A6F88 /* STDSAuthenticationResponse.h in Headers */ = {isa = PBXBuildFile; fileRef = 4EA4AB3BA2FB41B3D5F38978 /* STDSAuthenticationResponse.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 4518AD83580FC222B883DAFB /* STDSSimulatorChecker.m in Sources */ = {isa = PBXBuildFile; fileRef = 14D60BDF245487FE3362BE07 /* STDSSimulatorChecker.m */; }; + 4587DADB1B2D98560E8BB181 /* error@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 1E3640C73D001D32CCC449F1 /* error@3x.png */; }; + 458BC5A3D0E7B4D05549474F /* STDSBrandingView.h in Headers */ = {isa = PBXBuildFile; fileRef = 2CFAC8D74AFFC2E61BAD82A5 /* STDSBrandingView.h */; }; + 480CD62EC91F4796D1D8D8DC /* STDSSecTypeUtilitiesTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 30D22985091381DFCDF077E9 /* STDSSecTypeUtilitiesTests.m */; }; + 48316C1AB1D9151C205A59A4 /* STDSChallengeResponseMessageExtensionObject.h in Headers */ = {isa = PBXBuildFile; fileRef = BAAC56EDE0AAD2E50E02E9AE /* STDSChallengeResponseMessageExtensionObject.h */; }; + 4A9423CC79312BC3A0DEC38B /* STDSChallengeRequestParametersTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 4EAD29E57D206B5D6BAF9099 /* STDSChallengeRequestParametersTest.m */; }; + 4C0E13298F08D391FE8600CF /* STDSJSONWebSignatureTests.m in Sources */ = {isa = PBXBuildFile; fileRef = E4C668442518B446D573AD24 /* STDSJSONWebSignatureTests.m */; }; + 4C267E9A30ADC18E71408ED5 /* STDSTextFieldCustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = 68ED2A0E11E1B27980E0C60A /* STDSTextFieldCustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 4C573A44AB0A9CB28D76C621 /* STDSJSONWebEncryption.h in Headers */ = {isa = PBXBuildFile; fileRef = 76EA9DF8572C992E30BCF8A3 /* STDSJSONWebEncryption.h */; }; + 4D73C2FBAC9B96ECB45BDC94 /* Stripe3DS2.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = CE4030C50383B488C97F4B56 /* Stripe3DS2.framework */; }; + 4D79F0CFC54E8B923E9238CF /* Stripe3DS2.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = CE4030C50383B488C97F4B56 /* Stripe3DS2.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + 4E35267801D94FB84816C519 /* STDSInvalidInputException.h in Headers */ = {isa = PBXBuildFile; fileRef = 853A361AB630A2BD402D1B47 /* STDSInvalidInputException.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 4E4E4BC1E2D041FA530336EE /* STDSSwiftTryCatch.m in Sources */ = {isa = PBXBuildFile; fileRef = 9946D28A86E85A6C84EB6C7B /* STDSSwiftTryCatch.m */; }; + 4FE8C076102C195D609977F5 /* STDSChallengeSelectionView.m in Sources */ = {isa = PBXBuildFile; fileRef = A4D460893CB5DDE52AA0AB85 /* STDSChallengeSelectionView.m */; }; + 511513A9180F9835B08CBE56 /* STDSProgressViewController.h in Headers */ = {isa = PBXBuildFile; fileRef = E629F85B783E42244E37DFA9 /* STDSProgressViewController.h */; }; + 53DBABFCA9B7E264366922F2 /* visa-logo@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 34FCE3CEEDC2D23116C11DAA /* visa-logo@3x.png */; }; + 574A7976213046F02F7F60C4 /* STDSProcessingView.m in Sources */ = {isa = PBXBuildFile; fileRef = 7AC1C5F6F1311F50D9C37291 /* STDSProcessingView.m */; }; + 57E58A3B88D9EEB9FAE9B383 /* STDSDeviceInformation.h in Headers */ = {isa = PBXBuildFile; fileRef = 1FBE88F6E2C3153A8315EDC6 /* STDSDeviceInformation.h */; }; + 59756023104E4FB886B48936 /* STDSTextChallengeView.h in Headers */ = {isa = PBXBuildFile; fileRef = 82C41E6DC2CC247ECC17F2B9 /* STDSTextChallengeView.h */; }; + 5B16918BF5F67B0F5FB1AFD5 /* ARes.json in Resources */ = {isa = PBXBuildFile; fileRef = 237977B021FE538E5E4FA35C /* ARes.json */; }; + 5C83A3A4631E51EEECBEAA0F /* STDSEllipticCurvePoint.h in Headers */ = {isa = PBXBuildFile; fileRef = 5049A72A6BEB558D8559A5EB /* STDSEllipticCurvePoint.h */; }; + 5DAC25EBAB19555229654E44 /* STDSAuthenticationResponseTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 28E32DA72CAFBF2AF8099151 /* STDSAuthenticationResponseTests.m */; }; + 5FE5FB933E2651C8D84FA477 /* STDSAuthenticationResponseObject.m in Sources */ = {isa = PBXBuildFile; fileRef = 5043E1CF6955176EC3D43F88 /* STDSAuthenticationResponseObject.m */; }; + 602C526C0B52DF81846DA664 /* STDSOSVersionChecker.m in Sources */ = {isa = PBXBuildFile; fileRef = F363439353A6DD554FC44AC2 /* STDSOSVersionChecker.m */; }; + 613EF855524F7861A6DDD8C1 /* STDSImageLoader.h in Headers */ = {isa = PBXBuildFile; fileRef = ECC649704CDDA50E73DB3C10 /* STDSImageLoader.h */; }; + 63AFB3C739DD987EE56F801F /* STDSDirectoryServerCertificate.h in Headers */ = {isa = PBXBuildFile; fileRef = 069990DA025BE9F33602E96A /* STDSDirectoryServerCertificate.h */; }; + 64D05353B50FAF3BB76D6527 /* STDSRuntimeErrorEvent.m in Sources */ = {isa = PBXBuildFile; fileRef = 1501A7A3010FFE53D00E318F /* STDSRuntimeErrorEvent.m */; }; + 65C3CC17DA8E8B98A5ACA320 /* cartes-bancaires-logo.png in Resources */ = {isa = PBXBuildFile; fileRef = 581BD6EEB19753FC873EB077 /* cartes-bancaires-logo.png */; }; + 675801746AD86B23F8560F59 /* Chevron@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 41CB752E5656578F2437E247 /* Chevron@3x.png */; }; + 67D431374DECE36B2606D944 /* STDSProtocolErrorEvent.h in Headers */ = {isa = PBXBuildFile; fileRef = 3A6D20D1DAB7D769E5E97528 /* STDSProtocolErrorEvent.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 68F314D63323954EFE309E49 /* STDSOSVersionChecker.h in Headers */ = {isa = PBXBuildFile; fileRef = FAE7026135B5049B732CEDEB /* STDSOSVersionChecker.h */; }; + 6B3AA85EE71FC42EF9BD999F /* STDSEllipticCurvePoint.m in Sources */ = {isa = PBXBuildFile; fileRef = 2EF5AA537586EF706E4056FD /* STDSEllipticCurvePoint.m */; }; + 6BA3F565B6C0B78F6DED8826 /* Localizable.strings in Resources */ = {isa = PBXBuildFile; fileRef = F31A6580385C6390910AFD93 /* Localizable.strings */; }; + 6DB54DBAEFD08967367B6BF7 /* STDSChallengeParameters.m in Sources */ = {isa = PBXBuildFile; fileRef = 23C929BF760735CC01E1E8F5 /* STDSChallengeParameters.m */; }; + 6E727D2B738D7A3527952D28 /* STDSTransaction.m in Sources */ = {isa = PBXBuildFile; fileRef = 5FDE2481364C454058BF2377 /* STDSTransaction.m */; }; + 72F430506E684D61BD5CFE3B /* STDSChallengeStatusReceiver.h in Headers */ = {isa = PBXBuildFile; fileRef = B7A75140FB62A71261CAF5EC /* STDSChallengeStatusReceiver.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 7337FB20DC0EF491B5922913 /* STDSChallengeResponse.h in Headers */ = {isa = PBXBuildFile; fileRef = 315AEE1534F9D9974DC1674B /* STDSChallengeResponse.h */; }; + 75150C4678B8207AA6E7D969 /* STDSBrandingView.m in Sources */ = {isa = PBXBuildFile; fileRef = F9C39A42ECBA58571EFA03C5 /* STDSBrandingView.m */; }; + 75CDFDABD7F9813563E3F618 /* Stripe3DS2-Bridging-Header.h in Headers */ = {isa = PBXBuildFile; fileRef = 0F024DEAD6628A0229856656 /* Stripe3DS2-Bridging-Header.h */; }; + 7706F5211DE7A4411CD108EA /* visa-white-logo@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = C7F1C55E51129933B02576E7 /* visa-white-logo@3x.png */; }; + 793D61AD55630DD336A141A7 /* STDSChallengeResponseMessageExtension.h in Headers */ = {isa = PBXBuildFile; fileRef = AA8BE982004398CF8AAA7D1E /* STDSChallengeResponseMessageExtension.h */; }; + 794D78D81790620AFA6AA5C3 /* amex-logo@3x.png in Resources */ = {isa = PBXBuildFile; fileRef = 947935869E5A08E35FA11E13 /* amex-logo@3x.png */; }; + 79557307ECD48D17C566997A /* STDSFooterCustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = BA96C9FFA48B85F9C42E8C68 /* STDSFooterCustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 7C20B044ACFD3A50FD4AC9A8 /* acs_challenge.html in Resources */ = {isa = PBXBuildFile; fileRef = 7971EB56716D5E2F59053B6D /* acs_challenge.html */; }; + 7C7073B54628ED836F422F1D /* STDSJailbreakChecker.h in Headers */ = {isa = PBXBuildFile; fileRef = 94943BEAB94E949AF9830EFA /* STDSJailbreakChecker.h */; }; + 7EC8E5523F29B0F01FE827DB /* STDSWebView.h in Headers */ = {isa = PBXBuildFile; fileRef = 581EEE576D24047004C828DF /* STDSWebView.h */; }; + 7ED98325E7E54A54BFE54D6C /* STDSAuthenticationRequestParameters.m in Sources */ = {isa = PBXBuildFile; fileRef = AB871F64FC72EBC7A91F96C1 /* STDSAuthenticationRequestParameters.m */; }; + 81B1C12449852CEB3C59793F /* ul-test.der in Resources */ = {isa = PBXBuildFile; fileRef = B6275E7099E55F0C04688890 /* ul-test.der */; }; + 8233B9F131602DD7143FF96F /* STDSBundleLocator.m in Sources */ = {isa = PBXBuildFile; fileRef = 9564F43EF7CF4F60C3C202CD /* STDSBundleLocator.m */; }; + 825ACBE0734BDCE746E66AB0 /* STDSWhitelistView.h in Headers */ = {isa = PBXBuildFile; fileRef = 33CEADC8006E456FD82CE134 /* STDSWhitelistView.h */; }; + 8322973CD09F2E1C88B6046E /* UIColor+DefaultColors.h in Headers */ = {isa = PBXBuildFile; fileRef = 4A567E08748C86FCE2A75FEA /* UIColor+DefaultColors.h */; }; + 83878337274D8C43D492EC45 /* STDSUICustomizationTests.m in Sources */ = {isa = PBXBuildFile; fileRef = CB3154597040B803ADE14F9E /* STDSUICustomizationTests.m */; }; + 847926A68A88BEB58C92D9BC /* STDSThreeDSProtocolVersion.h in Headers */ = {isa = PBXBuildFile; fileRef = 5E6A9565E97DF2544254AA94 /* STDSThreeDSProtocolVersion.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 88197445522F39DB1E9F6AE7 /* NSString+JWEHelpers.m in Sources */ = {isa = PBXBuildFile; fileRef = 43CB8F416E4F4CEA34391481 /* NSString+JWEHelpers.m */; }; + 884D59AF7FD70889276AFC02 /* STDSWebView.m in Sources */ = {isa = PBXBuildFile; fileRef = AA85BC717C46C1B95AF8C1E3 /* STDSWebView.m */; }; + 890E22971E6F7B0FA1C8D649 /* STDSJSONEncodable.h in Headers */ = {isa = PBXBuildFile; fileRef = E6A3C5D4C46E681B7D76B2BA /* STDSJSONEncodable.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 893713F4464A88FAB92BC2C6 /* Stripe3DS2.h in Headers */ = {isa = PBXBuildFile; fileRef = 28C6B4F6BA431B9342970FD6 /* Stripe3DS2.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 8A0872706BF4CDE72EF0C480 /* STDSConfigParameters.m in Sources */ = {isa = PBXBuildFile; fileRef = 89B5ED899439D1C1054CB448 /* STDSConfigParameters.m */; }; + 8A64AFBB2AC15E0E0F57ED25 /* STDSNotInitializedException.m in Sources */ = {isa = PBXBuildFile; fileRef = B7BD1E24EA9427121E148DFC /* STDSNotInitializedException.m */; }; + 8C9BD4B150F384111CBD3BD3 /* STDSErrorMessage.m in Sources */ = {isa = PBXBuildFile; fileRef = 6DDB6222A2EF31FDB7592368 /* STDSErrorMessage.m */; }; + 8E0501C3BB849C2D2D99F34E /* STDSLocalizedString.h in Headers */ = {isa = PBXBuildFile; fileRef = 41E0F638E776A7F3456BDA3E /* STDSLocalizedString.h */; }; + 8E27285FE76F8BECFD70286D /* STDSCompletionEvent.h in Headers */ = {isa = PBXBuildFile; fileRef = 10DB3CD8F2541AC06681F2C8 /* STDSCompletionEvent.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 8E833B7FCBC3A99072582FEA /* UIFont+DefaultFonts.h in Headers */ = {isa = PBXBuildFile; fileRef = 1E844798B2733C8511AC080E /* UIFont+DefaultFonts.h */; }; + 9146B2E3B89EF3F489D7E233 /* STDSIPAddress.m in Sources */ = {isa = PBXBuildFile; fileRef = CB00E968CF4DD07FB8BF2AAA /* STDSIPAddress.m */; }; + 955CBE0C979291F89567D8A7 /* STDSJSONWebEncryption.m in Sources */ = {isa = PBXBuildFile; fileRef = 99392D3F702EC6BF7CE15291 /* STDSJSONWebEncryption.m */; }; + 95A2D58051AADDBE16CCA0B6 /* STDSChallengeResponseImage.h in Headers */ = {isa = PBXBuildFile; fileRef = 10C45EE3433B731253E21E24 /* STDSChallengeResponseImage.h */; }; + 9673946B78A7763070E05CD0 /* STDSAuthenticationRequestParameters.h in Headers */ = {isa = PBXBuildFile; fileRef = C91B239583899192C7B26FCF /* STDSAuthenticationRequestParameters.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 97CE8B57A97D037E977E62F3 /* STDSBundleLocator.h in Headers */ = {isa = PBXBuildFile; fileRef = A137A31BEFCCB646FB754EE2 /* STDSBundleLocator.h */; }; + 98075195759ABB91B0D19847 /* STDSJSONWebEncryptionTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 97505E93BE83F233E69BB5A9 /* STDSJSONWebEncryptionTests.m */; }; + 9843F6AABF7B0A623E7DF979 /* STDSDeviceInformationManager.h in Headers */ = {isa = PBXBuildFile; fileRef = 7A637DA708BC866C2903E0EA /* STDSDeviceInformationManager.h */; }; + 987347CA74818CEB716B9DB3 /* UIFont+DefaultFonts.m in Sources */ = {isa = PBXBuildFile; fileRef = 634095EA97A4E48BF7CAA03F /* UIFont+DefaultFonts.m */; }; + 9B32E9A4CD2BDA61C730F5EB /* STDSJSONDecodable.h in Headers */ = {isa = PBXBuildFile; fileRef = B219360BFB325779485BF702 /* STDSJSONDecodable.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 9B7081D378D1441B98B81207 /* STDSTransactionTest.m in Sources */ = {isa = PBXBuildFile; fileRef = A4A3CCBE9A2E8C1A73D8214E /* STDSTransactionTest.m */; }; + 9C225B9C556C20D1A6F97180 /* STDSChallengeResponseSelectionInfoObject.h in Headers */ = {isa = PBXBuildFile; fileRef = 4CCA15E0819F9A9D89FEF9EA /* STDSChallengeResponseSelectionInfoObject.h */; }; + 9DC5612697835A383DC6606E /* STDSTransaction.h in Headers */ = {isa = PBXBuildFile; fileRef = F40CC91AA69F5296B0AA985C /* STDSTransaction.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 9E46777893FA2D6691F496D7 /* STDSChallengeResponseViewController.h in Headers */ = {isa = PBXBuildFile; fileRef = E45303DCFFDC390971E6E122 /* STDSChallengeResponseViewController.h */; }; + 9ED9C602C10682CD5DD91C12 /* NSLayoutConstraint+LayoutSupport.m in Sources */ = {isa = PBXBuildFile; fileRef = F99B290CF2B4987A4CFE5DCE /* NSLayoutConstraint+LayoutSupport.m */; }; + 9F0F6085FBD892BF5F14CF86 /* STDSWarning.h in Headers */ = {isa = PBXBuildFile; fileRef = 0BEB5006261C5E070FEE62BF /* STDSWarning.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 9F52F67A1A3A2199AEA598FC /* STDSEllipticCurvePointTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 0DA1EE83D91993BCAF13EC72 /* STDSEllipticCurvePointTests.m */; }; + 9F9BB9E7FA18FEDA44F7042E /* STDSDirectoryServerCertificate.m in Sources */ = {isa = PBXBuildFile; fileRef = 9754FA4F3CC6AA921787916B /* STDSDirectoryServerCertificate.m */; }; + A026717FFF299A7C1C51565F /* UIView+LayoutSupport.h in Headers */ = {isa = PBXBuildFile; fileRef = 99CB2B66DBD356B5D222EDA9 /* UIView+LayoutSupport.h */; }; + A33A549CF12209280E3B1DBA /* STDSAuthenticationRequestParametersTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 969DE2FA3845BB7939D3CD2E /* STDSAuthenticationRequestParametersTest.m */; }; + A39CFB56BF33B21A89987F9C /* STDSChallengeResponseObject+TestObjects.m in Sources */ = {isa = PBXBuildFile; fileRef = 99DB24A9B44979EA19347A76 /* STDSChallengeResponseObject+TestObjects.m */; }; + A4CFDBFC3099BD7E1A694E3E /* UIButton+CustomInitialization.m in Sources */ = {isa = PBXBuildFile; fileRef = AA29B74DD3963B1205AA1C0C /* UIButton+CustomInitialization.m */; }; + A5193DADCD0F48911F775D88 /* NSString+JWEHelpers.h in Headers */ = {isa = PBXBuildFile; fileRef = 018F4AA25D84E9CBD2546BB5 /* NSString+JWEHelpers.h */; }; + A56AE2CD941D7ADAC7FC8BCC /* STDSBase64URLEncodingTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 4B1E3760B1369DB1A33C9DFE /* STDSBase64URLEncodingTests.m */; }; + A6909D6C1A68963504733939 /* STDSDebuggerChecker.m in Sources */ = {isa = PBXBuildFile; fileRef = C29F78CC37253B0D33FF86F2 /* STDSDebuggerChecker.m */; }; + A78F19DA3D146A258FB9DE3C /* STDSACSNetworkingManager.h in Headers */ = {isa = PBXBuildFile; fileRef = D827473D1858B2BED85BDFA1 /* STDSACSNetworkingManager.h */; }; + A8A86B9BCE9702D74E0D2DB6 /* Stripe3DS2.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = CE4030C50383B488C97F4B56 /* Stripe3DS2.framework */; }; + A8CAD8276ED8057A760147CB /* NSData+JWEHelpers.m in Sources */ = {isa = PBXBuildFile; fileRef = 1FC15225A5E83948EF546B71 /* NSData+JWEHelpers.m */; }; + A97E1DF84D381952A0FB4C1C /* amex.der in Resources */ = {isa = PBXBuildFile; fileRef = D8B7EE0F935EF44EB2EF7D45 /* amex.der */; }; + AA4D157F31A246890D446981 /* STDSChallengeResponseSelectionInfoObject.m in Sources */ = {isa = PBXBuildFile; fileRef = F046DE266BAAE3DA0AC43785 /* STDSChallengeResponseSelectionInfoObject.m */; }; + AA94519687902E34E5243AEA /* STDSSpacerView.h in Headers */ = {isa = PBXBuildFile; fileRef = 72DBE992B33F45C059EDB597 /* STDSSpacerView.h */; }; + AC474013841CE1DED97F3FA8 /* STDSSimulatorChecker.h in Headers */ = {isa = PBXBuildFile; fileRef = A0ADFB365AD2DE9E84873BB1 /* STDSSimulatorChecker.h */; }; + AE1894FEA510AC394DE73C4B /* STDSTransaction+Private.h in Headers */ = {isa = PBXBuildFile; fileRef = 8A9F5C315222AFCD14C9342F /* STDSTransaction+Private.h */; }; + B012C4A8ACA1D064C40A1B63 /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 3C22D410C297E2C3AFE727A6 /* main.m */; }; + B1C2AA6259CE34B0042A1FB1 /* STDSDeviceInformationManagerTests.m in Sources */ = {isa = PBXBuildFile; fileRef = A4050F381183D6FDCC990D01 /* STDSDeviceInformationManagerTests.m */; }; + B31168F5FEC3464AC212DB85 /* STDSACSNetworkingManager.m in Sources */ = {isa = PBXBuildFile; fileRef = 2D6004F2D9880055C0C0BCF1 /* STDSACSNetworkingManager.m */; }; + B426123CC7FCA337E13304F9 /* NSDictionary+DecodingHelpersTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 5EE8BF51F49E161737406B3A /* NSDictionary+DecodingHelpersTest.m */; }; + B51D1C218F460F67B58A1F0D /* STDSProtocolErrorEvent.m in Sources */ = {isa = PBXBuildFile; fileRef = 3255706D2304FB551C97305F /* STDSProtocolErrorEvent.m */; }; + B84ED7FBD49865F23B774067 /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 8B59CC3349FD9BBF37624667 /* AppDelegate.m */; }; + B94E5A5BF5F22998E4CA9ED3 /* STDSEphemeralKeyPair.h in Headers */ = {isa = PBXBuildFile; fileRef = 20533E2C69C4B80BB33A4766 /* STDSEphemeralKeyPair.h */; }; + B9A456E52E62E9DAE9F424A7 /* STDSChallengeResponseMessageExtensionObject.m in Sources */ = {isa = PBXBuildFile; fileRef = 7D02FAFC403BC901931A629F /* STDSChallengeResponseMessageExtensionObject.m */; }; + BA00D11D792C99B7D5749F59 /* STDSJailbreakChecker.m in Sources */ = {isa = PBXBuildFile; fileRef = 1BFBB92DBE97E16DE9D18B4A /* STDSJailbreakChecker.m */; }; + BA65731CDB75C3124834586C /* STDSSynchronousLocationManagerTests.m in Sources */ = {isa = PBXBuildFile; fileRef = F02456AB660E709F332C0C7D /* STDSSynchronousLocationManagerTests.m */; }; + BB379E68231A7DA62F87E628 /* STDSNavigationBarCustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = E9F1A5A8E5922748A9BEC724 /* STDSNavigationBarCustomization.m */; }; + BF06B99FECD98C26F7B64665 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 08D577934984712C20C5E903 /* XCTest.framework */; }; + BF2651A9F3A31CB1AFEC44BB /* STDSCustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = A8E62EC973FC5C9905A40CA8 /* STDSCustomization.m */; }; + BF342613EE89845EB1C4B72A /* STDSChallengeRequestParameters.h in Headers */ = {isa = PBXBuildFile; fileRef = 762DEC6662DF8240FE19CFFC /* STDSChallengeRequestParameters.h */; }; + BFC5852E14724750E70DA2A6 /* STDSEphemeralKeyPairTests.m in Sources */ = {isa = PBXBuildFile; fileRef = BF80634E06E8D6F598CFC667 /* STDSEphemeralKeyPairTests.m */; }; + C2BA2E6ABDD8E80963FFC671 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 08D577934984712C20C5E903 /* XCTest.framework */; }; + C430332104547BA508416B41 /* STDSButtonCustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = 5489DCAA65624C6F966816B6 /* STDSButtonCustomization.m */; }; + C4BA75544A7F51355BF4B5F8 /* STDSDirectoryServer.h in Headers */ = {isa = PBXBuildFile; fileRef = 6AA43E7E90E4002DD56B7C3D /* STDSDirectoryServer.h */; }; + C6662AAA303ADB0DA6D47809 /* discover-logo.png in Resources */ = {isa = PBXBuildFile; fileRef = 0DE0C407AFDBD26ABDBE7496 /* discover-logo.png */; }; + CA617B6F5CDFB8FDEEE38636 /* STDSJSONWebSignature.h in Headers */ = {isa = PBXBuildFile; fileRef = 3AA2E6DA60350400C5270E08 /* STDSJSONWebSignature.h */; }; + CCC376FE7424029342A8D15E /* STDSChallengeResponseViewController.m in Sources */ = {isa = PBXBuildFile; fileRef = B064352BB2876CA7266C410A /* STDSChallengeResponseViewController.m */; }; + CEA1EB91858043A837638FE0 /* STDSSecTypeUtilities.h in Headers */ = {isa = PBXBuildFile; fileRef = C8F0E11AA6794BFB1715685E /* STDSSecTypeUtilities.h */; }; + D1427392DA8AD8F5B4118781 /* STDSACSNetworkingManagerTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6118F98576E9A39A7292C6C6 /* STDSACSNetworkingManagerTest.m */; }; + D189220981C7705913D93687 /* STDSJSONEncoderTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 6B9B49A5CF64DCB23CD0AEDE /* STDSJSONEncoderTest.m */; }; + D3432C5BF5211A60D5C15D9E /* STDSIntegrityChecker.h in Headers */ = {isa = PBXBuildFile; fileRef = C02DF839637771684BC57B3A /* STDSIntegrityChecker.h */; }; + D3AA14D1E059E90F5A965CC4 /* STDSSelectionCustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = 43540AB7D13C0C9A563C3F4D /* STDSSelectionCustomization.m */; }; + D41F091BE1579B801D1BBC20 /* STDSSelectionButton.m in Sources */ = {isa = PBXBuildFile; fileRef = B458FCA7DFDBF75E62A43BA1 /* STDSSelectionButton.m */; }; + D5706ACF84F0A993CE1885D4 /* STDSTextChallengeView.m in Sources */ = {isa = PBXBuildFile; fileRef = 04E6B6D3CAE363114723472F /* STDSTextChallengeView.m */; }; + D63E3DDC92DD46062A265D86 /* STDSChallengeParametersTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 89ED2133B61092FE1B478204 /* STDSChallengeParametersTests.m */; }; + D819C341ECEE1EFE6FE66085 /* STDSChallengeResponseImageObject.h in Headers */ = {isa = PBXBuildFile; fileRef = 14F83E51E99D453DF9C2DDE1 /* STDSChallengeResponseImageObject.h */; }; + D83D2FF20387FACC383A2A0F /* STDSAlreadyInitializedException.m in Sources */ = {isa = PBXBuildFile; fileRef = 27AC486DA2A80A43C1F517B2 /* STDSAlreadyInitializedException.m */; }; + D927789F89B413C08DE4D388 /* STDSThreeDS2ServiceTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 7FF8ACC073E814B7DEBA7647 /* STDSThreeDS2ServiceTests.m */; }; + D9F8BA88953A91DC082FA6DA /* STDSSelectionButton.h in Headers */ = {isa = PBXBuildFile; fileRef = C4777F15603AC755362BD846 /* STDSSelectionButton.h */; }; + DB96817598FCE73F5131437E /* STDSDirectoryServerCertificate+Internal.h in Headers */ = {isa = PBXBuildFile; fileRef = B015B937DD3657D62C61CE58 /* STDSDirectoryServerCertificate+Internal.h */; }; + DBDB8B56FFF5C4234705E698 /* STDSSecTypeUtilities.m in Sources */ = {isa = PBXBuildFile; fileRef = CF3DED43BD4E0226BFB0759D /* STDSSecTypeUtilities.m */; }; + DC0D9A57F2E312663C088FB8 /* STDSChallengeParameters.h in Headers */ = {isa = PBXBuildFile; fileRef = D03B98D5861739833B197D8F /* STDSChallengeParameters.h */; settings = {ATTRIBUTES = (Public, ); }; }; + DC32281BF0847A98B100C3A1 /* STDSConfigParameters.h in Headers */ = {isa = PBXBuildFile; fileRef = A1BFB8640032796CDA016765 /* STDSConfigParameters.h */; settings = {ATTRIBUTES = (Public, ); }; }; + DDB98914E1135E8D445545C8 /* STDSIPAddress.h in Headers */ = {isa = PBXBuildFile; fileRef = A3DFB001DB5BA7F1AFA7353A /* STDSIPAddress.h */; }; + DE32EAA8F5C2645652FCFB48 /* STDSSwiftTryCatch.h in Headers */ = {isa = PBXBuildFile; fileRef = E5BC34A661D9080A2EE239F8 /* STDSSwiftTryCatch.h */; settings = {ATTRIBUTES = (Public, ); }; }; + DEEF260365A4D9B0D83E557D /* NSString+EmptyChecking.m in Sources */ = {isa = PBXBuildFile; fileRef = D8D309E4FBE7E73D0BB83BF5 /* NSString+EmptyChecking.m */; }; + DF4D38DC0AF89EAAA8718AAE /* STDSUICustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = A55B2474CD1A39D70869B75B /* STDSUICustomization.m */; }; + E0644CE0260178023E643B23 /* STDSSynchronousLocationManager.m in Sources */ = {isa = PBXBuildFile; fileRef = A5DBDDEB5C5DAE2EA8C95CC3 /* STDSSynchronousLocationManager.m */; }; + E08AF96E690D75F6AB52021F /* STDSWarning.m in Sources */ = {isa = PBXBuildFile; fileRef = A502CCC5190F6B642EEC0CE6 /* STDSWarning.m */; }; + E19C8104FC1A9BAEA0BE8A9D /* UIColor+ThirteenSupport.m in Sources */ = {isa = PBXBuildFile; fileRef = 37ED28756222539D37554CC7 /* UIColor+ThirteenSupport.m */; }; + E4DA2CE8DC8C60DE40222185 /* STDSDirectoryServerCertificateTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 6AA75957DD13FA92FA866AF3 /* STDSDirectoryServerCertificateTests.m */; }; + E630DCBBEFAAD0435BEF989C /* STDSDeviceInformation.m in Sources */ = {isa = PBXBuildFile; fileRef = 56B1A52548593FE78D801734 /* STDSDeviceInformation.m */; }; + E70CFDC59731BA62DD89906C /* STDSWhitelistView.m in Sources */ = {isa = PBXBuildFile; fileRef = 615F1BD510661B38D6825128 /* STDSWhitelistView.m */; }; + E73028FD4537F6E48F927127 /* NSString+EmptyCheckingTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 57B8822D21AADE7DA51931CC /* NSString+EmptyCheckingTests.m */; }; + E7A72BB7CC8DE8DA7FA0DEC5 /* ErrorMessage.json in Resources */ = {isa = PBXBuildFile; fileRef = E12798864D5CFFB7157D4CF5 /* ErrorMessage.json */; }; + E7E424C5264A4DE4A1EC19B6 /* STDSChallengeSelectionView.h in Headers */ = {isa = PBXBuildFile; fileRef = EB910E98EDD6D3E00802793C /* STDSChallengeSelectionView.h */; }; + E90749131EDA0C86BBBFCE5C /* STDSChallengeResponseObject.h in Headers */ = {isa = PBXBuildFile; fileRef = EB1BDF12DBD6264C6199FFA7 /* STDSChallengeResponseObject.h */; }; + EA19A7AEC298F3EC010D1199 /* STDSTestJSONUtils.m in Sources */ = {isa = PBXBuildFile; fileRef = 5FF3235EDD908CC1500399A1 /* STDSTestJSONUtils.m */; }; + EA89021709AFB5F04961EB78 /* cartes-bancaires.der in Resources */ = {isa = PBXBuildFile; fileRef = 83838563F29179FB1E9F2E66 /* cartes-bancaires.der */; }; + EB791827A943AA61976D8C29 /* STDSIntegrityChecker.m in Sources */ = {isa = PBXBuildFile; fileRef = BB065D6900E95C5E90864C16 /* STDSIntegrityChecker.m */; }; + EC51189277D7D68D08C2A65C /* STDSButtonCustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = BD94CAE01A17E083E5617E56 /* STDSButtonCustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + ECF8755D2602E0E06DCB3210 /* STDSChallengeResponseObjectTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 5E401845EE5759CCD30786AA /* STDSChallengeResponseObjectTest.m */; }; + EE3854BC2A14F87E8D51B0D0 /* STDSInvalidInputException.m in Sources */ = {isa = PBXBuildFile; fileRef = 6FDAC320458399141EB161E2 /* STDSInvalidInputException.m */; }; + EEE77034728D2CFE38CC5A85 /* STDSLabelCustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = 01D320CD5ADA49F17B8BC1B0 /* STDSLabelCustomization.m */; }; + EFC72E766F9FD4EE18D309BC /* STDSJSONWebSignature.m in Sources */ = {isa = PBXBuildFile; fileRef = A1DC6B47C06B522B22EB19FF /* STDSJSONWebSignature.m */; }; + F04482C663F6DF8E522B0833 /* STDSCustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = AFF187D1D58405C736474042 /* STDSCustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + F0B9EFD359B1AB28A695CB48 /* STDSSynchronousLocationManager.h in Headers */ = {isa = PBXBuildFile; fileRef = C12AE31D386D4A193C00004B /* STDSSynchronousLocationManager.h */; }; + F13A032A4AF15BB61EA18A8D /* STDSChallengeInformationView.h in Headers */ = {isa = PBXBuildFile; fileRef = C893FD867C964775E68AE85F /* STDSChallengeInformationView.h */; }; + F28AD6CA8AB1A5DEE7A8F429 /* STDSDebuggerChecker.h in Headers */ = {isa = PBXBuildFile; fileRef = D122FAE093BC4F649CC6E7AB /* STDSDebuggerChecker.h */; }; + F35963DB86D90320CFA7BCB9 /* STDSExpandableInformationView.h in Headers */ = {isa = PBXBuildFile; fileRef = 939C45AE068CF4F5BEB4C138 /* STDSExpandableInformationView.h */; }; + F4E8353A470750D9BCEA3CD8 /* CRes.json in Resources */ = {isa = PBXBuildFile; fileRef = 8C6D262891811FB3FE0C947E /* CRes.json */; }; + F5513045A25210C524A05DF5 /* STDSSelectionCustomization.h in Headers */ = {isa = PBXBuildFile; fileRef = 7CD753D9BBFC378C1B491B00 /* STDSSelectionCustomization.h */; settings = {ATTRIBUTES = (Public, ); }; }; + F569B584D357DC5C55542015 /* STDSFooterCustomization.m in Sources */ = {isa = PBXBuildFile; fileRef = 6CFF141AC98F213122D037C6 /* STDSFooterCustomization.m */; }; + F939E05536AE838DC489C3AA /* STDSChallengeResponseViewControllerSnapshotTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 466DF80D33CEC20DF4E844A9 /* STDSChallengeResponseViewControllerSnapshotTests.m */; }; + FD6456BD86F07C446B9FB6C8 /* ec_test.der in Resources */ = {isa = PBXBuildFile; fileRef = 3E4C02D9D6978AD5C3D7E3D2 /* ec_test.der */; }; + FD8A81873C3BC7579ED07460 /* STDSCompletionEvent.m in Sources */ = {isa = PBXBuildFile; fileRef = 3B6B71C61CF5ADC7796441F5 /* STDSCompletionEvent.m */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 35B6793A7F3FB130F40209F1 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 8D219187B704575AD3CD3EC5 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 5907C55B1F111921112DF2BF; + remoteInfo = Stripe3DS2DemoUI; + }; + 52E1EB51359DA6E75D851D71 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 8D219187B704575AD3CD3EC5 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 57AEC53510AE0DC0539730F3; + remoteInfo = Stripe3DS2; + }; + 6C355B127D670833C76237D3 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 8D219187B704575AD3CD3EC5 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 57AEC53510AE0DC0539730F3; + remoteInfo = Stripe3DS2; + }; + 947ED275EAB25AD4CD701936 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 8D219187B704575AD3CD3EC5 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 57AEC53510AE0DC0539730F3; + remoteInfo = Stripe3DS2; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 00E415680C602ED1D6D8E71F /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + 4D79F0CFC54E8B923E9238CF /* Stripe3DS2.framework in Embed Frameworks */, + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + 43B749EC915FB6D2FF5ABEE0 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + 58FA2FE74AE67CD3CDAFD7E5 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + 726DD7176204D90BD7552B84 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 00658E077ECB223C90484AF7 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 018F4AA25D84E9CBD2546BB5 /* NSString+JWEHelpers.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSString+JWEHelpers.h"; sourceTree = ""; }; + 01D320CD5ADA49F17B8BC1B0 /* STDSLabelCustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSLabelCustomization.m; sourceTree = ""; }; + 02D762271DBCC1AE80E0F9D4 /* Stripe3DS2DemoUI.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Stripe3DS2DemoUI.app; sourceTree = BUILT_PRODUCTS_DIR; }; + 037C294011A01E4B4DCE5BB7 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 04AB48F014A8D5BA56FEF463 /* discover.der */ = {isa = PBXFileReference; path = discover.der; sourceTree = ""; }; + 04E6B6D3CAE363114723472F /* STDSTextChallengeView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSTextChallengeView.m; sourceTree = ""; }; + 0681C043F732148587737233 /* STDSJSONEncoder.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSJSONEncoder.m; sourceTree = ""; }; + 069990DA025BE9F33602E96A /* STDSDirectoryServerCertificate.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSDirectoryServerCertificate.h; sourceTree = ""; }; + 06C5207432FB07BD71703BCC /* STDSJSONEncoder.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSJSONEncoder.h; sourceTree = ""; }; + 08D577934984712C20C5E903 /* XCTest.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = XCTest.framework; path = Platforms/iPhoneOS.platform/Developer/Library/Frameworks/XCTest.framework; sourceTree = DEVELOPER_DIR; }; + 0BEB5006261C5E070FEE62BF /* STDSWarning.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSWarning.h; sourceTree = ""; }; + 0DA1EE83D91993BCAF13EC72 /* STDSEllipticCurvePointTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSEllipticCurvePointTests.m; sourceTree = ""; }; + 0DE0C407AFDBD26ABDBE7496 /* discover-logo.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "discover-logo.png"; sourceTree = ""; }; + 0E38A775DB9F529427E900CF /* hr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = hr; path = hr.lproj/Localizable.strings; sourceTree = ""; }; + 0E3C05BE55CF3086738AA182 /* Stripe3DS2DemoUITests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2DemoUITests-Debug.xcconfig"; sourceTree = ""; }; + 0F024DEAD6628A0229856656 /* Stripe3DS2-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Stripe3DS2-Bridging-Header.h"; sourceTree = ""; }; + 10C45EE3433B731253E21E24 /* STDSChallengeResponseImage.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseImage.h; sourceTree = ""; }; + 10DB3CD8F2541AC06681F2C8 /* STDSCompletionEvent.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSCompletionEvent.h; sourceTree = ""; }; + 1104D5855D28E49E104CA004 /* mastercard.der */ = {isa = PBXFileReference; path = mastercard.der; sourceTree = ""; }; + 138E82072FC6572612A8E248 /* STDSAuthenticationResponseObject.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSAuthenticationResponseObject.h; sourceTree = ""; }; + 13ED1CC076C6B3FB49C9197A /* it */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = it; path = it.lproj/Localizable.strings; sourceTree = ""; }; + 14D60BDF245487FE3362BE07 /* STDSSimulatorChecker.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSimulatorChecker.m; sourceTree = ""; }; + 14F83E51E99D453DF9C2DDE1 /* STDSChallengeResponseImageObject.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseImageObject.h; sourceTree = ""; }; + 1501A7A3010FFE53D00E318F /* STDSRuntimeErrorEvent.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSRuntimeErrorEvent.m; sourceTree = ""; }; + 157970FF5B4A817041E3D668 /* ja */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ja; path = ja.lproj/Localizable.strings; sourceTree = ""; }; + 172AF5C85EDBD92A1030E361 /* STDSDeviceInformationParameter.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSDeviceInformationParameter.h; sourceTree = ""; }; + 18D88DE3E34106F934015BF9 /* mt */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = mt; path = mt.lproj/Localizable.strings; sourceTree = ""; }; + 1BBADD412A7C2D1FF35A7C86 /* es */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = es; path = es.lproj/Localizable.strings; sourceTree = ""; }; + 1BFBB92DBE97E16DE9D18B4A /* STDSJailbreakChecker.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSJailbreakChecker.m; sourceTree = ""; }; + 1D6269F8B91341C387A911DB /* STDSStackView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSStackView.m; sourceTree = ""; }; + 1E3640C73D001D32CCC449F1 /* error@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "error@3x.png"; sourceTree = ""; }; + 1E844798B2733C8511AC080E /* UIFont+DefaultFonts.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "UIFont+DefaultFonts.h"; sourceTree = ""; }; + 1F0DADEABA0B043A6CA36DD6 /* STDSDemoViewController.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSDemoViewController.h; sourceTree = ""; }; + 1F1FD11407319293954C6D81 /* STDSAlreadyInitializedException.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSAlreadyInitializedException.h; sourceTree = ""; }; + 1FBE88F6E2C3153A8315EDC6 /* STDSDeviceInformation.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSDeviceInformation.h; sourceTree = ""; }; + 1FC15225A5E83948EF546B71 /* NSData+JWEHelpers.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSData+JWEHelpers.m"; sourceTree = ""; }; + 20533E2C69C4B80BB33A4766 /* STDSEphemeralKeyPair.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSEphemeralKeyPair.h; sourceTree = ""; }; + 210B22FF4DCB0C6E7C763EAB /* STDSUICustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSUICustomization.h; sourceTree = ""; }; + 2192708AD201F80F6E1A39F7 /* tr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = tr; path = tr.lproj/Localizable.strings; sourceTree = ""; }; + 237977B021FE538E5E4FA35C /* ARes.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = ARes.json; sourceTree = ""; }; + 23A3DDCE911F8C43CF74CDF4 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 23C929BF760735CC01E1E8F5 /* STDSChallengeParameters.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeParameters.m; sourceTree = ""; }; + 24CE086326AF3DE336BF4F4C /* STDSChallengeResponseObject.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeResponseObject.m; sourceTree = ""; }; + 25C7D18868DDADEF4CB6220C /* STDSDeviceInformationParameter.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDeviceInformationParameter.m; sourceTree = ""; }; + 26C20D4B77372845627D6466 /* NSError+Stripe3DS2.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSError+Stripe3DS2.m"; sourceTree = ""; }; + 278F2C40987F5218BD031E3D /* STDSErrorMessageTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSErrorMessageTest.m; sourceTree = ""; }; + 27AC486DA2A80A43C1F517B2 /* STDSAlreadyInitializedException.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSAlreadyInitializedException.m; sourceTree = ""; }; + 28C6B4F6BA431B9342970FD6 /* Stripe3DS2.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = Stripe3DS2.h; sourceTree = ""; }; + 28E32DA72CAFBF2AF8099151 /* STDSAuthenticationResponseTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSAuthenticationResponseTests.m; sourceTree = ""; }; + 2C2666DF7D23EBC3FFE7CCF1 /* de */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = de; path = de.lproj/Localizable.strings; sourceTree = ""; }; + 2CB591796654445B8434A501 /* STDSErrorMessage+Internal.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "STDSErrorMessage+Internal.m"; sourceTree = ""; }; + 2CFAC8D74AFFC2E61BAD82A5 /* STDSBrandingView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSBrandingView.h; sourceTree = ""; }; + 2D6004F2D9880055C0C0BCF1 /* STDSACSNetworkingManager.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSACSNetworkingManager.m; sourceTree = ""; }; + 2EF5AA537586EF706E4056FD /* STDSEllipticCurvePoint.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSEllipticCurvePoint.m; sourceTree = ""; }; + 2FCABBF51CBA7F9FEF757B4C /* STDSNotInitializedException.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSNotInitializedException.h; sourceTree = ""; }; + 30D22985091381DFCDF077E9 /* STDSSecTypeUtilitiesTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSecTypeUtilitiesTests.m; sourceTree = ""; }; + 315AEE1534F9D9974DC1674B /* STDSChallengeResponse.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponse.h; sourceTree = ""; }; + 3255706D2304FB551C97305F /* STDSProtocolErrorEvent.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSProtocolErrorEvent.m; sourceTree = ""; }; + 33CEADC8006E456FD82CE134 /* STDSWhitelistView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSWhitelistView.h; sourceTree = ""; }; + 34FCE3CEEDC2D23116C11DAA /* visa-logo@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "visa-logo@3x.png"; sourceTree = ""; }; + 371F8076630A8839C1F6A508 /* hu */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = hu; path = hu.lproj/Localizable.strings; sourceTree = ""; }; + 37ED28756222539D37554CC7 /* UIColor+ThirteenSupport.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UIColor+ThirteenSupport.m"; sourceTree = ""; }; + 382C583385FAECA91366769E /* STDSDemoViewController.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDemoViewController.m; sourceTree = ""; }; + 38572D4E32BF3670F1C83409 /* lv-LV */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "lv-LV"; path = "lv-LV.lproj/Localizable.strings"; sourceTree = ""; }; + 38F3A7D625E4B0D3506A37F6 /* sv */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = sv; path = sv.lproj/Localizable.strings; sourceTree = ""; }; + 3A6D20D1DAB7D769E5E97528 /* STDSProtocolErrorEvent.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSProtocolErrorEvent.h; sourceTree = ""; }; + 3A8CD68A006F970DDC54469A /* Project-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Debug.xcconfig"; sourceTree = ""; }; + 3A9F828840A3B361842E38DE /* STDSChallengeResponseImageObject.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeResponseImageObject.m; sourceTree = ""; }; + 3AA2E6DA60350400C5270E08 /* STDSJSONWebSignature.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSJSONWebSignature.h; sourceTree = ""; }; + 3B6B71C61CF5ADC7796441F5 /* STDSCompletionEvent.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSCompletionEvent.m; sourceTree = ""; }; + 3B9A821F35B93898A7AC1ADF /* sl-SI */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "sl-SI"; path = "sl-SI.lproj/Localizable.strings"; sourceTree = ""; }; + 3C22D410C297E2C3AFE727A6 /* main.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = main.m; sourceTree = ""; }; + 3E4C02D9D6978AD5C3D7E3D2 /* ec_test.der */ = {isa = PBXFileReference; path = ec_test.der; sourceTree = ""; }; + 3F017BF6ECE0EBE4E0DFCF9C /* zh-Hans */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-Hans"; path = "zh-Hans.lproj/Localizable.strings"; sourceTree = ""; }; + 406D88ADED27D64DED2002A2 /* es-419 */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "es-419"; path = "es-419.lproj/Localizable.strings"; sourceTree = ""; }; + 40DC4913154C3D3A27E35207 /* ca-ES */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ca-ES"; path = "ca-ES.lproj/Localizable.strings"; sourceTree = ""; }; + 41CB752E5656578F2437E247 /* Chevron@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "Chevron@3x.png"; sourceTree = ""; }; + 41E0F638E776A7F3456BDA3E /* STDSLocalizedString.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSLocalizedString.h; sourceTree = ""; }; + 43540AB7D13C0C9A563C3F4D /* STDSSelectionCustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSelectionCustomization.m; sourceTree = ""; }; + 43CB8F416E4F4CEA34391481 /* NSString+JWEHelpers.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSString+JWEHelpers.m"; sourceTree = ""; }; + 461F938CDE7ED6D06D9D2700 /* STDSException+Internal.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSException+Internal.h"; sourceTree = ""; }; + 461FDC130846625171164A9E /* et-EE */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "et-EE"; path = "et-EE.lproj/Localizable.strings"; sourceTree = ""; }; + 466DF80D33CEC20DF4E844A9 /* STDSChallengeResponseViewControllerSnapshotTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeResponseViewControllerSnapshotTests.m; sourceTree = ""; }; + 474F39AA3F47D84F8D54E5B7 /* STDSWarningTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSWarningTests.m; sourceTree = ""; }; + 47C29E9F3D2A62676B424CD1 /* STDSThreeDSProtocolVersion+Private.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSThreeDSProtocolVersion+Private.h"; sourceTree = ""; }; + 4869D6E14F01EDB960CB3065 /* STDSDeviceInformationParameter+Private.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSDeviceInformationParameter+Private.h"; sourceTree = ""; }; + 49EF79D69693F937FEC46906 /* fil */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fil; path = fil.lproj/Localizable.strings; sourceTree = ""; }; + 4A567E08748C86FCE2A75FEA /* UIColor+DefaultColors.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "UIColor+DefaultColors.h"; sourceTree = ""; }; + 4B1E3760B1369DB1A33C9DFE /* STDSBase64URLEncodingTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSBase64URLEncodingTests.m; sourceTree = ""; }; + 4BC4D9FD1D3869D491CC2CF8 /* fr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fr; path = fr.lproj/Localizable.strings; sourceTree = ""; }; + 4C158B552B0CBE635A2AF7BB /* el-GR */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "el-GR"; path = "el-GR.lproj/Localizable.strings"; sourceTree = ""; }; + 4CCA15E0819F9A9D89FEF9EA /* STDSChallengeResponseSelectionInfoObject.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseSelectionInfoObject.h; sourceTree = ""; }; + 4DB91F58CF23E72636C69C41 /* Project-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Release.xcconfig"; sourceTree = ""; }; + 4DD55BECC3FB85216FE46218 /* UIButton+CustomInitialization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "UIButton+CustomInitialization.h"; sourceTree = ""; }; + 4EA4AB3BA2FB41B3D5F38978 /* STDSAuthenticationResponse.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSAuthenticationResponse.h; sourceTree = ""; }; + 4EAD29E57D206B5D6BAF9099 /* STDSChallengeRequestParametersTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeRequestParametersTest.m; sourceTree = ""; }; + 4EB746D4E46ABD2452A154AE /* Stripe3DS2Tests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = Stripe3DS2Tests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 5043E1CF6955176EC3D43F88 /* STDSAuthenticationResponseObject.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSAuthenticationResponseObject.m; sourceTree = ""; }; + 5049A72A6BEB558D8559A5EB /* STDSEllipticCurvePoint.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSEllipticCurvePoint.h; sourceTree = ""; }; + 51E4A57094A671EB14380882 /* STDSSpacerView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSpacerView.m; sourceTree = ""; }; + 52D6E4F76CBA258163E57DF3 /* STDSEphemeralKeyPair+Testing.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSEphemeralKeyPair+Testing.h"; sourceTree = ""; }; + 52EEBC7D74623E1A1D7E3219 /* STDSChallengeResponseSelectionInfo.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseSelectionInfo.h; sourceTree = ""; }; + 5316D0759F7F2ED0BF1D3B2E /* Stripe3DS2DemoUI-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2DemoUI-Debug.xcconfig"; sourceTree = ""; }; + 5463EB1CC8E85BE3BD12DEFF /* STDSErrorMessage+Internal.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSErrorMessage+Internal.h"; sourceTree = ""; }; + 5489DCAA65624C6F966816B6 /* STDSButtonCustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSButtonCustomization.m; sourceTree = ""; }; + 56B1A52548593FE78D801734 /* STDSDeviceInformation.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDeviceInformation.m; sourceTree = ""; }; + 57B8822D21AADE7DA51931CC /* NSString+EmptyCheckingTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSString+EmptyCheckingTests.m"; sourceTree = ""; }; + 581BD6EEB19753FC873EB077 /* cartes-bancaires-logo.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "cartes-bancaires-logo.png"; sourceTree = ""; }; + 581EEE576D24047004C828DF /* STDSWebView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSWebView.h; sourceTree = ""; }; + 5862FA7E6300A2169A3CD3D7 /* mastercard-logo@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "mastercard-logo@3x.png"; sourceTree = ""; }; + 58C38FF9205C9B7D79C51A37 /* sk-SK */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "sk-SK"; path = "sk-SK.lproj/Localizable.strings"; sourceTree = ""; }; + 5B05A609753EE71502409082 /* NSData+JWEHelpers.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSData+JWEHelpers.h"; sourceTree = ""; }; + 5D93713CADD5589B5E6F6A0D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 5E401845EE5759CCD30786AA /* STDSChallengeResponseObjectTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeResponseObjectTest.m; sourceTree = ""; }; + 5E6A9565E97DF2544254AA94 /* STDSThreeDSProtocolVersion.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSThreeDSProtocolVersion.h; sourceTree = ""; }; + 5EAA444FA7C82BDA4AD907BF /* STDSThreeDS2Service.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSThreeDS2Service.h; sourceTree = ""; }; + 5EE8BF51F49E161737406B3A /* NSDictionary+DecodingHelpersTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSDictionary+DecodingHelpersTest.m"; sourceTree = ""; }; + 5F551808E65CDADD9B3A9CB8 /* STDSConfigParametersTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSConfigParametersTests.m; sourceTree = ""; }; + 5FDE2481364C454058BF2377 /* STDSTransaction.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSTransaction.m; sourceTree = ""; }; + 5FF3235EDD908CC1500399A1 /* STDSTestJSONUtils.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSTestJSONUtils.m; sourceTree = ""; }; + 60693BC560C4927ACFD2E6C3 /* STDSChallengeRequestParameters.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeRequestParameters.m; sourceTree = ""; }; + 6118F98576E9A39A7292C6C6 /* STDSACSNetworkingManagerTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSACSNetworkingManagerTest.m; sourceTree = ""; }; + 615F1BD510661B38D6825128 /* STDSWhitelistView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSWhitelistView.m; sourceTree = ""; }; + 62773D5C85C567F28BCE0DA5 /* id */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = id; path = id.lproj/Localizable.strings; sourceTree = ""; }; + 634095EA97A4E48BF7CAA03F /* UIFont+DefaultFonts.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UIFont+DefaultFonts.m"; sourceTree = ""; }; + 645D0BB927B7F8A0D37EE04D /* STDSErrorMessage.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSErrorMessage.h; sourceTree = ""; }; + 6690476C1BEA59C37576FB36 /* pt-BR */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pt-BR"; path = "pt-BR.lproj/Localizable.strings"; sourceTree = ""; }; + 68ED2A0E11E1B27980E0C60A /* STDSTextFieldCustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSTextFieldCustomization.h; sourceTree = ""; }; + 6932DD26C7C16938DFA0E198 /* STDSLabelCustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSLabelCustomization.h; sourceTree = ""; }; + 6940D75CA36F7DB9FB56196B /* Stripe3DS2DemoUI-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2DemoUI-Release.xcconfig"; sourceTree = ""; }; + 694DF6F47767A651907D55D4 /* ru */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ru; path = ru.lproj/Localizable.strings; sourceTree = ""; }; + 6AA43E7E90E4002DD56B7C3D /* STDSDirectoryServer.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSDirectoryServer.h; sourceTree = ""; }; + 6AA75957DD13FA92FA866AF3 /* STDSDirectoryServerCertificateTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDirectoryServerCertificateTests.m; sourceTree = ""; }; + 6B487D8DD39DAC17042A3F04 /* nb */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = nb; path = nb.lproj/Localizable.strings; sourceTree = ""; }; + 6B9B49A5CF64DCB23CD0AEDE /* STDSJSONEncoderTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSJSONEncoderTest.m; sourceTree = ""; }; + 6CFF141AC98F213122D037C6 /* STDSFooterCustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSFooterCustomization.m; sourceTree = ""; }; + 6DDB6222A2EF31FDB7592368 /* STDSErrorMessage.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSErrorMessage.m; sourceTree = ""; }; + 6F030410FDABFF833CD8FE46 /* STDSException.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSException.m; sourceTree = ""; }; + 6FDAC320458399141EB161E2 /* STDSInvalidInputException.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSInvalidInputException.m; sourceTree = ""; }; + 71494F94F36F4B731A27D182 /* lt-LT */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "lt-LT"; path = "lt-LT.lproj/Localizable.strings"; sourceTree = ""; }; + 72DBE992B33F45C059EDB597 /* STDSSpacerView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSSpacerView.h; sourceTree = ""; }; + 762DEC6662DF8240FE19CFFC /* STDSChallengeRequestParameters.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeRequestParameters.h; sourceTree = ""; }; + 76EA9DF8572C992E30BCF8A3 /* STDSJSONWebEncryption.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSJSONWebEncryption.h; sourceTree = ""; }; + 77646CEC7EE754F47EDBDA4F /* STDSThreeDS2Service.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSThreeDS2Service.m; sourceTree = ""; }; + 7859D635964B2854A07B5285 /* cs-CZ */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "cs-CZ"; path = "cs-CZ.lproj/Localizable.strings"; sourceTree = ""; }; + 7971EB56716D5E2F59053B6D /* acs_challenge.html */ = {isa = PBXFileReference; lastKnownFileType = text.html; path = acs_challenge.html; sourceTree = ""; }; + 7A637DA708BC866C2903E0EA /* STDSDeviceInformationManager.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSDeviceInformationManager.h; sourceTree = ""; }; + 7AC1C5F6F1311F50D9C37291 /* STDSProcessingView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSProcessingView.m; sourceTree = ""; }; + 7CD753D9BBFC378C1B491B00 /* STDSSelectionCustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSSelectionCustomization.h; sourceTree = ""; }; + 7D02FAFC403BC901931A629F /* STDSChallengeResponseMessageExtensionObject.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeResponseMessageExtensionObject.m; sourceTree = ""; }; + 7FF8ACC073E814B7DEBA7647 /* STDSThreeDS2ServiceTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSThreeDS2ServiceTests.m; sourceTree = ""; }; + 82C41E6DC2CC247ECC17F2B9 /* STDSTextChallengeView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSTextChallengeView.h; sourceTree = ""; }; + 83838563F29179FB1E9F2E66 /* cartes-bancaires.der */ = {isa = PBXFileReference; path = "cartes-bancaires.der"; sourceTree = ""; }; + 83BDD516620860DC08B36B96 /* nl */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = nl; path = nl.lproj/Localizable.strings; sourceTree = ""; }; + 84967ED0D9B206B3F5AA4F02 /* STDSProgressViewController.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSProgressViewController.m; sourceTree = ""; }; + 853A361AB630A2BD402D1B47 /* STDSInvalidInputException.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSInvalidInputException.h; sourceTree = ""; }; + 869733153BCA6BE2EB7A452F /* UIColor+DefaultColors.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UIColor+DefaultColors.m"; sourceTree = ""; }; + 89B5ED899439D1C1054CB448 /* STDSConfigParameters.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSConfigParameters.m; sourceTree = ""; }; + 89ED2133B61092FE1B478204 /* STDSChallengeParametersTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeParametersTests.m; sourceTree = ""; }; + 8A6035A8211B3D50D10EB947 /* bg-BG */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "bg-BG"; path = "bg-BG.lproj/Localizable.strings"; sourceTree = ""; }; + 8A9F5C315222AFCD14C9342F /* STDSTransaction+Private.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSTransaction+Private.h"; sourceTree = ""; }; + 8AC648332A706809F1BDFD59 /* STDSThreeDSProtocolVersion.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSThreeDSProtocolVersion.m; sourceTree = ""; }; + 8B0F0485B7048DDAA7DA8F9B /* UIView+LayoutSupport.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UIView+LayoutSupport.m"; sourceTree = ""; }; + 8B59CC3349FD9BBF37624667 /* AppDelegate.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = AppDelegate.m; sourceTree = ""; }; + 8B8D429F1C03603DB218700F /* da */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = da; path = da.lproj/Localizable.strings; sourceTree = ""; }; + 8C6D262891811FB3FE0C947E /* CRes.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = CRes.json; sourceTree = ""; }; + 8F72413F99F8E50FD0FA085D /* UIViewController+Stripe3DS2.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "UIViewController+Stripe3DS2.h"; sourceTree = ""; }; + 939C45AE068CF4F5BEB4C138 /* STDSExpandableInformationView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSExpandableInformationView.h; sourceTree = ""; }; + 947935869E5A08E35FA11E13 /* amex-logo@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "amex-logo@3x.png"; sourceTree = ""; }; + 94943BEAB94E949AF9830EFA /* STDSJailbreakChecker.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSJailbreakChecker.h; sourceTree = ""; }; + 95641ECBE1AE1CC198013405 /* STDSProcessingView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSProcessingView.h; sourceTree = ""; }; + 9564F43EF7CF4F60C3C202CD /* STDSBundleLocator.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSBundleLocator.m; sourceTree = ""; }; + 9580A82A59EB2B0103EDF47A /* STDSDeviceInformationManager.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDeviceInformationManager.m; sourceTree = ""; }; + 969DE2FA3845BB7939D3CD2E /* STDSAuthenticationRequestParametersTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSAuthenticationRequestParametersTest.m; sourceTree = ""; }; + 97163B5E9598CA3A298920B9 /* STDSTextFieldCustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSTextFieldCustomization.m; sourceTree = ""; }; + 97505E93BE83F233E69BB5A9 /* STDSJSONWebEncryptionTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSJSONWebEncryptionTests.m; sourceTree = ""; }; + 9754FA4F3CC6AA921787916B /* STDSDirectoryServerCertificate.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDirectoryServerCertificate.m; sourceTree = ""; }; + 99392D3F702EC6BF7CE15291 /* STDSJSONWebEncryption.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSJSONWebEncryption.m; sourceTree = ""; }; + 99451064136843047C7881AD /* UIViewController+Stripe3DS2.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UIViewController+Stripe3DS2.m"; sourceTree = ""; }; + 9946D28A86E85A6C84EB6C7B /* STDSSwiftTryCatch.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSwiftTryCatch.m; sourceTree = ""; }; + 99CB2B66DBD356B5D222EDA9 /* UIView+LayoutSupport.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "UIView+LayoutSupport.h"; sourceTree = ""; }; + 99DB24A9B44979EA19347A76 /* STDSChallengeResponseObject+TestObjects.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "STDSChallengeResponseObject+TestObjects.m"; sourceTree = ""; }; + 9A8B0001705400C661678617 /* UIColor+ThirteenSupport.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "UIColor+ThirteenSupport.h"; sourceTree = ""; }; + 9F4376BB07FD1EE783249AED /* STDSStripe3DS2Error.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSStripe3DS2Error.m; sourceTree = ""; }; + 9F5C194C81AF1F5578698A27 /* STDSChallengeResponseObject+TestObjects.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSChallengeResponseObject+TestObjects.h"; sourceTree = ""; }; + A09A89BBE7360F93195641C9 /* STDSImageLoader.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSImageLoader.m; sourceTree = ""; }; + A0ADFB365AD2DE9E84873BB1 /* STDSSimulatorChecker.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSSimulatorChecker.h; sourceTree = ""; }; + A137A31BEFCCB646FB754EE2 /* STDSBundleLocator.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSBundleLocator.h; sourceTree = ""; }; + A1BFB8640032796CDA016765 /* STDSConfigParameters.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSConfigParameters.h; sourceTree = ""; }; + A1DC6B47C06B522B22EB19FF /* STDSJSONWebSignature.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSJSONWebSignature.m; sourceTree = ""; }; + A3DFB001DB5BA7F1AFA7353A /* STDSIPAddress.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSIPAddress.h; sourceTree = ""; }; + A4050F381183D6FDCC990D01 /* STDSDeviceInformationManagerTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDeviceInformationManagerTests.m; sourceTree = ""; }; + A4620559FBB8A42E15044E32 /* AppDelegate.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = AppDelegate.h; sourceTree = ""; }; + A4A3CCBE9A2E8C1A73D8214E /* STDSTransactionTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSTransactionTest.m; sourceTree = ""; }; + A4D460893CB5DDE52AA0AB85 /* STDSChallengeSelectionView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeSelectionView.m; sourceTree = ""; }; + A502CCC5190F6B642EEC0CE6 /* STDSWarning.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSWarning.m; sourceTree = ""; }; + A55B2474CD1A39D70869B75B /* STDSUICustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSUICustomization.m; sourceTree = ""; }; + A5DBDDEB5C5DAE2EA8C95CC3 /* STDSSynchronousLocationManager.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSynchronousLocationManager.m; sourceTree = ""; }; + A6778F8CE36F769DF608F932 /* NSError+Stripe3DS2.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSError+Stripe3DS2.h"; sourceTree = ""; }; + A70071543985284E0D9DAC64 /* visa.der */ = {isa = PBXFileReference; path = visa.der; sourceTree = ""; }; + A8A77DB1C711B8696B2E5FEF /* NSString+EmptyChecking.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSString+EmptyChecking.h"; sourceTree = ""; }; + A8E62EC973FC5C9905A40CA8 /* STDSCustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSCustomization.m; sourceTree = ""; }; + A9CD7EC589C5CC6CE5CF9310 /* Stripe3DS2-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2-Debug.xcconfig"; sourceTree = ""; }; + AA29B74DD3963B1205AA1C0C /* UIButton+CustomInitialization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UIButton+CustomInitialization.m"; sourceTree = ""; }; + AA85BC717C46C1B95AF8C1E3 /* STDSWebView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSWebView.m; sourceTree = ""; }; + AA8A113E655E23381CB3BDA4 /* Stripe3DS2DemoUITests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = Stripe3DS2DemoUITests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + AA8BE982004398CF8AAA7D1E /* STDSChallengeResponseMessageExtension.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseMessageExtension.h; sourceTree = ""; }; + AB871F64FC72EBC7A91F96C1 /* STDSAuthenticationRequestParameters.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSAuthenticationRequestParameters.m; sourceTree = ""; }; + AE1BBF3A441B1F782B766F61 /* STDSExpandableInformationView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSExpandableInformationView.m; sourceTree = ""; }; + AE53BAA72835AFA3B35C58D3 /* STDSStripe3DS2Error.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSStripe3DS2Error.h; sourceTree = ""; }; + AF389752CCEB6FA734AAE31E /* zh-Hant */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-Hant"; path = "zh-Hant.lproj/Localizable.strings"; sourceTree = ""; }; + AFF187D1D58405C736474042 /* STDSCustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSCustomization.h; sourceTree = ""; }; + B015B937DD3657D62C61CE58 /* STDSDirectoryServerCertificate+Internal.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "STDSDirectoryServerCertificate+Internal.h"; sourceTree = ""; }; + B064352BB2876CA7266C410A /* STDSChallengeResponseViewController.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeResponseViewController.m; sourceTree = ""; }; + B1A7D41498B79E35BD724681 /* ms-MY */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ms-MY"; path = "ms-MY.lproj/Localizable.strings"; sourceTree = ""; }; + B219360BFB325779485BF702 /* STDSJSONDecodable.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSJSONDecodable.h; sourceTree = ""; }; + B458FCA7DFDBF75E62A43BA1 /* STDSSelectionButton.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSelectionButton.m; sourceTree = ""; }; + B6275E7099E55F0C04688890 /* ul-test.der */ = {isa = PBXFileReference; path = "ul-test.der"; sourceTree = ""; }; + B71A1C110DCC23A9CE929837 /* STDSException.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSException.h; sourceTree = ""; }; + B7A75140FB62A71261CAF5EC /* STDSChallengeStatusReceiver.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeStatusReceiver.h; sourceTree = ""; }; + B7AE3B4732D203134FE096FE /* STDSStackView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSStackView.h; sourceTree = ""; }; + B7BD1E24EA9427121E148DFC /* STDSNotInitializedException.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSNotInitializedException.m; sourceTree = ""; }; + B8619CA38E2A5B49DBF8546B /* STDSRuntimeException.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSRuntimeException.h; sourceTree = ""; }; + B97D91CB54F48F64CAC9E378 /* vi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = vi; path = vi.lproj/Localizable.strings; sourceTree = ""; }; + BA96C9FFA48B85F9C42E8C68 /* STDSFooterCustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSFooterCustomization.h; sourceTree = ""; }; + BAAC56EDE0AAD2E50E02E9AE /* STDSChallengeResponseMessageExtensionObject.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseMessageExtensionObject.h; sourceTree = ""; }; + BB065D6900E95C5E90864C16 /* STDSIntegrityChecker.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSIntegrityChecker.m; sourceTree = ""; }; + BC3EE020DC9358FF56389BBE /* pt-PT */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pt-PT"; path = "pt-PT.lproj/Localizable.strings"; sourceTree = ""; }; + BD94CAE01A17E083E5617E56 /* STDSButtonCustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSButtonCustomization.h; sourceTree = ""; }; + BF80634E06E8D6F598CFC667 /* STDSEphemeralKeyPairTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSEphemeralKeyPairTests.m; sourceTree = ""; }; + C02DF839637771684BC57B3A /* STDSIntegrityChecker.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSIntegrityChecker.h; sourceTree = ""; }; + C12AE31D386D4A193C00004B /* STDSSynchronousLocationManager.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSSynchronousLocationManager.h; sourceTree = ""; }; + C29F78CC37253B0D33FF86F2 /* STDSDebuggerChecker.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDebuggerChecker.m; sourceTree = ""; }; + C4777F15603AC755362BD846 /* STDSSelectionButton.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSSelectionButton.h; sourceTree = ""; }; + C566D6E444FCA4BCEC65F3D2 /* NSDictionary+DecodingHelpers.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSDictionary+DecodingHelpers.m"; sourceTree = ""; }; + C78AA1D7AD2BAB52EE58D078 /* NSDictionary+DecodingHelpers.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSDictionary+DecodingHelpers.h"; sourceTree = ""; }; + C7F1C55E51129933B02576E7 /* visa-white-logo@3x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "visa-white-logo@3x.png"; sourceTree = ""; }; + C893FD867C964775E68AE85F /* STDSChallengeInformationView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeInformationView.h; sourceTree = ""; }; + C8F0E11AA6794BFB1715685E /* STDSSecTypeUtilities.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSSecTypeUtilities.h; sourceTree = ""; }; + C91B239583899192C7B26FCF /* STDSAuthenticationRequestParameters.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSAuthenticationRequestParameters.h; sourceTree = ""; }; + CB00E968CF4DD07FB8BF2AAA /* STDSIPAddress.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSIPAddress.m; sourceTree = ""; }; + CB3154597040B803ADE14F9E /* STDSUICustomizationTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSUICustomizationTests.m; sourceTree = ""; }; + CD33421F13A675DCFE597FFB /* STDSRuntimeException.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSRuntimeException.m; sourceTree = ""; }; + CE4030C50383B488C97F4B56 /* Stripe3DS2.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Stripe3DS2.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + CECC1E22D0F0336039B435DE /* zh-HK */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-HK"; path = "zh-HK.lproj/Localizable.strings"; sourceTree = ""; }; + CF3DED43BD4E0226BFB0759D /* STDSSecTypeUtilities.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSecTypeUtilities.m; sourceTree = ""; }; + D03B98D5861739833B197D8F /* STDSChallengeParameters.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeParameters.h; sourceTree = ""; }; + D122FAE093BC4F649CC6E7AB /* STDSDebuggerChecker.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSDebuggerChecker.h; sourceTree = ""; }; + D24777EE9931075405F370DB /* STDSChallengeInformationView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeInformationView.m; sourceTree = ""; }; + D3F5A6D5A680F008C00A2D69 /* nn-NO */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "nn-NO"; path = "nn-NO.lproj/Localizable.strings"; sourceTree = ""; }; + D7A9D36FEF7E97CF735D82B8 /* Stripe3DS2DemoUITests-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2DemoUITests-Release.xcconfig"; sourceTree = ""; }; + D827473D1858B2BED85BDFA1 /* STDSACSNetworkingManager.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSACSNetworkingManager.h; sourceTree = ""; }; + D8B7EE0F935EF44EB2EF7D45 /* amex.der */ = {isa = PBXFileReference; path = amex.der; sourceTree = ""; }; + D8D309E4FBE7E73D0BB83BF5 /* NSString+EmptyChecking.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSString+EmptyChecking.m"; sourceTree = ""; }; + DE7D85369E012B15702D8DF9 /* STDSRuntimeErrorEvent.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSRuntimeErrorEvent.h; sourceTree = ""; }; + E0949399E96663964091C0EF /* NSLayoutConstraint+LayoutSupport.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSLayoutConstraint+LayoutSupport.h"; sourceTree = ""; }; + E12798864D5CFFB7157D4CF5 /* ErrorMessage.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = ErrorMessage.json; sourceTree = ""; }; + E1630D876436E43547FF69CE /* STDSNavigationBarCustomization.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSNavigationBarCustomization.h; sourceTree = ""; }; + E45303DCFFDC390971E6E122 /* STDSChallengeResponseViewController.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseViewController.h; sourceTree = ""; }; + E4C668442518B446D573AD24 /* STDSJSONWebSignatureTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSJSONWebSignatureTests.m; sourceTree = ""; }; + E5BC34A661D9080A2EE239F8 /* STDSSwiftTryCatch.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSSwiftTryCatch.h; sourceTree = ""; }; + E629F85B783E42244E37DFA9 /* STDSProgressViewController.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSProgressViewController.h; sourceTree = ""; }; + E64C700DED163CB77DCDEA78 /* fr-CA */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "fr-CA"; path = "fr-CA.lproj/Localizable.strings"; sourceTree = ""; }; + E6A3C5D4C46E681B7D76B2BA /* STDSJSONEncodable.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSJSONEncodable.h; sourceTree = ""; }; + E9F1A5A8E5922748A9BEC724 /* STDSNavigationBarCustomization.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSNavigationBarCustomization.m; sourceTree = ""; }; + EB1BDF12DBD6264C6199FFA7 /* STDSChallengeResponseObject.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeResponseObject.h; sourceTree = ""; }; + EB910E98EDD6D3E00802793C /* STDSChallengeSelectionView.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSChallengeSelectionView.h; sourceTree = ""; }; + ECC649704CDDA50E73DB3C10 /* STDSImageLoader.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSImageLoader.h; sourceTree = ""; }; + EF5219762616ACF204F08C19 /* en */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = en; path = en.lproj/Localizable.strings; sourceTree = ""; }; + F02456AB660E709F332C0C7D /* STDSSynchronousLocationManagerTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSSynchronousLocationManagerTests.m; sourceTree = ""; }; + F046DE266BAAE3DA0AC43785 /* STDSChallengeResponseSelectionInfoObject.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSChallengeResponseSelectionInfoObject.m; sourceTree = ""; }; + F112967E9FE8EE02FFFDC313 /* en-GB */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "en-GB"; path = "en-GB.lproj/Localizable.strings"; sourceTree = ""; }; + F139E48FDEFD921FF410892F /* fi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fi; path = fi.lproj/Localizable.strings; sourceTree = ""; }; + F2BC7014D26158AC9D74CC67 /* Stripe3DS2Tests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2Tests-Debug.xcconfig"; sourceTree = ""; }; + F363439353A6DD554FC44AC2 /* STDSOSVersionChecker.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSOSVersionChecker.m; sourceTree = ""; }; + F40CC91AA69F5296B0AA985C /* STDSTransaction.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSTransaction.h; sourceTree = ""; }; + F4D6DB90E3D01495C1F9BFE9 /* ro-RO */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ro-RO"; path = "ro-RO.lproj/Localizable.strings"; sourceTree = ""; }; + F5686FB8EFA3AE3B39F85BBC /* Stripe3DS2Tests-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2Tests-Release.xcconfig"; sourceTree = ""; }; + F86F34D1339F7467D0FDAC75 /* ko */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ko; path = ko.lproj/Localizable.strings; sourceTree = ""; }; + F8F6FFED1A0966A49B78F252 /* Stripe3DS2-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Stripe3DS2-Release.xcconfig"; sourceTree = ""; }; + F99B290CF2B4987A4CFE5DCE /* NSLayoutConstraint+LayoutSupport.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSLayoutConstraint+LayoutSupport.m"; sourceTree = ""; }; + F9C39A42ECBA58571EFA03C5 /* STDSBrandingView.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSBrandingView.m; sourceTree = ""; }; + F9D521B45783D36C8E83F0EA /* STDSEphemeralKeyPair.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSEphemeralKeyPair.m; sourceTree = ""; }; + FAE7026135B5049B732CEDEB /* STDSOSVersionChecker.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STDSOSVersionChecker.h; sourceTree = ""; }; + FC52C3C844BDA981EE34BAB9 /* STDSDeviceInformationParameterTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STDSDeviceInformationParameterTests.m; sourceTree = ""; }; + FE1DE9D978E6E682CB094128 /* pl-PL */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pl-PL"; path = "pl-PL.lproj/Localizable.strings"; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 16B0A41D6C0F157DDF229397 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + C2BA2E6ABDD8E80963FFC671 /* XCTest.framework in Frameworks */, + 4D73C2FBAC9B96ECB45BDC94 /* Stripe3DS2.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 8ECF69FED0DDA991C821CF6A /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 1469551DB874336B68F6C39D /* Stripe3DS2.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + A56B40D43D552FDE77670CB5 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + BF06B99FECD98C26F7B64665 /* XCTest.framework in Frameworks */, + A8A86B9BCE9702D74E0D2DB6 /* Stripe3DS2.framework in Frameworks */, + 3C6D16D8E7B5BC865D956A0B /* iOSSnapshotTestCase in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + C65BFA70E847549921E39F4E /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 148D1BA6A2E8B2B60CA9951F /* Stripe3DS2 */ = { + isa = PBXGroup; + children = ( + E4F3BE80D5F61EAC1694F954 /* include */, + A3DDF4E193616FA951B1B120 /* Resources */, + 5D93713CADD5589B5E6F6A0D /* Info.plist */, + 5B05A609753EE71502409082 /* NSData+JWEHelpers.h */, + 1FC15225A5E83948EF546B71 /* NSData+JWEHelpers.m */, + C78AA1D7AD2BAB52EE58D078 /* NSDictionary+DecodingHelpers.h */, + C566D6E444FCA4BCEC65F3D2 /* NSDictionary+DecodingHelpers.m */, + A6778F8CE36F769DF608F932 /* NSError+Stripe3DS2.h */, + 26C20D4B77372845627D6466 /* NSError+Stripe3DS2.m */, + E0949399E96663964091C0EF /* NSLayoutConstraint+LayoutSupport.h */, + F99B290CF2B4987A4CFE5DCE /* NSLayoutConstraint+LayoutSupport.m */, + A8A77DB1C711B8696B2E5FEF /* NSString+EmptyChecking.h */, + D8D309E4FBE7E73D0BB83BF5 /* NSString+EmptyChecking.m */, + 018F4AA25D84E9CBD2546BB5 /* NSString+JWEHelpers.h */, + 43CB8F416E4F4CEA34391481 /* NSString+JWEHelpers.m */, + D827473D1858B2BED85BDFA1 /* STDSACSNetworkingManager.h */, + 2D6004F2D9880055C0C0BCF1 /* STDSACSNetworkingManager.m */, + 138E82072FC6572612A8E248 /* STDSAuthenticationResponseObject.h */, + 5043E1CF6955176EC3D43F88 /* STDSAuthenticationResponseObject.m */, + 2CFAC8D74AFFC2E61BAD82A5 /* STDSBrandingView.h */, + F9C39A42ECBA58571EFA03C5 /* STDSBrandingView.m */, + A137A31BEFCCB646FB754EE2 /* STDSBundleLocator.h */, + 9564F43EF7CF4F60C3C202CD /* STDSBundleLocator.m */, + C893FD867C964775E68AE85F /* STDSChallengeInformationView.h */, + D24777EE9931075405F370DB /* STDSChallengeInformationView.m */, + 762DEC6662DF8240FE19CFFC /* STDSChallengeRequestParameters.h */, + 60693BC560C4927ACFD2E6C3 /* STDSChallengeRequestParameters.m */, + 315AEE1534F9D9974DC1674B /* STDSChallengeResponse.h */, + 10C45EE3433B731253E21E24 /* STDSChallengeResponseImage.h */, + 14F83E51E99D453DF9C2DDE1 /* STDSChallengeResponseImageObject.h */, + 3A9F828840A3B361842E38DE /* STDSChallengeResponseImageObject.m */, + AA8BE982004398CF8AAA7D1E /* STDSChallengeResponseMessageExtension.h */, + BAAC56EDE0AAD2E50E02E9AE /* STDSChallengeResponseMessageExtensionObject.h */, + 7D02FAFC403BC901931A629F /* STDSChallengeResponseMessageExtensionObject.m */, + EB1BDF12DBD6264C6199FFA7 /* STDSChallengeResponseObject.h */, + 24CE086326AF3DE336BF4F4C /* STDSChallengeResponseObject.m */, + 52EEBC7D74623E1A1D7E3219 /* STDSChallengeResponseSelectionInfo.h */, + 4CCA15E0819F9A9D89FEF9EA /* STDSChallengeResponseSelectionInfoObject.h */, + F046DE266BAAE3DA0AC43785 /* STDSChallengeResponseSelectionInfoObject.m */, + E45303DCFFDC390971E6E122 /* STDSChallengeResponseViewController.h */, + B064352BB2876CA7266C410A /* STDSChallengeResponseViewController.m */, + EB910E98EDD6D3E00802793C /* STDSChallengeSelectionView.h */, + A4D460893CB5DDE52AA0AB85 /* STDSChallengeSelectionView.m */, + D122FAE093BC4F649CC6E7AB /* STDSDebuggerChecker.h */, + C29F78CC37253B0D33FF86F2 /* STDSDebuggerChecker.m */, + 1FBE88F6E2C3153A8315EDC6 /* STDSDeviceInformation.h */, + 56B1A52548593FE78D801734 /* STDSDeviceInformation.m */, + 7A637DA708BC866C2903E0EA /* STDSDeviceInformationManager.h */, + 9580A82A59EB2B0103EDF47A /* STDSDeviceInformationManager.m */, + 172AF5C85EDBD92A1030E361 /* STDSDeviceInformationParameter.h */, + 25C7D18868DDADEF4CB6220C /* STDSDeviceInformationParameter.m */, + 4869D6E14F01EDB960CB3065 /* STDSDeviceInformationParameter+Private.h */, + 6AA43E7E90E4002DD56B7C3D /* STDSDirectoryServer.h */, + 069990DA025BE9F33602E96A /* STDSDirectoryServerCertificate.h */, + 9754FA4F3CC6AA921787916B /* STDSDirectoryServerCertificate.m */, + B015B937DD3657D62C61CE58 /* STDSDirectoryServerCertificate+Internal.h */, + 5049A72A6BEB558D8559A5EB /* STDSEllipticCurvePoint.h */, + 2EF5AA537586EF706E4056FD /* STDSEllipticCurvePoint.m */, + 20533E2C69C4B80BB33A4766 /* STDSEphemeralKeyPair.h */, + F9D521B45783D36C8E83F0EA /* STDSEphemeralKeyPair.m */, + 52D6E4F76CBA258163E57DF3 /* STDSEphemeralKeyPair+Testing.h */, + 5463EB1CC8E85BE3BD12DEFF /* STDSErrorMessage+Internal.h */, + 2CB591796654445B8434A501 /* STDSErrorMessage+Internal.m */, + 461F938CDE7ED6D06D9D2700 /* STDSException+Internal.h */, + 939C45AE068CF4F5BEB4C138 /* STDSExpandableInformationView.h */, + AE1BBF3A441B1F782B766F61 /* STDSExpandableInformationView.m */, + ECC649704CDDA50E73DB3C10 /* STDSImageLoader.h */, + A09A89BBE7360F93195641C9 /* STDSImageLoader.m */, + C02DF839637771684BC57B3A /* STDSIntegrityChecker.h */, + BB065D6900E95C5E90864C16 /* STDSIntegrityChecker.m */, + A3DFB001DB5BA7F1AFA7353A /* STDSIPAddress.h */, + CB00E968CF4DD07FB8BF2AAA /* STDSIPAddress.m */, + 94943BEAB94E949AF9830EFA /* STDSJailbreakChecker.h */, + 1BFBB92DBE97E16DE9D18B4A /* STDSJailbreakChecker.m */, + 76EA9DF8572C992E30BCF8A3 /* STDSJSONWebEncryption.h */, + 99392D3F702EC6BF7CE15291 /* STDSJSONWebEncryption.m */, + 3AA2E6DA60350400C5270E08 /* STDSJSONWebSignature.h */, + A1DC6B47C06B522B22EB19FF /* STDSJSONWebSignature.m */, + 41E0F638E776A7F3456BDA3E /* STDSLocalizedString.h */, + FAE7026135B5049B732CEDEB /* STDSOSVersionChecker.h */, + F363439353A6DD554FC44AC2 /* STDSOSVersionChecker.m */, + 95641ECBE1AE1CC198013405 /* STDSProcessingView.h */, + 7AC1C5F6F1311F50D9C37291 /* STDSProcessingView.m */, + E629F85B783E42244E37DFA9 /* STDSProgressViewController.h */, + 84967ED0D9B206B3F5AA4F02 /* STDSProgressViewController.m */, + C8F0E11AA6794BFB1715685E /* STDSSecTypeUtilities.h */, + CF3DED43BD4E0226BFB0759D /* STDSSecTypeUtilities.m */, + C4777F15603AC755362BD846 /* STDSSelectionButton.h */, + B458FCA7DFDBF75E62A43BA1 /* STDSSelectionButton.m */, + A0ADFB365AD2DE9E84873BB1 /* STDSSimulatorChecker.h */, + 14D60BDF245487FE3362BE07 /* STDSSimulatorChecker.m */, + 72DBE992B33F45C059EDB597 /* STDSSpacerView.h */, + 51E4A57094A671EB14380882 /* STDSSpacerView.m */, + B7AE3B4732D203134FE096FE /* STDSStackView.h */, + 1D6269F8B91341C387A911DB /* STDSStackView.m */, + C12AE31D386D4A193C00004B /* STDSSynchronousLocationManager.h */, + A5DBDDEB5C5DAE2EA8C95CC3 /* STDSSynchronousLocationManager.m */, + 82C41E6DC2CC247ECC17F2B9 /* STDSTextChallengeView.h */, + 04E6B6D3CAE363114723472F /* STDSTextChallengeView.m */, + 8AC648332A706809F1BDFD59 /* STDSThreeDSProtocolVersion.m */, + 47C29E9F3D2A62676B424CD1 /* STDSThreeDSProtocolVersion+Private.h */, + 8A9F5C315222AFCD14C9342F /* STDSTransaction+Private.h */, + 581EEE576D24047004C828DF /* STDSWebView.h */, + AA85BC717C46C1B95AF8C1E3 /* STDSWebView.m */, + 33CEADC8006E456FD82CE134 /* STDSWhitelistView.h */, + 615F1BD510661B38D6825128 /* STDSWhitelistView.m */, + 0F024DEAD6628A0229856656 /* Stripe3DS2-Bridging-Header.h */, + 4DD55BECC3FB85216FE46218 /* UIButton+CustomInitialization.h */, + AA29B74DD3963B1205AA1C0C /* UIButton+CustomInitialization.m */, + 4A567E08748C86FCE2A75FEA /* UIColor+DefaultColors.h */, + 869733153BCA6BE2EB7A452F /* UIColor+DefaultColors.m */, + 9A8B0001705400C661678617 /* UIColor+ThirteenSupport.h */, + 37ED28756222539D37554CC7 /* UIColor+ThirteenSupport.m */, + 1E844798B2733C8511AC080E /* UIFont+DefaultFonts.h */, + 634095EA97A4E48BF7CAA03F /* UIFont+DefaultFonts.m */, + 99CB2B66DBD356B5D222EDA9 /* UIView+LayoutSupport.h */, + 8B0F0485B7048DDAA7DA8F9B /* UIView+LayoutSupport.m */, + 8F72413F99F8E50FD0FA085D /* UIViewController+Stripe3DS2.h */, + 99451064136843047C7881AD /* UIViewController+Stripe3DS2.m */, + ); + path = Stripe3DS2; + sourceTree = ""; + }; + 18B6E1F5666C797A11E2BE03 = { + isa = PBXGroup; + children = ( + EB6FA022FDF5831C6D162E11 /* Project */, + 280289821F799A8274082E14 /* Frameworks */, + 398BBCD5E4B06CA5006CCC40 /* Products */, + ); + sourceTree = ""; + }; + 20B0FECAB4F18569649603E9 /* BuildConfigurations */ = { + isa = PBXGroup; + children = ( + 3A8CD68A006F970DDC54469A /* Project-Debug.xcconfig */, + 4DB91F58CF23E72636C69C41 /* Project-Release.xcconfig */, + A9CD7EC589C5CC6CE5CF9310 /* Stripe3DS2-Debug.xcconfig */, + F8F6FFED1A0966A49B78F252 /* Stripe3DS2-Release.xcconfig */, + 5316D0759F7F2ED0BF1D3B2E /* Stripe3DS2DemoUI-Debug.xcconfig */, + 6940D75CA36F7DB9FB56196B /* Stripe3DS2DemoUI-Release.xcconfig */, + 0E3C05BE55CF3086738AA182 /* Stripe3DS2DemoUITests-Debug.xcconfig */, + D7A9D36FEF7E97CF735D82B8 /* Stripe3DS2DemoUITests-Release.xcconfig */, + F2BC7014D26158AC9D74CC67 /* Stripe3DS2Tests-Debug.xcconfig */, + F5686FB8EFA3AE3B39F85BBC /* Stripe3DS2Tests-Release.xcconfig */, + ); + path = BuildConfigurations; + sourceTree = ""; + }; + 280289821F799A8274082E14 /* Frameworks */ = { + isa = PBXGroup; + children = ( + 08D577934984712C20C5E903 /* XCTest.framework */, + ); + name = Frameworks; + sourceTree = ""; + }; + 398BBCD5E4B06CA5006CCC40 /* Products */ = { + isa = PBXGroup; + children = ( + CE4030C50383B488C97F4B56 /* Stripe3DS2.framework */, + 02D762271DBCC1AE80E0F9D4 /* Stripe3DS2DemoUI.app */, + AA8A113E655E23381CB3BDA4 /* Stripe3DS2DemoUITests.xctest */, + 4EB746D4E46ABD2452A154AE /* Stripe3DS2Tests.xctest */, + ); + name = Products; + sourceTree = ""; + }; + 47CD3B0E4EAA05F75DA05252 /* CertificateFiles */ = { + isa = PBXGroup; + children = ( + D8B7EE0F935EF44EB2EF7D45 /* amex.der */, + 83838563F29179FB1E9F2E66 /* cartes-bancaires.der */, + 04AB48F014A8D5BA56FEF463 /* discover.der */, + 3E4C02D9D6978AD5C3D7E3D2 /* ec_test.der */, + 1104D5855D28E49E104CA004 /* mastercard.der */, + B6275E7099E55F0C04688890 /* ul-test.der */, + A70071543985284E0D9DAC64 /* visa.der */, + ); + path = CertificateFiles; + sourceTree = ""; + }; + 5E482DAA27888E537EBB24B1 /* Images */ = { + isa = PBXGroup; + children = ( + 947935869E5A08E35FA11E13 /* amex-logo@3x.png */, + 581BD6EEB19753FC873EB077 /* cartes-bancaires-logo.png */, + 41CB752E5656578F2437E247 /* Chevron@3x.png */, + 0DE0C407AFDBD26ABDBE7496 /* discover-logo.png */, + 1E3640C73D001D32CCC449F1 /* error@3x.png */, + 5862FA7E6300A2169A3CD3D7 /* mastercard-logo@3x.png */, + 34FCE3CEEDC2D23116C11DAA /* visa-logo@3x.png */, + C7F1C55E51129933B02576E7 /* visa-white-logo@3x.png */, + ); + path = Images; + sourceTree = ""; + }; + 79E03C9F2FB6092200EE737F /* Stripe3DS2Tests */ = { + isa = PBXGroup; + children = ( + AD446D5D3D8D07D491373E24 /* JSON */, + 23A3DDCE911F8C43CF74CDF4 /* Info.plist */, + 5EE8BF51F49E161737406B3A /* NSDictionary+DecodingHelpersTest.m */, + 57B8822D21AADE7DA51931CC /* NSString+EmptyCheckingTests.m */, + 6118F98576E9A39A7292C6C6 /* STDSACSNetworkingManagerTest.m */, + 969DE2FA3845BB7939D3CD2E /* STDSAuthenticationRequestParametersTest.m */, + 28E32DA72CAFBF2AF8099151 /* STDSAuthenticationResponseTests.m */, + 4B1E3760B1369DB1A33C9DFE /* STDSBase64URLEncodingTests.m */, + 89ED2133B61092FE1B478204 /* STDSChallengeParametersTests.m */, + 4EAD29E57D206B5D6BAF9099 /* STDSChallengeRequestParametersTest.m */, + 5E401845EE5759CCD30786AA /* STDSChallengeResponseObjectTest.m */, + 5F551808E65CDADD9B3A9CB8 /* STDSConfigParametersTests.m */, + A4050F381183D6FDCC990D01 /* STDSDeviceInformationManagerTests.m */, + FC52C3C844BDA981EE34BAB9 /* STDSDeviceInformationParameterTests.m */, + 6AA75957DD13FA92FA866AF3 /* STDSDirectoryServerCertificateTests.m */, + 0DA1EE83D91993BCAF13EC72 /* STDSEllipticCurvePointTests.m */, + BF80634E06E8D6F598CFC667 /* STDSEphemeralKeyPairTests.m */, + 278F2C40987F5218BD031E3D /* STDSErrorMessageTest.m */, + 6B9B49A5CF64DCB23CD0AEDE /* STDSJSONEncoderTest.m */, + 97505E93BE83F233E69BB5A9 /* STDSJSONWebEncryptionTests.m */, + E4C668442518B446D573AD24 /* STDSJSONWebSignatureTests.m */, + 30D22985091381DFCDF077E9 /* STDSSecTypeUtilitiesTests.m */, + F02456AB660E709F332C0C7D /* STDSSynchronousLocationManagerTests.m */, + 5FF3235EDD908CC1500399A1 /* STDSTestJSONUtils.m */, + 7FF8ACC073E814B7DEBA7647 /* STDSThreeDS2ServiceTests.m */, + A4A3CCBE9A2E8C1A73D8214E /* STDSTransactionTest.m */, + CB3154597040B803ADE14F9E /* STDSUICustomizationTests.m */, + 474F39AA3F47D84F8D54E5B7 /* STDSWarningTests.m */, + ); + path = Stripe3DS2Tests; + sourceTree = ""; + }; + 82BC8EB365BF993731432954 /* Sources */ = { + isa = PBXGroup; + children = ( + A4620559FBB8A42E15044E32 /* AppDelegate.h */, + 8B59CC3349FD9BBF37624667 /* AppDelegate.m */, + 3C22D410C297E2C3AFE727A6 /* main.m */, + 9F5C194C81AF1F5578698A27 /* STDSChallengeResponseObject+TestObjects.h */, + 99DB24A9B44979EA19347A76 /* STDSChallengeResponseObject+TestObjects.m */, + 1F0DADEABA0B043A6CA36DD6 /* STDSDemoViewController.h */, + 382C583385FAECA91366769E /* STDSDemoViewController.m */, + ); + path = Sources; + sourceTree = ""; + }; + 8CF6AFFC9FD5638E0D1E988E /* Stripe3DS2DemoUI */ = { + isa = PBXGroup; + children = ( + 98160E2F8248A228E5EC73F8 /* Resources */, + 82BC8EB365BF993731432954 /* Sources */, + 037C294011A01E4B4DCE5BB7 /* Info.plist */, + ); + path = Stripe3DS2DemoUI; + sourceTree = ""; + }; + 98160E2F8248A228E5EC73F8 /* Resources */ = { + isa = PBXGroup; + children = ( + 7971EB56716D5E2F59053B6D /* acs_challenge.html */, + ); + path = Resources; + sourceTree = ""; + }; + A3DDF4E193616FA951B1B120 /* Resources */ = { + isa = PBXGroup; + children = ( + 47CD3B0E4EAA05F75DA05252 /* CertificateFiles */, + 5E482DAA27888E537EBB24B1 /* Images */, + F31A6580385C6390910AFD93 /* Localizable.strings */, + ); + path = Resources; + sourceTree = ""; + }; + AD446D5D3D8D07D491373E24 /* JSON */ = { + isa = PBXGroup; + children = ( + 237977B021FE538E5E4FA35C /* ARes.json */, + 8C6D262891811FB3FE0C947E /* CRes.json */, + E12798864D5CFFB7157D4CF5 /* ErrorMessage.json */, + ); + path = JSON; + sourceTree = ""; + }; + C169A9439FD628D620B218AA /* Stripe3DS2DemoUITests */ = { + isa = PBXGroup; + children = ( + 00658E077ECB223C90484AF7 /* Info.plist */, + 466DF80D33CEC20DF4E844A9 /* STDSChallengeResponseViewControllerSnapshotTests.m */, + ); + path = Stripe3DS2DemoUITests; + sourceTree = ""; + }; + E4F3BE80D5F61EAC1694F954 /* include */ = { + isa = PBXGroup; + children = ( + 1F1FD11407319293954C6D81 /* STDSAlreadyInitializedException.h */, + 27AC486DA2A80A43C1F517B2 /* STDSAlreadyInitializedException.m */, + C91B239583899192C7B26FCF /* STDSAuthenticationRequestParameters.h */, + AB871F64FC72EBC7A91F96C1 /* STDSAuthenticationRequestParameters.m */, + 4EA4AB3BA2FB41B3D5F38978 /* STDSAuthenticationResponse.h */, + BD94CAE01A17E083E5617E56 /* STDSButtonCustomization.h */, + 5489DCAA65624C6F966816B6 /* STDSButtonCustomization.m */, + D03B98D5861739833B197D8F /* STDSChallengeParameters.h */, + 23C929BF760735CC01E1E8F5 /* STDSChallengeParameters.m */, + B7A75140FB62A71261CAF5EC /* STDSChallengeStatusReceiver.h */, + 10DB3CD8F2541AC06681F2C8 /* STDSCompletionEvent.h */, + 3B6B71C61CF5ADC7796441F5 /* STDSCompletionEvent.m */, + A1BFB8640032796CDA016765 /* STDSConfigParameters.h */, + 89B5ED899439D1C1054CB448 /* STDSConfigParameters.m */, + AFF187D1D58405C736474042 /* STDSCustomization.h */, + A8E62EC973FC5C9905A40CA8 /* STDSCustomization.m */, + 645D0BB927B7F8A0D37EE04D /* STDSErrorMessage.h */, + 6DDB6222A2EF31FDB7592368 /* STDSErrorMessage.m */, + B71A1C110DCC23A9CE929837 /* STDSException.h */, + 6F030410FDABFF833CD8FE46 /* STDSException.m */, + BA96C9FFA48B85F9C42E8C68 /* STDSFooterCustomization.h */, + 6CFF141AC98F213122D037C6 /* STDSFooterCustomization.m */, + 853A361AB630A2BD402D1B47 /* STDSInvalidInputException.h */, + 6FDAC320458399141EB161E2 /* STDSInvalidInputException.m */, + B219360BFB325779485BF702 /* STDSJSONDecodable.h */, + E6A3C5D4C46E681B7D76B2BA /* STDSJSONEncodable.h */, + 06C5207432FB07BD71703BCC /* STDSJSONEncoder.h */, + 0681C043F732148587737233 /* STDSJSONEncoder.m */, + 6932DD26C7C16938DFA0E198 /* STDSLabelCustomization.h */, + 01D320CD5ADA49F17B8BC1B0 /* STDSLabelCustomization.m */, + E1630D876436E43547FF69CE /* STDSNavigationBarCustomization.h */, + E9F1A5A8E5922748A9BEC724 /* STDSNavigationBarCustomization.m */, + 2FCABBF51CBA7F9FEF757B4C /* STDSNotInitializedException.h */, + B7BD1E24EA9427121E148DFC /* STDSNotInitializedException.m */, + 3A6D20D1DAB7D769E5E97528 /* STDSProtocolErrorEvent.h */, + 3255706D2304FB551C97305F /* STDSProtocolErrorEvent.m */, + DE7D85369E012B15702D8DF9 /* STDSRuntimeErrorEvent.h */, + 1501A7A3010FFE53D00E318F /* STDSRuntimeErrorEvent.m */, + B8619CA38E2A5B49DBF8546B /* STDSRuntimeException.h */, + CD33421F13A675DCFE597FFB /* STDSRuntimeException.m */, + 7CD753D9BBFC378C1B491B00 /* STDSSelectionCustomization.h */, + 43540AB7D13C0C9A563C3F4D /* STDSSelectionCustomization.m */, + AE53BAA72835AFA3B35C58D3 /* STDSStripe3DS2Error.h */, + 9F4376BB07FD1EE783249AED /* STDSStripe3DS2Error.m */, + E5BC34A661D9080A2EE239F8 /* STDSSwiftTryCatch.h */, + 9946D28A86E85A6C84EB6C7B /* STDSSwiftTryCatch.m */, + 68ED2A0E11E1B27980E0C60A /* STDSTextFieldCustomization.h */, + 97163B5E9598CA3A298920B9 /* STDSTextFieldCustomization.m */, + 5EAA444FA7C82BDA4AD907BF /* STDSThreeDS2Service.h */, + 77646CEC7EE754F47EDBDA4F /* STDSThreeDS2Service.m */, + 5E6A9565E97DF2544254AA94 /* STDSThreeDSProtocolVersion.h */, + F40CC91AA69F5296B0AA985C /* STDSTransaction.h */, + 5FDE2481364C454058BF2377 /* STDSTransaction.m */, + 210B22FF4DCB0C6E7C763EAB /* STDSUICustomization.h */, + A55B2474CD1A39D70869B75B /* STDSUICustomization.m */, + 0BEB5006261C5E070FEE62BF /* STDSWarning.h */, + A502CCC5190F6B642EEC0CE6 /* STDSWarning.m */, + 28C6B4F6BA431B9342970FD6 /* Stripe3DS2.h */, + ); + path = include; + sourceTree = ""; + }; + EB6FA022FDF5831C6D162E11 /* Project */ = { + isa = PBXGroup; + children = ( + 20B0FECAB4F18569649603E9 /* BuildConfigurations */, + 148D1BA6A2E8B2B60CA9951F /* Stripe3DS2 */, + 8CF6AFFC9FD5638E0D1E988E /* Stripe3DS2DemoUI */, + C169A9439FD628D620B218AA /* Stripe3DS2DemoUITests */, + 79E03C9F2FB6092200EE737F /* Stripe3DS2Tests */, + ); + name = Project; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXHeadersBuildPhase section */ + A8117E5A795D5342E5AEF24C /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 3192FCC841152A7E5E6F19D6 /* STDSAlreadyInitializedException.h in Headers */, + 9673946B78A7763070E05CD0 /* STDSAuthenticationRequestParameters.h in Headers */, + 435EFC9119D4B42B2C6A6F88 /* STDSAuthenticationResponse.h in Headers */, + EC51189277D7D68D08C2A65C /* STDSButtonCustomization.h in Headers */, + DC0D9A57F2E312663C088FB8 /* STDSChallengeParameters.h in Headers */, + 72F430506E684D61BD5CFE3B /* STDSChallengeStatusReceiver.h in Headers */, + 8E27285FE76F8BECFD70286D /* STDSCompletionEvent.h in Headers */, + DC32281BF0847A98B100C3A1 /* STDSConfigParameters.h in Headers */, + F04482C663F6DF8E522B0833 /* STDSCustomization.h in Headers */, + 2BBA82878804711E06CBFCCB /* STDSErrorMessage.h in Headers */, + 24D49F833E1ED28DE557F219 /* STDSException.h in Headers */, + 79557307ECD48D17C566997A /* STDSFooterCustomization.h in Headers */, + 4E35267801D94FB84816C519 /* STDSInvalidInputException.h in Headers */, + 9B32E9A4CD2BDA61C730F5EB /* STDSJSONDecodable.h in Headers */, + 890E22971E6F7B0FA1C8D649 /* STDSJSONEncodable.h in Headers */, + 2224463CE5FB99D4BEE6A479 /* STDSJSONEncoder.h in Headers */, + 0BA0F0CADC4CFF419E1F7EFC /* STDSLabelCustomization.h in Headers */, + 3B430F172A3AD6AA9AFDFC04 /* STDSNavigationBarCustomization.h in Headers */, + 2CB6DF5CCCBC1EB9ABD03143 /* STDSNotInitializedException.h in Headers */, + 67D431374DECE36B2606D944 /* STDSProtocolErrorEvent.h in Headers */, + 15D4A2F07EA477B84CD48B15 /* STDSRuntimeErrorEvent.h in Headers */, + 2C6DB6516699B4EFADDF3360 /* STDSRuntimeException.h in Headers */, + F5513045A25210C524A05DF5 /* STDSSelectionCustomization.h in Headers */, + 335553A547A3CA80B3901CCF /* STDSStripe3DS2Error.h in Headers */, + DE32EAA8F5C2645652FCFB48 /* STDSSwiftTryCatch.h in Headers */, + 4C267E9A30ADC18E71408ED5 /* STDSTextFieldCustomization.h in Headers */, + 31C636A25BF83316EE7EB57D /* STDSThreeDS2Service.h in Headers */, + 847926A68A88BEB58C92D9BC /* STDSThreeDSProtocolVersion.h in Headers */, + 9DC5612697835A383DC6606E /* STDSTransaction.h in Headers */, + 0615DD02C0B022AE207DADF0 /* STDSUICustomization.h in Headers */, + 9F0F6085FBD892BF5F14CF86 /* STDSWarning.h in Headers */, + 893713F4464A88FAB92BC2C6 /* Stripe3DS2.h in Headers */, + 425849564519D6454DDA3424 /* NSData+JWEHelpers.h in Headers */, + 1134F81C7D015BA5EA52BFD1 /* NSDictionary+DecodingHelpers.h in Headers */, + 3617B07ABD719F1A5F8302FA /* NSError+Stripe3DS2.h in Headers */, + 28AD7EC6F6DA2AECD7F61BA7 /* NSLayoutConstraint+LayoutSupport.h in Headers */, + 30B25163BEDEB99CDD101B5F /* NSString+EmptyChecking.h in Headers */, + A5193DADCD0F48911F775D88 /* NSString+JWEHelpers.h in Headers */, + A78F19DA3D146A258FB9DE3C /* STDSACSNetworkingManager.h in Headers */, + 2B2787A7103C0D0AB5F00B78 /* STDSAuthenticationResponseObject.h in Headers */, + 458BC5A3D0E7B4D05549474F /* STDSBrandingView.h in Headers */, + 97CE8B57A97D037E977E62F3 /* STDSBundleLocator.h in Headers */, + F13A032A4AF15BB61EA18A8D /* STDSChallengeInformationView.h in Headers */, + BF342613EE89845EB1C4B72A /* STDSChallengeRequestParameters.h in Headers */, + 7337FB20DC0EF491B5922913 /* STDSChallengeResponse.h in Headers */, + 95A2D58051AADDBE16CCA0B6 /* STDSChallengeResponseImage.h in Headers */, + D819C341ECEE1EFE6FE66085 /* STDSChallengeResponseImageObject.h in Headers */, + 793D61AD55630DD336A141A7 /* STDSChallengeResponseMessageExtension.h in Headers */, + 48316C1AB1D9151C205A59A4 /* STDSChallengeResponseMessageExtensionObject.h in Headers */, + E90749131EDA0C86BBBFCE5C /* STDSChallengeResponseObject.h in Headers */, + 3C8C3BCDDDDEF6B9E150D4E1 /* STDSChallengeResponseSelectionInfo.h in Headers */, + 9C225B9C556C20D1A6F97180 /* STDSChallengeResponseSelectionInfoObject.h in Headers */, + 9E46777893FA2D6691F496D7 /* STDSChallengeResponseViewController.h in Headers */, + E7E424C5264A4DE4A1EC19B6 /* STDSChallengeSelectionView.h in Headers */, + F28AD6CA8AB1A5DEE7A8F429 /* STDSDebuggerChecker.h in Headers */, + 57E58A3B88D9EEB9FAE9B383 /* STDSDeviceInformation.h in Headers */, + 9843F6AABF7B0A623E7DF979 /* STDSDeviceInformationManager.h in Headers */, + 3844F0E21742F43BCB32499A /* STDSDeviceInformationParameter+Private.h in Headers */, + 2938D9CC41899DC78D01EF9F /* STDSDeviceInformationParameter.h in Headers */, + C4BA75544A7F51355BF4B5F8 /* STDSDirectoryServer.h in Headers */, + DB96817598FCE73F5131437E /* STDSDirectoryServerCertificate+Internal.h in Headers */, + 63AFB3C739DD987EE56F801F /* STDSDirectoryServerCertificate.h in Headers */, + 5C83A3A4631E51EEECBEAA0F /* STDSEllipticCurvePoint.h in Headers */, + 3343DA94A5032627843A343C /* STDSEphemeralKeyPair+Testing.h in Headers */, + B94E5A5BF5F22998E4CA9ED3 /* STDSEphemeralKeyPair.h in Headers */, + 34256504F0E0E519D586B7CE /* STDSErrorMessage+Internal.h in Headers */, + 149FE0C2910F08910B250BD8 /* STDSException+Internal.h in Headers */, + F35963DB86D90320CFA7BCB9 /* STDSExpandableInformationView.h in Headers */, + DDB98914E1135E8D445545C8 /* STDSIPAddress.h in Headers */, + 613EF855524F7861A6DDD8C1 /* STDSImageLoader.h in Headers */, + D3432C5BF5211A60D5C15D9E /* STDSIntegrityChecker.h in Headers */, + 4C573A44AB0A9CB28D76C621 /* STDSJSONWebEncryption.h in Headers */, + CA617B6F5CDFB8FDEEE38636 /* STDSJSONWebSignature.h in Headers */, + 7C7073B54628ED836F422F1D /* STDSJailbreakChecker.h in Headers */, + 8E0501C3BB849C2D2D99F34E /* STDSLocalizedString.h in Headers */, + 68F314D63323954EFE309E49 /* STDSOSVersionChecker.h in Headers */, + 09F0B1945CC12FB1215119FD /* STDSProcessingView.h in Headers */, + 511513A9180F9835B08CBE56 /* STDSProgressViewController.h in Headers */, + CEA1EB91858043A837638FE0 /* STDSSecTypeUtilities.h in Headers */, + D9F8BA88953A91DC082FA6DA /* STDSSelectionButton.h in Headers */, + AC474013841CE1DED97F3FA8 /* STDSSimulatorChecker.h in Headers */, + AA94519687902E34E5243AEA /* STDSSpacerView.h in Headers */, + 206B93BDE91CFED74A053B87 /* STDSStackView.h in Headers */, + F0B9EFD359B1AB28A695CB48 /* STDSSynchronousLocationManager.h in Headers */, + 59756023104E4FB886B48936 /* STDSTextChallengeView.h in Headers */, + 326AD1DD7BB8A2A6DA66EC67 /* STDSThreeDSProtocolVersion+Private.h in Headers */, + AE1894FEA510AC394DE73C4B /* STDSTransaction+Private.h in Headers */, + 7EC8E5523F29B0F01FE827DB /* STDSWebView.h in Headers */, + 825ACBE0734BDCE746E66AB0 /* STDSWhitelistView.h in Headers */, + 75CDFDABD7F9813563E3F618 /* Stripe3DS2-Bridging-Header.h in Headers */, + 0A2BC6A9E388B242C46C09D9 /* UIButton+CustomInitialization.h in Headers */, + 8322973CD09F2E1C88B6046E /* UIColor+DefaultColors.h in Headers */, + 27F3EA1BB7E7B56A1A8524BA /* UIColor+ThirteenSupport.h in Headers */, + 8E833B7FCBC3A99072582FEA /* UIFont+DefaultFonts.h in Headers */, + A026717FFF299A7C1C51565F /* UIView+LayoutSupport.h in Headers */, + 41246FCC0695BABE1489C53E /* UIViewController+Stripe3DS2.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXHeadersBuildPhase section */ + +/* Begin PBXNativeTarget section */ + 57AEC53510AE0DC0539730F3 /* Stripe3DS2 */ = { + isa = PBXNativeTarget; + buildConfigurationList = 5C9BBB6FD4175090CFCAA126 /* Build configuration list for PBXNativeTarget "Stripe3DS2" */; + buildPhases = ( + A8117E5A795D5342E5AEF24C /* Headers */, + 559A957A4AA29D178C3E3D8F /* Sources */, + 563147EAF54FC4A425A06D46 /* Resources */, + 726DD7176204D90BD7552B84 /* Embed Frameworks */, + C65BFA70E847549921E39F4E /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = Stripe3DS2; + productName = Stripe3DS2; + productReference = CE4030C50383B488C97F4B56 /* Stripe3DS2.framework */; + productType = "com.apple.product-type.framework"; + }; + 5907C55B1F111921112DF2BF /* Stripe3DS2DemoUI */ = { + isa = PBXNativeTarget; + buildConfigurationList = 37237B2A5629D2257C454AD1 /* Build configuration list for PBXNativeTarget "Stripe3DS2DemoUI" */; + buildPhases = ( + 5D618F152EF19018F5C7E579 /* Sources */, + C4175027D57AF53025CB9719 /* Resources */, + 00E415680C602ED1D6D8E71F /* Embed Frameworks */, + 8ECF69FED0DDA991C821CF6A /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + C7515969CD69027C276886AB /* PBXTargetDependency */, + ); + name = Stripe3DS2DemoUI; + productName = Stripe3DS2DemoUI; + productReference = 02D762271DBCC1AE80E0F9D4 /* Stripe3DS2DemoUI.app */; + productType = "com.apple.product-type.application"; + }; + 7DA168BC86CE957505FA091B /* Stripe3DS2DemoUITests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 5F64A3CED17753DA2643AA63 /* Build configuration list for PBXNativeTarget "Stripe3DS2DemoUITests" */; + buildPhases = ( + 7A43CA17E1E2D826DCBDD88D /* Sources */, + F3AE035E64AB50AB8B90A58E /* Resources */, + 58FA2FE74AE67CD3CDAFD7E5 /* Embed Frameworks */, + A56B40D43D552FDE77670CB5 /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + 0AE2FB81071258D69C9FCAD4 /* PBXTargetDependency */, + 026B55E92602E24EE7139E1F /* PBXTargetDependency */, + ); + name = Stripe3DS2DemoUITests; + packageProductDependencies = ( + 0118969F6608A92583CC6C98 /* iOSSnapshotTestCase */, + ); + productName = Stripe3DS2DemoUITests; + productReference = AA8A113E655E23381CB3BDA4 /* Stripe3DS2DemoUITests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; + F2D50FA32F27498F56CD08DD /* Stripe3DS2Tests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 271DFDD5ABBB719F3BD1AB56 /* Build configuration list for PBXNativeTarget "Stripe3DS2Tests" */; + buildPhases = ( + 0D8CCC419762E2FD4F945EB0 /* Sources */, + 4FDEAB55B960B360B76B3F41 /* Resources */, + 43B749EC915FB6D2FF5ABEE0 /* Embed Frameworks */, + 16B0A41D6C0F157DDF229397 /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + 798525809F1F67D0C6634927 /* PBXTargetDependency */, + ); + name = Stripe3DS2Tests; + productName = Stripe3DS2Tests; + productReference = 4EB746D4E46ABD2452A154AE /* Stripe3DS2Tests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 8D219187B704575AD3CD3EC5 /* Project object */ = { + isa = PBXProject; + attributes = { + TargetAttributes = { + 7DA168BC86CE957505FA091B = { + TestTargetID = 5907C55B1F111921112DF2BF; + }; + }; + }; + buildConfigurationList = 57B08C966701355644CF5CEB /* Build configuration list for PBXProject "Stripe3DS2" */; + compatibilityVersion = "Xcode 13.0"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + Base, + "bg-BG", + "ca-ES", + "cs-CZ", + da, + de, + "el-GR", + en, + "en-GB", + es, + "es-419", + "et-EE", + fi, + fil, + fr, + "fr-CA", + hr, + hu, + id, + it, + ja, + ko, + "lt-LT", + "lv-LV", + "ms-MY", + mt, + nb, + nl, + "nn-NO", + "pl-PL", + "pt-BR", + "pt-PT", + "ro-RO", + ru, + "sk-SK", + "sl-SI", + sv, + tr, + vi, + "zh-HK", + "zh-Hans", + "zh-Hant", + ); + mainGroup = 18B6E1F5666C797A11E2BE03; + packageReferences = ( + 176A29DF97FAFE939C7F667B /* XCRemoteSwiftPackageReference "ios-snapshot-test-case" */, + ); + productRefGroup = 398BBCD5E4B06CA5006CCC40 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 57AEC53510AE0DC0539730F3 /* Stripe3DS2 */, + F2D50FA32F27498F56CD08DD /* Stripe3DS2Tests */, + 5907C55B1F111921112DF2BF /* Stripe3DS2DemoUI */, + 7DA168BC86CE957505FA091B /* Stripe3DS2DemoUITests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 4FDEAB55B960B360B76B3F41 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 5B16918BF5F67B0F5FB1AFD5 /* ARes.json in Resources */, + F4E8353A470750D9BCEA3CD8 /* CRes.json in Resources */, + E7A72BB7CC8DE8DA7FA0DEC5 /* ErrorMessage.json in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 563147EAF54FC4A425A06D46 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + A97E1DF84D381952A0FB4C1C /* amex.der in Resources */, + EA89021709AFB5F04961EB78 /* cartes-bancaires.der in Resources */, + 3D674B4EE5ABACD65EB21A1E /* discover.der in Resources */, + FD6456BD86F07C446B9FB6C8 /* ec_test.der in Resources */, + 081347606D48F64161D95B45 /* mastercard.der in Resources */, + 81B1C12449852CEB3C59793F /* ul-test.der in Resources */, + 125427F1322501AA12EAEBE6 /* visa.der in Resources */, + 675801746AD86B23F8560F59 /* Chevron@3x.png in Resources */, + 794D78D81790620AFA6AA5C3 /* amex-logo@3x.png in Resources */, + 65C3CC17DA8E8B98A5ACA320 /* cartes-bancaires-logo.png in Resources */, + C6662AAA303ADB0DA6D47809 /* discover-logo.png in Resources */, + 4587DADB1B2D98560E8BB181 /* error@3x.png in Resources */, + 0A0D73B3004DC5854D2BC639 /* mastercard-logo@3x.png in Resources */, + 53DBABFCA9B7E264366922F2 /* visa-logo@3x.png in Resources */, + 7706F5211DE7A4411CD108EA /* visa-white-logo@3x.png in Resources */, + 6BA3F565B6C0B78F6DED8826 /* Localizable.strings in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + C4175027D57AF53025CB9719 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 7C20B044ACFD3A50FD4AC9A8 /* acs_challenge.html in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + F3AE035E64AB50AB8B90A58E /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 0D8CCC419762E2FD4F945EB0 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + B426123CC7FCA337E13304F9 /* NSDictionary+DecodingHelpersTest.m in Sources */, + E73028FD4537F6E48F927127 /* NSString+EmptyCheckingTests.m in Sources */, + D1427392DA8AD8F5B4118781 /* STDSACSNetworkingManagerTest.m in Sources */, + A33A549CF12209280E3B1DBA /* STDSAuthenticationRequestParametersTest.m in Sources */, + 5DAC25EBAB19555229654E44 /* STDSAuthenticationResponseTests.m in Sources */, + A56AE2CD941D7ADAC7FC8BCC /* STDSBase64URLEncodingTests.m in Sources */, + D63E3DDC92DD46062A265D86 /* STDSChallengeParametersTests.m in Sources */, + 4A9423CC79312BC3A0DEC38B /* STDSChallengeRequestParametersTest.m in Sources */, + ECF8755D2602E0E06DCB3210 /* STDSChallengeResponseObjectTest.m in Sources */, + 32BE38D04DD76CA4490389C6 /* STDSConfigParametersTests.m in Sources */, + B1C2AA6259CE34B0042A1FB1 /* STDSDeviceInformationManagerTests.m in Sources */, + 2B09B7FDF8C4AA551F7EE18E /* STDSDeviceInformationParameterTests.m in Sources */, + E4DA2CE8DC8C60DE40222185 /* STDSDirectoryServerCertificateTests.m in Sources */, + 9F52F67A1A3A2199AEA598FC /* STDSEllipticCurvePointTests.m in Sources */, + BFC5852E14724750E70DA2A6 /* STDSEphemeralKeyPairTests.m in Sources */, + 1767509FDC216602A5E43F0E /* STDSErrorMessageTest.m in Sources */, + D189220981C7705913D93687 /* STDSJSONEncoderTest.m in Sources */, + 98075195759ABB91B0D19847 /* STDSJSONWebEncryptionTests.m in Sources */, + 4C0E13298F08D391FE8600CF /* STDSJSONWebSignatureTests.m in Sources */, + 480CD62EC91F4796D1D8D8DC /* STDSSecTypeUtilitiesTests.m in Sources */, + BA65731CDB75C3124834586C /* STDSSynchronousLocationManagerTests.m in Sources */, + EA19A7AEC298F3EC010D1199 /* STDSTestJSONUtils.m in Sources */, + D927789F89B413C08DE4D388 /* STDSThreeDS2ServiceTests.m in Sources */, + 9B7081D378D1441B98B81207 /* STDSTransactionTest.m in Sources */, + 83878337274D8C43D492EC45 /* STDSUICustomizationTests.m in Sources */, + 30632F7C730BF8D23B607CF7 /* STDSWarningTests.m in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 559A957A4AA29D178C3E3D8F /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + A8CAD8276ED8057A760147CB /* NSData+JWEHelpers.m in Sources */, + 0D4B42EC5019D213BDBC84E7 /* NSDictionary+DecodingHelpers.m in Sources */, + 089DAFECA444861762781974 /* NSError+Stripe3DS2.m in Sources */, + 9ED9C602C10682CD5DD91C12 /* NSLayoutConstraint+LayoutSupport.m in Sources */, + DEEF260365A4D9B0D83E557D /* NSString+EmptyChecking.m in Sources */, + 88197445522F39DB1E9F6AE7 /* NSString+JWEHelpers.m in Sources */, + B31168F5FEC3464AC212DB85 /* STDSACSNetworkingManager.m in Sources */, + 5FE5FB933E2651C8D84FA477 /* STDSAuthenticationResponseObject.m in Sources */, + 75150C4678B8207AA6E7D969 /* STDSBrandingView.m in Sources */, + 8233B9F131602DD7143FF96F /* STDSBundleLocator.m in Sources */, + 14BDBD98F8E15576403255C9 /* STDSChallengeInformationView.m in Sources */, + 2970D4880EBFAEFABA2E8EBD /* STDSChallengeRequestParameters.m in Sources */, + 2AB06EC20B22306F1DDF9F4B /* STDSChallengeResponseImageObject.m in Sources */, + B9A456E52E62E9DAE9F424A7 /* STDSChallengeResponseMessageExtensionObject.m in Sources */, + 09DFAF7B38EAB76BC66AC9C8 /* STDSChallengeResponseObject.m in Sources */, + AA4D157F31A246890D446981 /* STDSChallengeResponseSelectionInfoObject.m in Sources */, + CCC376FE7424029342A8D15E /* STDSChallengeResponseViewController.m in Sources */, + 4FE8C076102C195D609977F5 /* STDSChallengeSelectionView.m in Sources */, + A6909D6C1A68963504733939 /* STDSDebuggerChecker.m in Sources */, + E630DCBBEFAAD0435BEF989C /* STDSDeviceInformation.m in Sources */, + 3F2624C33291FCE00D596768 /* STDSDeviceInformationManager.m in Sources */, + 0EEFDE04392FD017EA0A017A /* STDSDeviceInformationParameter.m in Sources */, + 9F9BB9E7FA18FEDA44F7042E /* STDSDirectoryServerCertificate.m in Sources */, + 6B3AA85EE71FC42EF9BD999F /* STDSEllipticCurvePoint.m in Sources */, + 05647ABCFBDBE1209D5044AC /* STDSEphemeralKeyPair.m in Sources */, + 2E2022723BC7A4B2DD5ECE76 /* STDSErrorMessage+Internal.m in Sources */, + 4142A3AB3E384F91A1E5550B /* STDSExpandableInformationView.m in Sources */, + 9146B2E3B89EF3F489D7E233 /* STDSIPAddress.m in Sources */, + 136255493F3F0E930FBA8606 /* STDSImageLoader.m in Sources */, + EB791827A943AA61976D8C29 /* STDSIntegrityChecker.m in Sources */, + 955CBE0C979291F89567D8A7 /* STDSJSONWebEncryption.m in Sources */, + EFC72E766F9FD4EE18D309BC /* STDSJSONWebSignature.m in Sources */, + BA00D11D792C99B7D5749F59 /* STDSJailbreakChecker.m in Sources */, + 602C526C0B52DF81846DA664 /* STDSOSVersionChecker.m in Sources */, + 574A7976213046F02F7F60C4 /* STDSProcessingView.m in Sources */, + 1156B25EBE0135B627E174D2 /* STDSProgressViewController.m in Sources */, + DBDB8B56FFF5C4234705E698 /* STDSSecTypeUtilities.m in Sources */, + D41F091BE1579B801D1BBC20 /* STDSSelectionButton.m in Sources */, + 4518AD83580FC222B883DAFB /* STDSSimulatorChecker.m in Sources */, + 3F30A8F418870D176A626F00 /* STDSSpacerView.m in Sources */, + 3909D8028AC485BCCF17B48B /* STDSStackView.m in Sources */, + E0644CE0260178023E643B23 /* STDSSynchronousLocationManager.m in Sources */, + D5706ACF84F0A993CE1885D4 /* STDSTextChallengeView.m in Sources */, + 1B20B454D0C309613A1FAF68 /* STDSThreeDSProtocolVersion.m in Sources */, + 884D59AF7FD70889276AFC02 /* STDSWebView.m in Sources */, + E70CFDC59731BA62DD89906C /* STDSWhitelistView.m in Sources */, + A4CFDBFC3099BD7E1A694E3E /* UIButton+CustomInitialization.m in Sources */, + 2086DFD1FC02783FB106AD35 /* UIColor+DefaultColors.m in Sources */, + E19C8104FC1A9BAEA0BE8A9D /* UIColor+ThirteenSupport.m in Sources */, + 987347CA74818CEB716B9DB3 /* UIFont+DefaultFonts.m in Sources */, + 390691CA0EAF81418B0C65DB /* UIView+LayoutSupport.m in Sources */, + 08ECC7E70E7478E76793ED1E /* UIViewController+Stripe3DS2.m in Sources */, + D83D2FF20387FACC383A2A0F /* STDSAlreadyInitializedException.m in Sources */, + 7ED98325E7E54A54BFE54D6C /* STDSAuthenticationRequestParameters.m in Sources */, + C430332104547BA508416B41 /* STDSButtonCustomization.m in Sources */, + 6DB54DBAEFD08967367B6BF7 /* STDSChallengeParameters.m in Sources */, + FD8A81873C3BC7579ED07460 /* STDSCompletionEvent.m in Sources */, + 8A0872706BF4CDE72EF0C480 /* STDSConfigParameters.m in Sources */, + BF2651A9F3A31CB1AFEC44BB /* STDSCustomization.m in Sources */, + 8C9BD4B150F384111CBD3BD3 /* STDSErrorMessage.m in Sources */, + 3C88AE37A760B91E93D423CF /* STDSException.m in Sources */, + F569B584D357DC5C55542015 /* STDSFooterCustomization.m in Sources */, + EE3854BC2A14F87E8D51B0D0 /* STDSInvalidInputException.m in Sources */, + 0D3D9C95CB14A610EAAAEF6E /* STDSJSONEncoder.m in Sources */, + EEE77034728D2CFE38CC5A85 /* STDSLabelCustomization.m in Sources */, + BB379E68231A7DA62F87E628 /* STDSNavigationBarCustomization.m in Sources */, + 8A64AFBB2AC15E0E0F57ED25 /* STDSNotInitializedException.m in Sources */, + B51D1C218F460F67B58A1F0D /* STDSProtocolErrorEvent.m in Sources */, + 64D05353B50FAF3BB76D6527 /* STDSRuntimeErrorEvent.m in Sources */, + 01706B4660728A5BFAC12840 /* STDSRuntimeException.m in Sources */, + D3AA14D1E059E90F5A965CC4 /* STDSSelectionCustomization.m in Sources */, + 2C5621AABECF6192F92A20CF /* STDSStripe3DS2Error.m in Sources */, + 4E4E4BC1E2D041FA530336EE /* STDSSwiftTryCatch.m in Sources */, + 284F72D3FD8C06C0E5974086 /* STDSTextFieldCustomization.m in Sources */, + 24F626CD0F93B7AF59B1D6AA /* STDSThreeDS2Service.m in Sources */, + 6E727D2B738D7A3527952D28 /* STDSTransaction.m in Sources */, + DF4D38DC0AF89EAAA8718AAE /* STDSUICustomization.m in Sources */, + E08AF96E690D75F6AB52021F /* STDSWarning.m in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 5D618F152EF19018F5C7E579 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + B84ED7FBD49865F23B774067 /* AppDelegate.m in Sources */, + A39CFB56BF33B21A89987F9C /* STDSChallengeResponseObject+TestObjects.m in Sources */, + 128B64380D6724F016EE8D56 /* STDSDemoViewController.m in Sources */, + B012C4A8ACA1D064C40A1B63 /* main.m in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 7A43CA17E1E2D826DCBDD88D /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + F939E05536AE838DC489C3AA /* STDSChallengeResponseViewControllerSnapshotTests.m in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 026B55E92602E24EE7139E1F /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = Stripe3DS2DemoUI; + target = 5907C55B1F111921112DF2BF /* Stripe3DS2DemoUI */; + targetProxy = 35B6793A7F3FB130F40209F1 /* PBXContainerItemProxy */; + }; + 0AE2FB81071258D69C9FCAD4 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = Stripe3DS2; + target = 57AEC53510AE0DC0539730F3 /* Stripe3DS2 */; + targetProxy = 52E1EB51359DA6E75D851D71 /* PBXContainerItemProxy */; + }; + 798525809F1F67D0C6634927 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = Stripe3DS2; + target = 57AEC53510AE0DC0539730F3 /* Stripe3DS2 */; + targetProxy = 6C355B127D670833C76237D3 /* PBXContainerItemProxy */; + }; + C7515969CD69027C276886AB /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = Stripe3DS2; + target = 57AEC53510AE0DC0539730F3 /* Stripe3DS2 */; + targetProxy = 947ED275EAB25AD4CD701936 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin PBXVariantGroup section */ + F31A6580385C6390910AFD93 /* Localizable.strings */ = { + isa = PBXVariantGroup; + children = ( + 8A6035A8211B3D50D10EB947 /* bg-BG */, + 40DC4913154C3D3A27E35207 /* ca-ES */, + 7859D635964B2854A07B5285 /* cs-CZ */, + 8B8D429F1C03603DB218700F /* da */, + 2C2666DF7D23EBC3FFE7CCF1 /* de */, + 4C158B552B0CBE635A2AF7BB /* el-GR */, + EF5219762616ACF204F08C19 /* en */, + F112967E9FE8EE02FFFDC313 /* en-GB */, + 1BBADD412A7C2D1FF35A7C86 /* es */, + 406D88ADED27D64DED2002A2 /* es-419 */, + 461FDC130846625171164A9E /* et-EE */, + F139E48FDEFD921FF410892F /* fi */, + 49EF79D69693F937FEC46906 /* fil */, + 4BC4D9FD1D3869D491CC2CF8 /* fr */, + E64C700DED163CB77DCDEA78 /* fr-CA */, + 0E38A775DB9F529427E900CF /* hr */, + 371F8076630A8839C1F6A508 /* hu */, + 62773D5C85C567F28BCE0DA5 /* id */, + 13ED1CC076C6B3FB49C9197A /* it */, + 157970FF5B4A817041E3D668 /* ja */, + F86F34D1339F7467D0FDAC75 /* ko */, + 71494F94F36F4B731A27D182 /* lt-LT */, + 38572D4E32BF3670F1C83409 /* lv-LV */, + B1A7D41498B79E35BD724681 /* ms-MY */, + 18D88DE3E34106F934015BF9 /* mt */, + 6B487D8DD39DAC17042A3F04 /* nb */, + 83BDD516620860DC08B36B96 /* nl */, + D3F5A6D5A680F008C00A2D69 /* nn-NO */, + FE1DE9D978E6E682CB094128 /* pl-PL */, + 6690476C1BEA59C37576FB36 /* pt-BR */, + BC3EE020DC9358FF56389BBE /* pt-PT */, + F4D6DB90E3D01495C1F9BFE9 /* ro-RO */, + 694DF6F47767A651907D55D4 /* ru */, + 58C38FF9205C9B7D79C51A37 /* sk-SK */, + 3B9A821F35B93898A7AC1ADF /* sl-SI */, + 38F3A7D625E4B0D3506A37F6 /* sv */, + 2192708AD201F80F6E1A39F7 /* tr */, + B97D91CB54F48F64CAC9E378 /* vi */, + 3F017BF6ECE0EBE4E0DFCF9C /* zh-Hans */, + AF389752CCEB6FA734AAE31E /* zh-Hant */, + CECC1E22D0F0336039B435DE /* zh-HK */, + ); + name = Localizable.strings; + sourceTree = ""; + }; +/* End PBXVariantGroup section */ + +/* Begin XCBuildConfiguration section */ + 03D567A9F86C4B204A11D604 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 6940D75CA36F7DB9FB56196B /* Stripe3DS2DemoUI-Release.xcconfig */; + buildSettings = { + INFOPLIST_FILE = Stripe3DS2DemoUI/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.Stripe3DS2DemoUI; + PRODUCT_NAME = Stripe3DS2DemoUI; + SDKROOT = iphoneos; + }; + name = Release; + }; + 380615C693D7BF5A8884139A /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 5316D0759F7F2ED0BF1D3B2E /* Stripe3DS2DemoUI-Debug.xcconfig */; + buildSettings = { + INFOPLIST_FILE = Stripe3DS2DemoUI/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.Stripe3DS2DemoUI; + PRODUCT_NAME = Stripe3DS2DemoUI; + SDKROOT = iphoneos; + }; + name = Debug; + }; + 437A50F3FA4CE0127490A271 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 4DB91F58CF23E72636C69C41 /* Project-Release.xcconfig */; + buildSettings = { + }; + name = Release; + }; + 7C7BC68E9581B5EDB7602612 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = D7A9D36FEF7E97CF735D82B8 /* Stripe3DS2DemoUITests-Release.xcconfig */; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = Stripe3DS2DemoUITests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.Stripe3DS2DemoUITests; + PRODUCT_NAME = Stripe3DS2DemoUITests; + SDKROOT = iphoneos; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Stripe3DS2DemoUI.app/Stripe3DS2DemoUI"; + TEST_TARGET_NAME = Stripe3DS2DemoUI; + }; + name = Release; + }; + 872D92C1EE444AB448EB744C /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = F5686FB8EFA3AE3B39F85BBC /* Stripe3DS2Tests-Release.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = Stripe3DS2Tests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.Stripe3DS2Tests; + PRODUCT_NAME = Stripe3DS2Tests; + SDKROOT = iphoneos; + }; + name = Release; + }; + 8CD9CB7B7E57B44EE65EF421 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 3A8CD68A006F970DDC54469A /* Project-Debug.xcconfig */; + buildSettings = { + }; + name = Debug; + }; + A084DA5D39CBD41FD4E5294E /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = F2BC7014D26158AC9D74CC67 /* Stripe3DS2Tests-Debug.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = Stripe3DS2Tests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.Stripe3DS2Tests; + PRODUCT_NAME = Stripe3DS2Tests; + SDKROOT = iphoneos; + }; + name = Debug; + }; + C34CBE7C8AA7B8ADB823B098 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 0E3C05BE55CF3086738AA182 /* Stripe3DS2DemoUITests-Debug.xcconfig */; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = Stripe3DS2DemoUITests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.Stripe3DS2DemoUITests; + PRODUCT_NAME = Stripe3DS2DemoUITests; + SDKROOT = iphoneos; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Stripe3DS2DemoUI.app/Stripe3DS2DemoUI"; + TEST_TARGET_NAME = Stripe3DS2DemoUI; + }; + name = Debug; + }; + D4CD9EDC1DA055683F427D24 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = F8F6FFED1A0966A49B78F252 /* Stripe3DS2-Release.xcconfig */; + buildSettings = { + INFOPLIST_FILE = Stripe3DS2/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-3ds2"; + PRODUCT_NAME = Stripe3DS2; + SDKROOT = iphoneos; + }; + name = Release; + }; + FCA19B8EBB1E9578613A65BE /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = A9CD7EC589C5CC6CE5CF9310 /* Stripe3DS2-Debug.xcconfig */; + buildSettings = { + INFOPLIST_FILE = Stripe3DS2/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-3ds2"; + PRODUCT_NAME = Stripe3DS2; + SDKROOT = iphoneos; + }; + name = Debug; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 271DFDD5ABBB719F3BD1AB56 /* Build configuration list for PBXNativeTarget "Stripe3DS2Tests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + A084DA5D39CBD41FD4E5294E /* Debug */, + 872D92C1EE444AB448EB744C /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 37237B2A5629D2257C454AD1 /* Build configuration list for PBXNativeTarget "Stripe3DS2DemoUI" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 380615C693D7BF5A8884139A /* Debug */, + 03D567A9F86C4B204A11D604 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 57B08C966701355644CF5CEB /* Build configuration list for PBXProject "Stripe3DS2" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 8CD9CB7B7E57B44EE65EF421 /* Debug */, + 437A50F3FA4CE0127490A271 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 5C9BBB6FD4175090CFCAA126 /* Build configuration list for PBXNativeTarget "Stripe3DS2" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + FCA19B8EBB1E9578613A65BE /* Debug */, + D4CD9EDC1DA055683F427D24 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 5F64A3CED17753DA2643AA63 /* Build configuration list for PBXNativeTarget "Stripe3DS2DemoUITests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + C34CBE7C8AA7B8ADB823B098 /* Debug */, + 7C7BC68E9581B5EDB7602612 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + +/* Begin XCRemoteSwiftPackageReference section */ + 176A29DF97FAFE939C7F667B /* XCRemoteSwiftPackageReference "ios-snapshot-test-case" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/uber/ios-snapshot-test-case"; + requirement = { + kind = upToNextMajorVersion; + minimumVersion = 8.0.0; + }; + }; +/* End XCRemoteSwiftPackageReference section */ + +/* Begin XCSwiftPackageProductDependency section */ + 0118969F6608A92583CC6C98 /* iOSSnapshotTestCase */ = { + isa = XCSwiftPackageProductDependency; + productName = iOSSnapshotTestCase; + }; +/* End XCSwiftPackageProductDependency section */ + }; + rootObject = 8D219187B704575AD3CD3EC5 /* Project object */; +} diff --git a/Stripe3DS2/Stripe3DS2.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/Stripe3DS2/Stripe3DS2.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2.xcscheme b/Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2.xcscheme new file mode 100644 index 00000000..14b59e6f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2.xcscheme @@ -0,0 +1,86 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2DemoUI.xcscheme b/Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2DemoUI.xcscheme new file mode 100644 index 00000000..b45aa0a9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2.xcodeproj/xcshareddata/xcschemes/Stripe3DS2DemoUI.xcscheme @@ -0,0 +1,107 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Stripe3DS2/Stripe3DS2/Info.plist b/Stripe3DS2/Stripe3DS2/Info.plist new file mode 100644 index 00000000..e1fe4cfb --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + FMWK + CFBundleShortVersionString + 1.0 + CFBundleVersion + $(CURRENT_PROJECT_VERSION) + + diff --git a/Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.h b/Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.h new file mode 100644 index 00000000..71159625 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.h @@ -0,0 +1,20 @@ +// +// NSData+JWEHelpers.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface NSData (JWEHelpers) + +- (nullable NSString *)_stds_base64URLEncodedString; +- (nullable NSString *)_stds_base64URLDecodedString; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.m b/Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.m new file mode 100644 index 00000000..c1bbab4e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSData+JWEHelpers.m @@ -0,0 +1,35 @@ +// +// NSData+JWEHelpers.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "NSData+JWEHelpers.h" + +#import "NSString+JWEHelpers.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation NSData (STDSJSONWebEncryption) + +- (nullable NSString *)_stds_base64URLEncodedString { + // ref. https://tools.ietf.org/html/draft-ietf-jose-json-web-signature-41#appendix-C + NSString *unpaddedBase64EncodedString = [[[[self base64EncodedStringWithOptions:0] + stringByTrimmingCharactersInSet:[NSCharacterSet characterSetWithCharactersInString:@"="]] // remove extra padding + stringByReplacingOccurrencesOfString:@"+" withString:@"-"] // replace "+" character w/ "-" + stringByReplacingOccurrencesOfString:@"/" withString:@"_"]; // replace "/" character w/ "_" + + return unpaddedBase64EncodedString; +} + +- (nullable NSString *)_stds_base64URLDecodedString { + return [[[self base64EncodedStringWithOptions:0] + stringByReplacingOccurrencesOfString:@"-" withString:@"+"] // replace "-" character w/ "+" + stringByReplacingOccurrencesOfString:@"_" withString:@"/"]; // replace "_" character w/ "/" +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.h b/Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.h new file mode 100644 index 00000000..70371f9f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.h @@ -0,0 +1,47 @@ +// +// NSDictionary+DecodingHelpers.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSJSONDecodable.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + Errors are populated according to the following rules: + - If the field is required and... + - the value is nil or empty -> STDSErrorCodeJSONFieldMissing + - the value is the wrong type -> STDSErrorCodeJSONFieldInvalid + - validator returns NO -> STDSErrorCodeJSONFieldInvalid + + - If the field is not required and... + - the value is nil -> valid, no error + - the value is empty -> STDSErrorCodeJSONFieldInvalid + - the value is the wrong type -> STDSErrorCodeJSONFieldInvalid + - validator returns NO -> STDSErrorCodeJSONFieldInvalid + */ +@interface NSDictionary (DecodingHelpers) + +/// Convenience method to extract an NSArray and populate it with instances of arrayElementType. +/// If isRequired is YES, returns nil without error if the key is not present +- (nullable NSArray *)_stds_arrayForKey:(NSString *)key arrayElementType:(Class)arrayElementType required:(BOOL)isRequired error:(NSError **)error; + +- (nullable NSURL *)_stds_urlForKey:(NSString *)key required:(BOOL)isRequired error:(NSError **)error; + +- (nullable NSDictionary *)_stds_dictionaryForKey:(NSString *)key required:(BOOL)isRequired error:(NSError **)error; + +- (nullable NSNumber *)_stds_boolForKey:(NSString *)key required:(BOOL)isRequired error:(NSError **)error; + +/// Convenience method that calls `_stpStringForKey:validator:required:error:`, passing nil for the validator argument +- (nullable NSString *)_stds_stringForKey:(NSString *)key required:(BOOL)isRequired error:(NSError **)error; + +- (nullable NSString *)_stds_stringForKey:(NSString *)key validator:(nullable BOOL (^)(NSString *))validatorBlock required:(BOOL)isRequired error:(NSError **)error; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.m b/Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.m new file mode 100644 index 00000000..4d2e2c09 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSDictionary+DecodingHelpers.m @@ -0,0 +1,147 @@ +// +// NSDictionary+DecodingHelpers.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "NSDictionary+DecodingHelpers.h" + +#import "NSError+Stripe3DS2.h" + +@implementation NSDictionary (DecodingHelpers) + +#pragma mark - NSArray + +- (nullable NSArray *)_stds_arrayForKey:(NSString *)key arrayElementType:(Class)arrayElementType required:(BOOL)isRequired error:(NSError * _Nullable __autoreleasing * _Nullable)error { + id value = self[key]; + + // Missing? + if (value == nil || ([value isKindOfClass:[NSArray class]] && ((NSArray *)value).count == 0)) { + if (isRequired && error) { + *error = [NSError _stds_missingJSONFieldError:key]; + } + return nil; + } + + // Invalid type or value? + if (![value isKindOfClass:[NSArray class]]) { + if (error) { + *error = [NSError _stds_invalidJSONFieldError:key]; + } + return nil; + } + + NSMutableArray *returnArray = [NSMutableArray new]; + for (id json in value) { + if (![json isKindOfClass:[NSDictionary class]]) { + if (error) { + *error = [NSError _stds_invalidJSONFieldError:key]; + } + return nil; + } + id element = [arrayElementType decodedObjectFromJSON:json error:error]; + if (element) { + [returnArray addObject:element]; + } + } + + return returnArray; +} + +#pragma mark - NSURL + +- (nullable NSURL *)_stds_urlForKey:(NSString *)key required:(BOOL)isRequired error:(NSError * _Nullable __autoreleasing *)error { + NSString *urlRawString = [self _stds_stringForKey:key validator:^BOOL (NSString *value) { + return [NSURL URLWithString:value] != nil; + } required:isRequired error:error]; + + if (urlRawString) { + return [NSURL URLWithString:urlRawString]; + } else { + return nil; + } +} + +#pragma mark - NSDictionary + +- (nullable NSDictionary *)_stds_dictionaryForKey:(NSString *)key required:(BOOL)isRequired error:(NSError * _Nullable __autoreleasing *)error { + id value = self[key]; + + // Missing? + if (value == nil) { + if (error && isRequired) { + *error = [NSError _stds_missingJSONFieldError:key]; + } + return nil; + } + + // Invalid type? + if (![value isKindOfClass:[NSDictionary class]]) { + if (error) { + *error = [NSError _stds_invalidJSONFieldError:key]; + } + return nil; + } + + return value; +} + +#pragma mark - NSString + +- (nullable NSString *)_stds_stringForKey:(NSString *)key required:(BOOL)isRequired error:(NSError * _Nullable __autoreleasing * _Nullable)error { + return [self _stds_stringForKey:key validator:nil required:isRequired error:error]; +} + +- (nullable NSString *)_stds_stringForKey:(NSString *)key validator:(nullable BOOL (^)(NSString * _Nonnull))validatorBlock required:(BOOL)isRequired error:(NSError * _Nullable __autoreleasing * _Nullable)error { + id value = self[key]; + + // Missing? + if (value == nil || ([value isKindOfClass:[NSString class]] && ((NSString *)value).length == 0)) { + if (error) { + if (isRequired) { + *error = [NSError _stds_missingJSONFieldError:key]; + } else if (value != nil) { + *error = [NSError _stds_invalidJSONFieldError:key]; + } + } + return nil; + } + + // Invalid type or value? + if (![value isKindOfClass:[NSString class]] || (validatorBlock && !validatorBlock(value))) { + if (error) { + *error = [NSError _stds_invalidJSONFieldError:key]; + } + return nil; + } + + return value; +} + +#pragma mark - NSURL + +- (NSNumber *)_stds_boolForKey:(NSString *)key required:(BOOL)isRequired error:(NSError * _Nullable __autoreleasing *)error { + id value = self[key]; + + // Missing? + if (value == nil) { + if (error && isRequired) { + *error = [NSError _stds_missingJSONFieldError:key]; + } + return nil; + } + + // Invalid type? + if (![value isKindOfClass:[NSNumber class]]) { + if (error) { + *error = [NSError _stds_invalidJSONFieldError:key]; + } + return nil; + } + + return value; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.h b/Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.h new file mode 100644 index 00000000..6af09054 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.h @@ -0,0 +1,32 @@ +// +// NSError+Stripe3DS2.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface NSError (Stripe3DS2) + + +/// Represents an error where a JSON field value is not valid (e.g. expected 'Y' or 'N' but received something else). ++ (instancetype)_stds_invalidJSONFieldError:(NSString *)fieldName; + +/// Represents an error where a JSON field was either required or conditionally required but missing, empty, or null. ++ (instancetype)_stds_missingJSONFieldError:(NSString *)fieldName; + +/// Represents an error where a network request timed out. ++ (instancetype)_stds_timedOutError; + +// We explicitly do not provide any more info here based on security recommendations +// "the recipient MUST NOT distinguish between format, padding, and length errors of encrypted keys" +// https://tools.ietf.org/html/rfc7516#section-11.5 ++ (instancetype)_stds_jweError; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.m b/Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.m new file mode 100644 index 00000000..cd60c4cc --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSError+Stripe3DS2.m @@ -0,0 +1,38 @@ +// +// NSError+Stripe3DS2.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "NSError+Stripe3DS2.h" +#import "STDSLocalizedString.h" + +#import "STDSStripe3DS2Error.h" + +@implementation NSError (Stripe3DS2) + ++ (instancetype)_stds_invalidJSONFieldError:(NSString *)fieldName { + return [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:STDSErrorCodeJSONFieldInvalid + userInfo:@{STDSStripe3DS2ErrorFieldKey: fieldName}]; +} + ++ (instancetype)_stds_missingJSONFieldError:(NSString *)fieldName { + return [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:STDSErrorCodeJSONFieldMissing + userInfo:@{STDSStripe3DS2ErrorFieldKey: fieldName}]; +} + ++ (instancetype)_stds_timedOutError { + return [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:STDSErrorCodeTimeout + userInfo:@{NSLocalizedDescriptionKey : STDSLocalizedString(@"Timeout", @"Error description for when a network request times out. English value is as required by UL certification.")}]; +} + ++ (instancetype)_stds_jweError { + return [[NSError alloc] initWithDomain:STDSStripe3DS2ErrorDomain code:STDSErrorCodeDecryptionVerification userInfo:nil]; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.h b/Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.h new file mode 100644 index 00000000..66596c81 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.h @@ -0,0 +1,53 @@ +// +// NSLayoutConstraint+LayoutSupport.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface NSLayoutConstraint (LayoutSupport) + +/** + Provides an NSLayoutConstraint where the `NSLayoutAttributeTop` is equal for both views, with a multiplier of 1, and a constant of 0. + + @param view1 The view to constrain. + @param view2 The view to constraint to. + @return An NSLayoutConstraint that is constraining the first view to the second at the top. + */ ++ (NSLayoutConstraint *)_stds_topConstraintWithItem:(id)view1 toItem:(id)view2; + +/** + Provides an NSLayoutConstraint where the `NSLayoutAttributeLeft` is equal for both views, with a multiplier of 1, and a constant of 0. + + @param view1 The view to constrain. + @param view2 The view to constraint to. + @return An NSLayoutConstraint that is constraining the first view to the second on the left. + */ ++ (NSLayoutConstraint *)_stds_leftConstraintWithItem:(id)view1 toItem:(id)view2; + +/** + Provides an NSLayoutConstraint where the `NSLayoutAttributeRight` is equal for both views, with a multiplier of 1, and a constant of 0. + + @param view1 The view to constrain. + @param view2 The view to constraint to. + @return An NSLayoutConstraint that is constraining the first view to the second on the right. + */ ++ (NSLayoutConstraint *)_stds_rightConstraintWithItem:(id)view1 toItem:(id)view2; + +/** + Provides an NSLayoutConstraint where the `NSLayoutAttributeBottom` is equal for both views, with a multiplier of 1, and a constant of 0. + + @param view1 The view to constrain. + @param view2 The view to constraint to. + @return An NSLayoutConstraint that is constraining the first view to the second at the bottom. + */ ++ (NSLayoutConstraint *)_stds_bottomConstraintWithItem:(id)view1 toItem:(id)view2; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.m b/Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.m new file mode 100644 index 00000000..8b73b5fe --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSLayoutConstraint+LayoutSupport.m @@ -0,0 +1,30 @@ +// +// NSLayoutConstraint+LayoutSupport.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "NSLayoutConstraint+LayoutSupport.h" + +@implementation NSLayoutConstraint (LayoutSupport) + + ++ (NSLayoutConstraint *)_stds_topConstraintWithItem:(id)view1 toItem:(id)view2 { + return [NSLayoutConstraint constraintWithItem:view1 attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:view2 attribute:NSLayoutAttributeTop multiplier:1 constant:0]; +} + ++ (NSLayoutConstraint *)_stds_leftConstraintWithItem:(id)view1 toItem:(id)view2 { + return [NSLayoutConstraint constraintWithItem:view1 attribute:NSLayoutAttributeLeft relatedBy:NSLayoutRelationEqual toItem:view2 attribute:NSLayoutAttributeLeft multiplier:1 constant:0]; +} + ++ (NSLayoutConstraint *)_stds_rightConstraintWithItem:(id)view1 toItem:(id)view2 { + return [NSLayoutConstraint constraintWithItem:view1 attribute:NSLayoutAttributeRight relatedBy:NSLayoutRelationEqual toItem:view2 attribute:NSLayoutAttributeRight multiplier:1 constant:0]; +} + ++ (NSLayoutConstraint *)_stds_bottomConstraintWithItem:(id)view1 toItem:(id)view2 { + return [NSLayoutConstraint constraintWithItem:view1 attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:view2 attribute:NSLayoutAttributeBottom multiplier:1 constant:0]; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.h b/Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.h new file mode 100644 index 00000000..aa019bf0 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.h @@ -0,0 +1,19 @@ +// +// NSString+EmptyChecking.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface NSString (EmptyChecking) + ++ (BOOL)_stds_isStringEmpty:(NSString *)string; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.m b/Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.m new file mode 100644 index 00000000..bf749718 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSString+EmptyChecking.m @@ -0,0 +1,25 @@ +// +// NSString+EmptyChecking.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "NSString+EmptyChecking.h" + +@implementation NSString (EmptyChecking) + ++ (BOOL)_stds_isStringEmpty:(NSString *)string { + if (string.length == 0) { + return YES; + } + + if(![string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]].length) { + return YES; + } + + return NO; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.h b/Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.h new file mode 100644 index 00000000..acb9afde --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.h @@ -0,0 +1,21 @@ +// +// NSString+JWEHelpers.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface NSString (JWEHelpers) + +- (nullable NSString *)_stds_base64URLEncodedString; +- (nullable NSString *)_stds_base64URLDecodedString; +- (nullable NSData *)_stds_base64URLDecodedData; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.m b/Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.m new file mode 100644 index 00000000..2c8d5ad8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/NSString+JWEHelpers.m @@ -0,0 +1,54 @@ +// +// NSString+JWEHelpers.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "NSString+JWEHelpers.h" + +#import "NSData+JWEHelpers.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation NSString (JWEHelpers) + +- (nullable NSString *)_stds_base64URLEncodedString { + return [[self dataUsingEncoding:NSUTF8StringEncoding] _stds_base64URLEncodedString]; +} + +- (nullable NSString *)_stds_base64URLDecodedString { + NSData *data = [self _stds_base64URLDecodedData]; + return data != nil ? [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding] : nil; +} + +// ref. https://tools.ietf.org/html/draft-ietf-jose-json-web-signature-41#appendix-C +- (nullable NSData *)_stds_base64URLDecodedData { + NSCharacterSet *illegalBase64Chars = [NSCharacterSet characterSetWithCharactersInString:@"+/ \n"]; // TC_SDK_10556_001 & TC_SDK_10557_001 & TC_SDK_10558_001 & TC_SDK_10559_001 + if ([self hasSuffix:@"="] || [self rangeOfCharacterFromSet:illegalBase64Chars].location != NSNotFound) { + return nil; // invalid base64url string TC_SDK_10554_001 & TC_SDK_10555_001 + } + NSMutableString *decodedString = [[[self stringByReplacingOccurrencesOfString:@"-" withString:@"+"] // replace "-" character w/ "+" + stringByReplacingOccurrencesOfString:@"_" withString:@"/"] mutableCopy]; // replace "_" character w/ "/"]; + + switch (decodedString.length % 4) { + case 0: + break; // no padding needed + case 2: + [decodedString appendString:@"=="]; // pad with 2 + break; + case 3: + [decodedString appendString:@"="]; // pad with 1 + break; + default: + return nil; // invalid base64url string + + } + + return [[NSData alloc] initWithBase64EncodedString:decodedString options:0]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/amex.der b/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/amex.der new file mode 100644 index 0000000000000000000000000000000000000000..61a1cfd1f0490b32bdbe4e29d7a11602cbc2458f GIT binary patch literal 1237 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 literal 0 HcmV?d00001 diff --git a/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/mastercard.der b/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/mastercard.der new file mode 100644 index 0000000000000000000000000000000000000000..6e136e13312275677887b3949a1212807ae4ba48 GIT binary patch literal 1465 zcmXqLV%=)c#5{KaGZP~dlYrWii5b6V{a>z`xic-q@LGTYFB_*;n@8JsUPeZ4RtAGA zLv903Hs(+kHesgFU_)U8K@f*an9Db@xFog6Ik70kP{TkCB*-l+j}TPwOi9fv$tXZu*;J)OU7b)WK_?YT4z^28wtXOp7|kb@*#lQNM%a z;)6j47n~QY`r+^(%=Uey=?Ol2APQWV%D0P zZcMYXZLe|pE}48K*gE~f-3Y@}*CoM5TUWR%MI}isIlR61>w~yAA`ITg4YT>uwOape zyQ_Tlo{vDs)-bE@8?4uMY;9$2s8Ov>&w9U1No%YA6}|TiCQEK`?r2)0|Moi5oba9- zjJH_=qOMK|bga_ZSz7RMy1>eBHnTf=nV1Al4ug@0BpDtUY7Aa||$*HDhsm2Dzh9+rg zrm3l^CKgE+CZ^ zah+t-Y^loyjHUw)|I zknspz7Cz-e?UcpS_;OcA|5?i3bgS&V_mO(Vg{g86*X^^}>!Mj3*vKHw6z;Yv$obuzNtN;cw}*&s`0a*%3s@?y_Gz=-Ru>oeco|d=X|UconHG6N+hu_drWnSvCL)ir0GS39XO{*pxD)5$6dHLp7f9lgE)1HXN>gwd#+Ppt0ymsB>p4=-xR`sc9 z{Mo;7`)99?39R?73$x2yv69rQm7Ofd@=ZlMXJ=bRy=s(2uK209ZQJt#f>=M)FosTe zk~KBTGe35=fX=<6^54%W+J5?^Xc`?KlHH>?P5Sw}{#%L3OXT*t-IiJSLGbaLy6wyv Z$pt^d{!ZZZv~EmwSjr^LyGZbx8UUfUE+GH_ literal 0 HcmV?d00001 diff --git a/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/ul-test.der b/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/ul-test.der new file mode 100644 index 0000000000000000000000000000000000000000..11e5f29b3be5d00b2cf10c91929e8b9b913730d7 GIT binary patch literal 1578 zcmXqLVpB6{V&PlB%*4pV#KxFWq`|h!fR~L^tIebBJ1-+6H!FidjUl%ICmVAp3!5;L zpO2xYfjWr8#iJ1FqYzS*m{**bT#}ierx2W)Tw0V_QmK%VS(aH06fsmVkOL{@<`D-e z4OTFA(FH3?RS5F;4^eP-gsEW0slq@`oY&COz|hFh$iT$J&?rir*T?{gYanSTZXgP> zg_(!1xTGkvAXT?CN3SHcxWu4|NeS66jI0dIO-%d@KyfanCMHIPRXabQo`34Xu0-+D zExlUva#l>eeBx{2v=>o#Ma``}n-83P#I@?oN_jq&sJ8Iwkxtbvmu5O0b(ffSf8IPz z$?7Cd-O2ajE|iH(EL1aR57||ySu3wve~Pg{V{+QVptWBCNWKJS_dQR{BrT z#W_lAj#_2jd>(tF+oSKP-;ouc=PZxhD7?JaM8jWVXz(sQn_7V@MT8@$CSf7Z*prn z&ZKFv6-9M;=JShYn-`>3n2$w_MP%03nmRb&ddmBF(GFfU;zNk zG>i;;qLg-sy~7h6wI7uI9y_+P*6ki*?e}x zT7Iz@t$p*v~I@fof=2IJmntF5#Pc)MSV^5_RTl%FD&ae+f^QExo!HBSGrHH z+=}abAK;unxBl$q|CcX6k4(|N6*|>uS>21S3CAzk)xY}Nt1c^%r6Az-^!2wLnS!>j z)5_|l7M``Y)oeNPY2}{z>kF;^)bndK?#p7^_j>Vr0qoATi{ET1pEc{|U&|W- z3->tYaDVhjHc#;j`PwHiuVcdPv*opuzDpJCUUHx}NcG+2=-%s17hVPJ6TeqwaC7aV z(1Z;J?{_=9FwM*2<0e_`6DKPHgq7dR-|6s|Qhgy0TxI z$jm>zu5Yo_hT4BwsduVn(*(aRp2~N`Cvi^stjeN#&W`lkN>!I3|8>Gz}y4ehraX88qOJ##l)UF^&6#Jfp*q9m5xNM6gT`kiUv dZH?xLYoB|y_O7}APsU??|HLivPSg2*006^ahiCu* literal 0 HcmV?d00001 diff --git a/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/visa.der b/Stripe3DS2/Stripe3DS2/Resources/CertificateFiles/visa.der new file mode 100644 index 0000000000000000000000000000000000000000..a77735dfa6e224cfa40334bb41a402771358f9a8 GIT binary patch literal 1458 zcmXqLVqIs@#4>FGGZP~dlK^|x$yo=l9^BkpUN_DD3jbOIUN%mxHjlRNyo`+8tPBQ) zhTI06Y|No7Y{E>T!G^pB+#n8@FiV(cu%n^Afi8&4Evy!nS)8cgnOBlpl$ThNnV*-K zqY#{0RF;{Xs^C~$oS&Qt7By5dPylIQ7M28SNOjK7%}p%=D)KBYF3rqKS8#STkQ3)M zG&eLfFaiQ&LxU)BULyk_*Vw=S%r$6SXeePI1~P_SnBOBaJtHSEFQr%^C^0WN!%)zG zA0)#e%;B7$lV6mWl22U^s2V6kJRoD7Qf#DGRGg@nnwMNuSpf31UKubH^pf*)4VoC0 zkVBG@m4Ug5k)Hu5&c)Qk$jGqie!Jx7^BZelz9>B6e9d^#tEW!3Rc6Q4X2%{{zFo4q zJ(4|3Y|gBi?>g$2c(E*aV93r}_~2qf%#u?j1#-Jf9Zh3wyqSed8urDS#X`cw2BP@4w4i)ZT3Vu&!ZCg4d#1`PT&5&tGOWH+s4!`Ney7{$<^_ z4Z91D*_LQY#|x||*{Ew|6dSUi_m^7shTa9+nV1#%5$>i8Kf|P=)ag7~3Q>N=gc>^!4+Tiwod^4(1u?!Id$!DM6K` z<`w1VgX2d(6&OjmddaCp2J#@gm05r(szK!bVXaU-eJ86bhq;p$e0Q(B-rwG3 zzynes%*gnkg~@=ypz#!lE6>t+$e?k*fyM&01uAfBTrsTyrEk6Dq8x+9NedeL7c};? zHFoBtB!b+Iq1wvW!cspu&lYFS&`Gup4c5sog{UBy*>v)4L7CJZ=*yDKw9MqhlGLEo zvV0&7OsYPa#U%!OAiwZ~yvD-J#Ja#h9mE%8Q818e!k@flK??a;#8^cB%+7M&>K$>t z{QsQ`eKX&a7!;k(+F<*$D`#5PLr zd%IkNOE{<5&bj5&t`;>;@o*#a% zW~9xpy22j1iglla+2ilKPoLWJHH=sD)AjjBt{FOamYQgo{NIsN&NN5*TBJM2jh#h8 z9i}$|E=iwAurJp<{Fd=X*_`SN0g`ez8;v*_m|bM&RX8NiC>L`nFxjc#<&_p(qj1~a zWKj{9SO4X!4i7Hxk0@&sVSU@w^sFzyR&! VxmlLhu5Qh(dR3q=eoivq001R~|EB-| literal 0 HcmV?d00001 diff --git a/Stripe3DS2/Stripe3DS2/Resources/Images/Chevron@3x.png b/Stripe3DS2/Stripe3DS2/Resources/Images/Chevron@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..75ec6e4be309010192af53c7cddc38fcc73a3127 GIT binary patch literal 454 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 literal 0 HcmV?d00001 diff --git a/Stripe3DS2/Stripe3DS2/Resources/Images/discover-logo.png b/Stripe3DS2/Stripe3DS2/Resources/Images/discover-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..09ca3812d06ea40c08f804a9a987fb801cf1bb5f GIT binary patch literal 3220 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 literal 0 HcmV?d00001 diff --git a/Stripe3DS2/Stripe3DS2/Resources/Images/mastercard-logo@3x.png b/Stripe3DS2/Stripe3DS2/Resources/Images/mastercard-logo@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..1dffc3835699f2ce57950ff752be611cabe4a5f8 GIT binary patch literal 3860 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 W{mWrvxU%2?0000j67##+{2Vpo| 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 D6gm&x literal 0 HcmV?d00001 diff --git a/Stripe3DS2/Stripe3DS2/Resources/Images/visa-white-logo@3x.png b/Stripe3DS2/Stripe3DS2/Resources/Images/visa-white-logo@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..3a0dccda3fa1ae6bf184bc965c84d33a39acc44a GIT binary patch literal 2088 zcmV+@2-o+CP)sU-+ysKypU)Kay#)V{UE7*uPht<*LQwM>dpK}t2P)?O4DH{Dtac3{un;6F;iIQC{T2BI+gyvem)Y ziR4Nrv74&hhL4|pAi0m_Dr@;4oNzug!3-P{E3`x8#6b)~CkNj&l53GKZlqcZb4u6h zTRzsPkb+6*iS}rRt{8~v z*q}~O55%Kyew-CMaazEXm>2qcMyXFosf#G!;%r26o^tWM#ir5mi15{;0)`?r=P)YA zI(?Cd5A@Crt+>B z#QQbL>oQ!OCyiPk2$2wD7+))yE>MIywb~IOfUau4_3+Y#R&!TurfEdTPFz5gcz+Qq z_A-x(KbvLmsLE73t%G<=GJEz}2VYIbiE~19VKJgjvJ0QtR%_lx==Q)jzWfO!{}F>= zanQxrkcm;Zj9)KVRDOpn1Aa1Jq=iXi2>F=~-$)PlRFZdzi?0?Nqy9mS5S(~~NSTTv zntW{eOcyCY7F*fg4jsBPJEpjYdn@UffkBa(ZHTDb7$s@B`dHVRe#;O>b{AhXUuXxZ zeM1lLc9M62PrRj5p<2crmP2T7?nJebK9BZaX(>2Mk0J;(g<;m9lL%$vh8j!o_4AV;Z-Kff4 zMQ3!VM>aGZ^*!PUuGQvF48tV?Y?dHqCHZ2GppOPdlL&UWt{FwRkiL{0YG_Mv9kobG z)q1|WhA>wP)J?DW8nMc5qJ}VxVH&w!de0Z71nx+3K$IZTeB~`2^m}nA@kA~>{=Yq) z1QUpVk`gVn+C($6YVT|@$_HUA$z5F7XKHQk#As5;y#?lSM=tb;JSN zu|2}bHkT*#U=6`J(dLCdO@MU>1n?Z$b-dn zrb)L)sND;UDvkje`XjF>)ke);hKo!fkrGLnkF1>-&b-jeMb{Bb>0cW)%V=*cI_VxF zjZ1%^yR{R!iQq)RjSCy9YNKd|!2&Bu6s~_$!80ZsB}~3oFuLRp4zM~;o!C3CgLg1B zAW`bmeMb-PEC5$A7d17nay)z4snBD-(neLco)T@G*hLJ#pFtT$9z`3O+Y}j?fi~WY zONc;UzHU)#1=AusO0#idg!sB|Ka1MOQ;#B_y5X%9kcJIdj>TAtWMq)3@Kb8?1kLx8 zp-||K)u0ypMaj)&ZU({Jcx@TtRU<#zl#A9Cc~5n0oR}bjPGEPlA{+X_je-H8+Nf@z zvS5CG5J{RJwTB1WII*2*x)H@7v?b*Zdw3Uxyi=ZzFRC*MS|hF9(_>N=~ zi|~QGm5=lgT{u=UpjF2Oc2kqgbi1VXN3H7Ybq;Knwr`pKiG;E-+;yAhy2&ikdwGD6 z6?`%12RBut{4vcJE}7CGE&M!(m>9a-_ZY%3R}tAweUIDJ#)&%9MzhjUq}jKF(*=yP z^x#8$9-L1ywbovt-N`ogl0_SLD@NV+zsE0VVtB4jq7BlFs`Z@p)k1GYi{_@&Kp)K~ z67p;8q9Ql>)tpw$#~E{oas6rEo34CshE#pXykX>$8p<%TD6wg>VgEq%l`pvcM2bz S*;>~C0000 + +@class STDSChallengeRequestParameters; +@class STDSErrorMessage; +@protocol STDSChallengeResponse; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSACSNetworkingManager : NSObject + +- (instancetype)initWithURL:(NSURL *)acsURL + sdkContentEncryptionKey:(NSData *)sdkCEK + acsContentEncryptionKey:(NSData *)acsCEK + acsTransactionIdentifier:(NSString *)acsTransactionID; + +- (void)submitChallengeRequest:(STDSChallengeRequestParameters *)request withCompletion:(void (^)(id _Nullable, NSError * _Nullable))completion; + +- (void)sendErrorMessage:(STDSErrorMessage *)errorMessage; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSACSNetworkingManager.m b/Stripe3DS2/Stripe3DS2/STDSACSNetworkingManager.m new file mode 100644 index 00000000..f7c8de55 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSACSNetworkingManager.m @@ -0,0 +1,169 @@ +// +// STDSACSNetworkingManager..m +// Stripe3DS2 +// +// Created by Cameron Sabol on 4/3/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSACSNetworkingManager.h" + +#import "STDSChallengeRequestParameters.h" +#import "STDSChallengeResponseObject.h" +#import "STDSJSONEncoder.h" +#import "STDSJSONWebEncryption.h" +#import "STDSStripe3DS2Error.h" +#import "STDSErrorMessage+Internal.h" +#import "NSError+Stripe3DS2.h" + +NS_ASSUME_NONNULL_BEGIN + +/// [Req 239] requires us to abort if the ACS does not respond with the CRes message within 10 seconds. +static const NSTimeInterval kTimeoutInterval = 10; + +@implementation STDSACSNetworkingManager { + NSURL *_acsURL; + NSData *_sdkContentEncryptionKey; + NSData *_acsContentEncryptionKey; + NSString *_acsTransactionIdentifier; + + NSURLSession *_urlSession; + NSURLSessionTask * _Nullable _currentTask; +} + +- (instancetype)initWithURL:(NSURL *)acsURL + sdkContentEncryptionKey:(NSData *)sdkCEK + acsContentEncryptionKey:(NSData *)acsCEK + acsTransactionIdentifier:(nonnull NSString *)acsTransactionID { + self = [super init]; + if (self) { + _acsURL = acsURL; + _sdkContentEncryptionKey = sdkCEK; + _acsContentEncryptionKey = acsCEK; + _acsTransactionIdentifier = [acsTransactionID copy]; + _urlSession = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration ephemeralSessionConfiguration] + delegate:nil + delegateQueue:[NSOperationQueue mainQueue]]; + } + + return self; +} + +- (void)dealloc { + [_urlSession finishTasksAndInvalidate]; +} + +- (void)submitChallengeRequest:(STDSChallengeRequestParameters *)request withCompletion:(void (^)(id _Nullable, NSError * _Nullable))completion { + if (_currentTask != nil) { + dispatch_async(dispatch_get_main_queue(), ^{ + completion(nil, [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:STDSErrorCodeAssertionFailed + userInfo:@{@"assertion": [NSString stringWithFormat:@"%@ is not intended to handle multiple concurrent tasks.", NSStringFromClass([self class])]}]); + }); + return; + } + + NSDictionary *requestJSON = [STDSJSONEncoder dictionaryForObject:request]; + NSError *encryptionError = nil; + NSString *encryptedRequest = [STDSJSONWebEncryption directEncryptJSON:requestJSON + withContentEncryptionKey:_sdkContentEncryptionKey + forACSTransactionID:_acsTransactionIdentifier + error:&encryptionError]; + + if (encryptedRequest != nil) { + NSMutableURLRequest *urlRequest = [[NSMutableURLRequest alloc] initWithURL:_acsURL]; + urlRequest.HTTPMethod = @"POST"; + urlRequest.timeoutInterval = kTimeoutInterval; + [urlRequest setValue:@"application/jose;charset=UTF-8" forHTTPHeaderField:@"Content-Type"]; + __weak __typeof(self) weakSelf = self; + NSURLSessionUploadTask *requestTask = [_urlSession uploadTaskWithRequest:[urlRequest copy] fromData:[encryptedRequest dataUsingEncoding:NSUTF8StringEncoding] completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) { + + __typeof(self) strongSelf = weakSelf; + if (strongSelf == nil) { + return; + } + + strongSelf->_currentTask = nil; + + if (data != nil) { + NSError *decryptionError = nil; + NSDictionary *decrypted = [STDSJSONWebEncryption decryptData:data + withContentEncryptionKey:strongSelf->_acsContentEncryptionKey + error:&decryptionError]; + if (decrypted != nil) { + NSError *challengeResponseError = nil; + id response = [strongSelf decodeJSON:decrypted error:&challengeResponseError]; + completion(response, challengeResponseError); + } else { + completion(nil, decryptionError); + } + } else { + if (error.code == NSURLErrorTimedOut) { + // We convert timeout errors for convenience, since the SDK must treat them differently from generic network errors. + error = [NSError _stds_timedOutError]; + } + completion(nil, error); + } + + }]; + _currentTask = requestTask; + [requestTask resume]; + } else { + dispatch_async(dispatch_get_main_queue(), ^{ + completion(nil, encryptionError); + }); + } +} + +- (void)sendErrorMessage:(STDSErrorMessage *)errorMessage { + NSDictionary *requestJSON = [STDSJSONEncoder dictionaryForObject:errorMessage]; + NSMutableURLRequest *urlRequest = [[NSMutableURLRequest alloc] initWithURL:_acsURL]; + urlRequest.HTTPMethod = @"POST"; + [urlRequest setValue:@"application/JSON; charset = UTF-8" forHTTPHeaderField:@"Content-Type"]; + NSURLSessionUploadTask *requestTask = [_urlSession uploadTaskWithRequest:[urlRequest copy] + fromData:[NSJSONSerialization dataWithJSONObject:requestJSON options:0 error:nil] + completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) { + // no-op + }]; + [requestTask resume]; +} + +#pragma mark - Helpers + +/** + Returns an STDSChallengeResponseObject instance decoded from the given dict, or populates the error argument. + */ +- (nullable id)decodeJSON:(NSDictionary *)dict error:(NSError * _Nullable *)outError { + NSString *kErrorMessageType = @"Erro"; + NSString *kChallengeResponseType = @"CRes"; + NSString *messageType = dict[@"messageType"]; + NSError *error; + id decodedObject; + + if ([messageType isEqualToString:kErrorMessageType]) { + // Error message type + STDSErrorMessage *errorMessage = [STDSErrorMessage decodedObjectFromJSON:dict error:&error]; + if (errorMessage) { + error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:STDSErrorCodeReceivedErrorMessage + userInfo:@{STDSStripe3DS2ErrorMessageErrorKey: errorMessage}]; + } + } else if ([messageType isEqualToString:kChallengeResponseType]) { + // CRes message type + decodedObject = [STDSChallengeResponseObject decodedObjectFromJSON:dict error:&error]; + } else { + // Unknown message type + error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:STDSErrorCodeUnknownMessageType + userInfo:nil]; + } + + if (error && outError) { + *outError = error; + } + return decodedObject; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.h b/Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.h new file mode 100644 index 00000000..78b2c491 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.h @@ -0,0 +1,20 @@ +// +// STDSAuthenticationResponseObject.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 5/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSAuthenticationResponse.h" +#import "STDSJSONDecodable.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSAuthenticationResponseObject : NSObject + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.m b/Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.m new file mode 100644 index 00000000..4a9bd175 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSAuthenticationResponseObject.m @@ -0,0 +1,100 @@ +// +// STDSAuthenticationResponseObject.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 5/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSAuthenticationResponseObject.h" + +#import "NSDictionary+DecodingHelpers.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSAuthenticationResponseObject + +@synthesize acsOperatorID = _acsOperatorID; +@synthesize acsReferenceNumber = _acsReferenceNumber; +@synthesize acsSignedContent = _acsSignedContent; +@synthesize acsTransactionID = _acsTransactionID; +@synthesize acsURL = _acsURL; +@synthesize cardholderInfo = _cardholderInfo; +@synthesize status = _status; +@synthesize challengeRequired = _challengeRequired; +@synthesize directoryServerReferenceNumber = _directoryServerReferenceNumber; +@synthesize directoryServerTransactionID = _directoryServerTransactionID; +@synthesize protocolVersion = _protocolVersion; +@synthesize sdkTransactionID = _sdkTransactionID; +@synthesize threeDSServerTransactionID = _threeDSServerTransactionID; +@synthesize willUseDecoupledAuthentication = _willUseDecoupledAuthentication; + ++ (nullable instancetype)decodedObjectFromJSON:(nullable NSDictionary *)json error:(NSError **)outError { + if (json == nil) { + return nil; + } + STDSAuthenticationResponseObject *response = [[self alloc] init]; + NSError *error = nil; + + response->_threeDSServerTransactionID = [[json _stds_stringForKey:@"threeDSServerTransID" required:YES error:&error] copy]; + NSString *transStatusString = [json _stds_stringForKey:@"transStatus" required:NO error:&error]; + response->_status = [self statusTypeForString:transStatusString]; + response->_challengeRequired = (response->_status == STDSACSStatusTypeChallengeRequired); + response->_willUseDecoupledAuthentication = [[json _stds_boolForKey:@"acsDecConInd" required:NO error:&error] boolValue]; + response->_acsOperatorID = [[json _stds_stringForKey:@"acsOperatorID" required:NO error:&error] copy]; + response->_acsReferenceNumber = [[json _stds_stringForKey:@"acsReferenceNumber" required:NO error:&error] copy]; + response->_acsSignedContent = [[json _stds_stringForKey:@"acsSignedContent" required:NO error:&error] copy]; + response->_acsTransactionID = [[json _stds_stringForKey:@"acsTransID" required:YES error:&error] copy]; + response->_acsURL = [json _stds_urlForKey:@"acsURL" required:NO error:&error]; + response->_cardholderInfo = [[json _stds_stringForKey:@"cardholderInfo" required:NO error:&error] copy]; + response->_directoryServerReferenceNumber = [[json _stds_stringForKey:@"dsReferenceNumber" required:NO error:&error] copy]; + response->_directoryServerTransactionID = [[json _stds_stringForKey:@"dsTransID" required:NO error:&error] copy]; + response->_protocolVersion = [[json _stds_stringForKey:@"messageVersion" required:YES error:&error] copy]; + response->_sdkTransactionID = [[json _stds_stringForKey:@"sdkTransID" required:YES error:&error] copy]; + + if (error != nil) { + if (outError != nil) { + *outError = error; + } + + return nil; + } + + return response; +} + ++ (STDSACSStatusType)statusTypeForString:(NSString *)statusString { + if ([statusString isEqualToString:@"Y"]) { + return STDSACSStatusTypeAuthenticated; + } + if ([statusString isEqualToString:@"C"]) { + return STDSACSStatusTypeChallengeRequired; + } + if ([statusString isEqualToString:@"D"]) { + return STDSACSStatusTypeDecoupledAuthentication; + } + if ([statusString isEqualToString:@"N"]) { + return STDSACSStatusTypeNotAuthenticated; + } + if ([statusString isEqualToString:@"A"]) { + return STDSACSStatusTypeProofGenerated; + } + if ([statusString isEqualToString:@"U"]) { + return STDSACSStatusTypeError; + } + if ([statusString isEqualToString:@"R"]) { + return STDSACSStatusTypeRejected; + } + if ([statusString isEqualToString:@"I"]) { + return STDSACSStatusTypeInformationalOnly; + } + return STDSACSStatusTypeUnknown; +} + +@end + +id _Nullable STDSAuthenticationResponseFromJSON(NSDictionary *json) { + return [STDSAuthenticationResponseObject decodedObjectFromJSON:json error:NULL]; +} + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSBrandingView.h b/Stripe3DS2/Stripe3DS2/STDSBrandingView.h new file mode 100644 index 00000000..310e18a6 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSBrandingView.h @@ -0,0 +1,23 @@ +// +// STDSBrandingView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSBrandingView: UIView + +/// The issuer image to present in the branding view. +@property (nonatomic, strong) UIImage *issuerImage; + +/// The payment system image to present in the branding view. +@property (nonatomic, strong) UIImage *paymentSystemImage; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSBrandingView.m b/Stripe3DS2/Stripe3DS2/STDSBrandingView.m new file mode 100644 index 00000000..9783cc76 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSBrandingView.m @@ -0,0 +1,132 @@ +// +// STDSBrandingView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSBrandingView.h" +#import "STDSStackView.h" +#import "UIView+LayoutSupport.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSBrandingView() + +@property (nonatomic, strong) STDSStackView *stackView; + +@property (nonatomic, strong) UIImageView *issuerImageView; +@property (nonatomic, strong) UIImageView *paymentSystemImageView; + +@property (nonatomic, strong) UIView *issuerView; +@property (nonatomic, strong) UIView *paymentSystemView; + +@end + +@implementation STDSBrandingView + +static const CGFloat kBrandingViewBottomPadding = 24; +static const CGFloat kBrandingViewSpacing = 16; +static const CGFloat kImageViewBorderWidth = 1; +static const CGFloat kImageViewHorizontalInset = 7; +static const CGFloat kImageViewVerticalInset = 19; +static const CGFloat kImageViewCornerRadius = 6; + +- (instancetype)initWithFrame:(CGRect)frame { + self = [super initWithFrame:frame]; + + if (self) { + [self _setupViewHierarchy]; + } + + return self; +} + +- (void)setPaymentSystemImage:(UIImage *)paymentSystemImage { + _paymentSystemImage = paymentSystemImage; + + self.paymentSystemImageView.image = paymentSystemImage; +} + +- (void)setIssuerImage:(UIImage *)issuerImage { + _issuerImage = issuerImage; + + self.issuerImageView.image = issuerImage; +} + +- (void)didMoveToWindow { + [super didMoveToWindow]; + + if (self.window.screen.nativeScale > 0) { + self.issuerView.layer.borderWidth = kImageViewBorderWidth / self.window.screen.nativeScale; + self.paymentSystemView.layer.borderWidth = kImageViewBorderWidth / self.window.screen.nativeScale; + } +} + +- (void)_setupViewHierarchy { + self.layoutMargins = UIEdgeInsetsMake(0, 0, kBrandingViewBottomPadding, 0); + + self.stackView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisHorizontal]; + [self addSubview:self.stackView]; + + [self.stackView _stds_pinToSuperviewBounds]; + + self.issuerImageView = [self _newBrandingImageView]; + self.issuerView = [self _newInsetViewWithImageView:self.issuerImageView]; + [self.stackView addArrangedSubview:self.issuerView]; + + [self.stackView addSpacer:kBrandingViewSpacing]; + + self.paymentSystemImageView = [self _newBrandingImageView]; + self.paymentSystemView = [self _newInsetViewWithImageView:self.paymentSystemImageView]; + [self.stackView addArrangedSubview:self.paymentSystemView]; + + NSLayoutConstraint *imageViewWidthConstraint = [NSLayoutConstraint constraintWithItem:self.issuerView attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeWidth multiplier:0.5 constant:0]; + // Setting the priority of the width constraint, so that the priority of the equal widths constraint below takes precedence, allowing both image views to take half of the remaining space equally. + imageViewWidthConstraint.priority = UILayoutPriorityDefaultHigh; + NSLayoutConstraint *width = [NSLayoutConstraint constraintWithItem:self.paymentSystemView attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:self.issuerView attribute:NSLayoutAttributeWidth multiplier:1 constant:0]; + + [NSLayoutConstraint activateConstraints:@[imageViewWidthConstraint, width]]; +} + +- (UIView *)_newInsetViewWithImageView:(UIImageView *)imageView { + UIView *insetView = [UIView new]; + insetView.layoutMargins = UIEdgeInsetsMake(kImageViewHorizontalInset, kImageViewVerticalInset, kImageViewHorizontalInset, kImageViewVerticalInset); + insetView.layer.cornerRadius = kImageViewCornerRadius; + insetView.backgroundColor = [UIColor whiteColor]; // Issuer images always expect a white background. + insetView.layer.masksToBounds = YES; + if (@available(iOS 12.0, *)) { + insetView.layer.borderColor = (self.traitCollection.userInterfaceStyle == UIUserInterfaceStyleLight) ? + [UIColor colorWithRed:(CGFloat)0.0 green:(CGFloat)57.0/(CGFloat)255.0 blue:(CGFloat)69.0/(CGFloat)255.0 alpha:(CGFloat)0.25].CGColor : + [UIColor colorWithRed:(CGFloat)195.0/(CGFloat)255.0 green:(CGFloat)214.0/(CGFloat)255.0 blue:(CGFloat)218.0/(CGFloat)255.0 alpha:(CGFloat)0.25].CGColor; + } else { + insetView.layer.borderColor = [UIColor colorWithRed:(CGFloat)0.0 green:(CGFloat)57.0/(CGFloat)255.0 blue:(CGFloat)69.0/(CGFloat)255.0 alpha:(CGFloat)0.25].CGColor; + } + + [insetView addSubview:imageView]; + [imageView _stds_pinToSuperviewBounds]; + + return insetView; +} + +- (UIImageView *)_newBrandingImageView { + UIImageView *imageView = [[UIImageView alloc] init]; + imageView.contentMode = UIViewContentModeScaleAspectFit; + + return imageView; +} + +- (void)traitCollectionDidChange:(UITraitCollection * _Nullable)previousTraitCollection { + if (@available(iOS 12.0, *)) { + CGColorRef borderColor = (self.traitCollection.userInterfaceStyle == UIUserInterfaceStyleLight) ? + [UIColor colorWithRed:(CGFloat)0.0 green:(CGFloat)57.0/(CGFloat)255.0 blue:(CGFloat)69.0/(CGFloat)255.0 alpha:(CGFloat)0.25].CGColor : + [UIColor colorWithRed:(CGFloat)195.0/(CGFloat)255.0 green:(CGFloat)214.0/(CGFloat)255.0 blue:(CGFloat)218.0/(CGFloat)255.0 alpha:(CGFloat)0.25].CGColor; + self.issuerView.layer.borderColor = borderColor; + self.paymentSystemView.layer.borderColor = borderColor; + } +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSBundleLocator.h b/Stripe3DS2/Stripe3DS2/STDSBundleLocator.h new file mode 100644 index 00000000..ed919fa5 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSBundleLocator.h @@ -0,0 +1,15 @@ +// +// STDSBundleLocator.h +// Stripe3DS2 +// +// Created by David Estes on 7/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@interface STDSBundleLocator : NSObject + ++ (NSBundle *)stdsResourcesBundle; + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSBundleLocator.m b/Stripe3DS2/Stripe3DS2/STDSBundleLocator.m new file mode 100644 index 00000000..55384068 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSBundleLocator.m @@ -0,0 +1,109 @@ +// +// STDSBundleLocator.m +// Stripe3DS2 +// +// Created by David Estes on 7/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// Based on STPBundleLocator.m in Stripe.framework +// + +#import "STDSBundleLocator.h" + +/** + Using a private class to ensure that it can't be subclassed, which may + change the result of `bundleForClass` + */ +@interface STDSBundleLocatorInternal : NSObject +@end +@implementation STDSBundleLocatorInternal +@end + +@implementation STDSBundleLocator + +// This is copied from SPM's resource_bundle_accessor.m ++ (NSBundle *)stdsSPMBundle { + NSString *bundleName = @"Stripe_Stripe"; + + NSArray *candidates = @[ + NSBundle.mainBundle.resourceURL, + [NSBundle bundleForClass:[self class]].resourceURL, + NSBundle.mainBundle.bundleURL + ]; + + for (NSURL* candiate in candidates) { + NSURL *bundlePath = [candiate URLByAppendingPathComponent:[NSString stringWithFormat:@"%@.bundle", bundleName]]; + + NSBundle *bundle = [NSBundle bundleWithURL:bundlePath]; + if (bundle != nil) { + return bundle; + } + } + + return nil; +} + ++ (NSBundle *)stdsResourcesBundle { + /** + First, find Stripe.framework. + Places to check: + 1. Stripe_Stripe3DS2.bundle (for SwiftPM) + 1. Stripe_Stripe.bundle (for SwiftPM) + 2. Stripe.bundle (for manual static installations, Fabric, and framework-less Cocoapods) + 3. Stripe.framework/Stripe.bundle (for framework-based Cocoapods) + 4. Stripe.framework (for Carthage, manual dynamic installations) + 5. main bundle (for people dragging all our files into their project) + **/ + + static NSBundle *ourBundle; + + static dispatch_once_t onceToken; + dispatch_once(&onceToken, ^{ +#ifdef SWIFTPM_MODULE_BUNDLE + ourBundle = SWIFTPM_MODULE_BUNDLE; +#endif + + if (ourBundle == nil) { + ourBundle = [STDSBundleLocator stdsSPMBundle]; + } + + if (ourBundle == nil) { + ourBundle = [NSBundle bundleWithPath:@"Stripe.bundle"]; + } + + if (ourBundle == nil) { + // This might be the same as the previous check if not using a dynamic framework + NSString *path = [[NSBundle bundleForClass:[STDSBundleLocatorInternal class]] pathForResource:@"Stripe" ofType:@"bundle"]; + ourBundle = [NSBundle bundleWithPath:path]; + } + + if (ourBundle == nil) { + // This will be the same as mainBundle if not using a dynamic framework + ourBundle = [NSBundle bundleForClass:[STDSBundleLocatorInternal class]]; + } + + if (ourBundle == nil) { + ourBundle = [NSBundle mainBundle]; + } + + // Once we've found Stripe.framework, seek around to find Stripe3DS2.bundle. + // Try to find Stripe3DS2 bundle within our current bundle + NSString *stdsBundlePath = [[ourBundle bundlePath] stringByAppendingPathComponent:@"Stripe3DS2.bundle"]; + NSBundle *stdsBundle = [NSBundle bundleWithPath:stdsBundlePath]; + if (stdsBundle != nil) { + ourBundle = stdsBundle; + } + // If it's not there, it might be a level up from us? + // (CocoaPods arranges us this way, as an example.) + if (stdsBundle == nil) { + NSString *stdsBundlePath = [[[ourBundle bundlePath] stringByDeletingLastPathComponent] stringByAppendingPathComponent:@"Stripe3DS2.bundle"]; + stdsBundle = [NSBundle bundleWithPath:stdsBundlePath]; + if (stdsBundle != nil) { + ourBundle = stdsBundle; + } + } + }); + + return ourBundle; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.h b/Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.h new file mode 100644 index 00000000..2111ea3f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.h @@ -0,0 +1,25 @@ +// +// STDSChallengeInformationView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSLabelCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeInformationView: UIView + +@property (nonatomic, strong, nullable) NSString *headerText; +@property (nonatomic, strong, nullable) UIImage *textIndicatorImage; +@property (nonatomic, strong, nullable) NSString *challengeInformationText; +@property (nonatomic, strong, nullable) NSString *challengeInformationLabel; + +@property (nonatomic, strong, nullable) STDSLabelCustomization *labelCustomization; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.m b/Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.m new file mode 100644 index 00000000..6796dfe3 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeInformationView.m @@ -0,0 +1,137 @@ +// +// STDSChallengeInformationView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeInformationView.h" +#import "STDSStackView.h" +#import "STDSSpacerView.h" +#import "UIView+LayoutSupport.h" +#import "NSString+EmptyChecking.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeInformationView () + +@property (nonatomic, strong) STDSStackView *informationStackView; +@property (nonatomic, strong) STDSStackView *indicatorStackView; + +@property (nonatomic, strong) UILabel *headerLabel; +@property (nonatomic, strong) UIImageView *textIndicatorImageView; +@property (nonatomic, strong) UILabel *textLabel; +@property (nonatomic, strong) UILabel *informationLabel; +@property (nonatomic, strong) UIView *indicatorStackViewSpacerView; +@property (nonatomic, strong) UIView *indicatorImageTextSpacerView; + +@end + +@implementation STDSChallengeInformationView + +static const CGFloat kHeaderTextBottomPadding = 8; +static const CGFloat kInformationTextBottomPadding = 20; +static const CGFloat kChallengeInformationViewBottomPadding = 6; +static const CGFloat kTextIndicatorHorizontalPadding = 8; + +- (instancetype)initWithFrame:(CGRect)frame { + self = [super initWithFrame:frame]; + + if (self) { + [self _setupViewHierarchy]; + } + + return self; +} + +- (void)setHeaderText:(NSString * _Nullable)headerText { + _headerText = headerText; + + self.headerLabel.text = headerText; + self.headerLabel.hidden = [NSString _stds_isStringEmpty:headerText]; +} + +- (void)setTextIndicatorImage:(UIImage * _Nullable)textIndicatorImage { + _textIndicatorImage = textIndicatorImage; + + self.textIndicatorImageView.image = textIndicatorImage; + self.textIndicatorImageView.hidden = textIndicatorImage == nil; + self.indicatorImageTextSpacerView.hidden = textIndicatorImage == nil; +} + +- (void)setChallengeInformationText:(NSString * _Nullable)challengeInformationText { + _challengeInformationText = challengeInformationText; + + self.textLabel.text = challengeInformationText; + self.textLabel.hidden = [NSString _stds_isStringEmpty:challengeInformationText]; +} + +- (void)setChallengeInformationLabel:(NSString * _Nullable)challengeInformationLabel { + _challengeInformationLabel = challengeInformationLabel; + + self.informationLabel.text = challengeInformationLabel; + self.informationLabel.hidden = [NSString _stds_isStringEmpty:challengeInformationLabel]; + self.indicatorStackViewSpacerView.hidden = self.informationLabel.hidden; +} + +- (void)_setupViewHierarchy { + self.layoutMargins = UIEdgeInsetsMake(0, 0, kChallengeInformationViewBottomPadding, 0); + + self.headerLabel = [self _newInformationLabel]; + + self.textIndicatorImageView = [[UIImageView alloc] init]; + self.textIndicatorImageView.contentMode = UIViewContentModeTop; + self.textIndicatorImageView.hidden = YES; + + self.textLabel = [self _newInformationLabel]; + self.informationLabel = [self _newInformationLabel]; + + self.indicatorStackView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisHorizontal]; + + [self.indicatorStackView addArrangedSubview:self.textIndicatorImageView]; + self.indicatorImageTextSpacerView = [[STDSSpacerView alloc] initWithLayoutAxis:STDSStackViewLayoutAxisHorizontal dimension:kTextIndicatorHorizontalPadding]; + self.indicatorImageTextSpacerView.hidden = YES; + [self.indicatorStackView addArrangedSubview:self.indicatorImageTextSpacerView]; + [self.indicatorStackView addArrangedSubview:self.textLabel]; + + self.informationStackView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + [self.informationStackView addArrangedSubview:self.headerLabel]; + [self.informationStackView addSpacer:kHeaderTextBottomPadding]; + [self.informationStackView addArrangedSubview:self.indicatorStackView]; + self.indicatorStackViewSpacerView = [[STDSSpacerView alloc] initWithLayoutAxis:STDSStackViewLayoutAxisVertical dimension:kInformationTextBottomPadding]; + [self.informationStackView addArrangedSubview:self.indicatorStackViewSpacerView]; + [self.informationStackView addArrangedSubview:self.informationLabel]; + + [self addSubview:self.informationStackView]; + + [self.informationStackView _stds_pinToSuperviewBounds]; + + NSLayoutConstraint *imageViewWidthConstraint = [NSLayoutConstraint constraintWithItem:self.textIndicatorImageView attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1.0 constant:35]; + [NSLayoutConstraint activateConstraints:@[imageViewWidthConstraint]]; +} + +- (void)setLabelCustomization:(STDSLabelCustomization * _Nullable)labelCustomization { + _labelCustomization = labelCustomization; + + self.headerLabel.font = labelCustomization.headingFont; + self.headerLabel.textColor = labelCustomization.headingTextColor; + + self.textLabel.font = labelCustomization.font; + self.textLabel.textColor = labelCustomization.textColor; + + self.informationLabel.font = labelCustomization.font; + self.informationLabel.textColor = labelCustomization.textColor; +} + +- (UILabel *)_newInformationLabel { + UILabel *label = [[UILabel alloc] init]; + label.numberOfLines = 0; + label.hidden = YES; + + return label; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.h b/Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.h new file mode 100644 index 00000000..d0809963 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.h @@ -0,0 +1,138 @@ +// +// STDSChallengeRequestParameters.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 4/1/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSJSONEncodable.h" + +@class STDSChallengeParameters; + +typedef NS_ENUM(NSInteger, STDSChallengeCancelType) { + /// The cardholder selected "Cancel" from the UI + STDSChallengeCancelTypeCardholderSelectedCancel, + + /// The transaction timed out + STDSChallengeCancelTypeTransactionTimedOut, +}; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeRequestParameters : NSObject + +/** + Convenience initializer to create parameters for the first Challenge Request for a transaction. + */ +- (instancetype)initWithChallengeParameters:(STDSChallengeParameters *)challengeParams + transactionIdentifier:(NSString *)transactionIdentifier + messageVersion:(NSString *)messageVersion; + +/** + Designated initializer for `STDSChallengeRequestParameters` + */ +- (instancetype)initWithThreeDSServerTransactionIdentifier:(NSString *)threeDSServerTransactionIdentifier + acsTransactionIdentifier:(NSString *)acsTransactionIdentifier + messageVersion:(NSString *)messageVersion + sdkTransactionIdentifier:(NSString *)sdkTransactionIdentifier + requestorAppUrl:(NSString *)requestorAppUrl + sdkCounterStoA:(NSInteger)sdkCounterStoA NS_DESIGNATED_INITIALIZER; + +/** + Returns a new instance of STDSChallengeRequestParameters using the receiver, copying over the properties that are invariant across all CReqs for a given transaction and incrementing sdkCounterStoA. + */ +- (instancetype)nextChallengeRequestParametersByIncrementCounter; + +- (instancetype)init NS_UNAVAILABLE; + +#pragma mark - Required Properties + +/** + Universally unique transaction identifier assigned by the 3DS SDK to identify a single transaction. + */ +@property (nonatomic, readonly) NSString *sdkTransactionIdentifier; + +/** + Transaction identifier assigned by the 3DS Server to uniquely identify + a transaction. + */ +@property (nonatomic, readonly) NSString *threeDSServerTransactionIdentifier; + +/** + Transaction identifier assigned by the Access Control Server (ACS) + to uniquely identify a transaction. + */ +@property (nonatomic, readonly) NSString *acsTransactionIdentifier; + +/** + Identifies the type of message - always "CReq" + */ +@property (nonatomic, readonly) NSString *messageType; + +/** + The protocol version that is supported by the SDK and used for the transaction. + */ +@property (nonatomic, readonly) NSString *messageVersion; + +/** + Counter used as a security measure in the 3DS SDK to ACS secure channel. + */ +@property (nonatomic, readonly) NSString *sdkCounterStoA; + +#pragma mark - Optional/Conditional Properties + +/** + The URL for the application that is requesting 3DS2 verification. + This property can be optionally set and will be included with the + messages sent to the Directory Server during the challenge flow. + */ +@property (nonatomic, copy, nullable) NSString *threeDSRequestorAppURL; + +/** + A STDSChallengeCancelType wrapped in NSNumber, indicating that the authentication has been canceled. + */ +@property (nonatomic, copy, nullable) NSNumber *challengeCancel; + +/** + Contains the data that the Cardholder entered into the Native UI text field. + + @note The setter converts empty strings to nil. + */ +@property (nonatomic, copy, nullable) NSString *challengeDataEntry; + +/** + Data that the Cardholder entered into the HTML UI. + */ +@property (nonatomic, copy, nullable) NSString *challengeHTMLDataEntry; + +/** + Data necessary to support requirements not otherwise defined in the 3- D Secure message. + */ +@property (nonatomic, copy, nullable) NSArray *messageExtension; + +/** + A BOOL indiciating that Cardholder has completed the authentication as requested by selecting the Continue button in an Out- of-Band (OOB) authentication method. + */ +@property (nonatomic, nullable) NSNumber *oobContinue; + +/** + Indicator to resend the challenge information code to the Cardholder. + */ +@property (nonatomic, copy, nullable) NSString *resendChallenge; + +/** + Indicator confirming whether whitelisting was opted by the cardholder. + */ +@property (nonatomic, copy, nullable) NSString *whitelistingDataEntry; + +/** + Indicator informing that the Cardholder submits an empty response (no data entered in the UI). + */ +@property (nonatomic, copy, nullable) NSString *challengeNoEntry; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.m b/Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.m new file mode 100644 index 00000000..18fcb8db --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeRequestParameters.m @@ -0,0 +1,105 @@ +// +// STDSChallengeRequestParameters.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 4/1/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeRequestParameters.h" + +#import "STDSChallengeParameters.h" + +@implementation STDSChallengeRequestParameters + +- (instancetype)initWithChallengeParameters:(STDSChallengeParameters *)challengeParams + transactionIdentifier:(NSString *)transactionIdentifier + messageVersion:(NSString *)messageVersion { + return [self initWithThreeDSServerTransactionIdentifier:challengeParams.threeDSServerTransactionID + acsTransactionIdentifier:challengeParams.acsTransactionID + messageVersion:messageVersion + sdkTransactionIdentifier:transactionIdentifier + requestorAppUrl:challengeParams.threeDSRequestorAppURL + sdkCounterStoA:0]; +} + +- (instancetype)initWithThreeDSServerTransactionIdentifier:(NSString *)threeDSServerTransactionIdentifier + acsTransactionIdentifier:(NSString *)acsTransactionIdentifier + messageVersion:(NSString *)messageVersion + sdkTransactionIdentifier:(NSString *)sdkTransactionIdentifier + requestorAppUrl:(NSString *)requestorAppUrl + sdkCounterStoA:(NSInteger)sdkCounterStoA { + self = [super init]; + if (self) { + _messageType = @"CReq"; + _threeDSServerTransactionIdentifier = [threeDSServerTransactionIdentifier copy]; + _acsTransactionIdentifier = [acsTransactionIdentifier copy]; + _messageVersion = [messageVersion copy]; + _sdkTransactionIdentifier = [sdkTransactionIdentifier copy]; + _threeDSRequestorAppURL = [requestorAppUrl copy]; + _sdkCounterStoA = [NSString stringWithFormat:@"%03ld", (long)sdkCounterStoA]; + } + return self; +} + +- (instancetype)nextChallengeRequestParametersByIncrementCounter { + NSInteger incrementedCounter = [self.sdkCounterStoA intValue] + 1; + return [[STDSChallengeRequestParameters alloc] initWithThreeDSServerTransactionIdentifier:self.threeDSServerTransactionIdentifier + acsTransactionIdentifier:self.acsTransactionIdentifier + messageVersion:self.messageVersion + sdkTransactionIdentifier:self.sdkTransactionIdentifier + requestorAppUrl:self.threeDSRequestorAppURL // TC_SDK_10209_001 + sdkCounterStoA:incrementedCounter]; +} + +- (void)setChallengeDataEntry:(NSString *)challengeDataEntry { + // [Req 40] ...if the cardholder has submitted the response without entering any data in the UI, the Challenge Data Entry field shall not be present in the CReq message. + if (challengeDataEntry.length == 0) { + _challengeDataEntry = nil; + _challengeNoEntry = @"Y"; + } else { + _challengeDataEntry = [challengeDataEntry copy]; + _challengeNoEntry = nil; + } +} + +#pragma mark - Helpers + +- (nullable NSString *)challengeCancelString { + if (self.challengeCancel == nil) { + return nil; + } + + STDSChallengeCancelType challengeCancelType = (STDSChallengeCancelType)[self.challengeCancel integerValue]; + switch (challengeCancelType) { + case STDSChallengeCancelTypeCardholderSelectedCancel: + return @"01"; + case STDSChallengeCancelTypeTransactionTimedOut: + return @"08"; + } + return @"07"; // Unknown +} + +#pragma mark - STDSJSONEncodable + ++ (NSDictionary *)propertyNamesToJSONKeysMapping { + return @{ + NSStringFromSelector(@selector(threeDSServerTransactionIdentifier)): @"threeDSServerTransID", + NSStringFromSelector(@selector(acsTransactionIdentifier)): @"acsTransID", + NSStringFromSelector(@selector(threeDSRequestorAppURL)): @"threeDSRequestorAppURL", + NSStringFromSelector(@selector(challengeCancelString)): @"challengeCancel", + NSStringFromSelector(@selector(challengeDataEntry)): @"challengeDataEntry", + NSStringFromSelector(@selector(challengeHTMLDataEntry)): @"challengeHTMLDataEntry", + NSStringFromSelector(@selector(challengeNoEntry)): @"challengeNoEntry", + NSStringFromSelector(@selector(messageExtension)): @"messageExtension", + NSStringFromSelector(@selector(messageVersion)): @"messageVersion", + NSStringFromSelector(@selector(messageType)): @"messageType", + NSStringFromSelector(@selector(oobContinue)): @"oobContinue", + NSStringFromSelector(@selector(resendChallenge)): @"resendChallenge", + NSStringFromSelector(@selector(sdkTransactionIdentifier)): @"sdkTransID", + NSStringFromSelector(@selector(sdkCounterStoA)): @"sdkCounterStoA", + NSStringFromSelector(@selector(whitelistingDataEntry)): @"whitelistingDataEntry", + }; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponse.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponse.h new file mode 100644 index 00000000..58606d25 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponse.h @@ -0,0 +1,145 @@ +// +// STDSChallengeResponse.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponseSelectionInfo.h" +#import "STDSChallengeResponseMessageExtension.h" +#import "STDSChallengeResponseImage.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + The `STDSACSUIType` enum defines the type of UI to be presented. + */ +typedef NS_ENUM(NSInteger, STDSACSUIType) { + + /// No UI associated with the response. + STDSACSUITypeNone = 0, + + /// Text challenge response UI. + STDSACSUITypeText = 1, + + /// Single-select challenge response UI. + STDSACSUITypeSingleSelect = 2, + + /// Multi-select challenge response UI. + STDSACSUITypeMultiSelect = 3, + + /// Out Of Band challenge response UI. + STDSACSUITypeOOB = 4, + + /// HTML challenge response UI. + STDSACSUITypeHTML = 5, +}; + +/// A protocol that represents the information contained within a challenge response. +@protocol STDSChallengeResponse + +/// Universally unique transaction identifier assigned by the 3DS Server to identify a single transaction. +@property (nonatomic, readonly) NSString *threeDSServerTransactionID; + +/// Counter used as a security measure in the ACS to 3DS SDK secure channel. +@property (nonatomic, readonly) NSString *acsCounterACStoSDK; + +/// Universally unique transaction identifier assigned by the ACS to identify a single transaction. +@property (nonatomic, readonly) NSString *acsTransactionID; + +/// HTML provided by the ACS in the Challenge Response message. Utilised when HTML is specified in the ACS UI Type during the Cardholder challenge. +@property (nonatomic, readonly, nullable) NSString *acsHTML; + +/// Optional HTML provided by the ACS in the CRes message to be utilised in the Out of Band flow when the HTML is specified in the ACS UI Type during the Cardholder challenge, displayed when the app is moved to the foreground. +@property (nonatomic, readonly, nullable) NSString *acsHTMLRefresh; + +/// User interface type that the 3DS SDK will render, which includes the specific data mapping and requirements. +@property (nonatomic, readonly) STDSACSUIType acsUIType; + +/** + Indicator of the state of the ACS challenge cycle and whether the challenge has completed or will require additional messages. Shall be populated in all Challenge Response messages to convey the current state of the transaction. + + - Note: + If set to YES, the ACS will populate the Transaction Status in the Challenge Response message. + */ +@property (nonatomic, readonly) BOOL challengeCompletionIndicator; + +/// Header text that for the challenge information screen that is being presented. +@property (nonatomic, readonly, nullable) NSString *challengeInfoHeader; + +/// Label to modify the Challenge Data Entry field provided by the Issuer. +@property (nonatomic, readonly, nullable) NSString *challengeInfoLabel; + +/// Text provided by the ACS/Issuer to Cardholder during the Challenge Message exchange. +@property (nonatomic, readonly, nullable) NSString *challengeInfoText; + +/// Text provided by the ACS/Issuer to Cardholder during OOB authentication to replace Challenge Information Text and Challenge Information Text Indicator +@property (nonatomic, readonly, nullable) NSString *challengeAdditionalInfoText; + +/// Indicates when the Issuer/ACS would like a warning icon or similar visual indicator to draw attention to the “Challenge Information Text” that is being displayed. +@property (nonatomic, readonly) BOOL showChallengeInfoTextIndicator; + +/// Selection information that will be presented to the Cardholder if the option is single or multi-select. The variables will be sent in a JSON Array and parsed by the SDK for display in the user interface. +@property (nonatomic, readonly, nullable) NSArray> *challengeSelectInfo; + +/// Label displayed to the Cardholder for the content in Expandable Information Text. +@property (nonatomic, readonly, nullable) NSString *expandInfoLabel; + +/// Text provided by the Issuer from the ACS to be displayed to the Cardholder for additional information and the format will be an expandable text field. +@property (nonatomic, readonly, nullable) NSString *expandInfoText; + +/// Sent in the initial Challenge Response message from the ACS to the 3DS SDK to provide the URL(s) of the Issuer logo or image to be used in the Native UI. +@property (nonatomic, readonly, nullable) id issuerImage; + +/// Data necessary to support requirements not otherwise defined in the 3-D Secure message are carried in a Message Extension. +@property (nonatomic, readonly, nullable) NSArray> *messageExtensions; + +/// Identifies the type of message that is passed. +@property (nonatomic, readonly) NSString *messageType; + +/// Protocol version identifier. This shall be the Protocol Version Number of the specification utilised by the system creating this message. The Message Version Number is set by the 3DS Server which originates the protocol with the AReq message. The Message Version Number does not change during a 3DS transaction. +@property (nonatomic, readonly) NSString *messageVersion; + +/// Mobile Deep link to an authentication app used in the out-of-band authentication. The App URL will open the appropriate location within the authentication app. +@property (nonatomic, readonly, nullable) NSURL *oobAppURL; + +/// Label to be displayed for the link to the OOB App URL. For example: “oobAppLabel”: “Click here to open Your Bank App” +@property (nonatomic, readonly, nullable) NSString *oobAppLabel; + +/// Label to be used in the UI for the button that the user selects when they have completed the OOB authentication. +@property (nonatomic, readonly, nullable) NSString *oobContinueLabel; + +/// Sent in the initial Challenge Response message from the ACS to the 3DS SDK to provide the URL(s) of the DS or Payment System logo or image to be used in the Native UI. +@property (nonatomic, readonly, nullable) id paymentSystemImage; + +/// Label to be used in the UI for the button that the user selects when they would like to have the authentication information present. +@property (nonatomic, readonly, nullable) NSString *resendInformationLabel; + +/// Universally unique transaction identifier assigned by the 3DS SDK to identify a single transaction. +@property (nonatomic, readonly) NSString *sdkTransactionID; + +/** + Label to be used in the UI for the button that the user selects when they have completed the authentication. + + - Note: + This is not used for OOB authentication. + */ +@property (nonatomic, readonly, nullable) NSString *submitAuthenticationLabel; + +/// Text provided by the ACS/Issuer to Cardholder during a Whitelisting transaction. For example, “Would you like to add this Merchant to your whitelist?” +@property (nonatomic, readonly, nullable) NSString *whitelistingInfoText; + +/// Label to be displayed to the Cardholder for the "why" information section. +@property (nonatomic, readonly, nullable) NSString *whyInfoLabel; + +/// Text provided by the Issuer to be displayed to the Cardholder to explain why the Cardholder is being asked to perform the authentication task. +@property (nonatomic, readonly, nullable) NSString *whyInfoText; + +/// Indicates the state of the associated Transaction. +@property (nonatomic, readonly, nullable) NSString *transactionStatus; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImage.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImage.h new file mode 100644 index 00000000..9128d77d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImage.h @@ -0,0 +1,27 @@ +// +// STDSChallengeResponseImage.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/// A protocol used to represent information about an individual image resource inside of a challenge response. +@protocol STDSChallengeResponseImage + +/// A medium density image to display as the issuer image. +@property (nonatomic, readonly, nullable) NSURL *mediumDensityURL; + +/// A high density image to display as the issuer image. +@property (nonatomic, readonly, nullable) NSURL *highDensityURL; + +/// An extra-high density image to display as the issuer image. +@property (nonatomic, readonly, nullable) NSURL *extraHighDensityURL; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.h new file mode 100644 index 00000000..59e21208 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.h @@ -0,0 +1,23 @@ +// +// STDSChallengeResponseImageObject.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponseImage.h" + +#import "STDSJSONDecodable.h" + +NS_ASSUME_NONNULL_BEGIN + +/// An object used to represent information about an individual image resource inside of a challenge response. +@interface STDSChallengeResponseImageObject: NSObject + +- (instancetype)initWithMediumDensityURL:(NSURL * _Nullable)mediumDensityURL highDensityURL:(NSURL * _Nullable)highDensityURL extraHighDensityURL:(NSURL * _Nullable)extraHighDensityURL; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.m b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.m new file mode 100644 index 00000000..a3ea6840 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseImageObject.m @@ -0,0 +1,53 @@ +// +// STDSChallengeResponseImageObject.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeResponseImageObject.h" + +#import "NSDictionary+DecodingHelpers.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeResponseImageObject() + +@property (nonatomic, nullable) NSURL *mediumDensityURL; +@property (nonatomic, nullable) NSURL *highDensityURL; +@property (nonatomic, nullable) NSURL *extraHighDensityURL; + +@end + +@implementation STDSChallengeResponseImageObject + +- (instancetype)initWithMediumDensityURL:(NSURL * _Nullable)mediumDensityURL highDensityURL:(NSURL * _Nullable)highDensityURL extraHighDensityURL:(NSURL * _Nullable)extraHighDensityURL { + self = [super init]; + + if (self) { + _mediumDensityURL = mediumDensityURL; + _highDensityURL = highDensityURL; + _extraHighDensityURL = extraHighDensityURL; + } + + return self; +} + ++ (nullable instancetype)decodedObjectFromJSON:(nullable NSDictionary *)json error:(NSError * _Nullable __autoreleasing * _Nullable)outError { + if (json == nil) { + return nil; + } + + NSURL *mediumDensityURL = [json _stds_urlForKey:@"medium" required:NO error:nil]; + NSURL *highDensityURL = [json _stds_urlForKey:@"high" required:NO error:nil]; + NSURL *extraHighDensityURL = [json _stds_urlForKey:@"extraHigh" required:NO error:nil]; + + return [[STDSChallengeResponseImageObject alloc] initWithMediumDensityURL:mediumDensityURL + highDensityURL:highDensityURL + extraHighDensityURL:extraHighDensityURL]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtension.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtension.h new file mode 100644 index 00000000..c26425f9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtension.h @@ -0,0 +1,30 @@ +// +// STDSChallengeResponseMessageExtension.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/// A protocol that encapsulates an individual message extension inside of a challenge response. +@protocol STDSChallengeResponseMessageExtension + +/// The name of the extension data set as defined by the extension owner. +@property (nonatomic, readonly) NSString *name; + +/// A unique identifier for the extension. +@property (nonatomic, readonly) NSString *identifier; + +/// A Boolean value indicating whether the recipient must understand the contents of the extension to interpret the entire message. +@property (nonatomic, readonly, getter = isCriticalityIndicator) BOOL criticalityIndicator; + +/// The data carried in the extension. +@property (nonatomic, readonly) NSDictionary *data; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.h new file mode 100644 index 00000000..5ece844b --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.h @@ -0,0 +1,21 @@ +// +// STDSChallengeResponseMessageExtensionObject.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponseMessageExtension.h" + +#import "STDSJSONDecodable.h" + +NS_ASSUME_NONNULL_BEGIN + +/// An object used to represent an individual message extension inside of a challenge response. +@interface STDSChallengeResponseMessageExtensionObject: NSObject + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.m b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.m new file mode 100644 index 00000000..54e58d42 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseMessageExtensionObject.m @@ -0,0 +1,72 @@ +// +// STDSChallengeResponseMessageExtensionObject.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeResponseMessageExtensionObject.h" + +#import "NSDictionary+DecodingHelpers.h" +#import "NSError+Stripe3DS2.h" + +NS_ASSUME_NONNULL_BEGIN + +static const NSInteger kMaximumStringFieldLength = 64; +static const NSInteger kMaximumDataFieldLength = 8059; + +@implementation STDSChallengeResponseMessageExtensionObject + +@synthesize name = _name; +@synthesize identifier = _identifier; +@synthesize criticalityIndicator = _criticalityIndicator; +@synthesize data = _data; + +- (instancetype)initWithName:(NSString *)name identifier:(NSString *)identifier criticalityIndicator:(BOOL)criticalityIndicator data:(NSDictionary *)data { + self = [super init]; + if (self) { + _name = [name copy]; + _identifier = [identifier copy]; + _criticalityIndicator = criticalityIndicator; + _data = data; + } + return self; +} + ++ (nullable instancetype)decodedObjectFromJSON:(nullable NSDictionary *)json error:(NSError * _Nullable __autoreleasing * _Nullable)outError { + if (json == nil) { + return nil; + } + NSError *error; + + NSString *name = [json _stds_stringForKey:@"name" validator:^BOOL (NSString *value) { + return value.length <= kMaximumStringFieldLength; + }required:YES error:&error]; + NSString *identifier = [json _stds_stringForKey:@"id" validator:^BOOL (NSString *value) { + return value.length <= kMaximumStringFieldLength; + } required:YES error:&error]; + BOOL criticalityIndicator= [json _stds_boolForKey:@"criticalityIndicator" required:YES error:&error].boolValue; + NSDictionary *data = [json _stds_dictionaryForKey:@"data" required:YES error:&error]; + // The spec requires data to be "Maximum 8059 characters" + if (data && [NSJSONSerialization dataWithJSONObject:data options:0 error:nil].length > kMaximumDataFieldLength) { + error = [NSError _stds_invalidJSONFieldError:@"data"]; + } + + if (error) { + if (outError) { + *outError = error; + } + return nil; + } + + if (data != nil) { + return [[STDSChallengeResponseMessageExtensionObject alloc] initWithName:name identifier:identifier criticalityIndicator:criticalityIndicator data:data]; + } else { + return nil; + } +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.h new file mode 100644 index 00000000..1f245286 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.h @@ -0,0 +1,49 @@ +// +// STDSChallengeResponseObject.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponse.h" +#import "STDSJSONDecodable.h" + +NS_ASSUME_NONNULL_BEGIN + +/// An object used to represent a challenge response from the ACS. +@interface STDSChallengeResponseObject: NSObject + +- (instancetype)initWithThreeDSServerTransactionID:(NSString *)threeDSServerTransactionID + acsCounterACStoSDK:(NSString *)acsCounterACStoSDK + acsTransactionID:(NSString *)acsTransactionID + acsHTML:(NSString * _Nullable)acsHTML + acsHTMLRefresh:(NSString * _Nullable)acsHTMLRefresh + acsUIType:(STDSACSUIType)acsUIType + challengeCompletionIndicator:(BOOL)challengeCompletionIndicator + challengeInfoHeader:(NSString * _Nullable)challengeInfoHeader + challengeInfoLabel:(NSString * _Nullable)challengeInfoLabel + challengeInfoText:(NSString * _Nullable)challengeInfoText + challengeAdditionalInfoText:(NSString * _Nullable)challengeAdditionalInfoText + showChallengeInfoTextIndicator:(BOOL)showChallengeInfoTextIndicator + challengeSelectInfo:(NSArray> * _Nullable)challengeSelectInfo + expandInfoLabel:(NSString * _Nullable)expandInfoLabel + expandInfoText:(NSString * _Nullable)expandInfoText + issuerImage:(id _Nullable)issuerImage + messageExtensions:(NSArray> * _Nullable)messageExtensions + messageVersion:(NSString *)messageVersion + oobAppURL:(NSURL * _Nullable)oobAppURL + oobAppLabel:(NSString * _Nullable)oobAppLabel + oobContinueLabel:(NSString * _Nullable)oobContinueLabel + paymentSystemImage:(id _Nullable)paymentSystemImage + resendInformationLabel:(NSString * _Nullable)resendInformationLabel + sdkTransactionID:(NSString *)sdkTransactionID + submitAuthenticationLabel:(NSString * _Nullable)submitAuthenticationLabel + whitelistingInfoText:(NSString * _Nullable)whitelistingInfoText + whyInfoLabel:(NSString * _Nullable)whyInfoLabel + whyInfoText:(NSString * _Nullable)whyInfoText + transactionStatus:(NSString * _Nullable)transactionStatus; +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.m b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.m new file mode 100644 index 00000000..7d952bb9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseObject.m @@ -0,0 +1,321 @@ +// +// STDSChallengeResponseObject.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeResponseObject.h" + +#import "NSDictionary+DecodingHelpers.h" +#import "NSError+Stripe3DS2.h" +#import "STDSChallengeResponseSelectionInfoObject.h" +#import "STDSChallengeResponseImageObject.h" +#import "STDSChallengeResponseMessageExtensionObject.h" +#import "NSString+JWEHelpers.h" +#import "STDSStripe3DS2Error.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSChallengeResponseObject + +@synthesize threeDSServerTransactionID = _threeDSServerTransactionID; +@synthesize acsCounterACStoSDK = _acsCounterACStoSDK; +@synthesize acsTransactionID = _acsTransactionID; +@synthesize acsHTML = _acsHTML; +@synthesize acsHTMLRefresh = _acsHTMLRefresh; +@synthesize acsUIType = _acsUIType; +@synthesize challengeCompletionIndicator = _challengeCompletionIndicator; +@synthesize challengeInfoHeader = _challengeInfoHeader; +@synthesize challengeInfoLabel = _challengeInfoLabel; +@synthesize challengeInfoText = _challengeInfoText; +@synthesize challengeAdditionalInfoText = _challengeAdditionalInfoText; +@synthesize showChallengeInfoTextIndicator = _showChallengeInfoTextIndicator; +@synthesize challengeSelectInfo = _challengeSelectInfo; +@synthesize expandInfoLabel = _expandInfoLabel; +@synthesize expandInfoText = _expandInfoText; +@synthesize issuerImage = _issuerImage; +@synthesize messageExtensions = _messageExtensions; +@synthesize messageType = _messageType; +@synthesize messageVersion = _messageVersion; +@synthesize oobAppURL = _oobAppURL; +@synthesize oobAppLabel = _oobAppLabel; +@synthesize oobContinueLabel = _oobContinueLabel; +@synthesize paymentSystemImage = _paymentSystemImage; +@synthesize resendInformationLabel = _resendInformationLabel; +@synthesize sdkTransactionID = _sdkTransactionID; +@synthesize submitAuthenticationLabel = _submitAuthenticationLabel; +@synthesize whitelistingInfoText = _whitelistingInfoText; +@synthesize whyInfoLabel = _whyInfoLabel; +@synthesize whyInfoText = _whyInfoText; +@synthesize transactionStatus = _transactionStatus; + +- (NSString *)description { + return [NSString stringWithFormat:@"%@ -- completion: %@, count: %@", [super description], @(self.challengeCompletionIndicator), self.acsCounterACStoSDK]; +} + +- (instancetype)initWithThreeDSServerTransactionID:(NSString *)threeDSServerTransactionID + acsCounterACStoSDK:(NSString *)acsCounterACStoSDK + acsTransactionID:(NSString *)acsTransactionID + acsHTML:(NSString * _Nullable)acsHTML + acsHTMLRefresh:(NSString * _Nullable)acsHTMLRefresh + acsUIType:(STDSACSUIType)acsUIType + challengeCompletionIndicator:(BOOL)challengeCompletionIndicator + challengeInfoHeader:(NSString * _Nullable)challengeInfoHeader + challengeInfoLabel:(NSString * _Nullable)challengeInfoLabel + challengeInfoText:(NSString * _Nullable)challengeInfoText + challengeAdditionalInfoText:(NSString * _Nullable)challengeAdditionalInfoText + showChallengeInfoTextIndicator:(BOOL)showChallengeInfoTextIndicator + challengeSelectInfo:(NSArray> * _Nullable)challengeSelectInfo + expandInfoLabel:(NSString * _Nullable)expandInfoLabel + expandInfoText:(NSString * _Nullable)expandInfoText + issuerImage:(id _Nullable)issuerImage + messageExtensions:(NSArray> * _Nullable)messageExtensions + messageVersion:(NSString *)messageVersion + oobAppURL:(NSURL * _Nullable)oobAppURL + oobAppLabel:(NSString * _Nullable)oobAppLabel + oobContinueLabel:(NSString * _Nullable)oobContinueLabel + paymentSystemImage:(id _Nullable)paymentSystemImage + resendInformationLabel:(NSString * _Nullable)resendInformationLabel + sdkTransactionID:(NSString *)sdkTransactionID + submitAuthenticationLabel:(NSString * _Nullable)submitAuthenticationLabel + whitelistingInfoText:(NSString * _Nullable)whitelistingInfoText + whyInfoLabel:(NSString * _Nullable)whyInfoLabel + whyInfoText:(NSString * _Nullable)whyInfoText + transactionStatus:(NSString * _Nullable)transactionStatus { + self = [super init]; + + if (self) { + _threeDSServerTransactionID = [threeDSServerTransactionID copy]; + _acsCounterACStoSDK = [acsCounterACStoSDK copy]; + _acsTransactionID = [acsTransactionID copy]; + _acsHTML = [acsHTML copy]; + _acsHTMLRefresh = [acsHTMLRefresh copy]; + _acsUIType = acsUIType; + _challengeCompletionIndicator = challengeCompletionIndicator; + _challengeInfoHeader = [challengeInfoHeader copy]; + _challengeInfoLabel = [challengeInfoLabel copy]; + _challengeInfoText = [challengeInfoText copy]; + _challengeAdditionalInfoText = [challengeAdditionalInfoText copy]; + _showChallengeInfoTextIndicator = showChallengeInfoTextIndicator; + _challengeSelectInfo = [challengeSelectInfo copy]; + _expandInfoLabel = [expandInfoLabel copy]; + _expandInfoText = [expandInfoText copy]; + _issuerImage = issuerImage; + _messageExtensions = [messageExtensions copy]; + _messageType = @"CRes"; + _messageVersion = [messageVersion copy]; + _oobAppURL = oobAppURL; + _oobAppLabel = [oobAppLabel copy]; + _oobContinueLabel = [oobContinueLabel copy]; + _paymentSystemImage = paymentSystemImage; + _resendInformationLabel = [resendInformationLabel copy]; + _sdkTransactionID = [sdkTransactionID copy]; + _submitAuthenticationLabel = [submitAuthenticationLabel copy]; + _whitelistingInfoText = [whitelistingInfoText copy]; + _whyInfoLabel = [whyInfoLabel copy]; + _whyInfoText = [whyInfoText copy]; + _transactionStatus = [transactionStatus copy]; + } + + return self; +} + +#pragma mark Private Helpers + ++ (NSDictionary *)acsUITypeStringMapping { + return @{ + @"01": @(STDSACSUITypeText), + @"02": @(STDSACSUITypeSingleSelect), + @"03": @(STDSACSUITypeMultiSelect), + @"04": @(STDSACSUITypeOOB), + @"05": @(STDSACSUITypeHTML), + }; +} + +/// The message extension identifiers that we support. ++ (NSSet *)supportedMessageExtensions { + return [NSSet new]; +} + +#pragma mark STDSJSONDecodable + ++ (nullable instancetype)decodedObjectFromJSON:(nullable NSDictionary *)json error:(NSError **)outError { + if (json == nil) { + return nil; + } + NSError *error; + +#pragma mark Required + NSString *threeDSServerTransactionID = [json _stds_stringForKey:@"threeDSServerTransID" validator:^BOOL (NSString *value) { + return [[NSUUID alloc] initWithUUIDString:value] != nil; + } required:YES error:&error]; + NSString *acsCounterACStoSDK = [json _stds_stringForKey:@"acsCounterAtoS" required:YES error:&error]; + NSString *acsTransactionID = [json _stds_stringForKey:@"acsTransID" required:YES error:&error]; + NSString *challengeCompletionIndicatorRawString = [json _stds_stringForKey:@"challengeCompletionInd" validator:^BOOL (NSString *value) { + return [value isEqualToString:@"N"] || [value isEqualToString:@"Y"]; + } required:YES error:&error]; + // There is only one valid messageType value for this object (@"CRes"), so we don't store it. + [json _stds_stringForKey:@"messageType" validator:^BOOL (NSString *value) { + return [value isEqualToString:@"CRes"]; + } required:YES error:&error]; + NSString *messageVersion = [json _stds_stringForKey:@"messageVersion" required:YES error:&error]; + NSString *sdkTransactionID = [json _stds_stringForKey:@"sdkTransID" required:YES error:&error]; + + BOOL challengeCompletionIndicator = challengeCompletionIndicatorRawString.boolValue; + + STDSACSUIType acsUIType = STDSACSUITypeNone; + if (!challengeCompletionIndicator) { + NSString *acsUITypeRawString = [json _stds_stringForKey:@"acsUiType" validator:^BOOL (NSString *value) { + return [self acsUITypeStringMapping][value] != nil; + } required:YES error:&error]; + + acsUIType = [self acsUITypeStringMapping][acsUITypeRawString].integerValue; + } + + if (error) { + // We failed to populate a required field + if (outError) { + *outError = error; + } + return nil; + } + + // At this point all the above values are valid: e.g. raw string representations of a BOOL or enum will map to a valid value. + +#pragma mark Conditional + NSString *encodedAcsHTML = [json _stds_stringForKey:@"acsHTML" required:(acsUIType == STDSACSUITypeHTML) error: &error]; + NSString *acsHTML = [encodedAcsHTML _stds_base64URLDecodedString]; + if (encodedAcsHTML && !acsHTML) { + // html was not valid base64url + error = [NSError _stds_invalidJSONFieldError:@"acsHTML"]; + } + + NSArray> *challengeSelectInfo = [json _stds_arrayForKey:@"challengeSelectInfo" + arrayElementType:[STDSChallengeResponseSelectionInfoObject class] + required:(acsUIType == STDSACSUITypeSingleSelect || acsUIType == STDSACSUITypeMultiSelect) + error:&error]; + NSString *oobContinueLabel = [json _stds_stringForKey:@"oobContinueLabel" required:(acsUIType == STDSACSUITypeOOB) error:&error]; + NSString *submitAuthenticationLabel = [json _stds_stringForKey:@"submitAuthenticationLabel" required:(acsUIType == STDSACSUITypeText || acsUIType == STDSACSUITypeSingleSelect || acsUIType == STDSACSUITypeMultiSelect || acsUIType == STDSACSUITypeText) error:&error]; + +#pragma mark Optional + NSArray> *messageExtensions = [json _stds_arrayForKey:@"messageExtension" + arrayElementType:[STDSChallengeResponseMessageExtensionObject class] + required:NO + error:&error]; + NSMutableArray *unrecognizedMessageExtensionIdentifiers = [NSMutableArray new]; + for (id messageExtension in messageExtensions) { + if (messageExtension.criticalityIndicator && ![[self supportedMessageExtensions] containsObject:messageExtension.identifier]) { + [unrecognizedMessageExtensionIdentifiers addObject:messageExtension.identifier]; + } + } + if (unrecognizedMessageExtensionIdentifiers.count > 0) { + error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain code:STDSErrorCodeUnrecognizedCriticalMessageExtension userInfo:@{STDSStripe3DS2UnrecognizedCriticalMessageExtensionsKey: unrecognizedMessageExtensionIdentifiers}]; + } + if (messageExtensions.count > 10) { + error = [NSError _stds_invalidJSONFieldError:@"messageExtension"]; + } + + NSString *encodedAcsHTMLRefresh = [json _stds_stringForKey:@"acsHTMLRefresh" required:NO error: &error]; + NSString *acsHTMLRefresh = [encodedAcsHTMLRefresh _stds_base64URLDecodedString]; + if (encodedAcsHTMLRefresh && !acsHTMLRefresh) { + // html was not valid base64url + error = [NSError _stds_invalidJSONFieldError:@"acsHTMLRefresh"]; + } + + BOOL infoLabelRequired = NO; + BOOL headerRequired = NO; + BOOL infoTextRequired = NO; + switch (acsUIType) { + case STDSACSUITypeNone: + break; // no-op + case STDSACSUITypeText: + case STDSACSUITypeSingleSelect: + case STDSACSUITypeMultiSelect: + infoLabelRequired = YES; // TC_SDK_10270_001 & TC_SDK_10276_001 & TC_SDK_10284_001 + headerRequired = YES; // TC_SDK_10268_001 & TC_SDK_10273_001 & TC_SDK_10282_001 + infoTextRequired = YES; // TC_SDK_10272_001 & TC_SDK_10278_001 & TC_SDK_10286_001 + break; + case STDSACSUITypeOOB: + + break; + case STDSACSUITypeHTML: + break; // no-op + } + + + NSString *challengeInfoLabel = [json _stds_stringForKey:@"challengeInfoLabel" validator:nil required:infoLabelRequired error:&error]; + NSString *challengeInfoHeader = [json _stds_stringForKey:@"challengeInfoHeader" required: (oobContinueLabel != nil) || headerRequired error:&error]; // TC_SDK_10292_001 + NSString *challengeInfoText = [json _stds_stringForKey:@"challengeInfoText" required:(oobContinueLabel != nil) || infoTextRequired error:&error]; // TC_SDK_10292_001 + NSString *challengeAdditionalInfoText = [json _stds_stringForKey:@"challengeAddInfo" required:NO error:&error]; + if (!error && submitAuthenticationLabel && (!challengeInfoLabel || !challengeInfoHeader || !challengeInfoText)) { + error = [NSError _stds_missingJSONFieldError:@"challengeInfoLabel or challengeInfoText"]; + } + + NSString *showChallengeInfoTextIndicatorRawString; + if (json[@"challengeInfoTextIndicator"]) { + showChallengeInfoTextIndicatorRawString = [json _stds_stringForKey:@"challengeInfoTextIndicator" validator:^BOOL (NSString *value) { + return [value isEqualToString:@"N"] || [value isEqualToString:@"Y"]; + } required:NO error:&error]; + } + BOOL showChallengeInfoTextIndicator = showChallengeInfoTextIndicatorRawString ? showChallengeInfoTextIndicatorRawString.boolValue : NO; // If the field is missing, we shouldn't show the indicator + NSString *expandInfoLabel = [json _stds_stringForKey:@"expandInfoLabel" required:NO error:&error]; + NSString *expandInfoText = [json _stds_stringForKey:@"expandInfoText" required:NO error:&error]; + NSURL *oobAppURL = [json _stds_urlForKey:@"oobAppURL" required:NO error:&error]; + NSString *oobAppLabel = [json _stds_stringForKey:@"oobAppURL" required:NO error:&error]; + NSDictionary *issuerImageJSON = [json _stds_dictionaryForKey:@"issuerImage" required:NO error:&error]; + STDSChallengeResponseImageObject *issuerImage = [STDSChallengeResponseImageObject decodedObjectFromJSON:issuerImageJSON error:&error]; + NSDictionary *paymentSystemImageJSON = [json _stds_dictionaryForKey:@"psImage" required:NO error:&error]; + STDSChallengeResponseImageObject *paymentSystemImage = [STDSChallengeResponseImageObject decodedObjectFromJSON:paymentSystemImageJSON error:&error]; + NSString *resendInformationLabel = [json _stds_stringForKey:@"resendInformationLabel" required:NO error:&error]; + NSString *whitelistingInfoText = [json _stds_stringForKey:@"whitelistingInfoText" required:NO error:&error]; + if (whitelistingInfoText.length > 64) { + // TC_SDK_10199_001 + error = [NSError _stds_invalidJSONFieldError:@"whitelisting text is greater than 64 characters"]; + } + NSString *whyInfoLabel = [json _stds_stringForKey:@"whyInfoLabel" required:NO error:&error]; + NSString *whyInfoText = [json _stds_stringForKey:@"whyInfoText" required:NO error:&error]; + NSString *transactionStatus = [json _stds_stringForKey:@"transStatus" required:challengeCompletionIndicator error:&error]; + + if (error) { + if (outError) { + *outError = error; + } + return nil; + } + + return [[self alloc] initWithThreeDSServerTransactionID:threeDSServerTransactionID + acsCounterACStoSDK:acsCounterACStoSDK + acsTransactionID:acsTransactionID + acsHTML:acsHTML + acsHTMLRefresh:acsHTMLRefresh + acsUIType:acsUIType + challengeCompletionIndicator:challengeCompletionIndicator + challengeInfoHeader:challengeInfoHeader + challengeInfoLabel:challengeInfoLabel + challengeInfoText:challengeInfoText + challengeAdditionalInfoText:challengeAdditionalInfoText + showChallengeInfoTextIndicator:showChallengeInfoTextIndicator + challengeSelectInfo:challengeSelectInfo + expandInfoLabel:expandInfoLabel + expandInfoText:expandInfoText + issuerImage:issuerImage + messageExtensions:messageExtensions + messageVersion:messageVersion + oobAppURL:oobAppURL + oobAppLabel:oobAppLabel + oobContinueLabel:oobContinueLabel + paymentSystemImage:paymentSystemImage + resendInformationLabel:resendInformationLabel + sdkTransactionID:sdkTransactionID + submitAuthenticationLabel:submitAuthenticationLabel + whitelistingInfoText:whitelistingInfoText + whyInfoLabel:whyInfoLabel + whyInfoText:whyInfoText + transactionStatus:transactionStatus]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfo.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfo.h new file mode 100644 index 00000000..a482c8c1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfo.h @@ -0,0 +1,24 @@ +// +// STDSChallengeResponseSelectionInfo.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/// A protocol that encapsulates information about an individual selection inside of a challenge response. +@protocol STDSChallengeResponseSelectionInfo + +/// The name of the selection option. +@property (nonatomic, readonly) NSString *name; + +/// The value of the selection option. +@property (nonatomic, readonly) NSString *value; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.h new file mode 100644 index 00000000..6e34b6c6 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.h @@ -0,0 +1,21 @@ +// +// STDSChallengeResponseSelectionInfoObject.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponseSelectionInfo.h" + +NS_ASSUME_NONNULL_BEGIN + +/// An object used to represent information about an individual selection inside of a challenge response. +@interface STDSChallengeResponseSelectionInfoObject: NSObject + +- (instancetype)initWithName:(NSString *)name value:(NSString *)value; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.m b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.m new file mode 100644 index 00000000..8aa22ee0 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseSelectionInfoObject.m @@ -0,0 +1,46 @@ +// +// STDSChallengeResponseSelectionInfoObject.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeResponseSelectionInfoObject.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeResponseSelectionInfoObject() + +@property (nonatomic, strong) NSString *name; +@property (nonatomic, strong) NSString *value; + +@end + +@implementation STDSChallengeResponseSelectionInfoObject + +- (instancetype)initWithName:(NSString *)name value:(NSString *)value { + self = [super init]; + + if (self) { + _name = name; + _value = value; + } + + return self; +} + ++ (nullable instancetype)decodedObjectFromJSON:(nullable NSDictionary *)json error:(NSError * _Nullable __autoreleasing * _Nullable)outError { + if (json == nil) { + return nil; + } + + NSString *name = [json allKeys].firstObject; + NSString *value = [json objectForKey:name]; + + return [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:name value:value]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.h b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.h new file mode 100644 index 00000000..e1ace0cc --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.h @@ -0,0 +1,80 @@ +// +// STDSChallengeResponseViewController.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponse.h" +#import "STDSUICustomization.h" +#import "STDSImageLoader.h" +#import "STDSDirectoryServer.h" + +@class STDSChallengeResponseViewController; + +NS_ASSUME_NONNULL_BEGIN + +@protocol STDSChallengeResponseViewControllerDelegate + +/** + Called when the user taps the Submit button after entering text in the Text flow (STDSACSUITypeText) + */ +- (void)challengeResponseViewController:(STDSChallengeResponseViewController *)viewController didSubmitInput:(NSString *)userInput + whitelistSelection: (id) whitelistSelection; + +/** + Called when the user taps the Submit button after selecting one or more options in the Single-Select (STDSACSUITypeSingleSelect) or Multi-Select (STDSACSUITypeMultiSelect) flow. + */ +- (void)challengeResponseViewController:(STDSChallengeResponseViewController *)viewController didSubmitSelection:(NSArray> *)selection whitelistSelection: (id) whitelistSelection; + +/** + Called when the user submits an HTML form. + */ +- (void)challengeResponseViewController:(STDSChallengeResponseViewController *)viewController didSubmitHTMLForm:(NSString *)form; + +/** + Called when the user taps the Continue button from an Out-of-Band flow (STDSACSUITypeOOB). + */ +- (void)challengeResponseViewControllerDidOOBContinue:(STDSChallengeResponseViewController *)viewController whitelistSelection: (id) whitelistSelection; + +/** + Called when the user taps the Cancel button. + */ +- (void)challengeResponseViewControllerDidCancel:(STDSChallengeResponseViewController *)viewController; + +/** + Called when the user taps the Resend button. + */ +- (void)challengeResponseViewControllerDidRequestResend:(STDSChallengeResponseViewController *)viewController; + +@end + +@protocol STDSChallengeResponseViewControllerPresentationDelegate + +- (void)dismissChallengeResponseViewController:(STDSChallengeResponseViewController *)viewController; + +@end + +@interface STDSChallengeResponseViewController : UIViewController + +@property (nonatomic, weak) id delegate; + +@property (nonatomic, nullable, weak) id presentationDelegate; + +/// Use setChallengeResponser:animated: to update this value +@property (nonatomic, strong, readonly) id response; + +- (instancetype)initWithUICustomization:(STDSUICustomization * _Nullable)uiCustomization imageLoader:(STDSImageLoader *)imageLoader directoryServer:(STDSDirectoryServer)directoryServer; + +/// If `setLoading` was called beforehand, this waits until the loading spinner has been shown for at least 1 second before displaying the challenge responseself.processingView.isHidden. +- (void)setChallengeResponse:(id)response animated:(BOOL)animated; + +- (void)setLoading; + +- (void)dismiss; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.m b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.m new file mode 100644 index 00000000..7f313c8a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeResponseViewController.m @@ -0,0 +1,571 @@ +// +// STDSChallengeResponseViewController.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +@import WebKit; + +#import "STDSBundleLocator.h" +#import "STDSLocalizedString.h" +#import "STDSChallengeResponseViewController.h" +#import "STDSImageLoader.h" +#import "STDSStackView.h" +#import "STDSBrandingView.h" +#import "STDSChallengeInformationView.h" +#import "STDSChallengeSelectionView.h" +#import "STDSTextChallengeView.h" +#import "STDSWhitelistView.h" +#import "STDSExpandableInformationView.h" +#import "STDSWebView.h" +#import "STDSProcessingView.h" +#import "UIView+LayoutSupport.h" +#import "NSString+EmptyChecking.h" +#import "UIColor+DefaultColors.h" +#import "UIButton+CustomInitialization.h" +#import "UIFont+DefaultFonts.h" +#import "UIViewController+Stripe3DS2.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeResponseViewController() + +@property (nonatomic, strong, nullable) id response; +@property (nonatomic) STDSDirectoryServer directoryServer; +/// Used to track how long we've been showing a loading spinner. Nil if we are not showing a spinner. +@property (nonatomic, strong, nullable) NSDate *loadingStartDate; +@property (nonatomic, strong, nullable) STDSUICustomization *uiCustomization; +@property (nonatomic, strong) STDSImageLoader *imageLoader; +@property (nonatomic, strong) NSTimer *processingTimer; +@property (nonatomic, getter=isLoading) BOOL loading; +@property (nonatomic, strong) STDSProcessingView *processingView; +@property (nonatomic, strong, nullable) UIScrollView *scrollView; +@property (nonatomic, strong, nullable) STDSWebView *webView; +@property (nonatomic, strong, nullable) STDSChallengeInformationView *challengeInformationView; +@property (nonatomic, strong) UITapGestureRecognizer *tapOutsideKeyboardGestureRecognizer; + +// User input views +@property (nonatomic, strong) STDSChallengeSelectionView *challengeSelectionView; +@property (nonatomic, strong) STDSTextChallengeView *textChallengeView; +@property (nonatomic, strong) STDSWhitelistView *whitelistView; +@property (nonatomic, strong) UIStackView *buttonStackView; +@end + +@implementation STDSChallengeResponseViewController + +static const NSTimeInterval kInterstepProcessingTime = 1.0; +static const NSTimeInterval kDefaultTransitionAnimationDuration = 0.3; +static const CGFloat kBrandingViewHeight = 107; +static const CGFloat kContentHorizontalInset = 16; +static const CGFloat kExpandableContentHorizontalInset = 27; +static const CGFloat kContentViewTopPadding = 16; +static const CGFloat kContentViewBottomPadding = 26; +static const CGFloat kExpandableContentViewTopPadding = 28; + +static NSString * const kHTMLStringLoadingURL = @"about:blank"; + +- (instancetype)initWithUICustomization:(STDSUICustomization * _Nullable)uiCustomization imageLoader:(STDSImageLoader *)imageLoader directoryServer:(STDSDirectoryServer)directoryServer { + self = [super initWithNibName:nil bundle:nil]; + + if (self) { + _uiCustomization = uiCustomization; + _imageLoader = imageLoader; + _tapOutsideKeyboardGestureRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(_didTapOutsideKeyboard:)]; + _directoryServer = directoryServer; + } + + return self; +} + +- (void)viewDidLoad { + [super viewDidLoad]; + [self _stds_setupNavigationBarElementsWithCustomization:_uiCustomization cancelButtonSelector:@selector(_cancelButtonTapped:)]; + self.view.backgroundColor = self.uiCustomization.backgroundColor; + + [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(_keyboardDidShow:) name:UIKeyboardDidShowNotification object:nil]; + [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(_keyboardWillHide:) name:UIKeyboardWillHideNotification object:nil]; + [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(_applicationWillEnterForeground:) name:UIApplicationWillEnterForegroundNotification object:nil]; + + NSString *imageName = STDSDirectoryServerImageName(self.directoryServer); + UIImage *dsImage = imageName ? [UIImage imageNamed:imageName inBundle:[STDSBundleLocator stdsResourcesBundle] compatibleWithTraitCollection:nil] : nil; + self.processingView = [[STDSProcessingView alloc] initWithCustomization:self.uiCustomization directoryServerLogo:dsImage]; + self.processingView.hidden = !self.isLoading; + + [self.view addSubview:self.processingView]; + [self.processingView _stds_pinToSuperviewBoundsWithoutMargin]; + + [self.view addGestureRecognizer:self.tapOutsideKeyboardGestureRecognizer]; +} + +- (UIStatusBarStyle)preferredStatusBarStyle { + return self.uiCustomization.preferredStatusBarStyle; +} + +#pragma mark - Public APIs + +- (void)setLoading { + [self _setLoading:YES]; +} + +- (void)setChallengeResponse:(id)response animated:(BOOL)animated { + BOOL isFirstChallengeResponse = _response == nil; + _response = response; + + [self.processingTimer invalidate]; + + if (isFirstChallengeResponse || !self.isLoading || !self.loadingStartDate) { + [self _displayChallengeResponseAnimated:animated]; + } else { + // Show the loading spinner for at least kDefaultProcessingTime seconds before displaying + NSTimeInterval timeSpentLoading = [[NSDate date] timeIntervalSinceDate:self.loadingStartDate]; + if (timeSpentLoading >= kInterstepProcessingTime) { + // loadingStartDate is nil if we called this method in between viewDidLoad and viewDidAppear. + // There is no time requirement for the initial CRes. + [self _displayChallengeResponseAnimated:animated]; + } else { + self.processingTimer = [NSTimer timerWithTimeInterval:(kInterstepProcessingTime - timeSpentLoading) target:self selector:@selector(_timerDidFire:) userInfo:@(animated) repeats:NO]; + [[NSRunLoop currentRunLoop] addTimer:self.processingTimer forMode:NSDefaultRunLoopMode]; + } + } +} + +- (void)dismiss { + if (self.presentationDelegate) { + [self.presentationDelegate dismissChallengeResponseViewController:self]; + } else { + [self dismissViewControllerAnimated:YES completion:nil]; + } +} + +#pragma mark - Private Helpers + +- (void)_setLoading:(BOOL)isLoading { + self.loading = isLoading; + if (!self.viewLoaded || isLoading == !self.processingView.isHidden) { + return; + } + + /* According to the specs [0], this should be set to NO during AReq/Ares and YES during CReq/CRes. + However, according to UL test feedback [1], the AReq/ARes and initial CReq/CRes processing views should be identical. + + [0]: EMV 3-D Secure Protocol and Core Functions Specification v2.1.0 4.2.1.1 + - "The 3DS SDK shall for the CReq/CRes message exchange...[Req 148] Not include the DS logo or any other design element in the Processing screen." + - "The 3DS SDK shall for the AReq/ARes message exchange...[Req 143] If requested, integrate the DS logo into the Processing screen." + + [1]: UL_PreCompTestReport_ID846_201906_1.0 + - "Visual test case TC_SDK_10022_001 - The test case is FAILED because the processing screen for step 1 and step 2 are not identical. Step 1 displays a 'DS logo' while step 2 does not. + + To pass certification, we'll show the DS logo during the initial CReq/CRes (when self.response == nil). + */ + self.processingView.shouldDisplayDSLogo = self.response == nil; + // If there's no response, the blur view has nothing to blur and looks better visually if it's just the background color + // EDIT Jan 2021: The challenge contents is hidden so this never looks good https://jira.corp.stripe.com/browse/MOBILESDK-153 + self.processingView.shouldDisplayBlurView = NO; // self.response != nil; + + if (isLoading) { + [self.view bringSubviewToFront:self.processingView]; + self.processingView.hidden = NO; + + self.loadingStartDate = [NSDate date]; + UIAccessibilityPostNotification(UIAccessibilityScreenChangedNotification, STDSLocalizedString(@"Loading", @"Spoken by VoiceOver when the challenge is loading.")); + } else { + self.processingView.hidden = YES; + self.loadingStartDate = nil; + } +} + +- (void)_timerDidFire:(NSTimer *)timer { + BOOL animated = ((NSNumber *)timer.userInfo).boolValue; + [self.processingTimer invalidate]; + [self _displayChallengeResponseAnimated:animated]; +} + +- (void)_setupViewHierarchy { + self.scrollView = [[UIScrollView alloc] init]; + self.scrollView.backgroundColor = self.uiCustomization.footerCustomization.backgroundColor; + self.scrollView.alwaysBounceVertical = YES; + [self.view addSubview:self.scrollView]; + [self.scrollView _stds_pinToSuperviewBoundsWithoutMargin]; + + STDSStackView *containerStackView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + [self.scrollView addSubview:containerStackView]; + [containerStackView _stds_pinToSuperviewBoundsWithoutMargin]; + + UIView *contentView = [UIView new]; + contentView.layoutMargins = UIEdgeInsetsMake(kContentViewTopPadding, kContentHorizontalInset, kContentViewBottomPadding, kContentHorizontalInset); + contentView.backgroundColor = self.uiCustomization.backgroundColor; + [containerStackView addArrangedSubview:contentView]; + + STDSStackView *contentStackView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + [contentView addSubview:contentStackView]; + [contentStackView _stds_pinToSuperviewBounds]; + + STDSBrandingView *brandingView = [self _newConfiguredBrandingView]; + STDSChallengeInformationView *challengeInformationView = [self _newConfiguredChallengeInformationView]; + self.challengeInformationView = challengeInformationView; + UIButton *actionButton = [self _newConfiguredActionButton]; + UIButton *resendButton = [self _newConfiguredResendButton]; + STDSTextChallengeView *textChallengeView = [self _newConfiguredTextChallengeView]; + self.textChallengeView = textChallengeView; + STDSChallengeSelectionView *challengeSelectionView = [self _newConfiguredChallengeSelectionView]; + self.challengeSelectionView = challengeSelectionView; + self.whitelistView = [self _newConfiguredWhitelistView]; + + UIView *expandableContentView = [UIView new]; + expandableContentView.layoutMargins = UIEdgeInsetsMake(kExpandableContentViewTopPadding, kExpandableContentHorizontalInset, 0, kExpandableContentHorizontalInset); + [containerStackView addArrangedSubview:expandableContentView]; + + STDSStackView *expandableContentStackView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + [expandableContentView addSubview:expandableContentStackView]; + [expandableContentStackView _stds_pinToSuperviewBounds]; + + STDSExpandableInformationView *whyInformationView = [self _newConfiguredWhyInformationView]; + STDSExpandableInformationView *expandableInformationView = [self _newConfiguredExpandableInformationView]; + + [contentStackView addArrangedSubview:brandingView]; + [contentStackView addArrangedSubview:challengeInformationView]; + [contentStackView addArrangedSubview:textChallengeView]; + [contentStackView addArrangedSubview:challengeSelectionView]; + + self.buttonStackView = [self _newSubmitButtonStackView]; + + [self.buttonStackView addArrangedSubview:actionButton]; + + [contentStackView addArrangedSubview:self.buttonStackView]; + + if (_response.acsUIType != STDSACSUITypeOOB && _response.acsUIType != STDSACSUITypeMultiSelect && _response.acsUIType != STDSACSUITypeSingleSelect) { + [self.buttonStackView addArrangedSubview:resendButton]; + } + if (!self.whitelistView.isHidden) { + [contentStackView addSpacer:10]; + } + [contentStackView addArrangedSubview:self.whitelistView]; + [expandableContentStackView addArrangedSubview:whyInformationView]; + [expandableContentStackView addArrangedSubview:expandableInformationView]; + + NSLayoutConstraint *contentViewWidth = [NSLayoutConstraint constraintWithItem:containerStackView attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:self.scrollView attribute:NSLayoutAttributeWidth multiplier:1 constant:0]; + NSLayoutConstraint *brandingViewHeightConstraint = [NSLayoutConstraint constraintWithItem:brandingView attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:kBrandingViewHeight]; + [NSLayoutConstraint activateConstraints:@[brandingViewHeightConstraint, contentViewWidth]]; + + [self _loadBrandingViewImages:brandingView]; +} + +- (void)_setupWebView { + self.webView = [[STDSWebView alloc] init]; + self.webView.navigationDelegate = self; + [self.view addSubview:self.webView]; + [self.webView _stds_pinToSuperviewBounds]; + [self.webView loadExternalResourceBlockingHTMLString:self.response.acsHTML]; +} + +- (void)_loadBrandingViewImages:(STDSBrandingView *)brandingView { + NSURL *issuerImageURL = [self _highestFideltyURLFromChallengeResponseImage:self.response.issuerImage]; + + if (issuerImageURL != nil) { + [self.imageLoader loadImageFromURL:issuerImageURL completion:^(UIImage * _Nullable image) { + brandingView.issuerImage = image; + }]; + } + + NSURL *paymentSystemImageURL = [self _highestFideltyURLFromChallengeResponseImage:self.response.paymentSystemImage]; + + if (paymentSystemImageURL != nil) { + [self.imageLoader loadImageFromURL:paymentSystemImageURL completion:^(UIImage * _Nullable image) { + brandingView.paymentSystemImage = image; + }]; + } +} + +- (NSURL * _Nullable)_highestFideltyURLFromChallengeResponseImage:(id )image { + return image.extraHighDensityURL ?: image.highDensityURL ?: image.mediumDensityURL; +} + +- (void)_displayChallengeResponseAnimated:(BOOL)animated { + if (self.response != nil) { + [self _setLoading:NO]; + + UIScrollView *existingScrollView = self.scrollView; + STDSWebView *existingWebView = self.webView; + + void (^transitionBlock)(UIView *, BOOL) = ^void(UIView *viewToTransition, BOOL animated) { + NSTimeInterval transitionTime = animated ? kDefaultTransitionAnimationDuration : 0; + viewToTransition.alpha = 0; + [UIView animateWithDuration:transitionTime animations:^{ + viewToTransition.alpha = 1; + } completion:^(BOOL finished) { + [existingScrollView removeFromSuperview]; + [existingWebView removeFromSuperview]; + [[NSNotificationCenter defaultCenter] postNotificationName:@"STDSChallengeResponseViewController.didDisplayChallengeResponse" object:self]; + UIAccessibilityPostNotification(UIAccessibilityScreenChangedNotification, self.navigationItem.titleView); + }]; + }; + + switch (self.response.acsUIType) { + case STDSACSUITypeNone: + break; + case STDSACSUITypeText: + case STDSACSUITypeSingleSelect: + case STDSACSUITypeMultiSelect: + case STDSACSUITypeOOB: + [self _setupViewHierarchy]; + + transitionBlock(self.scrollView, animated); + break; + case STDSACSUITypeHTML: + [self _setupWebView]; + + transitionBlock(self.webView, animated); + break; + } + } +} + +- (STDSBrandingView *)_newConfiguredBrandingView { + STDSBrandingView *brandingView = [[STDSBrandingView alloc] init]; + brandingView.hidden = self.response.issuerImage == nil && self.response.paymentSystemImage == nil; + + return brandingView; +} + +- (STDSChallengeInformationView *)_newConfiguredChallengeInformationView { + STDSChallengeInformationView *challengeInformationView = [[STDSChallengeInformationView alloc] init]; + challengeInformationView.headerText = self.response.challengeInfoHeader; + challengeInformationView.challengeInformationText = self.response.challengeInfoText; + challengeInformationView.challengeInformationLabel = self.response.challengeInfoLabel; + challengeInformationView.labelCustomization = self.uiCustomization.labelCustomization; + + if (self.response.showChallengeInfoTextIndicator) { + challengeInformationView.textIndicatorImage = [UIImage imageNamed:@"error" inBundle:[STDSBundleLocator stdsResourcesBundle] compatibleWithTraitCollection:nil]; + } + + return challengeInformationView; +} + +- (STDSTextChallengeView *)_newConfiguredTextChallengeView { + STDSTextChallengeView *textChallengeView = [[STDSTextChallengeView alloc] init]; + textChallengeView.hidden = self.response.acsUIType != STDSACSUITypeText; + textChallengeView.textFieldCustomization = self.uiCustomization.textFieldCustomization; + textChallengeView.textField.accessibilityLabel = self.response.challengeInfoLabel; + textChallengeView.backgroundColor = self.uiCustomization.backgroundColor; + + return textChallengeView; +} + +- (STDSChallengeSelectionView *)_newConfiguredChallengeSelectionView { + STDSChallengeSelectionStyle selectionStyle = self.response.acsUIType == STDSACSUITypeMultiSelect ? STDSChallengeSelectionStyleMulti : STDSChallengeSelectionStyleSingle; + STDSChallengeSelectionView *challengeSelectionView = [[STDSChallengeSelectionView alloc] initWithChallengeSelectInfo:self.response.challengeSelectInfo selectionStyle:selectionStyle]; + challengeSelectionView.hidden = self.response.acsUIType != STDSACSUITypeSingleSelect && self.response.acsUIType != STDSACSUITypeMultiSelect; + challengeSelectionView.labelCustomization = self.uiCustomization.labelCustomization; + challengeSelectionView.selectionCustomization = self.uiCustomization.selectionCustomization; + challengeSelectionView.backgroundColor = self.uiCustomization.backgroundColor; + + return challengeSelectionView; +} + +- (UIButton *)_newConfiguredActionButton { + STDSUICustomizationButtonType buttonType = STDSUICustomizationButtonTypeSubmit; + NSString *buttonTitle; + + switch (self.response.acsUIType) { + case STDSACSUITypeNone: + break; + case STDSACSUITypeText: + case STDSACSUITypeSingleSelect: + case STDSACSUITypeMultiSelect: { + buttonTitle = self.response.submitAuthenticationLabel; + + break; + } + case STDSACSUITypeOOB: { + buttonType = STDSUICustomizationButtonTypeContinue; + buttonTitle = self.response.oobContinueLabel; + + break; + } + case STDSACSUITypeHTML: + break; + } + + STDSButtonCustomization *buttonCustomization = [self.uiCustomization buttonCustomizationForButtonType:buttonType]; + UIButton *actionButton = [UIButton _stds_buttonWithTitle:buttonTitle customization:buttonCustomization]; + [actionButton addTarget:self action:@selector(_actionButtonTapped:) forControlEvents:UIControlEventTouchUpInside]; + actionButton.hidden = buttonTitle == nil || [NSString _stds_isStringEmpty:buttonTitle]; + actionButton.accessibilityIdentifier = @"Continue"; + + return actionButton; +} + +- (UIButton *)_newConfiguredResendButton { + STDSButtonCustomization *buttonCustomization = [self.uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeResend]; + + NSString *resendButtonTitle = self.response.resendInformationLabel; + UIButton *resendButton = [UIButton _stds_buttonWithTitle:resendButtonTitle customization:buttonCustomization]; + + resendButton.hidden = resendButtonTitle == nil || [NSString _stds_isStringEmpty:resendButtonTitle]; + [resendButton addTarget:self action:@selector(_resendButtonTapped:) forControlEvents:UIControlEventTouchUpInside]; + + return resendButton; +} + +- (STDSWhitelistView *)_newConfiguredWhitelistView { + STDSWhitelistView *whitelistView = [[STDSWhitelistView alloc] init]; + whitelistView.whitelistText = self.response.whitelistingInfoText; + whitelistView.labelCustomization = self.uiCustomization.labelCustomization; + whitelistView.selectionCustomization = self.uiCustomization.selectionCustomization; + whitelistView.hidden = whitelistView.whitelistText == nil; + whitelistView.accessibilityIdentifier = @"STDSWhitelistView"; + + return whitelistView; +} + +- (STDSExpandableInformationView *)_newConfiguredWhyInformationView { + STDSExpandableInformationView *whyInformationView = [[STDSExpandableInformationView alloc] init]; + whyInformationView.title = self.response.whyInfoLabel; + whyInformationView.text = self.response.whyInfoText; + whyInformationView.customization = self.uiCustomization.footerCustomization; + whyInformationView.hidden = whyInformationView.title == nil; + whyInformationView.backgroundColor = self.uiCustomization.footerCustomization.backgroundColor; + __weak typeof(self) weakSelf = self; + whyInformationView.didTap = ^{ + [weakSelf.textChallengeView endEditing:NO]; + }; + + return whyInformationView; +} + +- (STDSExpandableInformationView *)_newConfiguredExpandableInformationView { + + STDSExpandableInformationView *expandableInformationView = [[STDSExpandableInformationView alloc] init]; + expandableInformationView.title = self.response.expandInfoLabel; + expandableInformationView.text = self.response.expandInfoText; + expandableInformationView.customization = self.uiCustomization.footerCustomization; + expandableInformationView.hidden = expandableInformationView.title == nil; + expandableInformationView.backgroundColor = self.uiCustomization.footerCustomization.backgroundColor; + __weak typeof(self) weakSelf = self; + expandableInformationView.didTap = ^{ + [weakSelf.textChallengeView endEditing:NO]; + }; + + return expandableInformationView; +} + +- (UIStackView *)_newSubmitButtonStackView { + UIStackView *stackView = [[UIStackView alloc] init]; + stackView.axis = UILayoutConstraintAxisVertical; + stackView.distribution = UIStackViewDistributionFillEqually; + stackView.alignment = UIStackViewAlignmentFill; + stackView.spacing = 5; + stackView.translatesAutoresizingMaskIntoConstraints = NO; + + CGSize size = [UIScreen mainScreen].bounds.size; + if (size.width > size.height) { + // hack to detect landscape + stackView.axis = UILayoutConstraintAxisHorizontal; + stackView.alignment = UIStackViewAlignmentCenter; + } + return stackView; +} + +- (void)_keyboardDidShow:(NSNotification *)notification { + CGSize keyboardSize = [[[notification userInfo] objectForKey:UIKeyboardFrameEndUserInfoKey] CGRectValue].size; + UIEdgeInsets contentInsets = UIEdgeInsetsMake(self.scrollView.contentInset.top, 0.0, keyboardSize.height, 0.0); + self.scrollView.contentInset = contentInsets; + self.scrollView.scrollIndicatorInsets = contentInsets; +} + +- (void)_keyboardWillHide:(NSNotification *)notification { + UIEdgeInsets contentInsets = UIEdgeInsetsMake(self.scrollView.contentInset.top, 0.0, 0.0, 0.0); + self.scrollView.contentInset = contentInsets; + self.scrollView.scrollIndicatorInsets = contentInsets; +} + +- (void)_applicationWillEnterForeground:(NSNotification *)notification { + if (self.response.acsUIType == STDSACSUITypeOOB && self.response.challengeAdditionalInfoText) { + // [Req 316] When Challenge Additional Information Text is present, the SDK would replace the Challenge Information Text and Challenge Information Text Indicator with the Challenge Additional Information Text when the 3DS Requestor App is moved to the foreground. + self.challengeInformationView.challengeInformationText = self.response.challengeAdditionalInfoText; + self.challengeInformationView.textIndicatorImage = nil; + } else if (self.response.acsUIType == STDSACSUITypeHTML && self.response.acsHTMLRefresh) { + // [Req 317] When the ACS HTML Refresh element is present, the SDK replaces the ACS HTML with the contents of ACS HTML Refresh when the 3DS Requestor App is moved to the foreground. + [self.webView loadExternalResourceBlockingHTMLString:self.response.acsHTMLRefresh]; + } +} + +- (void)_didTapOutsideKeyboard:(UIGestureRecognizer *)gestureRecognizer { + // Note this doesn't fire if a subview handles the touch (e.g. UIControls, STDSExpandableInformationView) + [self.textChallengeView endEditing:NO]; +} + +#pragma mark - Button callbacks + +- (void)_cancelButtonTapped:(UIButton *)sender { + [self.textChallengeView endEditing:NO]; + [self.delegate challengeResponseViewControllerDidCancel:self]; +} + +- (void)_resendButtonTapped:(UIButton *)sender { + [self.textChallengeView endEditing:NO]; + [self.delegate challengeResponseViewControllerDidRequestResend:self]; +} + +- (void)_actionButtonTapped:(UIButton *)sender { + [self.textChallengeView endEditing:NO]; + switch (self.response.acsUIType) { + case STDSACSUITypeNone: + break; + case STDSACSUITypeText: { + [self.delegate challengeResponseViewController:self + didSubmitInput:self.textChallengeView.inputText + whitelistSelection:self.whitelistView.selectedResponse]; + break; + } + case STDSACSUITypeSingleSelect: + case STDSACSUITypeMultiSelect: { + [self.delegate challengeResponseViewController:self + didSubmitSelection:self.challengeSelectionView.currentlySelectedChallengeInfo + whitelistSelection:self.whitelistView.selectedResponse]; + break; + } + case STDSACSUITypeOOB: + [self.delegate challengeResponseViewControllerDidOOBContinue:self + whitelistSelection:self.whitelistView.selectedResponse]; + break; + case STDSACSUITypeHTML: + // No action button in this case, see WKNavigationDelegate. + break; + } +} + +#pragma mark - WKNavigationDelegate + +- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler { + + NSURLRequest *request = navigationAction.request; + + if ([request.URL.absoluteString isEqualToString:kHTMLStringLoadingURL]) { + return decisionHandler(WKNavigationActionPolicyAllow); + } else { + if (navigationAction.navigationType == WKNavigationTypeFormSubmitted || navigationAction.navigationType == WKNavigationTypeOther) { + // When the Cardholder’s response is returned as a parameter string, the form data is passed to the web view instance by triggering a location change to a specified (HTTPS://EMV3DS/challenge) URL with the challenge responses appended to the location URL as query parameters (for example, HTTPS://EMV3DS/challenge?city=Pittsburgh). The web view instance, because it monitors URL changes, receives the Cardholder’s responses as query parameters. + [self.delegate challengeResponseViewController:self didSubmitHTMLForm:request.URL.query]; + } + + return decisionHandler(WKNavigationActionPolicyCancel); + } +} + +- (void) viewWillTransitionToSize:(CGSize)size withTransitionCoordinator:(id)coordinator { + if (size.width > size.height) { + // hack to detect landscape + self.buttonStackView.axis = UILayoutConstraintAxisHorizontal; + self.buttonStackView.alignment = UIStackViewAlignmentCenter; + } else { + self.buttonStackView.axis = UILayoutConstraintAxisVertical; + self.buttonStackView.alignment = UIStackViewAlignmentFill; + } +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.h b/Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.h new file mode 100644 index 00000000..5ac5be0d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.h @@ -0,0 +1,35 @@ +// +// STDSChallengeSelectionView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/6/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponseSelectionInfo.h" +#import "STDSLabelCustomization.h" +#import "STDSSelectionCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +typedef NS_ENUM(NSInteger, STDSChallengeSelectionStyle) { + + /// A display style for selecting a single option. + STDSChallengeSelectionStyleSingle = 0, + + /// A display style for selection multiple options. + STDSChallengeSelectionStyleMulti = 1, +}; + +@interface STDSChallengeSelectionView : UIView + +@property (nonatomic, strong, readonly) NSArray> *currentlySelectedChallengeInfo; +@property (nonatomic, strong) STDSLabelCustomization *labelCustomization; +@property (nonatomic, strong) STDSSelectionCustomization *selectionCustomization; + +- (instancetype)initWithChallengeSelectInfo:(NSArray> *)challengeSelectInfo selectionStyle:(STDSChallengeSelectionStyle)selectionStyle; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.m b/Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.m new file mode 100644 index 00000000..0b4d15b3 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSChallengeSelectionView.m @@ -0,0 +1,255 @@ +// +// STDSChallengeSelectionView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/6/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSLocalizedString.h" +#import "STDSBundleLocator.h" +#import "STDSChallengeSelectionView.h" +#import "STDSStackView.h" +#import "UIView+LayoutSupport.h" +#import "STDSSelectionButton.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeResponseSelectionRow: STDSStackView + +typedef NS_ENUM(NSInteger, STDSChallengeResponseSelectionRowStyle) { + + /// A display style for showing a radio button. + STDSChallengeResponseSelectionRowStyleRadio = 0, + + /// A display style for shows a checkbox. + STDSChallengeResponseSelectionRowStyleCheckbox = 1, +}; + +typedef void (^STDSChallengeResponseRowSelectedBlock)(STDSChallengeResponseSelectionRow *); + +@property (nonatomic, strong, readonly) id challengeSelectInfo; +@property (nonatomic, getter=isSelected) BOOL selected; +@property (nonatomic, strong) STDSLabelCustomization *labelCustomization; +@property (nonatomic, strong) STDSSelectionCustomization *selectionCustomization; + +- (instancetype)initWithChallengeSelectInfo:(id)challengeSelectInfo rowStyle:(STDSChallengeResponseSelectionRowStyle)rowStyle rowSelectedBlock:(STDSChallengeResponseRowSelectedBlock)rowSelectedBlock; + +@end + +@interface STDSChallengeResponseSelectionRow() + +@property (nonatomic, strong) id challengeSelectInfo; +@property (nonatomic, strong) STDSChallengeResponseRowSelectedBlock rowSelectedBlock; +@property (nonatomic) STDSChallengeResponseSelectionRowStyle rowStyle; +@property (nonatomic, strong) STDSSelectionButton *selectionButton; +@property (nonatomic, strong) UILabel *valueLabel; +@property (nonatomic, strong) UITapGestureRecognizer *valueLabelTapRecognizer; + +@end + +@implementation STDSChallengeResponseSelectionRow + +- (instancetype)initWithChallengeSelectInfo:(id)challengeSelectInfo rowStyle:(STDSChallengeResponseSelectionRowStyle)rowStyle rowSelectedBlock:(STDSChallengeResponseRowSelectedBlock)rowSelectedBlock { + self = [super initWithAlignment:STDSStackViewLayoutAxisHorizontal]; + + if (self) { + _challengeSelectInfo = challengeSelectInfo; + _rowStyle = rowStyle; + _rowSelectedBlock = rowSelectedBlock; + self.isAccessibilityElement = YES; + self.accessibilityIdentifier = @"STDSChallengeResponseSelectionRow"; + + [self _setupViewHierarchy]; + } + + return self; +} + +- (void)_setupViewHierarchy { + self.selectionButton = [[STDSSelectionButton alloc] initWithCustomization:self.selectionCustomization]; + self.selectionButton.customization = self.selectionCustomization; + [self.selectionButton addTarget:self action:@selector(_rowWasSelected) forControlEvents:UIControlEventTouchUpInside]; + + if (self.rowStyle == STDSChallengeResponseSelectionRowStyleCheckbox) { + self.selectionButton.isCheckbox = YES; + } + + self.valueLabel = [[UILabel alloc] init]; + self.valueLabel.text = self.challengeSelectInfo.value; + self.valueLabel.userInteractionEnabled = YES; + self.valueLabel.numberOfLines = 0; + self.valueLabelTapRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(_rowWasSelected)]; + [self.valueLabel addGestureRecognizer:self.valueLabelTapRecognizer]; + + [self addArrangedSubview:self.selectionButton]; + [self addSpacer:15.0]; + [self addArrangedSubview:self.valueLabel]; + [self addArrangedSubview:[UIView new]]; +} + +- (void)_rowWasSelected { + self.rowSelectedBlock(self); +} + +- (BOOL)isSelected { + /// Placeholder until visual and interaction design is complete. + return self.selectionButton.isSelected; +} + +- (void)setSelected:(BOOL)selected { + /// Placeholder until visual and interaction design is complete. + self.selectionButton.selected = selected; +} + +- (void)setLabelCustomization:(STDSLabelCustomization *)labelCustomization { + _labelCustomization = labelCustomization; + + self.valueLabel.font = labelCustomization.font; + self.valueLabel.textColor = labelCustomization.textColor; +} + +- (void)setSelectionCustomization:(STDSSelectionCustomization *)selectionCustomization { + _selectionCustomization = selectionCustomization; + + self.selectionButton.customization = selectionCustomization; +} + +#pragma mark - UIAccessibility + +- (BOOL)accessibilityActivate { + self.rowSelectedBlock(self); + return YES; +} + +- (nullable NSString *)accessibilityLabel { + return self.valueLabel.text; +} + +- (nullable NSString *)accessibilityValue { + return self.selected ? STDSLocalizedString(@"Selected", @"Indicates that a button is selected.") : STDSLocalizedString(@"Unselected", @"Indicates that a button is not selected."); +} + +- (UIAccessibilityTraits)accessibilityTraits { + // remove the selected trait since we manually add that as an accessibilityValue above + return (self.selectionButton.accessibilityTraits & ~UIAccessibilityTraitSelected); +} + +@end + +@interface STDSChallengeSelectionView() + +@property (nonatomic, strong) STDSStackView *containerView; +@property (nonatomic, strong) NSArray *challengeSelectionRows; + +@property (nonatomic) STDSChallengeSelectionStyle selectionStyle; + +@end + +@implementation STDSChallengeSelectionView + +static const CGFloat kChallengeSelectionViewTopPadding = 5; +static const CGFloat kChallengeSelectionViewBottomPadding = 20; +static const CGFloat kChallengeSelectionViewInterRowVerticalPadding = 16; + +- (instancetype)initWithChallengeSelectInfo:(NSArray> *)challengeSelectInfo selectionStyle:(STDSChallengeSelectionStyle)selectionStyle { + self = [super init]; + + if (self) { + _selectionStyle = selectionStyle; + _challengeSelectionRows = [self _rowsForChallengeSelectInfo:challengeSelectInfo]; + + [self _setupViewHierarchy]; + } + + return self; +} + +- (void)_setupViewHierarchy { + self.containerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + + for (STDSChallengeResponseSelectionRow *selectionRow in self.challengeSelectionRows) { + [self.containerView addArrangedSubview:selectionRow]; + + if (selectionRow != self.challengeSelectionRows.lastObject) { + [self.containerView addSpacer:kChallengeSelectionViewInterRowVerticalPadding]; + } + } + + if (self.challengeSelectionRows.count > 0) { + self.layoutMargins = UIEdgeInsetsMake(kChallengeSelectionViewTopPadding, 0, kChallengeSelectionViewBottomPadding, 0); + } else { + self.layoutMargins = UIEdgeInsetsZero; + } + + [self addSubview:self.containerView]; + [self.containerView _stds_pinToSuperviewBounds]; +} + +- (NSArray *)_rowsForChallengeSelectInfo:(NSArray> *)challengeSelectInfo { + NSMutableArray *challengeRows = [NSMutableArray array]; + STDSChallengeResponseSelectionRowStyle rowStyle = self.selectionStyle == STDSChallengeSelectionStyleSingle ? STDSChallengeResponseSelectionRowStyleRadio : STDSChallengeResponseSelectionRowStyleCheckbox; + + for (id selectionInfo in challengeSelectInfo) { + __weak typeof(self) weakSelf = self; + STDSChallengeResponseSelectionRow *challengeRow = [[STDSChallengeResponseSelectionRow alloc] initWithChallengeSelectInfo:selectionInfo rowStyle:rowStyle rowSelectedBlock:^(STDSChallengeResponseSelectionRow * _Nonnull selectedRow) { + __strong typeof(self) strongSelf = weakSelf; + + [strongSelf _rowWasSelected:selectedRow]; + }]; + + if (selectionInfo == challengeSelectInfo.firstObject && self.selectionStyle == STDSChallengeSelectionStyleSingle) { + challengeRow.selected = YES; + } + + [challengeRows addObject:challengeRow]; + } + + return [challengeRows copy]; +} + +- (void)_rowWasSelected:(STDSChallengeResponseSelectionRow *)selectedRow { + switch (self.selectionStyle) { + case STDSChallengeSelectionStyleSingle: + for (STDSChallengeResponseSelectionRow *row in self.challengeSelectionRows) { + row.selected = row == selectedRow; + } + + break; + case STDSChallengeSelectionStyleMulti: + selectedRow.selected = !selectedRow.isSelected; + break; + } +} + +- (NSArray> *)currentlySelectedChallengeInfo { + NSMutableArray *selectedChallengeInfo = [NSMutableArray array]; + + for (STDSChallengeResponseSelectionRow *selectionRow in self.challengeSelectionRows) { + if (selectionRow.isSelected) { + [selectedChallengeInfo addObject:selectionRow.challengeSelectInfo]; + } + } + + return [selectedChallengeInfo copy]; +} + +- (void)setLabelCustomization:(STDSLabelCustomization *)labelCustomization { + _labelCustomization = labelCustomization; + + for (STDSChallengeResponseSelectionRow *row in self.challengeSelectionRows) { + row.labelCustomization = labelCustomization; + } +} + +- (void)setSelectionCustomization:(STDSSelectionCustomization *)selectionCustomization { + _selectionCustomization = selectionCustomization; + + for (STDSChallengeResponseSelectionRow *row in self.challengeSelectionRows) { + row.selectionCustomization = selectionCustomization; + } +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.h b/Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.h new file mode 100644 index 00000000..a50689f9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.h @@ -0,0 +1,19 @@ +// +// STDSDebuggerChecker.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDebuggerChecker : NSObject + ++ (BOOL)processIsCurrentlyAttachedToDebugger; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.m b/Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.m new file mode 100644 index 00000000..cbe3a059 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDebuggerChecker.m @@ -0,0 +1,54 @@ +// +// STDSDebuggerChecker.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSDebuggerChecker.h" + +#include +#include +#include +#include +#include + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSDebuggerChecker + +// This checking code has been lifted from the apple documentation on how to determine if you're attached to a debugger: https://developer.apple.com/library/archive/qa/qa1361/_index.html ++ (BOOL)processIsCurrentlyAttachedToDebugger { + int junk; + int mib[4]; + struct kinfo_proc info; + size_t size; + + // Initialize the flags so that, if sysctl fails for some bizarre + // reason, we get a predictable result. + + info.kp_proc.p_flag = 0; + + // Initialize mib, which tells sysctl the info we want, in this case + // we're looking for information about a specific process ID. + + mib[0] = CTL_KERN; + mib[1] = KERN_PROC; + mib[2] = KERN_PROC_PID; + mib[3] = getpid(); + + // Call sysctl. + + size = sizeof(info); + junk = sysctl(mib, sizeof(mib) / sizeof(*mib), &info, &size, NULL, 0); + assert(junk == 0); + + // We're being debugged if the P_TRACED flag is set. + + return ( (info.kp_proc.p_flag & P_TRACED) != 0 ); +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDeviceInformation.h b/Stripe3DS2/Stripe3DS2/STDSDeviceInformation.h new file mode 100644 index 00000000..d73e1872 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDeviceInformation.h @@ -0,0 +1,21 @@ +// +// STDSDeviceInformation.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDeviceInformation : NSObject + +- (instancetype)initWithDictionary:(NSDictionary *)deviceInformationDict; + +@property (nonatomic, copy, readonly) NSDictionary *dictionaryValue; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDeviceInformation.m b/Stripe3DS2/Stripe3DS2/STDSDeviceInformation.m new file mode 100644 index 00000000..92425435 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDeviceInformation.m @@ -0,0 +1,30 @@ +// +// STDSDeviceInformation.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSDeviceInformation.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSDeviceInformation + +- (instancetype)initWithDictionary:(NSDictionary *)deviceInformationDict { + self = [super init]; + if (self) { + _dictionaryValue = [deviceInformationDict copy]; + } + + return self; +} + +- (NSString *)description { + return [NSString stringWithFormat:@"%@ : %@", [super description], _dictionaryValue]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.h b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.h new file mode 100644 index 00000000..b23abfed --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.h @@ -0,0 +1,23 @@ +// +// STDSDeviceInformationManager.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class STDSDeviceInformation; +@class STDSWarning; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDeviceInformationManager : NSObject + ++ (STDSDeviceInformation *)deviceInformationWithWarnings:(NSArray *)warnings + ignoringRestrictions:(BOOL)ignoreRestrictions; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.m b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.m new file mode 100644 index 00000000..5bd65e90 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationManager.m @@ -0,0 +1,65 @@ +// +// STDSDeviceInformationManager.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSDeviceInformationManager.h" + +#import "STDSDeviceInformation.h" +#import "STDSDeviceInformationParameter.h" +#import "STDSWarning.h" + +NS_ASSUME_NONNULL_BEGIN + +// TC_SDK_10089_001, Req 2 & 5 +static const NSString * const k3DSDataVersion = @"1.4"; + +static const NSString * const kDataVersionKey = @"DV"; +static const NSString * const kDeviceDataKey = @"DD"; +static const NSString * const kDeviceParameterNotAvailableKey = @"DPNA"; +static const NSString * const kDeviceWarningsKey = @"SW"; + +@implementation STDSDeviceInformationManager + ++ (STDSDeviceInformation *)deviceInformationWithWarnings:(NSArray *)warnings + ignoringRestrictions:(BOOL)ignoreRestrictions { + NSMutableDictionary *deviceInformation = [NSMutableDictionary dictionaryWithObject:k3DSDataVersion forKey:kDataVersionKey]; + + for (STDSDeviceInformationParameter *parameter in [STDSDeviceInformationParameter allParameters]) { + + [parameter collectIgnoringRestrictions:ignoreRestrictions withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + if (collected) { + NSMutableDictionary *deviceData = deviceInformation[kDeviceDataKey]; + if (deviceData == nil) { + deviceData = [NSMutableDictionary dictionary]; + deviceInformation[kDeviceDataKey] = deviceData; + } + deviceData[identifier] = value; + } else { + NSMutableDictionary *notAvailableData = deviceInformation[kDeviceParameterNotAvailableKey]; + if (notAvailableData == nil) { + notAvailableData = [NSMutableDictionary dictionary]; + deviceInformation[kDeviceParameterNotAvailableKey] = notAvailableData; + } + notAvailableData[identifier] = value; + } + }]; + } + + NSMutableArray *warningIDs = [NSMutableArray arrayWithCapacity:warnings.count]; + for (STDSWarning *warning in warnings) { + [warningIDs addObject:warning.identifier]; + } + if (warningIDs.count > 0) { + deviceInformation[kDeviceWarningsKey] = [warningIDs copy]; + } + + return [[STDSDeviceInformation alloc] initWithDictionary:deviceInformation]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter+Private.h b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter+Private.h new file mode 100644 index 00000000..22d9f1a1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter+Private.h @@ -0,0 +1,76 @@ +// +// STDSDeviceInformationParameter+Private.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSDeviceInformationParameter.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDeviceInformationParameter (Private) + +- (instancetype)initWithIdentifier:(NSString *)identifier + permissionCheck:(nullable BOOL (^)(void))permissionCheck + valueCheck:(id _Nullable (^)(void))valueCheck; + +/// Platform: Platform that the device is using ++ (instancetype)platform; +/// Device Model: Mobile device manufacturer and model ++ (instancetype)deviceModel; +/// OS Name: Operating system name ++ (instancetype)OSName; +/// OS Version: Operating system version ++ (instancetype)OSVersion; +/// Locale: Device locale set by the user ++ (instancetype)locale; +/// Time zone: Device time zone ++ (instancetype)timeZone; +/// Advertising ID: Unique ID available for adertising and fraud detection purposes ++ (instancetype)advertisingID; +/// Screen Resolution: Pixel width and pixel height ++ (instancetype)screenResolution; +/// Device Name: User-assigned device name ++ (instancetype)deviceName; +/// IP Address: IP address of device ++ (instancetype)IPAddress; +/// Latitude: Device physical location latitude ++ (instancetype)latitude; +/// Longitude: Device physical location longitude ++ (instancetype)longitude; + +/// Identifier for Vendor: Alphanumeric string that uniquely ideitifies a device to the app's vendor ++ (instancetype)identiferForVendor; +/// UserInterfaceIdiom: Style of interface to use on the current device ++ (instancetype)userInterfaceIdiom; + +/// familyNames: an array of font family names available on the system ++ (instancetype)familyNames; +/// fontNamesForFamilyName: an array of font names available in a particular font family using the system font family ++ (instancetype)fontNamesForFamilyName; +/// systemFont: System font ++ (instancetype)systemFont; +/// labelFontSize: standard font size used for labels ++ (instancetype)labelFontSize; +/// buttonFontSize: standard font size used for buttons ++ (instancetype)buttonFontSize; +/// smallSystemFontSize: size of the standard small system font ++ (instancetype)smallSystemFontSize; +/// systemFontSize: size of the standard system font ++ (instancetype)systemFontSize; + +/// systemLocale: the ID of the generic locale that contains fixed "backstop" settings that provide values for otherwise undefined keys ++ (instancetype)systemLocale; +/// availableLocaleIdentifiers: an array of NSString objecgts, each of which identifies a locale available on the system ++ (instancetype)availableLocaleIdentifiers; +/// preferredLanguages: the user's language preference order as an array of strings ++ (instancetype)preferredLanguages; + +/// defaultTimeZone: the default time zone for the current application ++ (instancetype)defaultTimeZone; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.h b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.h new file mode 100644 index 00000000..46a3f66b --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.h @@ -0,0 +1,24 @@ +// +// STDSDeviceInformationParameter.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDeviceInformationParameter : NSObject + ++ (NSArray *)allParameters; + +/// Returns a UUID unique to the app version ++ (NSString *)sdkAppIdentifier; + +- (void)collectIgnoringRestrictions:(BOOL)ignoreRestrictions withHandler:(void (^)(BOOL, NSString *, id))handler; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.m b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.m new file mode 100644 index 00000000..4633a111 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDeviceInformationParameter.m @@ -0,0 +1,449 @@ +// +// STDSDeviceInformationParameter.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSDeviceInformationParameter.h" + +#import +#import + +#import "STDSIPAddress.h" +#import "STDSSynchronousLocationManager.h" + +NS_ASSUME_NONNULL_BEGIN + +// Code value to use if the parameter is restricted by the region or market +static const NSString * const kParameterRestrictedCode = @"RE01"; +// Code value to use if the platform version does not support the parameter or the parameter has been deprecated +static const NSString * const kParameterUnavailableCode = @"RE02"; +// Code value to use if parameter collection not possible without prompting the user for permission +static const NSString * const kParameterMissingPermissionsCode = @"RE03"; +// Code value to use if parameter value returned is null or blank +static const NSString * const kParameterNilCode = @"RE04"; + +@implementation STDSDeviceInformationParameter +{ + NSString *_identifier; + BOOL (^ _Nullable _permissionCheck)(void); + id (^_valueCheck)(void); +} + +- (instancetype)initWithIdentifier:(NSString *)identifier + permissionCheck:(nullable BOOL (^)(void))permissionCheck + valueCheck:(id (^)(void))valueCheck { + self = [super init]; + if (self) { + _identifier = [identifier copy]; + _permissionCheck = [permissionCheck copy]; + _valueCheck = [valueCheck copy]; + } + + return self; +} + +- (BOOL)_hasPermissions { + if (_permissionCheck == nil) { + return YES; + } + return _permissionCheck(); +} + +- (BOOL)_isRestricted { + static NSSet *sApprovedParameters = nil; + static dispatch_once_t onceToken; + dispatch_once(&onceToken, ^{ + sApprovedParameters = [NSSet setWithObjects: + // platform + @"C001", + // device model + @"C002", + // OS name + @"C003", + // OS version + @"C004", + // locale + @"C005", + // time zone + @"C006", + // advertising id (i.e. hardware id) + @"C007", + // screen solution + @"C008", + nil + ]; + }); + + return ![sApprovedParameters containsObject:_identifier]; +} + +- (void)collectIgnoringRestrictions:(BOOL)ignoreRestrictions withHandler:(void (^)(BOOL, NSString *, id))handler { + if (!ignoreRestrictions && [self _isRestricted]) { + handler(NO, _identifier, kParameterRestrictedCode); + return; + } else if (![self _hasPermissions]) { + handler(NO, _identifier, kParameterMissingPermissionsCode); + return; + } + + NSAssert(_valueCheck != nil, @"STDSDeviceInformationParameter should not have nil _valueCheck."); + id value = _valueCheck != nil ? _valueCheck() : nil; + + handler(value != nil, _identifier, value ?: kParameterUnavailableCode); +} + ++ (NSArray *)allParameters { + static NSArray *allParameters = nil; + static dispatch_once_t onceToken; + dispatch_once(&onceToken, ^{ + allParameters = @[ + +#pragma mark - Common Parameters + + [STDSDeviceInformationParameter platform], + [STDSDeviceInformationParameter deviceModel], + [STDSDeviceInformationParameter OSName], + [STDSDeviceInformationParameter OSVersion], + [STDSDeviceInformationParameter locale], + [STDSDeviceInformationParameter timeZone], + [STDSDeviceInformationParameter advertisingID], + [STDSDeviceInformationParameter screenResolution], + [STDSDeviceInformationParameter deviceName], + [STDSDeviceInformationParameter IPAddress], + [STDSDeviceInformationParameter latitude], + [STDSDeviceInformationParameter longitude], + [STDSDeviceInformationParameter applicationPackageName], + [STDSDeviceInformationParameter sdkAppId], + [STDSDeviceInformationParameter sdkVersion], + + +#pragma mark - iOS-Specific Parameters + + [STDSDeviceInformationParameter identiferForVendor], + [STDSDeviceInformationParameter userInterfaceIdiom], + [STDSDeviceInformationParameter familyNames], + [STDSDeviceInformationParameter fontNamesForFamilyName], + [STDSDeviceInformationParameter systemFont], + [STDSDeviceInformationParameter labelFontSize], + [STDSDeviceInformationParameter buttonFontSize], + [STDSDeviceInformationParameter smallSystemFontSize], + [STDSDeviceInformationParameter systemFontSize], + [STDSDeviceInformationParameter systemLocale], + [STDSDeviceInformationParameter availableLocaleIdentifiers], + [STDSDeviceInformationParameter preferredLanguages], + [STDSDeviceInformationParameter defaultTimeZone], + [STDSDeviceInformationParameter appStoreReciptURL], + ]; + + + }); + + return allParameters; +} + ++ (instancetype)platform { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C001" + permissionCheck:nil + valueCheck:^id _Nullable{ + return @"iOS"; + }]; +} + ++ (instancetype)deviceModel { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C002" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [[UIDevice currentDevice] model]; + }]; +} + ++ (instancetype)OSName { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C003" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [[UIDevice currentDevice] systemName]; + }]; +} + ++ (instancetype)OSVersion { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C004" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [[UIDevice currentDevice] systemVersion]; + }]; +} + ++ (instancetype)locale { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C005" + permissionCheck:nil + valueCheck:^id _Nullable{ + NSLocale *locale = [NSLocale currentLocale]; + NSString *language = locale.languageCode; + NSString *country = locale.countryCode; + if (language != nil && country != nil) { + return [@[language, country] componentsJoinedByString:@"-"]; + } else { + return nil; + } + }]; +} + ++ (instancetype)timeZone { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C006" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [NSTimeZone localTimeZone].name; + }]; +} + ++ (instancetype)advertisingID { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C007" + permissionCheck:nil + valueCheck:^id _Nullable{ + // Actually collecting advertisingIdentifier would require our users to tell Apple they're using it during app submission. + // advertisingIdentifier returns all zeros when the user has limited ad tracking. + return @"00000000-0000-0000-0000-000000000000"; + }]; +} + ++ (instancetype)screenResolution { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C008" + permissionCheck:nil + valueCheck:^id _Nullable{ + CGRect boundsInPixels = [UIScreen mainScreen].nativeBounds; + return [NSString stringWithFormat:@"%ldx%ld", (long)boundsInPixels.size.width, (long)boundsInPixels.size.height]; + + }]; +} + ++ (instancetype)deviceName +{ + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C009" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [UIDevice currentDevice].localizedModel; + }]; +} + ++ (instancetype)IPAddress { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C010" + permissionCheck:nil + valueCheck:^id _Nullable{ + return STDSCurrentDeviceIPAddress(); + }]; +} + ++ (instancetype)latitude { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C011" + permissionCheck:^BOOL{ + return [STDSSynchronousLocationManager hasPermissions]; + } + valueCheck:^id _Nullable{ + CLLocation *location = [[STDSSynchronousLocationManager sharedManager] deviceLocation]; + return location != nil ? @(location.coordinate.latitude).stringValue : nil; + }]; +} + ++ (instancetype)longitude { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C012" + permissionCheck:^BOOL{ + return [STDSSynchronousLocationManager hasPermissions]; + } + valueCheck:^id _Nullable{ + CLLocation *location = [[STDSSynchronousLocationManager sharedManager] deviceLocation]; + return location != nil ? @(location.coordinate.longitude).stringValue : nil; + }]; +} + ++ (instancetype)applicationPackageName { + /* + The unique package name/bundle identifier of the application in which the + 3DS SDK is embedded. + • iOS: obtained from the [NSBundle mainBundle] bundleIdentifier + property. + */ + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C013" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [[NSBundle mainBundle] bundleIdentifier]; + }]; +} + + ++ (instancetype)sdkAppId { + /* + Universally unique ID that is created for each installation of the 3DS + Requestor App on a Consumer Device. + Note: This should be the same ID that is passed to the Requestor App in + the AuthenticationRequestParameters object (Refer to Section + 4.12.1 in the EMV 3DS SDK Specification). + */ + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C014" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [STDSDeviceInformationParameter sdkAppIdentifier]; + }]; +} + + ++ (instancetype)sdkVersion { + /* + 3DS SDK version as applied by the implementer and stored securely in the + SDK (refer to Req 58 in the EMV 3DS SDK Specification). + */ + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"C015" + permissionCheck:nil + valueCheck:^id _Nullable{ + return @"2.2.0"; + }]; +} + ++ (instancetype)identiferForVendor { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I001" + permissionCheck:nil + valueCheck:^id _Nullable{ + // N.B. This can return nil if the device is locked + // We've decided to mark this case and similar as parameter unavailable, + // even though we have permission and the device _can_ provide it when + // it's in a different state + return [UIDevice currentDevice].identifierForVendor.UUIDString; + }]; +} + ++ (instancetype)userInterfaceIdiom { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I002" + permissionCheck:nil + valueCheck:^id _Nullable{ + return @([UIDevice currentDevice].userInterfaceIdiom).stringValue; + }]; +} + ++ (instancetype)familyNames { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I003" + permissionCheck:nil + valueCheck:^id _Nullable{ + return UIFont.familyNames; + }]; +} + ++ (instancetype)fontNamesForFamilyName { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I004" + permissionCheck:nil + valueCheck:^id _Nullable{ + NSArray *fontNames = [UIFont fontNamesForFamilyName:[UIFont systemFontOfSize:[UIFont systemFontSize]].familyName]; + if (fontNames.count == 0) { + return @[@""]; // Workaround for TC_SDK_10176_001 + } + return fontNames; + }]; +} + ++ (instancetype)systemFont { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I005" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [UIFont systemFontOfSize:[UIFont systemFontSize]].fontName; + }]; +} + ++ (instancetype)labelFontSize { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I006" + permissionCheck:nil + valueCheck:^id _Nullable{ + return @([UIFont labelFontSize]).stringValue; + }]; +} + ++ (instancetype)buttonFontSize { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I007" + permissionCheck:nil + valueCheck:^id _Nullable{ + return @([UIFont buttonFontSize]).stringValue; + }]; +} + ++ (instancetype)smallSystemFontSize { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I008" + permissionCheck:nil + valueCheck:^id _Nullable{ + return @([UIFont smallSystemFontSize]).stringValue; + }]; +} + ++ (instancetype)systemFontSize { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I009" + permissionCheck:nil + valueCheck:^id _Nullable{ + return @([UIFont systemFontSize]).stringValue; + }]; +} + ++ (instancetype)systemLocale { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I010" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [NSLocale currentLocale].localeIdentifier; + }]; +} + ++ (instancetype)availableLocaleIdentifiers { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I011" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [NSLocale availableLocaleIdentifiers]; + }]; +} + ++ (instancetype)preferredLanguages { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I012" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [NSLocale preferredLanguages]; + }]; +} + ++ (instancetype)defaultTimeZone { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I013" + permissionCheck:nil + valueCheck:^id _Nullable{ + return [NSTimeZone defaultTimeZone].name; + }]; +} + ++ (instancetype)appStoreReciptURL { + return [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"I014" + permissionCheck:nil + valueCheck:^id _Nullable { + NSString *appStoreReceiptURL = [[NSBundle mainBundle] appStoreReceiptURL].absoluteString; + if (appStoreReceiptURL) { + return appStoreReceiptURL; + } + return kParameterNilCode; + }]; +} + ++ (NSString *)sdkAppIdentifier { + static NSString * const appIdentifierKeyPrefix = @"STDSStripe3DS2AppIdentifierKey"; + NSString *appVersion = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleShortVersionString"] ?: @""; + NSString *appIdentifierUserDefaultsKey = [appIdentifierKeyPrefix stringByAppendingString:appVersion]; + NSString *appIdentifier = [[NSUserDefaults standardUserDefaults] stringForKey:appIdentifierUserDefaultsKey]; + if (appIdentifier == nil) { + appIdentifier = [[NSUUID UUID] UUIDString].lowercaseString; + // Clean up any previous app identifiers + NSSet *previousKeys = [[[NSUserDefaults standardUserDefaults] dictionaryRepresentation] keysOfEntriesPassingTest:^BOOL (NSString *key, id obj, BOOL *stop) { + return [key hasPrefix:appIdentifierKeyPrefix] && ![key isEqualToString:appIdentifierUserDefaultsKey]; + }]; + for (NSString *key in previousKeys) { + [[NSUserDefaults standardUserDefaults] removeObjectForKey:key]; + } + } + [[NSUserDefaults standardUserDefaults] setObject:appIdentifier forKey:appIdentifierUserDefaultsKey]; + return appIdentifier; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDirectoryServer.h b/Stripe3DS2/Stripe3DS2/STDSDirectoryServer.h new file mode 100644 index 00000000..e658e38a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDirectoryServer.h @@ -0,0 +1,134 @@ +// +// Header.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import + +NS_ASSUME_NONNULL_BEGIN +typedef NS_ENUM(NSInteger, STDSDirectoryServer) { + STDSDirectoryServerULTestRSA, + STDSDirectoryServerULTestEC, + STDSDirectoryServerSTPTestRSA, + STDSDirectoryServerSTPTestEC, + STDSDirectoryServerAmex, + STDSDirectoryServerCartesBancaires, + STDSDirectoryServerDiscover, + STDSDirectoryServerMastercard, + STDSDirectoryServerVisa, + STDSDirectoryServerCustom, + STDSDirectoryServerUnknown, +}; + +static NSString * const kULTestRSADirectoryServerID = @"F055545342"; +static NSString * const kULTestECDirectoryServerID = @"F155545342"; + +static NSString * const kSTDSTestRSADirectoryServerID = @"ul_test"; +static NSString * const kSTDSTestECDirectoryServerID = @"ec_test"; + +static NSString * const kSTDSAmexDirectoryServerID = @"A000000025"; +static NSString * const kSTDSCartesBancairesServerID = @"A000000042"; +static NSString * const kSTDSDiscoverDirectoryServerID = @"A000000324"; +static NSString * const kSTDSDiscoverDirectoryServerID_2 = @"A000000152"; +static NSString * const kSTDSMastercardDirectoryServerID = @"A000000004"; +static NSString * const kSTDSVisaDirectoryServerID = @"A000000003"; + + +/// Returns the typed directory server enum or STDSDirectoryServerUnknown if the directoryServerID is not recognized +NS_INLINE STDSDirectoryServer STDSDirectoryServerForID(NSString *directoryServerID) { + if ([directoryServerID isEqualToString:kULTestRSADirectoryServerID]) { + return STDSDirectoryServerULTestRSA; + } else if ([directoryServerID isEqualToString:kULTestECDirectoryServerID]) { + return STDSDirectoryServerULTestEC; + } else if ([directoryServerID isEqualToString:kSTDSTestRSADirectoryServerID]) { + return STDSDirectoryServerSTPTestRSA; + } else if ([directoryServerID isEqualToString:kSTDSTestECDirectoryServerID]) { + return STDSDirectoryServerSTPTestEC; + } else if ([directoryServerID isEqualToString:kSTDSAmexDirectoryServerID]) { + return STDSDirectoryServerAmex; + } else if ([directoryServerID isEqualToString:kSTDSDiscoverDirectoryServerID] || [directoryServerID isEqualToString:kSTDSDiscoverDirectoryServerID_2]) { + return STDSDirectoryServerDiscover; + } else if ([directoryServerID isEqualToString:kSTDSMastercardDirectoryServerID]) { + return STDSDirectoryServerMastercard; + } else if ([directoryServerID isEqualToString:kSTDSVisaDirectoryServerID]) { + return STDSDirectoryServerVisa; + } else if ([directoryServerID isEqualToString:kSTDSCartesBancairesServerID]) { + return STDSDirectoryServerCartesBancaires; + } + + return STDSDirectoryServerUnknown; +} + +/// Returns the directory server ID or nil for STDSDirectoryServerUnknown +NS_INLINE NSString * _Nullable STDSDirectoryServerIdentifier(STDSDirectoryServer directoryServer) { + switch (directoryServer) { + case STDSDirectoryServerULTestRSA: + return kULTestRSADirectoryServerID; + + case STDSDirectoryServerULTestEC: + return kULTestECDirectoryServerID; + + case STDSDirectoryServerSTPTestRSA: + return kSTDSTestRSADirectoryServerID; + + case STDSDirectoryServerSTPTestEC: + return kSTDSTestECDirectoryServerID; + + case STDSDirectoryServerAmex: + return kSTDSAmexDirectoryServerID; + + case STDSDirectoryServerDiscover: + return kSTDSDiscoverDirectoryServerID; + + case STDSDirectoryServerMastercard: + return kSTDSMastercardDirectoryServerID; + + case STDSDirectoryServerVisa: + return kSTDSVisaDirectoryServerID; + + case STDSDirectoryServerCartesBancaires: + return kSTDSCartesBancairesServerID; + + case STDSDirectoryServerCustom: + return nil; + + case STDSDirectoryServerUnknown: + return nil; + } +} + +/// Returns the directory server image name if one exists +NS_INLINE NSString * _Nullable STDSDirectoryServerImageName(STDSDirectoryServer directoryServer) { + switch (directoryServer) { + case STDSDirectoryServerAmex: + return @"amex-logo"; + case STDSDirectoryServerDiscover: + return @"discover-logo"; + case STDSDirectoryServerMastercard: + return @"mastercard-logo"; + case STDSDirectoryServerCartesBancaires: + return @"cartes-bancaires-logo"; + // just default to an arbitrary logo for the test servers + case STDSDirectoryServerULTestEC: + case STDSDirectoryServerULTestRSA: + case STDSDirectoryServerSTPTestRSA: + case STDSDirectoryServerSTPTestEC: + case STDSDirectoryServerVisa: + if (@available(iOS 13.0, *)) { + if ([[UITraitCollection currentTraitCollection] userInterfaceStyle] == UIUserInterfaceStyleDark) { + return @"visa-white-logo"; + } + } + return @"visa-logo"; + case STDSDirectoryServerCustom: + case STDSDirectoryServerUnknown: + return nil; + + } +} + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate+Internal.h b/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate+Internal.h new file mode 100644 index 00000000..8f45d98a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate+Internal.h @@ -0,0 +1,22 @@ +// +// STDSDirectoryServerCertificate+Internal.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 4/2/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDirectoryServerCertificate.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDirectoryServerCertificate (Internal) + +/// Verifies the certificate chain represented by certificates where each element is a base64 encoded (NOT base64url) certificate ++ (BOOL)_verifyCertificateChain:(NSArray *)certificates withRootCertificates:(NSArray *)rootCertificates; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.h b/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.h new file mode 100644 index 00000000..4ac91b09 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.h @@ -0,0 +1,43 @@ +// +// STDSDirectoryServerCertificate.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class STDSJSONWebSignature; + +#import "STDSDirectoryServer.h" + +NS_ASSUME_NONNULL_BEGIN + +typedef NS_ENUM(NSInteger, STDSDirectoryServerKeyType) { + STDSDirectoryServerKeyTypeRSA, + STDSDirectoryServerKeyTypeEC, + STDSDirectoryServerKeyTypeUnknown, +}; + +@interface STDSDirectoryServerCertificate : NSObject + ++ (nullable instancetype)certificateForDirectoryServer:(STDSDirectoryServer)directoryServer; + ++ (nullable instancetype)customCertificateWithData:(NSData *)certificateData; + ++ (nullable instancetype)customCertificateWithString:(NSString *)certificateString; + +@property (nonatomic, readonly) STDSDirectoryServerKeyType keyType; + +@property (nonatomic, readonly) SecKeyRef publicKey; + +@property (nonatomic, readonly, copy) NSString *certificateString; + +- (nullable NSData *)encryptDataUsingRSA_OAEP_SHA256:(NSData *)plaintext; + ++ (BOOL)verifyJSONWebSignature:(STDSJSONWebSignature *)jws withRootCertificates:(NSArray *)rootCertificates; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.m b/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.m new file mode 100644 index 00000000..08d3d082 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSDirectoryServerCertificate.m @@ -0,0 +1,350 @@ +// +// STDSDirectoryServerCertificate.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSDirectoryServerCertificate.h" +#import "STDSDirectoryServerCertificate+Internal.h" + +#import "NSData+JWEHelpers.h" +#import "NSString+JWEHelpers.h" +#import "STDSEllipticCurvePoint.h" +#import "STDSJSONWebSignature.h" +#import "STDSSecTypeUtilities.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDirectoryServerCertificate () +{ + SecCertificateRef _certificate; + STDSDirectoryServer _directoryServer; +} + +- (instancetype)_initForDirectoryServer:(STDSDirectoryServer)directoryServer; + +@end + +@implementation STDSDirectoryServerCertificate + +- (instancetype)_initWithCertificate:(SecCertificateRef _Nullable)certificate forDirectorySever:(STDSDirectoryServer)directoryServer { + self = [super init]; + if (self) { + _certificate = certificate; + switch (directoryServer) { + + case STDSDirectoryServerULTestRSA: { + /** + UL provides the following, which is PKCS#8, but Security framework wants PKCS#1. Luckily all we have to do is remove the first 32 characters which are just a header to convert + @"-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAr/O0BfXWngO9OJDBsqdR\n5U2h28jrX6Y+LlblTBaYeT2tW7+ca3YzTFXA8duVUwdlWxl3JZCOOeL1feVP6g0TNOHVCkCnirVDLkcozod4aSkNvx+929aDr1ithqhruf0skBc2sMZGBBCNpso6XGzyAf2uZ2+9DvXoKIUYgcr7PQmL2Y0awyQN7KCRcusaotYNz2mOPrL/hAv6hTexkNrQ\nKzFcPwCuc6kN6aNjD+p2CJ51/5p02SNS70nPOmwmg63j6f3n7xVykQ56kNc1l5B5xOpeHJmqk3+hyF1dF/47rQmMFicN41QSvZ5AZJKgWlIn2VQROMkEHkF9ZBRLx1nF\nTwIDAQAB\n-----END PUBLIC KEY-----\n" + */ + static NSString * const kULTestRSAPublicKey = @"MIIBCgKCAQEAr/O0BfXWngO9OJDBsqdR\n5U2h28jrX6Y+LlblTBaYeT2tW7+ca3YzTFXA8duVUwdlWxl3JZCOOeL1feVP6g0TNOHVCkCnirVDLkcozod4aSkNvx+929aDr1ithqhruf0skBc2sMZGBBCNpso6XGzyAf2uZ2+9DvXoKIUYgcr7PQmL2Y0awyQN7KCRcusaotYNz2mOPrL/hAv6hTexkNrQ\nKzFcPwCuc6kN6aNjD+p2CJ51/5p02SNS70nPOmwmg63j6f3n7xVykQ56kNc1l5B5xOpeHJmqk3+hyF1dF/47rQmMFicN41QSvZ5AZJKgWlIn2VQROMkEHkF9ZBRLx1nF\nTwIDAQAB"; + + NSString *cleanedString = [[[kULTestRSAPublicKey stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]] componentsSeparatedByCharactersInSet:[NSCharacterSet newlineCharacterSet]] componentsJoinedByString:@""]; + + NSData *base64Decoded = [[NSData alloc] initWithBase64EncodedString:cleanedString options:0]; + NSDictionary *attributes = @{ + (__bridge NSString *)kSecAttrKeyType: (__bridge NSString *)kSecAttrKeyTypeRSA, + (__bridge NSString *)kSecAttrKeyClass: (__bridge NSString *)kSecAttrKeyClassPublic, + }; + CFErrorRef error = NULL; + SecKeyRef key = SecKeyCreateWithData((__bridge CFDataRef)base64Decoded, (__bridge CFDictionaryRef)attributes, &error); + if (key == NULL) { + return nil; + } + _publicKey = key; + } + break; + + case STDSDirectoryServerULTestEC: { + static NSString * const kULTestECPublicKey = @"MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEYktbLuAv0v52erE5LPscomKaOmQs\nvevxzOyn9k4sF1hqpBc5kUygzxA9Jl0R/2dTuk8ka7UCujk36xeUsLVpWA=="; + NSString *cleanedString = [[[kULTestECPublicKey stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]] componentsSeparatedByCharactersInSet:[NSCharacterSet newlineCharacterSet]] componentsJoinedByString:@""]; + NSData *base64Decoded = [[NSData alloc] initWithBase64EncodedString:cleanedString options:0]; + // This data is PEM encoded, to get to ec standard we take the last 65 bytes + if (base64Decoded.length >= 65) { + base64Decoded = [base64Decoded subdataWithRange:NSMakeRange(base64Decoded.length - 65, 65)]; + } + NSDictionary *attributes = @{ + (__bridge NSString *)kSecAttrKeyType: (__bridge NSString *)kSecAttrKeyTypeECSECPrimeRandom, + (__bridge NSString *)kSecAttrKeyClass: (__bridge NSString *)kSecAttrKeyClassPublic, + }; + CFErrorRef error = NULL; + SecKeyRef key = SecKeyCreateWithData((__bridge CFDataRef)base64Decoded, (__bridge CFDictionaryRef)attributes, &error); + if (key == NULL) { + return nil; + } + _publicKey = key; + } + break; + + case STDSDirectoryServerSTPTestRSA: + // fall-through + case STDSDirectoryServerSTPTestEC: + // fall-through + case STDSDirectoryServerAmex: + // fall-through + case STDSDirectoryServerCartesBancaires: + // fall-through + case STDSDirectoryServerDiscover: + // fall-through + case STDSDirectoryServerMastercard: + // fall-through + case STDSDirectoryServerVisa: + // fall-through + case STDSDirectoryServerCustom: + // fall-through + case STDSDirectoryServerUnknown: + NSAssert(certificate != NULL, @"Must provide a certificate"); + _publicKey = SecCertificateCopyKey(certificate); + } + _directoryServer = directoryServer; + + if (_publicKey == NULL) { + return nil; + } + } + + return self; +} + +- (instancetype)_initForDirectoryServer:(STDSDirectoryServer)directoryServer { + SecCertificateRef certificate = NULL; + + switch (directoryServer) { + case STDSDirectoryServerULTestRSA: + // fall-through + case STDSDirectoryServerULTestEC: + // The UL test servers don't actually have certificates, just hard-coded key values + break; + + case STDSDirectoryServerSTPTestRSA: + // fall-through + case STDSDirectoryServerSTPTestEC: + // fall-through + case STDSDirectoryServerAmex: + // fall-through + case STDSDirectoryServerCartesBancaires: + // fall-through + case STDSDirectoryServerDiscover: + // fall-through; + case STDSDirectoryServerMastercard: + // fall-through + case STDSDirectoryServerVisa: { + certificate = STDSCertificateForServer(directoryServer); + if (certificate == NULL) { + return nil; + } + } + break; + + case STDSDirectoryServerCustom: + return nil; + + case STDSDirectoryServerUnknown: + return nil; + } + return [self _initWithCertificate:certificate forDirectorySever:directoryServer]; +} + ++ (nullable instancetype)certificateForDirectoryServer:(STDSDirectoryServer)directoryServer { + return [[self alloc] _initForDirectoryServer:directoryServer]; +} + ++ (nullable instancetype)customCertificateWithData:(NSData *)certificateData { + SecCertificateRef certificate = STDSSecCertificateFromData(certificateData); + if (certificate == NULL) { + return nil; + } + return [[self alloc] _initWithCertificate:certificate forDirectorySever:STDSDirectoryServerCustom]; +} + ++ (nullable instancetype)customCertificateWithString:(NSString *)certificateString { + SecCertificateRef certificate = STDSSecCertificateFromString(certificateString); + if (certificate == NULL) { + return nil; + } + + return [[self alloc] _initWithCertificate:certificate forDirectorySever:STDSDirectoryServerCustom]; +} + +- (void)dealloc { + if (_certificate != NULL) { + CFRelease(_certificate); + } + if (_publicKey != NULL) { + CFRelease(_publicKey); + } +} + +- (NSString *)certificateString { + NSData *data = (NSData *)CFBridgingRelease(SecCertificateCopyData(_certificate)); + return [data base64EncodedStringWithOptions:0]; +} + +- (STDSDirectoryServerKeyType)keyType { + switch (_directoryServer) { + case STDSDirectoryServerULTestRSA: + return STDSDirectoryServerKeyTypeRSA; + + case STDSDirectoryServerULTestEC: + return STDSDirectoryServerKeyTypeEC; + + + case STDSDirectoryServerSTPTestRSA: + // fall-through + case STDSDirectoryServerSTPTestEC: + // fall-through + case STDSDirectoryServerAmex: + // fall-through + case STDSDirectoryServerCartesBancaires: + // fall-through + case STDSDirectoryServerDiscover: + // fall-through; + case STDSDirectoryServerMastercard: + // fall-through + case STDSDirectoryServerVisa: + // fall-through + case STDSDirectoryServerCustom: { + NSAssert(_certificate != NULL, @"Must have a valid certificate file"); + if (_certificate == NULL) { + return STDSDirectoryServerKeyTypeUnknown; + } + CFStringRef keyType = STDSSecCertificateCopyPublicKeyType(_certificate); + STDSDirectoryServerKeyType ret = STDSDirectoryServerKeyTypeUnknown; + if (keyType != NULL) { + if (CFStringCompare(keyType, kSecAttrKeyTypeRSA, 0) == kCFCompareEqualTo) { + ret = STDSDirectoryServerKeyTypeRSA; + } else if (CFStringCompare(keyType, kSecAttrKeyTypeECSECPrimeRandom, 0) == kCFCompareEqualTo) { + ret = STDSDirectoryServerKeyTypeEC; + } + + CFRelease(keyType); + } + return ret; + } + + case STDSDirectoryServerUnknown: + NSAssert(0, @"Should not have an STDSDirectoryServerCertificate instance withSTPDirectoryServerUnknown"); + return STDSDirectoryServerKeyTypeUnknown; + } +} + +- (nullable NSData *)encryptDataUsingRSA_OAEP_SHA256:(NSData *)plaintext { + NSAssert(_publicKey != NULL, @"STDSDirectoryServerCertificate should always have _publicKey"); + if (_publicKey == NULL) { + return nil; + } + + CFDataRef encryptedData = SecKeyCreateEncryptedData(_publicKey, + kSecKeyAlgorithmRSAEncryptionOAEPSHA256, + (CFDataRef)plaintext, + NULL); + return (NSData *)CFBridgingRelease(encryptedData); +} + ++ (BOOL)_verifyCertificateChain:(NSArray *)certificatesStrings withRootCertificates:(NSArray *)rootCertificateStrings { + if (certificatesStrings.count == 0 || rootCertificateStrings.count == 0) { + return NO; + } + + NSMutableArray *certificates = [[NSMutableArray alloc] initWithCapacity:certificatesStrings.count]; + for (NSString *certificateString in certificatesStrings) { + SecCertificateRef certificate = STDSSecCertificateFromString(certificateString); + if (certificate == NULL) { + return NO; + } + [certificates addObject:(id)CFBridgingRelease(certificate)]; + } + + NSMutableArray *rootCertificates = [[NSMutableArray alloc] initWithCapacity:rootCertificateStrings.count]; + for (NSString *certificateString in rootCertificateStrings) { + SecCertificateRef certificate = STDSSecCertificateFromString(certificateString); + if (certificate == NULL) { + return NO; + } + [rootCertificates addObject:(id)CFBridgingRelease(certificate)]; + } + + SecPolicyRef policy = SecPolicyCreateBasicX509(); + SecTrustRef trust; + OSStatus status = SecTrustCreateWithCertificates((__bridge CFTypeRef)certificates, + policy, + &trust); + if (policy) { + CFRelease(policy); + } + if (status != errSecSuccess) { + return NO; + } + if (rootCertificates.count > 0) { + status = SecTrustSetAnchorCertificates(trust, (__bridge CFTypeRef)rootCertificates); + if (status != errSecSuccess) { + return NO; + } + } + + if (@available(iOS 12.0, *)) { + CFErrorRef error = NULL; + + bool verified = SecTrustEvaluateWithError(trust, &error); + return (BOOL)verified; + } else { +#if TARGET_OS_MACCATALYST + return NO; +#else + // Fallback on earlier versions + dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0); + __block BOOL verified = NO; + dispatch_group_t group = dispatch_group_create(); + dispatch_group_enter(group); + dispatch_async(queue, ^{ + // SecTrustEvaluateAsyncWithError must be called from the same queue that is passed as an arg + SecTrustEvaluateAsyncWithError(trust, queue, ^(SecTrustRef _Nonnull trustRef, bool result, CFErrorRef _Nullable error) { + if (result) { + verified = YES; + } + dispatch_group_leave(group); + }); + }); + dispatch_group_wait(group, dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.2 * NSEC_PER_SEC))); // timeout after 200 ms + + return verified; +#endif + } +} + ++ (BOOL)verifyJSONWebSignature:(STDSJSONWebSignature *)jws withRootCertificates:(NSArray *)rootCertificates { + if (jws.certificateChain.count == 0 || ![self _verifyCertificateChain:jws.certificateChain withRootCertificates:rootCertificates]) { + return NO; + } + + switch (jws.algorithm) { + case STDSJSONWebSignatureAlgorithmES256: + return STDSVerifyEllipticCurveP256Signature(jws.ellipticCurvePoint.x, jws.ellipticCurvePoint.y, jws.digest, jws.signature); + + case STDSJSONWebSignatureAlgorithmPS256: { + if (jws.certificateChain.count == 0) { + return NO; + } + NSString *certificateString = [jws.certificateChain firstObject]; + SecCertificateRef certificate = STDSSecCertificateFromString(certificateString); + if (certificate == NULL) { + return NO; + } + + BOOL verified = STDSVerifyRSASignature(certificate, jws.digest, jws.signature); + CFRelease(certificate); + return verified; + } + + + case STDSJSONWebSignatureAlgorithmUnknown: + return NO; + } +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.h b/Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.h new file mode 100644 index 00000000..c856951d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.h @@ -0,0 +1,26 @@ +// +// STDSEllipticCurvePoint.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSEllipticCurvePoint : NSObject + +- (nullable instancetype)initWithX:(NSData *)x y:(NSData *)y; +- (nullable instancetype)initWithCertificateData:(NSData *)certificateData; +- (nullable instancetype)initWithKey:(SecKeyRef)key; +- (nullable instancetype)initWithJWK:(NSDictionary *)jwk; + +@property (nonatomic, readonly) NSData *x; +@property (nonatomic, readonly) NSData *y; + +@property (nonatomic, readonly) SecKeyRef publicKey; + +@end +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.m b/Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.m new file mode 100644 index 00000000..49fdf3e5 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSEllipticCurvePoint.m @@ -0,0 +1,90 @@ +// +// STDSEllipticCurvePoint.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSEllipticCurvePoint.h" + +#import "NSDictionary+DecodingHelpers.h" +#import "NSString+JWEHelpers.h" +#import "STDSSecTypeUtilities.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSEllipticCurvePoint + +- (nullable instancetype)initWithX:(NSData *)x y:(NSData *)y { + self = [super init]; + if (self) { + _x = x; + _y = y; + _publicKey = STDSSecKeyRefFromCoordinates(x, y); + if (_publicKey == NULL) { + return nil; + } + } + + return self; +} + +- (nullable instancetype)initWithKey:(SecKeyRef)key { + self = [super init]; + if (self) { + _publicKey = key; + CFErrorRef error = NULL; + NSData *keyData = (NSData *)CFBridgingRelease(SecKeyCopyExternalRepresentation(key, &error)); + if (keyData == nil) { + return nil; + } + + NSUInteger coordinateLength = (keyData.length - 1) / 2; // -1 because the first byte is formatting 0x04 + NSData *xData = [keyData subdataWithRange:NSMakeRange(1, coordinateLength)]; + NSData *yData = [keyData subdataWithRange:NSMakeRange(1 + coordinateLength, coordinateLength)]; + _x = xData; + _y = yData; + } + + return self; +} + +- (nullable instancetype)initWithCertificateData:(NSData *)certificateData { + SecCertificateRef certificate = STDSSecCertificateFromData(certificateData); + if (certificateData != NULL) { + SecKeyRef key = SecCertificateCopyKey(certificate); + CFRelease(certificate); + if (key != NULL) { + STDSEllipticCurvePoint *point = [self initWithKey:key]; + CFRelease(key); + return point; + } + } + return nil; +} + +- (nullable instancetype)initWithJWK:(NSDictionary *)jwk { + NSString *kty = [jwk _stds_stringForKey:@"kty" validator:^BOOL(NSString * _Nonnull val) { + return [val isEqualToString:@"EC"]; + } required:YES error:NULL]; + NSString *crv = [jwk _stds_stringForKey:@"crv" validator:^BOOL(NSString * _Nonnull val) { + return [val isEqualToString:@"P-256"]; + } required:YES error:NULL]; + + NSData *coordinateX = [[jwk _stds_stringForKey:@"x" required:YES error:NULL] _stds_base64URLDecodedData]; + NSData *coordinateY = [[jwk _stds_stringForKey:@"y" required:YES error:NULL] _stds_base64URLDecodedData]; + + if (kty == nil || + crv == nil || + coordinateX == nil || + coordinateY == nil + ) { + return nil; + } + return [self initWithX:coordinateX y:coordinateY]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair+Testing.h b/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair+Testing.h new file mode 100644 index 00000000..d8efb11d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair+Testing.h @@ -0,0 +1,19 @@ +// +// STDSEphemeralKeyPair+Testing.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 4/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSEphemeralKeyPair.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSEphemeralKeyPair (Testing) + ++ (nullable instancetype)testKeyPair; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.h b/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.h new file mode 100644 index 00000000..515fbb58 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.h @@ -0,0 +1,39 @@ +// +// STDSEphemeralKeyPair.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class STDSDirectoryServerCertificate; +@class STDSEllipticCurvePoint; + +#import "STDSDirectoryServer.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSEphemeralKeyPair : NSObject + +/// Creates a returns a new elliptic curve key pair using curve P-256 ++ (nullable instancetype)ephemeralKeyPair; + +- (instancetype)init NS_UNAVAILABLE; + +@property (nonatomic, readonly) NSString *publicKeyJWK; +@property (nonatomic, readonly) STDSEllipticCurvePoint *publicKeyCurvePoint; + +/** + Creates and returns a new secret key derived using Elliptic Curve Diffie-Hellman + and the certificate's public key (return nil on failure). + Per OpenSSL documentation: Never use a derived secret directly. Typically it is passed through some + hash function to produce a key (e.g. pass the secret as the first argument to STDSCreateConcatKDFWithSHA256) + */ +- (nullable NSData *)createSharedSecretWithEllipticCurveKey:(STDSEllipticCurvePoint *)ecKey; +- (nullable NSData *)createSharedSecretWithCertificate:(STDSDirectoryServerCertificate *)certificate; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.m b/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.m new file mode 100644 index 00000000..426ebc5a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSEphemeralKeyPair.m @@ -0,0 +1,108 @@ +// +// STDSEphemeralKeyPair.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSEphemeralKeyPair.h" + +#import "NSData+JWEHelpers.h" +#import "NSDictionary+DecodingHelpers.h" +#import "NSString+JWEHelpers.h" +#import "STDSDirectoryServerCertificate.h" +#import "STDSEllipticCurvePoint.h" +#import "STDSSecTypeUtilities.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSEphemeralKeyPair () +{ + SecKeyRef _privateKey; + SecKeyRef _publicKey; +} + +@end + + +@implementation STDSEphemeralKeyPair + +- (instancetype)_initWithPrivateKey:(SecKeyRef)privateKey publicKey:(SecKeyRef)publicKey { + self = [super init]; + if (self) { + _privateKey = privateKey; + _publicKey = publicKey; + } + + return self; +} + ++ (nullable instancetype)ephemeralKeyPair { + NSDictionary *parameters = @{ + (__bridge NSString *)kSecAttrKeyType: (__bridge NSString *)kSecAttrKeyTypeECSECPrimeRandom, + (__bridge NSString *)kSecAttrKeySizeInBits: @(256), + }; + CFErrorRef error = NULL; + SecKeyRef privateKey = SecKeyCreateRandomKey((__bridge CFDictionaryRef)parameters, &error); + + if (privateKey != NULL) { + SecKeyRef publicKey = SecKeyCopyPublicKey(privateKey); + return [[self alloc] _initWithPrivateKey:privateKey publicKey:publicKey]; + } + + return nil; +} + ++ (nullable instancetype)testKeyPair { + + // values from EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSData *d = [@"iyn--IbkBeNoPu8cN245L6pOQWt2lTH8V0Ds92jQmWA" _stds_base64URLDecodedData]; + NSData *x = [@"C1PL42i6kmNkM61aupEAgLJ4gF1ZRzcV7lqo1TG0mL4" _stds_base64URLDecodedData]; + NSData *y = [@"cNToWLSdcFQKG--PGVEUQrIHP8w6TcRyj0pyFx4-ZMc" _stds_base64URLDecodedData]; + + SecKeyRef privateKey = STDSPrivateSecKeyRefFromCoordinates(x, y, d); + if (privateKey == NULL) { + return nil; + } + SecKeyRef publicKey = SecKeyCopyPublicKey(privateKey); + if (publicKey == NULL) { + return nil; + } + + return [[STDSEphemeralKeyPair alloc] _initWithPrivateKey:privateKey publicKey:publicKey]; +} + +- (void)dealloc { + if (_privateKey != NULL) { + CFRelease(_privateKey); + } + if (_publicKey != NULL) { + CFRelease(_publicKey); + } +} + +- (NSString *)publicKeyJWK { + STDSEllipticCurvePoint *publicKeyCurvePoint = [[STDSEllipticCurvePoint alloc] initWithKey:_publicKey]; + return [NSString stringWithFormat:@"{\"kty\":\"EC\",\"crv\":\"P-256\",\"x\":\"%@\",\"y\":\"%@\"}", [publicKeyCurvePoint.x _stds_base64URLEncodedString], [publicKeyCurvePoint.y _stds_base64URLEncodedString]]; +} + +- (nullable NSData *)createSharedSecretWithEllipticCurveKey:(STDSEllipticCurvePoint *)ecKey { + return [self _createSharedSecretWithPrivateKey:_privateKey publicKey:ecKey.publicKey]; +} + +- (nullable NSData *)createSharedSecretWithCertificate:(STDSDirectoryServerCertificate *)certificate { + return [self _createSharedSecretWithPrivateKey:_privateKey publicKey:certificate.publicKey]; +} + +- (nullable NSData *)_createSharedSecretWithPrivateKey:(SecKeyRef)privateKey publicKey:(SecKeyRef)publicKey { + NSDictionary *params = @{(__bridge NSString *)kSecKeyKeyExchangeParameterRequestedSize: @(32)}; + CFErrorRef error = NULL; + CFDataRef secret = SecKeyCopyKeyExchangeResult(privateKey, kSecKeyAlgorithmECDHKeyExchangeStandard, publicKey, (__bridge CFDictionaryRef)params, &error); + + return (NSData *)CFBridgingRelease(secret); +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.h b/Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.h new file mode 100644 index 00000000..3a0730c3 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.h @@ -0,0 +1,40 @@ +// +// STDSErrorMessage+Internal.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 4/9/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSErrorMessage.h" + +NS_ASSUME_NONNULL_BEGIN + +// Constructors for the circumstances in which we are required to send an ErrorMessage to the ACS +@interface STDSErrorMessage (Internal) + +/// Received an invalid message type ++ (instancetype)errorForInvalidMessageWithACSTransactionID:(NSString *)acsTransactionID + messageVersion:(NSString *)messageVersion; + +/// Encountered an invalid field parsing a JSON response ++ (nullable instancetype)errorForJSONFieldInvalidWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion error:(NSError *)error; + +/// Encountered a missing field parsing a JSON response ++ (nullable instancetype)errorForJSONFieldMissingWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion error:(NSError *)error; + +/// Encountered an error decrypting a networking response ++ (instancetype)errorForDecryptionErrorWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion; + +/// Timed out ++ (instancetype)errorForTimeoutWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion; + ++ (instancetype)errorForUnrecognizedIDWithACSTransactionID:(NSString *)transactionID messageVersion:(NSString *)messageVersion; + +/// Encountered unrecognized critical message extension(s) ++ (instancetype)errorForUnrecognizedCriticalMessageExtensionsWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion error:(NSError *)error; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.m b/Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.m new file mode 100644 index 00000000..02eed1d0 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSErrorMessage+Internal.m @@ -0,0 +1,91 @@ +// +// STDSErrorMessage+Internal.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 4/9/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSErrorMessage+Internal.h" +#import "STDSStripe3DS2Error.h" + +@implementation STDSErrorMessage (Internal) + ++ (NSString *)_stringForErrorCode:(STDSErrorMessageCode)errorCode { + return [NSString stringWithFormat:@"%ld", (long)errorCode]; +} + ++ (instancetype)errorForInvalidMessageWithACSTransactionID:(nonnull NSString *)acsTransactionID messageVersion:(nonnull NSString *)messageVersion { + return [[[self class] alloc] initWithErrorCode:[self _stringForErrorCode:STDSErrorMessageCodeInvalidMessage] + errorComponent:@"C" + errorDescription:@"Message not recognized" + errorDetails:@"Unknown message type" + messageVersion:messageVersion + acsTransactionIdentifier:acsTransactionID + errorMessageType:@"CRes"]; + +} + ++ (nullable instancetype)errorForJSONFieldMissingWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion error:(NSError *)error { + return [[[self class] alloc] initWithErrorCode:[self _stringForErrorCode:STDSErrorMessageCodeRequiredDataElementMissing] + errorComponent:@"C" + errorDescription:@"Missing Field" + errorDetails:error.userInfo[STDSStripe3DS2ErrorFieldKey] + messageVersion:messageVersion + acsTransactionIdentifier:acsTransactionID + errorMessageType:@"CRes"]; +} + ++ (nullable instancetype)errorForJSONFieldInvalidWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion error:(NSError *)error { + return [[[self class] alloc] initWithErrorCode:[self _stringForErrorCode:STDSErrorMessageErrorInvalidDataElement] + errorComponent:@"C" + errorDescription:@"Invalid Field" + errorDetails:error.userInfo[STDSStripe3DS2ErrorFieldKey] + messageVersion:messageVersion + acsTransactionIdentifier:acsTransactionID + errorMessageType:@"CRes"]; +} + ++ (instancetype)errorForDecryptionErrorWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion { + return [[[self class] alloc] initWithErrorCode:[self _stringForErrorCode:STDSErrorMessageErrorDataDecryptionFailure] + errorComponent:@"C" + errorDescription:@"Response could not be decrypted." + errorDetails:@"Response could not be decrypted.s" + messageVersion:messageVersion + acsTransactionIdentifier:acsTransactionID + errorMessageType:@"CRes"]; +} + ++ (instancetype)errorForTimeoutWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion { + return [[[self class] alloc] initWithErrorCode:[self _stringForErrorCode:STDSErrorMessageErrorTimeout] + errorComponent:@"C" + errorDescription:@"Transaction timed out." + errorDetails:@"Transaction timed out." + messageVersion:messageVersion + acsTransactionIdentifier:acsTransactionID + errorMessageType:@"CRes"]; +} + ++ (instancetype)errorForUnrecognizedIDWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion { + return [[[self class] alloc] initWithErrorCode:[self _stringForErrorCode:STDSErrorMessageErrorTransactionIDNotRecognized] + errorComponent:@"C" + errorDescription:@"Unrecognized transaction ID" + errorDetails:@"Unrecognized transaction ID" + messageVersion:messageVersion + acsTransactionIdentifier:acsTransactionID + errorMessageType:@"CRes"]; +} + ++ (instancetype)errorForUnrecognizedCriticalMessageExtensionsWithACSTransactionID:(NSString *)acsTransactionID messageVersion:(NSString *)messageVersion error:(NSError *)error { + NSArray *unrecognizedIDs = error.userInfo[STDSStripe3DS2UnrecognizedCriticalMessageExtensionsKey]; + + return [[[self class] alloc] initWithErrorCode:[self _stringForErrorCode:STDSErrorMessageCodeUnrecognizedCriticalMessageExtension] + errorComponent:@"C" + errorDescription:@"Critical message extension not recognised." + errorDetails:[unrecognizedIDs componentsJoinedByString:@","] + messageVersion:messageVersion + acsTransactionIdentifier:acsTransactionID + errorMessageType:@"CRes"]; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSException+Internal.h b/Stripe3DS2/Stripe3DS2/STDSException+Internal.h new file mode 100644 index 00000000..3429aba9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSException+Internal.h @@ -0,0 +1,19 @@ +// +// STDSException+Internal.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSException.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSException (Internal) + ++ (instancetype)exceptionWithMessage:(NSString *)format, ...; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.h b/Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.h new file mode 100644 index 00000000..f6f3ee0e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.h @@ -0,0 +1,23 @@ +// +// STDSExpandableInformationView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSFooterCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSExpandableInformationView : UIView + +@property (nonatomic, strong, nullable) NSString *title; +@property (nonatomic, strong, nullable) NSString *text; +@property (nonatomic, strong, nullable) STDSFooterCustomization *customization; +@property (nonatomic, strong, nullable) void (^didTap)(void); + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.m b/Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.m new file mode 100644 index 00000000..9b1bbdfb --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSExpandableInformationView.m @@ -0,0 +1,150 @@ +// +// STDSExpandableInformationView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSLocalizedString.h" +#import "STDSBundleLocator.h" +#import "STDSExpandableInformationView.h" +#import "STDSStackView.h" +#import "UIView+LayoutSupport.h" +#import "NSString+EmptyChecking.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSExpandableInformationView() + +@property (nonatomic, strong) UIView *tappableView; +@property (nonatomic, strong) STDSStackView *textContainerView; +@property (nonatomic, strong) STDSStackView *imageViewStackView; +@property (nonatomic, strong) UIView *imageViewSpacerView; +@property (nonatomic, strong) UILabel *titleLabel; +@property (nonatomic, strong) UILabel *textLabel; +@property (nonatomic, strong) UIImageView *titleImageView; + +@end + +@implementation STDSExpandableInformationView + +static const CGFloat kTitleContainerSpacing = 20; +static const CGFloat kTextContainerSpacing = 13; +static const CGFloat kExpandableInformationViewBottomMargin = 30; +static const CGFloat kTitleImageViewRotationAnimationDuration = (CGFloat)0.2; + +- (instancetype)initWithFrame:(CGRect)frame { + self = [super initWithFrame:frame]; + + if (self) { + [self _setupViewHierarchy]; + self.accessibilityIdentifier = @"STDSExpandableInformationView"; + } + + return self; +} + +- (void)_setupViewHierarchy { + self.layoutMargins = UIEdgeInsetsMake(0, 0, kExpandableInformationViewBottomMargin, 0); + + self.titleLabel = [[UILabel alloc] init]; + // Set titleLabel as not an accessibility element because we make the + // container, which is the actual control, have the same accessibility label + // and accurately reflects that interactivity and state of the control + self.titleLabel.isAccessibilityElement = NO; + self.titleLabel.numberOfLines = 0; + + self.textLabel = [[UILabel alloc] init]; + self.textLabel.numberOfLines = 0; + + UIImage *chevronImage = [[UIImage imageNamed:@"Chevron" inBundle:[STDSBundleLocator stdsResourcesBundle] compatibleWithTraitCollection:nil] imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]; + self.titleImageView = [[UIImageView alloc] initWithImage:chevronImage]; + self.titleImageView.contentMode = UIViewContentModeScaleAspectFit; + + STDSStackView *containerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisHorizontal]; + [self addSubview:containerView]; + [containerView _stds_pinToSuperviewBounds]; + + STDSStackView *titleContainerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + [titleContainerView addArrangedSubview:self.titleLabel]; + + self.imageViewStackView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + self.imageViewSpacerView = [UIView new]; + [self.imageViewStackView addArrangedSubview:self.titleImageView]; + + [containerView addArrangedSubview:self.imageViewStackView]; + [containerView addSpacer:kTitleContainerSpacing]; + [containerView addArrangedSubview:titleContainerView]; + [containerView addArrangedSubview:[UIView new]]; + + self.textContainerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + self.textContainerView.hidden = YES; + [self.textContainerView addSpacer:kTextContainerSpacing]; + [self.textContainerView addArrangedSubview:self.textLabel]; + [titleContainerView addArrangedSubview:self.textContainerView]; + + UITapGestureRecognizer *expandTextTapRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(_toggleTextExpansion)]; + [containerView addGestureRecognizer:expandTextTapRecognizer]; + containerView.accessibilityTraits |= UIAccessibilityTraitButton; + containerView.isAccessibilityElement = YES; + self.tappableView = containerView; + [self _updateTappableViewAccessibilityValue]; +} + +- (void)setTitle:(NSString * _Nullable)title { + _title = title; + + self.titleLabel.text = title; + self.tappableView.accessibilityLabel = title; +} + +- (void)setText:(NSString * _Nullable)text { + _text = text; + + self.textLabel.text = text; +} + +- (void)_updateTappableViewAccessibilityValue { + if (self.textContainerView.isHidden) { + self.tappableView.accessibilityValue = STDSLocalizedString(@"Collapsed", @"Accessibility label for expandandable text control to indicate text is hidden."); + } else { + self.tappableView.accessibilityValue = STDSLocalizedString(@"Expanded", @"Accessibility label for expandandable text control to indicate that the UI has been expanded and additional text is available."); + } +} + +- (void)_toggleTextExpansion { + if (self.didTap) { + self.didTap(); + } + self.textContainerView.hidden = !self.textContainerView.hidden; + + CGFloat rotationValue = (CGFloat)M_PI_2; + if (self.textContainerView.isHidden) { + rotationValue = (CGFloat)0; + [self.imageViewStackView removeArrangedSubview:self.imageViewSpacerView]; + UIAccessibilityPostNotification(UIAccessibilityLayoutChangedNotification, self.titleLabel); + } else { + [self.imageViewStackView addArrangedSubview:self.imageViewSpacerView]; + UIAccessibilityPostNotification(UIAccessibilityLayoutChangedNotification, self.textLabel); + } + [self _updateTappableViewAccessibilityValue]; + + [UIView animateWithDuration:kTitleImageViewRotationAnimationDuration animations:^{ + self.titleImageView.transform = CGAffineTransformMakeRotation(rotationValue); + }]; +} + +- (void)setCustomization:(STDSFooterCustomization * _Nullable)customization { + self.titleLabel.font = customization.headingFont; + self.titleLabel.textColor = customization.headingTextColor; + + self.textLabel.font = customization.font; + self.textLabel.textColor = customization.textColor; + + self.titleImageView.tintColor = customization.chevronColor; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSIPAddress.h b/Stripe3DS2/Stripe3DS2/STDSIPAddress.h new file mode 100644 index 00000000..a6d5911f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSIPAddress.h @@ -0,0 +1,15 @@ +// +// STDSIPAddress.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +NSString * _Nullable STDSCurrentDeviceIPAddress(void); + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSIPAddress.m b/Stripe3DS2/Stripe3DS2/STDSIPAddress.m new file mode 100644 index 00000000..863eddb8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSIPAddress.m @@ -0,0 +1,43 @@ +// +// STDSIPAddress.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSIPAddress.h" + +#include +#include + +// source: https://zachwaugh.com/posts/programmatically-retrieving-ip-address-of-iphone +NSString * STDSCurrentDeviceIPAddress(void) { + NSString *address = nil; + struct ifaddrs *interfaces = NULL; + struct ifaddrs *temp_addr = NULL; + int success = 0; + + // retrieve the current interfaces - returns 0 on success + success = getifaddrs(&interfaces); + if (success == 0) { + // Loop through linked list of interfaces + temp_addr = interfaces; + while (temp_addr != NULL) { + if( temp_addr->ifa_addr->sa_family == AF_INET) { + // Check if interface is en0 which is the wifi connection on the iPhone + if ([[NSString stringWithUTF8String:temp_addr->ifa_name] isEqualToString:@"en0"]) { + // Get NSString from C String + address = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_addr)->sin_addr)]; + } + } + + temp_addr = temp_addr->ifa_next; + } + } + + // Free memory + freeifaddrs(interfaces); + + return address; +} diff --git a/Stripe3DS2/Stripe3DS2/STDSImageLoader.h b/Stripe3DS2/Stripe3DS2/STDSImageLoader.h new file mode 100644 index 00000000..15f1396a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSImageLoader.h @@ -0,0 +1,36 @@ +// +// STDSImageLoader.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/5/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import + +NS_ASSUME_NONNULL_BEGIN + +typedef void (^STDSImageDownloadCompletionBlock)(UIImage * _Nullable); + +@interface STDSImageLoader: NSObject + +/** + Initializes an `STDSImageLoader` with the given parameters. + + @param session The session to initialize the loader with. + @return Returns an initialized `STDSImageLoader` object. + */ +- (instancetype)initWithURLSession:(NSURLSession *)session; + +/** + Attempts to load an image from the specified URL. + + @param URL The URL to load an image for. + @param completion A completion block that is called when the image loading has finished. This will be called on the main queue. + */ +- (void)loadImageFromURL:(NSURL *)URL completion:(STDSImageDownloadCompletionBlock)completion; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSImageLoader.m b/Stripe3DS2/Stripe3DS2/STDSImageLoader.m new file mode 100644 index 00000000..1d87e44f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSImageLoader.m @@ -0,0 +1,50 @@ +// +// STDSImageLoader.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/5/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSImageLoader.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSImageLoader() + +@property (nonatomic, strong) NSURLSession *session; + +@end + +@implementation STDSImageLoader + +- (instancetype)initWithURLSession:(NSURLSession *)session { + self = [super init]; + + if (self) { + _session = session; + } + + return self; +} + +- (void)loadImageFromURL:(NSURL *)URL completion:(STDSImageDownloadCompletionBlock)completion { + NSURLSessionDataTask *dataTask = [self.session dataTaskWithURL:URL completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) { + UIImage *image; + + if (data != nil) { + image = [UIImage imageWithData:data]; + } + + [NSOperationQueue.mainQueue addOperationWithBlock:^{ + completion(image); + }]; + }]; + + [dataTask resume]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.h b/Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.h new file mode 100644 index 00000000..ed584dec --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.h @@ -0,0 +1,19 @@ +// +// STDSIntegrityChecker.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSIntegrityChecker : NSObject + ++ (BOOL)SDKIntegrityIsValid; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.m b/Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.m new file mode 100644 index 00000000..73ed1ea8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSIntegrityChecker.m @@ -0,0 +1,41 @@ +// +// STDSIntegrityChecker.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSIntegrityChecker.h" +#import "Stripe3DS2.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSIntegrityChecker + ++ (BOOL)SDKIntegrityIsValid { + + if (NSClassFromString(@"STDSIntegrityChecker") == [STDSIntegrityChecker class] && + NSClassFromString(@"STDSConfigParameters") == [STDSConfigParameters class] && + NSClassFromString(@"STDSThreeDS2Service") == [STDSThreeDS2Service class] && + NSClassFromString(@"STDSUICustomization") == [STDSUICustomization class] && + NSClassFromString(@"STDSWarning") == [STDSWarning class] && + NSClassFromString(@"STDSAlreadyInitializedException") == [STDSAlreadyInitializedException class] && + NSClassFromString(@"STDSNotInitializedException") == [STDSNotInitializedException class] && + NSClassFromString(@"STDSRuntimeException") == [STDSRuntimeException class] && + NSClassFromString(@"STDSErrorMessage") == [STDSErrorMessage class] && + NSClassFromString(@"STDSRuntimeErrorEvent") == [STDSRuntimeErrorEvent class] && + NSClassFromString(@"STDSProtocolErrorEvent") == [STDSProtocolErrorEvent class] && + NSClassFromString(@"STDSAuthenticationRequestParameters") == [STDSAuthenticationRequestParameters class] && + NSClassFromString(@"STDSChallengeParameters") == [STDSChallengeParameters class] && + NSClassFromString(@"STDSCompletionEvent") == [STDSCompletionEvent class] && + NSClassFromString(@"STDSTransaction") == [STDSTransaction class]) { + return YES; + } + + return NO; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.h b/Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.h new file mode 100644 index 00000000..2b2bc306 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.h @@ -0,0 +1,45 @@ +// +// STDSJSONWebEncryption.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/24/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDirectoryServer.h" + +@class STDSDirectoryServerCertificate; +@class STDSJSONWebSignature; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSJSONWebEncryption : NSObject + ++ (nullable NSString *)encryptJSON:(NSDictionary *)json + forDirectoryServer:(STDSDirectoryServer)directoryServer + error:(out NSError * _Nullable *)error; + ++ (nullable NSString *)encryptJSON:(NSDictionary *)json + withCertificate:(STDSDirectoryServerCertificate *)certificate + directoryServerID:(NSString *)directoryServerID + serverKeyID:(nullable NSString *)serverKeyID + error:(out NSError * _Nullable *)error; + ++ (nullable NSString *)directEncryptJSON:(NSDictionary *)json + withContentEncryptionKey:(NSData *)contentEncryptionKey + forACSTransactionID:(NSString *)acsTransactionID + error:(out NSError * _Nullable *)error; + ++ (nullable NSDictionary *)decryptData:(NSData *)data + withContentEncryptionKey:(NSData *)contentEncryptionKey + error:(out NSError * _Nullable *)error; + ++ (BOOL)verifyJSONWebSignature:(STDSJSONWebSignature *)jws forDirectoryServer:(STDSDirectoryServer)directoryServer; + ++ (BOOL)verifyJSONWebSignature:(STDSJSONWebSignature *)jws withCertificate:(STDSDirectoryServerCertificate *)certificate rootCertificates:(NSArray *)rootCertificates; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.m b/Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.m new file mode 100644 index 00000000..57023a5e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSJSONWebEncryption.m @@ -0,0 +1,407 @@ +// +// STDSJSONWebEncryption.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/24/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSJSONWebEncryption.h" + +#import +#import + +#import "NSData+JWEHelpers.h" +#import "NSError+Stripe3DS2.h" +#import "NSString+JWEHelpers.h" +#import "STDSDirectoryServerCertificate.h" +#import "STDSEphemeralKeyPair.h" +#import "STDSJSONWebSignature.h" +#import "STDSSecTypeUtilities.h" + +NS_ASSUME_NONNULL_BEGIN + +#pragma mark - _STPJSONWebEncryptionResult +@interface _STPJSONWebEncryptionResult : NSObject + +- (instancetype)initWithCiphertext:(NSData *)ciphertextData + initializationVector:(NSData *)initializationVector + hmacTag:(NSData *)hmacTag; + +@property (nonatomic, copy, readonly) NSData *ciphertextData; +@property (nonatomic, copy, readonly) NSData *initializationVector; +@property (nonatomic, copy, readonly) NSData *hmacTag; + +@end + +@implementation _STPJSONWebEncryptionResult + +- (instancetype)initWithCiphertext:(NSData *)ciphertextData + initializationVector:(NSData *)initializationVector + hmacTag:(NSData *)hmacTag { + self = [super init]; + if (self) { + _ciphertextData = [ciphertextData copy]; + _initializationVector = [initializationVector copy]; + _hmacTag = [hmacTag copy]; + } + + return self; +} + +@end + +#pragma mark - STDSJSONWebEncryption + +static const size_t kContentEncryptionKeyByteCount = 32; +static const size_t kHMACSubKeyByteCount = 16; +static const size_t kAESSubKeyByteCount = 16; + +@implementation STDSJSONWebEncryption + ++ (nullable NSString *)encryptJSON:(NSDictionary *)json + forDirectoryServer:(STDSDirectoryServer)directoryServer + error:(out NSError *__autoreleasing _Nullable * _Nullable)error { + + NSString *ciphertext = nil; + + STDSDirectoryServerCertificate *certificate = [STDSDirectoryServerCertificate certificateForDirectoryServer:directoryServer]; + NSString *directoryServerID = STDSDirectoryServerIdentifier(directoryServer); + + if (certificate != nil && directoryServerID != nil) { + + ciphertext = [self encryptJSON:json + withCertificate:certificate + directoryServerID:directoryServerID + serverKeyID:nil + error:error]; + } + + if (error != nil && ciphertext == nil) { + *error = *error ?: [NSError _stds_jweError]; + } + + return ciphertext; +} + ++ (nullable NSString *)encryptJSON:(NSDictionary *)json + withCertificate:(STDSDirectoryServerCertificate *)certificate + directoryServerID:(NSString *)directoryServerID + serverKeyID:(nullable NSString *)serverKeyID + error:(out NSError * _Nullable *)error { + + NSString *ciphertext = nil; + NSError *jsonError = nil; + NSData * jsonData = [NSJSONSerialization dataWithJSONObject:json options:0 error:&jsonError]; + + if (jsonData != nil) { + + switch (certificate.keyType) { + case STDSDirectoryServerKeyTypeRSA: + ciphertext = [self _RSAEncryptPlaintext:jsonData + withCertificate:certificate + serverKeyID:serverKeyID]; + break; + + case STDSDirectoryServerKeyTypeEC: + ciphertext = [self _directEncryptPlaintext:jsonData + withCertificate:certificate + forDirectoryServer:directoryServerID]; + break; + + case STDSDirectoryServerKeyTypeUnknown: + break; + } + } + + if (error != nil && ciphertext == nil) { + *error = jsonError ?: [NSError _stds_jweError]; + } + + return ciphertext; +} + ++ (nullable NSString *)directEncryptJSON:(NSDictionary *)json + withContentEncryptionKey:(NSData *)contentEncryptionKey + forACSTransactionID:(NSString *)acsTransactionID + error:(out NSError * _Nullable *)error { + NSString *ciphertext = nil; + + NSError *jsonError = nil; + + + NSData * jsonData = [NSJSONSerialization dataWithJSONObject:json options:0 error:&jsonError]; + + if (jsonData != nil) { + NSString *headerString = [NSString stringWithFormat:@"{\"kid\":\"%@\",\"enc\":\"A128CBC-HS256\",\"alg\":\"dir\"}", acsTransactionID]; + ciphertext = [self _directEncryptPlaintext:jsonData + withContentEncryptionKey:contentEncryptionKey + headerString:headerString]; + } + + if (error != nil && ciphertext == nil) { + *error = jsonError ?: [NSError _stds_jweError]; + } + + return ciphertext; +} + ++ (nullable NSData *)_hmacTagWithKey:(NSData *)hmacSubKey + headerString:(NSString *)headerString + initializationVector:(NSData *)initializationVector + cipherText:(NSData *)cipherText { + NSString *encodedHeaderString = [headerString _stds_base64URLEncodedString]; + + // Per JWE spec, the encoded header data is included as additional authenticated data but with ASCII encoding https://tools.ietf.org/html/rfc7516#section-5.1 + NSData *additionalAuthenticatedData = [encodedHeaderString dataUsingEncoding:NSASCIIStringEncoding]; + uint64_t AL = CFSwapInt64HostToBig(additionalAuthenticatedData.length*8); + + NSMutableData *d = [NSMutableData data]; + [d appendBytes:additionalAuthenticatedData.bytes length:additionalAuthenticatedData.length]; + [d appendBytes:initializationVector.bytes length:initializationVector.length]; + [d appendBytes:cipherText.bytes length:cipherText.length]; + [d appendBytes:&AL length:sizeof(AL)]; + + NSMutableData *M = [[NSMutableData alloc] initWithLength:CC_SHA256_DIGEST_LENGTH]; + + CCHmac(kCCHmacAlgSHA256, + hmacSubKey.bytes, + kHMACSubKeyByteCount, + d.bytes, + d.length, + M.mutableBytes); + + + NSData *hmacTag = [M subdataWithRange:NSMakeRange(0, 16)]; + + return hmacTag; +} + ++ (nullable _STPJSONWebEncryptionResult *)_encryptPlaintext:(NSData *)plaintextData + withKey:(NSData *)contentEncryptionKeyData + headerString:(NSString *)headerString { + // ecrypt JSON according to JWE (RFC 7516) using JWE Compact Serialization + // enc: A128CBC-HS256 + + // Ref: rfc7518 sec 5.2.3 https://www.rfc-editor.org/rfc/rfc7518.txt + + _STPJSONWebEncryptionResult *result = nil; + + static const size_t kInitializationVectorByteCount = 16; + + NSAssert(contentEncryptionKeyData.length == kContentEncryptionKeyByteCount, @"Must use a valid 256 content encryption key"); + if (contentEncryptionKeyData.length == kContentEncryptionKeyByteCount) { + + NSData *hmacSubKeyData = [contentEncryptionKeyData subdataWithRange:NSMakeRange(0, kHMACSubKeyByteCount)]; + NSData *aesSubKeyData = [contentEncryptionKeyData subdataWithRange:NSMakeRange(kHMACSubKeyByteCount, kAESSubKeyByteCount)]; + NSData *initializationVectorData = STDSCryptoRandomData(kInitializationVectorByteCount); + + + // pad with block size for AES + NSMutableData *ciphertextData = [NSMutableData dataWithLength:plaintextData.length + kCCBlockSizeAES128]; + size_t outLength; + CCCryptorStatus aesEncryptionResult = CCCrypt(kCCEncrypt, + kCCAlgorithmAES, + kCCOptionPKCS7Padding, + aesSubKeyData.bytes, + kAESSubKeyByteCount, + initializationVectorData.bytes, + plaintextData.bytes, + (size_t)plaintextData.length, + ciphertextData.mutableBytes, + ciphertextData.length, + &outLength); + if (aesEncryptionResult == kCCSuccess) { + ciphertextData.length = outLength; + + NSData *hmacTag = [self _hmacTagWithKey:hmacSubKeyData + headerString:headerString + initializationVector:initializationVectorData + cipherText:ciphertextData]; + + result = [[_STPJSONWebEncryptionResult alloc] initWithCiphertext:ciphertextData + initializationVector:initializationVectorData + hmacTag:hmacTag]; + } + } + + return result; +} + ++ (nullable NSString *)_RSAEncryptPlaintext:(NSData *)plaintextData + withCertificate:(STDSDirectoryServerCertificate *)certificate + serverKeyID:(nullable NSString *)serverKeyID { + NSData *contentEncryptionKey = STDSCryptoRandomData(kContentEncryptionKeyByteCount); + if (contentEncryptionKey != nil) { + NSString *headerString = nil; + + if (serverKeyID != nil) { + headerString = [NSString stringWithFormat:@"{\"enc\":\"A128CBC-HS256\",\"alg\":\"RSA-OAEP-256\",\"kid\":\"%@\"}", serverKeyID]; + } else { + headerString = @"{\"enc\":\"A128CBC-HS256\",\"alg\":\"RSA-OAEP-256\"}"; + + } + _STPJSONWebEncryptionResult *encryptedData = [self _encryptPlaintext:plaintextData + withKey:contentEncryptionKey + headerString:headerString]; + if (encryptedData != nil) { + NSData *encryptedCEK = [certificate encryptDataUsingRSA_OAEP_SHA256:contentEncryptionKey]; + if (encryptedCEK != nil) { + return [NSString stringWithFormat:@"%@.%@.%@.%@.%@", [headerString _stds_base64URLEncodedString], [encryptedCEK _stds_base64URLEncodedString], [encryptedData.initializationVector _stds_base64URLEncodedString], [encryptedData.ciphertextData _stds_base64URLEncodedString], [encryptedData.hmacTag _stds_base64URLEncodedString]]; + } + } + } + + return nil; +} + ++ (nullable NSString *)_directEncryptPlaintext:(NSData *)plaintextData + withCertificate:(STDSDirectoryServerCertificate *)certificate + forDirectoryServer:(NSString *)directoryServerID { + + STDSEphemeralKeyPair *ephemeralKeyPair = [STDSEphemeralKeyPair ephemeralKeyPair]; + NSData *rawSharedSecret = [ephemeralKeyPair createSharedSecretWithCertificate:certificate]; + NSData *contentEncryptionKey = STDSCreateConcatKDFWithSHA256(rawSharedSecret, kContentEncryptionKeyByteCount, directoryServerID); + + NSString *headerString = [NSString stringWithFormat:@"{\"enc\":\"A128CBC-HS256\",\"alg\":\"ECDH-ES\",\"epk\":%@}", ephemeralKeyPair.publicKeyJWK]; + + return [self _directEncryptPlaintext:plaintextData + withContentEncryptionKey:contentEncryptionKey + headerString:headerString]; +} + ++ (nullable NSString *)_directEncryptPlaintext:(NSData *)plaintextData + withContentEncryptionKey:(NSData *)contentEncryptionKey + headerString:(NSString *)headerString { + + _STPJSONWebEncryptionResult *encryptedData = [self _encryptPlaintext:plaintextData + withKey:contentEncryptionKey + headerString:headerString]; + if (encryptedData != nil) { + return [NSString stringWithFormat:@"%@..%@.%@.%@", [headerString _stds_base64URLEncodedString], [encryptedData.initializationVector _stds_base64URLEncodedString], [encryptedData.ciphertextData _stds_base64URLEncodedString], [encryptedData.hmacTag _stds_base64URLEncodedString]]; + } + + return nil; +} + +#pragma mark - Decryption + ++ (nullable NSDictionary *)decryptData:(NSData *)data + withContentEncryptionKey:(NSData *)contentEncryptionKey + error:(out NSError * _Nullable *)error { + NSString *jweString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]; + NSArray *jweComponents = [jweString componentsSeparatedByString:@"."]; + if (jweComponents.count != 5) { + + // Data may be JSON describing error + NSError *jsonError = nil; + NSDictionary *errorJSON = [NSJSONSerialization JSONObjectWithData:data options:0 error:&jsonError]; + if (errorJSON != nil) { + return errorJSON; + } + if (error != NULL) { + *error = jsonError ?: [NSError _stds_jweError]; + } + return nil; + } + + NSString *headerString = [jweComponents[0] _stds_base64URLDecodedString]; + NSData *initializationVector = [jweComponents[2] _stds_base64URLDecodedData]; + NSData *ciphertextData = [jweComponents[3] _stds_base64URLDecodedData]; + NSData *hmacTag = [jweComponents[4] _stds_base64URLDecodedData]; + + if (headerString == nil || + initializationVector == nil || + ciphertextData == nil || + hmacTag == nil + ) { + if (error != NULL) { + *error = [NSError _stds_jweError]; + } + return nil; + } + NSAssert(contentEncryptionKey.length == kContentEncryptionKeyByteCount, @"Must use a valid 256 content encryption key"); + if (contentEncryptionKey.length != kContentEncryptionKeyByteCount) { + if (error != NULL) { + *error = [NSError _stds_jweError]; + } + return nil; + } + + NSData *hmacSubKeyData = [contentEncryptionKey subdataWithRange:NSMakeRange(0, kHMACSubKeyByteCount)]; + NSData *aesSubKeyData = [contentEncryptionKey subdataWithRange:NSMakeRange(kHMACSubKeyByteCount, kAESSubKeyByteCount)]; + + // pad with block size for AES + NSMutableData *plaintextData = [NSMutableData dataWithLength:ciphertextData.length + kCCBlockSizeAES128]; + size_t outLength; + CCCryptorStatus aesDecryptionResult = CCCrypt(kCCDecrypt, + kCCAlgorithmAES, + kCCOptionPKCS7Padding, + aesSubKeyData.bytes, + kAESSubKeyByteCount, + initializationVector.bytes, + ciphertextData.bytes, + (size_t)ciphertextData.length, + plaintextData.mutableBytes, + plaintextData.length, + &outLength); + + if (aesDecryptionResult != kCCSuccess) { + if (error != NULL) { + *error = [NSError _stds_jweError]; + } + return nil; + } + + plaintextData.length = outLength; + NSData *calculatedHMACTag = [self _hmacTagWithKey:hmacSubKeyData + headerString:headerString + initializationVector:initializationVector + cipherText:ciphertextData]; + + if (![calculatedHMACTag isEqualToData:hmacTag]) { + if (error != NULL) { + *error = [NSError _stds_jweError]; + } + return nil; + } + + NSDictionary *decryptedJSON = [NSJSONSerialization JSONObjectWithData:plaintextData + options:0 + error:error]; + if (*error != NULL) { + *error = [NSError _stds_jweError]; + return nil; + } + + return decryptedJSON; +} + +#pragma mark - JSON Web Signature Verification + ++ (BOOL)verifyJSONWebSignature:(STDSJSONWebSignature *)jws forDirectoryServer:(STDSDirectoryServer)directoryServer { + STDSDirectoryServerCertificate *certificate = [STDSDirectoryServerCertificate certificateForDirectoryServer:directoryServer]; + NSString *certificateString = nil; + + if (directoryServer == STDSDirectoryServerULTestRSA || directoryServer == STDSDirectoryServerULTestEC) { + // for UL tests, the last certificate in the chain is the anchor/root + certificateString = jws.certificateChain.lastObject; + } else { + NSAssert(0, @"We shouldn't be using this path outside of UL testing"); + certificateString = certificate.certificateString; + } + + if (certificateString == nil) { + return NO; + } + + return [self verifyJSONWebSignature:jws withCertificate:certificate rootCertificates:@[certificateString]]; +} + ++ (BOOL)verifyJSONWebSignature:(STDSJSONWebSignature *)jws withCertificate:(__unused STDSDirectoryServerCertificate *)certificate rootCertificates:(NSArray *)rootCertificates { + return [STDSDirectoryServerCertificate verifyJSONWebSignature:jws withRootCertificates:rootCertificates]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.h b/Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.h new file mode 100644 index 00000000..d6f17a19 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.h @@ -0,0 +1,41 @@ +// +// STDSJSONWebSignature.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 4/2/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class STDSEllipticCurvePoint; + +NS_ASSUME_NONNULL_BEGIN + +typedef NS_ENUM(NSInteger, STDSJSONWebSignatureAlgorithm) { + STDSJSONWebSignatureAlgorithmES256, + STDSJSONWebSignatureAlgorithmPS256, + STDSJSONWebSignatureAlgorithmUnknown, +}; + +@interface STDSJSONWebSignature : NSObject + +- (nullable instancetype)initWithString:(NSString *)jwsString; +- (nullable instancetype)initWithString:(NSString *)jwsString allowNilKey:(BOOL)allowNilKey; + +@property (nonatomic, readonly) STDSJSONWebSignatureAlgorithm algorithm; + +@property (nonatomic, readonly) NSData *digest; +@property (nonatomic, readonly) NSData *signature; + +@property (nonatomic, readonly) NSData *payload; + +/// non-nil if algorithm == STDSJSONWebSignatureAlgorithmES256 +@property (nonatomic, nullable, readonly) STDSEllipticCurvePoint *ellipticCurvePoint; + +/// non-nil if algorithm == STDSJSONWebSignatureAlgorithmPS256, can be non-nil for algorithm == STDSJSONWebSignatureAlgorithmES256 +@property (nonatomic, nullable, readonly) NSArray *certificateChain; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.m b/Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.m new file mode 100644 index 00000000..2e294937 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSJSONWebSignature.m @@ -0,0 +1,88 @@ +// +// STDSJSONWebSignature.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 4/2/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSJSONWebSignature.h" + +#import "NSString+JWEHelpers.h" +#import "STDSEllipticCurvePoint.h" +#import "STDSSecTypeUtilities.h" + +NS_ASSUME_NONNULL_BEGIN + +static NSString * const kJWSAlgorithmKey = @"alg"; +static NSString * const kJWSAlgorithmES256 = @"ES256"; +static NSString * const kJWSAlgorithmPS256 = @"PS256"; + +static NSString * const kJWSCertificateChainKey = @"x5c"; + +@implementation STDSJSONWebSignature + +- (nullable instancetype)initWithString:(NSString *)jwsString { + return [self initWithString:jwsString allowNilKey:NO]; +} + +- (nullable instancetype)initWithString:(NSString *)jwsString allowNilKey:(BOOL)allowNilKey { + self = [super init]; + if (self) { + NSArray *components = [jwsString componentsSeparatedByString:@"."]; + if (components.count != 3) { + return nil; + } + NSData *headerData = [components[0] _stds_base64URLDecodedData]; + if (headerData == nil) { + return nil; + } + NSError *jsonError = nil; + NSDictionary * headerJSON = [NSJSONSerialization JSONObjectWithData:headerData options:0 error:&jsonError]; + if (headerJSON == nil) { + return nil; + } + + if (headerJSON[kJWSCertificateChainKey] != nil && ![headerJSON[kJWSCertificateChainKey] isKindOfClass:[NSArray class]]) { + return nil; + } + + _certificateChain = headerJSON[kJWSCertificateChainKey]; + + NSString *algorithm = headerJSON[kJWSAlgorithmKey]; + if ([algorithm compare:kJWSAlgorithmES256 options: NSCaseInsensitiveSearch] == NSOrderedSame) { + _algorithm = STDSJSONWebSignatureAlgorithmES256; + if (_certificateChain.count > 0) { + SecCertificateRef certificate = STDSSecCertificateFromString(_certificateChain.firstObject); + if (certificate != NULL) { + SecKeyRef key = SecCertificateCopyKey(certificate); + CFRelease(certificate); + if (key != NULL) { + _ellipticCurvePoint = [[STDSEllipticCurvePoint alloc] initWithKey:key]; + CFRelease(key); + } + } + if (_ellipticCurvePoint == nil && !allowNilKey) { + return nil; + } + } else if (!allowNilKey) { + return nil; + } + + } else if ([algorithm compare:kJWSAlgorithmPS256 options: NSCaseInsensitiveSearch] == NSOrderedSame) { + _algorithm = STDSJSONWebSignatureAlgorithmPS256; + } else { + return nil; + } + + _digest = [[@[components[0], components[1]] componentsJoinedByString:@"."] dataUsingEncoding:NSUTF8StringEncoding]; + _signature = [components[2] _stds_base64URLDecodedData]; + _payload = [components[1] _stds_base64URLDecodedData]; + } + + return self; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.h b/Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.h new file mode 100644 index 00000000..303575c5 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.h @@ -0,0 +1,19 @@ +// +// STDSJailbreakChecker.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSJailbreakChecker : NSObject + ++ (BOOL)isJailbroken; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.m b/Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.m new file mode 100644 index 00000000..6c0dcbf2 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSJailbreakChecker.m @@ -0,0 +1,31 @@ +// +// STDSJailbreakChecker.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSJailbreakChecker.h" + +@import UIKit; + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSJailbreakChecker + +// This was implemented under the following guidance: https://medium.com/@pinmadhon/how-to-check-your-app-is-installed-on-a-jailbroken-device-67fa0170cf56 ++ (BOOL)isJailbroken { + + // Check for existence of files that are common for jailbroken devices + if ([[NSFileManager defaultManager] fileExistsAtPath:@"/Applications/Cydia.app"] || + [[NSFileManager defaultManager] fileExistsAtPath:@"/Library/MobileSubstrate/MobileSubstrate.dylib"]) { + return YES; + } + + return NO; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSLocalizedString.h b/Stripe3DS2/Stripe3DS2/STDSLocalizedString.h new file mode 100644 index 00000000..7a3a76c2 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSLocalizedString.h @@ -0,0 +1,18 @@ +// +// STDSLocalizedString.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 7/9/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSBundleLocator.h" + +#ifndef STDSLocalizedString_h +#define STDSLocalizedString_h + +#define STDSLocalizedString(key, comment) \ +[[STDSBundleLocator stdsResourcesBundle] localizedStringForKey:(key) value:@"" table:nil] + + +#endif /* STDSLocalizedString_h */ diff --git a/Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.h b/Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.h new file mode 100644 index 00000000..72b79cf1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.h @@ -0,0 +1,19 @@ +// +// STDSOSVersionChecker.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSOSVersionChecker : NSObject + ++ (BOOL)isSupportedOSVersion; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.m b/Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.m new file mode 100644 index 00000000..f5f880e2 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSOSVersionChecker.m @@ -0,0 +1,25 @@ +// +// STDSOSVersionChecker.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSOSVersionChecker.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSOSVersionChecker + ++ (BOOL)isSupportedOSVersion { + if (@available(iOS 11, *)) { + return YES; + } else { + return NO; + } +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSProcessingView.h b/Stripe3DS2/Stripe3DS2/STDSProcessingView.h new file mode 100644 index 00000000..95364fc5 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSProcessingView.h @@ -0,0 +1,26 @@ +// +// STDSProcessingView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/19/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@class STDSUICustomization; + +@interface STDSProcessingView : UIView + +/// Defaults to NO +@property (nonatomic) BOOL shouldDisplayBlurView; +/// Defaults to YES +@property (nonatomic) BOOL shouldDisplayDSLogo; + +- (instancetype)initWithCustomization:(STDSUICustomization *)customization directoryServerLogo:(nullable UIImage *)directoryServerLogo; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSProcessingView.m b/Stripe3DS2/Stripe3DS2/STDSProcessingView.m new file mode 100644 index 00000000..3126f974 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSProcessingView.m @@ -0,0 +1,98 @@ +// +// STDSProcessingView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/19/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSProcessingView.h" +#import "STDSStackView.h" +#import "UIView+LayoutSupport.h" +#import "UIFont+DefaultFonts.h" +#import "STDSUICustomization.h" +#import "STDSBundleLocator.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSProcessingView() + +@property (nonatomic, strong) STDSStackView *imageStackView; +@property (nonatomic, strong) UIView *blurViewPlaceholder; +@property (nonatomic, strong) UIImageView *imageView; + +@end + +@implementation STDSProcessingView + +static const CGFloat kProcessingViewHorizontalMargin = 8; +static const CGFloat kProcessingViewTopPadding = 22; +static const CGFloat kProcessingViewBottomPadding = 36; + +- (instancetype)initWithCustomization:(STDSUICustomization *)customization directoryServerLogo:(nullable UIImage *)directoryServerLogo { + self = [super initWithFrame:CGRectZero]; + + if (self) { + _blurViewPlaceholder = [UIView new]; + _imageView = [[UIImageView alloc] initWithImage:directoryServerLogo]; + _imageView.contentMode = UIViewContentModeScaleAspectFit; + _shouldDisplayDSLogo = YES; + [self _setupViewHierarchyWithCustomization:customization]; + } + + return self; +} + +- (void)setShouldDisplayBlurView:(BOOL)shouldDisplayBlurView { + _shouldDisplayBlurView = shouldDisplayBlurView; + self.blurViewPlaceholder.hidden = shouldDisplayBlurView; +} + +- (void)setShouldDisplayDSLogo:(BOOL)shouldDisplayDSLogo { + _shouldDisplayDSLogo = shouldDisplayDSLogo; + self.imageView.hidden = !shouldDisplayDSLogo; +} + +- (void)_setupViewHierarchyWithCustomization:(STDSUICustomization *)customization { + self.blurViewPlaceholder.backgroundColor = customization.backgroundColor; + UIVisualEffectView *blurView = [[UIVisualEffectView alloc] initWithEffect:[UIBlurEffect effectWithStyle:customization.blurStyle]]; + blurView.translatesAutoresizingMaskIntoConstraints = NO; + + STDSStackView *containerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + containerView.backgroundColor = customization.backgroundColor; + containerView.layer.cornerRadius = 13; + containerView.translatesAutoresizingMaskIntoConstraints = NO; + + UIActivityIndicatorView *indicatorView = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:customization.activityIndicatorViewStyle]; + + [self addSubview:blurView]; + [blurView.contentView addSubview:self.blurViewPlaceholder]; + [self addSubview:containerView]; + + [self.blurViewPlaceholder _stds_pinToSuperviewBoundsWithoutMargin]; + [blurView _stds_pinToSuperviewBoundsWithoutMargin]; + + NSLayoutConstraint *centerXConstraint = [NSLayoutConstraint constraintWithItem:containerView attribute:NSLayoutAttributeCenterX relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeCenterX multiplier:1.0 constant:0]; + NSLayoutConstraint *centerYConstraint = [NSLayoutConstraint constraintWithItem:containerView attribute:NSLayoutAttributeCenterY relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeCenterY multiplier:1.0 constant:0]; + NSLayoutConstraint *widthConstraint = [NSLayoutConstraint constraintWithItem:containerView attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeWidth multiplier:1.0 constant:-kProcessingViewHorizontalMargin * 2]; + + [NSLayoutConstraint activateConstraints:@[ + centerXConstraint, + centerYConstraint, + widthConstraint, + [containerView.topAnchor constraintGreaterThanOrEqualToAnchor:self.topAnchor], + [self.bottomAnchor constraintGreaterThanOrEqualToAnchor:containerView.bottomAnchor], + ]]; + + [containerView addSpacer:kProcessingViewTopPadding]; + [containerView addArrangedSubview:self.imageView]; + [containerView addSpacer:20]; + [containerView addArrangedSubview:indicatorView]; + [containerView addSpacer:kProcessingViewBottomPadding]; + + [indicatorView startAnimating]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSProgressViewController.h b/Stripe3DS2/Stripe3DS2/STDSProgressViewController.h new file mode 100644 index 00000000..147fa4e4 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSProgressViewController.h @@ -0,0 +1,23 @@ +// +// STDSProgressViewController.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 5/6/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDirectoryServer.h" + +@class STDSImageLoader, STDSUICustomization; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSProgressViewController : UIViewController + +- (instancetype)initWithDirectoryServer:(STDSDirectoryServer)directoryServer uiCustomization:(STDSUICustomization * _Nullable)uiCustomization didCancel:(void (^)(void))didCancel; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSProgressViewController.m b/Stripe3DS2/Stripe3DS2/STDSProgressViewController.m new file mode 100644 index 00000000..d53fe7f1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSProgressViewController.m @@ -0,0 +1,55 @@ +// +// STDSProgressViewController.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 5/6/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSProgressViewController.h" + +#import "STDSBundleLocator.h" +#import "STDSUICustomization.h" +#import "UIViewController+Stripe3DS2.h" +#import "STDSProcessingView.h" + +@interface STDSProgressViewController() +@property (nonatomic, strong, nullable) STDSUICustomization *uiCustomization; +@property (nonatomic, strong) void (^didCancel)(void); +@property (nonatomic) STDSDirectoryServer directoryServer; +@end + +@implementation STDSProgressViewController + +- (instancetype)initWithDirectoryServer:(STDSDirectoryServer)directoryServer uiCustomization:(STDSUICustomization * _Nullable)uiCustomization didCancel:(void (^)(void))didCancel { + self = [super initWithNibName:nil bundle:nil]; + + if (self) { + _directoryServer = directoryServer; + _uiCustomization = uiCustomization; + _didCancel = didCancel; + } + + return self; +} + +- (void)loadView { + NSString *imageName = STDSDirectoryServerImageName(self.directoryServer); + UIImage *dsImage = imageName ? [UIImage imageNamed:imageName inBundle:[STDSBundleLocator stdsResourcesBundle] compatibleWithTraitCollection:nil] : nil; + self.view = [[STDSProcessingView alloc] initWithCustomization:self.uiCustomization directoryServerLogo:dsImage]; +} + +- (UIStatusBarStyle)preferredStatusBarStyle { + return self.uiCustomization.preferredStatusBarStyle; +} + +- (void)viewDidLoad { + [super viewDidLoad]; + [self _stds_setupNavigationBarElementsWithCustomization:self.uiCustomization cancelButtonSelector:@selector(_cancelButtonTapped:)]; +} + +- (void)_cancelButtonTapped:(UIButton *)sender { + self.didCancel(); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.h b/Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.h new file mode 100644 index 00000000..d0d416e6 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.h @@ -0,0 +1,49 @@ +// +// STDSSecTypeUtilities.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/28/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDirectoryServer.h" + +NS_ASSUME_NONNULL_BEGIN + +/// Returns the SecCertificateRef for the specified server or NULL if there's an error +SecCertificateRef _Nullable STDSCertificateForServer(STDSDirectoryServer server); + +/// Returns the public key in the certificate or NULL if there's an error +SecKeyRef _Nullable SecCertificateCopyKey(SecCertificateRef certificate); + +/// Returns one of the values defined for kSecAttrKeyType in or NULL +CFStringRef _Nullable STDSSecCertificateCopyPublicKeyType(SecCertificateRef certificate); + +/// Returns the hashed secret or nil +NSData * _Nullable STDSCreateConcatKDFWithSHA256(NSData *sharedSecret, NSUInteger keyLength, NSString *apv); + +/// Verifies the signature and payload using the Elliptic Curve P-256 with coordinateX and coordinateY +BOOL STDSVerifyEllipticCurveP256Signature(NSData *coordinateX, NSData *coordinateY, NSData *payload, NSData *signature); + +/// Verifies the signature and payload using RSA-PSS +BOOL STDSVerifyRSASignature(SecCertificateRef certificate, NSData *payload, NSData *signature); + +/// Returns data of length numBytes generated using CommonCrypto's CCRandomGenerateBytes or nil on failure +NSData * _Nullable STDSCryptoRandomData(size_t numBytes); + +/// Creates a certificate from base64encoded data +SecCertificateRef _Nullable STDSSecCertificateFromData(NSData *data); + +/// Creates a certificate from a PEM or DER encoded certificate string +SecCertificateRef _Nullable STDSSecCertificateFromString(NSString *certificateString); + +// Creates a public key using Elliptic Curve P-256 with coordinateX and coordinateY +SecKeyRef _Nullable STDSSecKeyRefFromCoordinates(NSData *coordinateX, NSData *coordinateY); + +// Creates a private key using Elliptic Curve P-256 with x, y, and d +SecKeyRef _Nullable STDSPrivateSecKeyRefFromCoordinates(NSData *x, NSData *y, NSData *d); + + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.m b/Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.m new file mode 100644 index 00000000..2a2331fb --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSecTypeUtilities.m @@ -0,0 +1,366 @@ +// +// STDSSecTypeUtilities.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/28/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSSecTypeUtilities.h" + +#import +#import +#import + +#import "STDSBundleLocator.h" +#import "STDSEllipticCurvePoint.h" + +NS_ASSUME_NONNULL_BEGIN + +SecCertificateRef _Nullable STDSCertificateForServer(STDSDirectoryServer server) { + static NSMutableDictionary *sCertificateData = nil; + static dispatch_once_t onceToken; + dispatch_once(&onceToken, ^{ + sCertificateData = [[NSMutableDictionary alloc] init]; + }); + + NSString *serverKey = nil; + switch (server) { + case STDSDirectoryServerULTestRSA: + serverKey = @"STDSDirectoryServerULTestRSA"; + break; + + case STDSDirectoryServerULTestEC: + serverKey = @"STDSDirectoryServerULTestEC"; + break; + + case STDSDirectoryServerSTPTestRSA: + serverKey = @"STDSDirectoryServerSTPTestRSA"; + break; + + case STDSDirectoryServerSTPTestEC: + serverKey = @"STDSDirectoryServerSTPTestEC"; + break; + + case STDSDirectoryServerAmex: + serverKey = @"STDSDirectoryServerAmex"; + break; + + case STDSDirectoryServerDiscover: + serverKey = @"STDSDirectoryServerDiscover"; + break; + + case STDSDirectoryServerMastercard: + serverKey = @"STDSDirectoryServerMastercard"; + break; + + case STDSDirectoryServerVisa: + serverKey = @"STDSDirectoryServerVisa"; + break; + + case STDSDirectoryServerCartesBancaires: + serverKey = @"STDSDirectoryServerCartesBancaires"; + break; + + case STDSDirectoryServerCustom: + break; + + case STDSDirectoryServerUnknown: + break; + } + + if (serverKey == nil) { + return NULL; + } + + NSData *certificateData = sCertificateData[serverKey]; + if (certificateData == nil) { + NSString *certificatePath = nil; + switch (server) { + case STDSDirectoryServerULTestRSA: + break; + + case STDSDirectoryServerULTestEC: + break; + + case STDSDirectoryServerSTPTestRSA: + certificatePath = [[STDSBundleLocator stdsResourcesBundle] pathForResource:@"ul-test" ofType:@"der"]; + break; + + case STDSDirectoryServerSTPTestEC: + certificatePath = [[STDSBundleLocator stdsResourcesBundle] pathForResource:@"ec_test" ofType:@"der"]; + break; + + case STDSDirectoryServerAmex: + certificatePath = [[STDSBundleLocator stdsResourcesBundle] pathForResource:@"amex" ofType:@"der"]; + break; + + case STDSDirectoryServerDiscover: + certificatePath = [[STDSBundleLocator stdsResourcesBundle] pathForResource:@"discover" ofType:@"der"]; + break; + + case STDSDirectoryServerMastercard: + certificatePath = [[STDSBundleLocator stdsResourcesBundle] pathForResource:@"mastercard" ofType:@"der"]; + break; + + case STDSDirectoryServerVisa: + certificatePath = [[STDSBundleLocator stdsResourcesBundle] pathForResource:@"visa" ofType:@"der"]; + break; + + case STDSDirectoryServerCartesBancaires: + certificatePath = [[STDSBundleLocator stdsResourcesBundle] pathForResource:@"cartes_bancaires" ofType:@"der"]; + break; + + case STDSDirectoryServerCustom: + break; + + case STDSDirectoryServerUnknown: + break; + } + + if (certificatePath != nil) { + certificateData = [NSData dataWithContentsOfFile:certificatePath]; + // cache the file data to limit file IO + sCertificateData[serverKey] = certificateData; + } + } + + // Note to Future: SecCertificateCreateWithData only works with DER formatted data. The other popular + // format for certificate files is PEM. These can be converted before adding to the SDK by invoking + // `openssl x509 -in certificate_PEM.crt -outform der -out certificate_DER.der` + return certificateData != nil ? SecCertificateCreateWithData(NULL, (CFDataRef)certificateData): NULL; +}; + +CFStringRef _Nullable STDSSecCertificateCopyPublicKeyType(SecCertificateRef certificate) { + CFStringRef ret = NULL; + + SecKeyRef key = SecCertificateCopyKey(certificate); + + if (key != NULL) { + CFDictionaryRef attributes = SecKeyCopyAttributes(key); + if (attributes == NULL) { + CFRelease(key); + return NULL; + } + + if (attributes != NULL) { + const void *keyType = CFDictionaryGetValue(attributes, kSecAttrKeyType); + if (keyType != NULL && CFGetTypeID(keyType) == CFStringGetTypeID()) { + ret = CFStringCreateCopy(kCFAllocatorDefault, (CFStringRef)keyType); + } + CFRelease(attributes); + } + CFRelease(key); + } + + return ret; +} + +SecCertificateRef _Nullable STDSSecCertificateFromString(NSString *certificateString) { + static NSString * const kCertificateAnchorPrefix = @"-----BEGIN CERTIFICATE-----"; + static NSString * const kCertificateAnchorSuffix = @"-----END CERTIFICATE-----"; + + // first remove newlines + NSString *certificateStringNoAnchors = [[[certificateString stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]] componentsSeparatedByCharactersInSet:[NSCharacterSet newlineCharacterSet]] componentsJoinedByString:@""]; + + // remove the begin/end certificate markers + NSUInteger fromIndex = [certificateStringNoAnchors hasPrefix:kCertificateAnchorPrefix] ? kCertificateAnchorPrefix.length : 0; + NSUInteger toIndex = [certificateStringNoAnchors hasSuffix:kCertificateAnchorSuffix] ? certificateStringNoAnchors.length - kCertificateAnchorSuffix.length : certificateStringNoAnchors.length; + certificateStringNoAnchors = [[certificateStringNoAnchors substringWithRange:NSMakeRange(fromIndex, toIndex - fromIndex)] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]]; + + if (certificateStringNoAnchors.length == 0) { + return NULL; + } + NSData *certificateData = [[NSData alloc] initWithBase64EncodedString:certificateStringNoAnchors options:0]; + if (certificateData == nil) { + return NULL; + } + + return STDSSecCertificateFromData(certificateData); +} + +SecCertificateRef _Nullable STDSSecCertificateFromData(NSData *data) { + SecCertificateRef certificate = SecCertificateCreateWithData(NULL, (__bridge CFDataRef)data); + return certificate; +} + +SecKeyRef _Nullable STDSPrivateSecKeyRefFromCoordinates(NSData *x, NSData *y, NSData *d) { + static unsigned char prefixBytes[] = {0x04}; + NSMutableData *bytes = [[NSMutableData alloc] initWithBytes:(void *)prefixBytes length:1]; + [bytes appendData:x]; + [bytes appendData:y]; + [bytes appendData:d]; + NSDictionary *attributes = @{ + (__bridge NSString *)kSecAttrKeyType: (__bridge NSString *)kSecAttrKeyTypeECSECPrimeRandom, + (__bridge NSString *)kSecAttrKeyClass: (__bridge NSString *)kSecAttrKeyClassPrivate, + (__bridge NSString *)kSecAttrKeySizeInBits: @(256), + }; + CFErrorRef error = NULL; + SecKeyRef key = SecKeyCreateWithData((__bridge CFDataRef)bytes, (__bridge CFDictionaryRef)attributes, &error); + return key; +} + +SecKeyRef _Nullable STDSSecKeyRefFromCoordinates(NSData *coordinateX, NSData *coordinateY) { + static unsigned char prefixBytes[] = {0x04}; + NSMutableData *bytes = [[NSMutableData alloc] initWithBytes:(void *)prefixBytes length:1]; + [bytes appendData:coordinateX]; + [bytes appendData:coordinateY]; + NSDictionary *attributes = @{ + (__bridge NSString *)kSecAttrKeyType: (__bridge NSString *)kSecAttrKeyTypeECSECPrimeRandom, + (__bridge NSString *)kSecAttrKeyClass: (__bridge NSString *)kSecAttrKeyClassPublic, + (__bridge NSString *)kSecAttrKeySizeInBits: @(256), + }; + CFErrorRef error = NULL; + SecKeyRef key = SecKeyCreateWithData((__bridge CFDataRef)bytes, (__bridge CFDictionaryRef)attributes, &error); + return key; +} + +// ref. https://crypto.stackexchange.com/questions/1795/how-can-i-convert-a-der-ecdsa-signature-to-asn-1/1797#1797 +NSData * _Nullable STDSDEREncodedSignature(NSData * signature) { + // make sure input signature is of correct R || S format + NSUInteger signatureLength = signature.length; + if (signatureLength == 0 || signatureLength % 2 != 0) { + return nil; + } + + static const uint8_t bytePrefix = 0x00; + + NSMutableData *rBytes = [[NSMutableData alloc] init]; + uint8_t firstRByte; + [signature getBytes:&firstRByte length:1]; + + if (firstRByte >= 0x80) { + // "Signed big-endian encoding of minimal length", we can't have the first bit be 1 because these are postive values + [rBytes appendBytes:&bytePrefix length:1]; + } + [rBytes appendBytes:signature.bytes length:signatureLength / 2]; + + NSMutableData *sBytes = [[NSMutableData alloc] init]; + uint8_t firstSByte; + [signature getBytes:&firstSByte range:NSMakeRange(signatureLength / 2, 1)]; + + if (firstSByte >= 0x80) { + // "Signed big-endian encoding of minimal length", we can't have the first bit be 1 because these are postive values + [sBytes appendBytes:&bytePrefix length:1]; + } + [sBytes appendBytes:(signature.bytes + (signatureLength / 2)) length:signatureLength / 2]; + + uint8_t rLength = (uint8_t)rBytes.length; + uint8_t sLength = (uint8_t)sBytes.length; + + static const uint8_t derBytePrefix = 0x30; + NSMutableData *derEncoded = [[NSMutableData alloc] initWithBytes:&derBytePrefix length:1]; + + static const uint8_t derSeparatorByte = 0x02; + // numBytes does not include the 0x30 byte + uint8_t numBytes = rLength + sLength + 2 + 2; // + 2 for separators, + 2 for r and s size bytes + [derEncoded appendBytes:&numBytes length:1]; + [derEncoded appendBytes:&derSeparatorByte length:1]; + + [derEncoded appendBytes:&rLength length:1]; + [derEncoded appendBytes:rBytes.bytes length:rBytes.length]; + + [derEncoded appendBytes:&derSeparatorByte length:1]; + + [derEncoded appendBytes:&sLength length:1]; + [derEncoded appendBytes:sBytes.bytes length:sBytes.length]; + + return [derEncoded copy]; +} + +BOOL STDSVerifyEllipticCurveP256Signature(NSData *coordinateX, NSData *coordinateY, NSData *payload, NSData *signature) { + BOOL ret = NO; + + // make P-256 curve key from coordinates + SecKeyRef key = STDSSecKeyRefFromCoordinates(coordinateX, coordinateY); + + if (key != NULL) { + size_t hashBytesSize = CC_SHA256_DIGEST_LENGTH; + unsigned char hashBytes[hashBytesSize]; + CC_SHA256(payload.bytes, (CC_LONG)payload.length, hashBytes); + CFErrorRef error = NULL; + NSData *derEncodedSignature = STDSDEREncodedSignature(signature); + if (derEncodedSignature == nil) { + CFRelease(key); + return NO; + } + ret = (BOOL)SecKeyVerifySignature(key, kSecKeyAlgorithmECDSASignatureDigestX962SHA256, (__bridge CFDataRef)[NSData dataWithBytes:hashBytes length:hashBytesSize], (__bridge CFDataRef)derEncodedSignature, &error); + CFRelease(key); + } + return ret; +} + +BOOL STDSVerifyRSASignature(SecCertificateRef certificate, NSData *payload, NSData *signature) { + BOOL ret = NO; + + SecKeyRef key = SecCertificateCopyKey(certificate); + if (key != NULL && signature) { + CFErrorRef error = NULL; + size_t hashBytesSize = CC_SHA256_DIGEST_LENGTH; + unsigned char hashBytes[hashBytesSize]; + CC_SHA256(payload.bytes, (CC_LONG)payload.length, hashBytes); + + ret = (BOOL)SecKeyVerifySignature(key, kSecKeyAlgorithmRSASignatureDigestPSSSHA256, (__bridge CFDataRef)[NSData dataWithBytes:hashBytes length:hashBytesSize], (__bridge CFDataRef)signature, &error); + CFRelease(key); + } + return ret; +} + +NSData * _Nullable STDSCryptoRandomData(size_t numBytes) { + void *randomBytes[numBytes]; + memset(randomBytes, 0, numBytes); + if (CCRandomGenerateBytes(randomBytes, numBytes) == kCCSuccess) { + NSData *data = [NSData dataWithBytes:randomBytes length:numBytes]; + return data; + } + return NULL; +} + +NSData * _Nullable _STPCreateKDFFormattedData(NSData *data) { + uint32_t bigEndianLength = CFSwapInt32HostToBig((uint32_t)data.length); + NSMutableData *encodedLength = [NSMutableData dataWithBytes:&bigEndianLength length:4]; + [encodedLength appendData:data]; + return [encodedLength copy]; +} + +NSData * _Nullable STDSCreateConcatKDFWithSHA256(NSData *sharedSecret, NSUInteger keyLength, NSString *apv) { + NSData *concatKDFData = nil; + + uint32_t bigEndianKeyLength = CFSwapInt32HostToBig((uint32_t)keyLength*8); + + // algorithmID and partyUInfo are intentionally empty strings based on the Core Spec + // section 6.2.3.3 which states that they should be null. The KDF standard, NIST.800-56A, + // requires that null values still have the length bytes set to 0. + NSData *algorithmID = _STPCreateKDFFormattedData([@"" dataUsingEncoding:NSASCIIStringEncoding]); + NSData *partyUInfo = _STPCreateKDFFormattedData([@"" dataUsingEncoding:NSUTF8StringEncoding]); + NSData *partyVInfo = _STPCreateKDFFormattedData([apv dataUsingEncoding:NSUTF8StringEncoding]); + NSData *suppPubInfo = [NSData dataWithBytes:&bigEndianKeyLength length:4]; + + if (algorithmID == nil || + partyUInfo == nil || + partyVInfo == nil || + suppPubInfo == nil + ) { + return nil; + } + + NSMutableData *otherInfo = [algorithmID mutableCopy]; + [otherInfo appendData:partyUInfo]; + [otherInfo appendData:partyVInfo]; + [otherInfo appendData:suppPubInfo]; + + const unsigned char roundOneBytes[4] = {0, 0, 0, 1}; + NSMutableData *roundOneHashInput = [[NSMutableData alloc] initWithBytes:roundOneBytes length:4]; + [roundOneHashInput appendData:sharedSecret]; + [roundOneHashInput appendData:otherInfo]; + + NSMutableData *roundOneHashOutput = [NSMutableData dataWithLength:CC_SHA256_DIGEST_LENGTH]; + + if (CC_SHA256(roundOneHashInput.bytes, (CC_LONG)roundOneHashInput.length, roundOneHashOutput.mutableBytes) != nil) { + concatKDFData = [NSData dataWithBytes:roundOneHashOutput.bytes length:MAX(keyLength, roundOneHashOutput.length)]; + } + + return concatKDFData; +} + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSSelectionButton.h b/Stripe3DS2/Stripe3DS2/STDSSelectionButton.h new file mode 100644 index 00000000..5cb78b7c --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSelectionButton.h @@ -0,0 +1,26 @@ +// +// STDSSelectionButton.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 6/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class STDSSelectionCustomization; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSSelectionButton : UIButton + +@property (nonatomic) STDSSelectionCustomization *customization; + +/// This control can either be styled as a radio button or a checkbox +@property (nonatomic) BOOL isCheckbox; + +- (instancetype)initWithCustomization:(STDSSelectionCustomization *)customization; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSSelectionButton.m b/Stripe3DS2/Stripe3DS2/STDSSelectionButton.m new file mode 100644 index 00000000..97bf19e4 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSelectionButton.m @@ -0,0 +1,167 @@ +// +// STDSSelectionButton.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 6/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSSelectionButton.h" + +#import "STDSSelectionCustomization.h" + +@interface _STDSSelectionButtonView: UIView +@property (nonatomic) BOOL isCheckbox; +@property (nonatomic) STDSSelectionCustomization *customization; +@property (nonatomic, getter = isSelected) BOOL selected; +@end + +@implementation _STDSSelectionButtonView + +- (instancetype)initWithFrame:(CGRect)frame { + self = [super initWithFrame:frame]; + if (self) { + self.opaque = NO; + } + + return self; +} + +- (void)setSelected:(BOOL)selected { + _selected = selected; + [self setNeedsDisplay]; +} + + +- (void)setCustomization:(STDSSelectionCustomization *)customization { + _customization = customization; + [self setNeedsDisplay]; +} + + +- (void)drawRect:(CGRect)rect { + if (self.isCheckbox) { + [self _drawCheckboxWithRect:rect]; + } else { + [self _drawRadioButtonWithRect:rect]; + } +} + +- (void)_drawRadioButtonWithRect:(CGRect)rect { + // Draw background + UIBezierPath *background = [UIBezierPath bezierPathWithOvalInRect:rect]; + if (self.isSelected) { + [self.customization.primarySelectedColor setFill]; + } else { + [self.customization.unselectedBackgroundColor setFill]; + } + [background fill]; + + // Draw unselected border + if (!self.isSelected) { + UIBezierPath *border = [UIBezierPath bezierPathWithOvalInRect:CGRectInset(rect, 0.5, 0.5)]; + [self.customization.unselectedBorderColor setStroke]; + [border stroke]; + } + + // Draw inner circle if selected + if (self.isSelected) { + CGRect selectedRect = CGRectInset(rect, 9, 9); + UIBezierPath *selected = [UIBezierPath bezierPathWithOvalInRect:selectedRect]; + [self.customization.secondarySelectedColor setFill]; + [selected fill]; + } +} + +- (void)_drawCheckboxWithRect:(CGRect)rect { + // Draw background + UIBezierPath *background = [UIBezierPath bezierPathWithRoundedRect:rect cornerRadius:8]; + if (self.isSelected) { + [self.customization.primarySelectedColor setFill]; + } else { + [self.customization.unselectedBackgroundColor setFill]; + } + [background fill]; + + // Draw unselected border + if (!self.isSelected) { + UIBezierPath *border = [UIBezierPath bezierPathWithRoundedRect:CGRectInset(rect, 0.5, 0.5) cornerRadius:8]; + border.lineWidth = 0.5; + [self.customization.unselectedBorderColor setStroke]; + [border stroke]; + } + + // Draw check mark if selected + if (self.isSelected) { + UIBezierPath *checkmark = [UIBezierPath bezierPath]; + [checkmark moveToPoint: CGPointMake(10, 15)]; + [checkmark addLineToPoint:CGPointMake(13.5, 18.5)]; + [checkmark addLineToPoint:CGPointMake(22, 10)]; + [self.customization.secondarySelectedColor setStroke]; + checkmark.lineWidth = 2; + [checkmark stroke]; + } +} + +@end + +static const CGFloat kMinimumTouchAreaDimension = 42.f; +static const CGFloat kContentSizeDimension = 30.f; + +@implementation STDSSelectionButton { + _STDSSelectionButtonView *_contentView; +} + +- (instancetype)initWithCustomization:(STDSSelectionCustomization *)customization { + self = [super init]; + if (self) { + _contentView = [[_STDSSelectionButtonView alloc] initWithFrame:CGRectMake(0, 0, kContentSizeDimension, kContentSizeDimension)]; + _contentView.userInteractionEnabled = NO; + [self addSubview:_contentView]; + self.customization = customization; + } + return self; +} + +- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event { + BOOL pointInside = [super pointInside:point withEvent:event]; + if (!pointInside && + self.enabled && + !self.isHidden && + (CGRectGetWidth(self.bounds) < kMinimumTouchAreaDimension || CGRectGetHeight(self.bounds) < kMinimumTouchAreaDimension) + ) { + // Make sure that we intercept touch events even outside our bounds if they are within the + // minimum touch area. Otherwise this button is too hard to tap + CGRect expandedBounds = CGRectInset(self.bounds, MIN(CGRectGetWidth(self.bounds) - kMinimumTouchAreaDimension, 0), MIN(CGRectGetHeight(self.bounds) < kMinimumTouchAreaDimension, 0)); + pointInside = CGRectContainsPoint(expandedBounds, point); + } + + return pointInside; +} + +- (CGSize)intrinsicContentSize { + return CGSizeMake(kContentSizeDimension, kContentSizeDimension); +} + +- (void)setSelected:(BOOL)selected { + [super setSelected:selected]; + _contentView.selected = selected; +} + +- (void)setCustomization:(STDSSelectionCustomization *)customization { + _contentView.customization = customization; +} + +- (STDSSelectionCustomization *)customization { + return _contentView.customization; +} + +- (void)setIsCheckbox:(BOOL)isCheckbox { + _contentView.isCheckbox = isCheckbox; +} + +- (BOOL)isCheckbox { + return _contentView.isCheckbox; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.h b/Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.h new file mode 100644 index 00000000..7b6e77d7 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.h @@ -0,0 +1,19 @@ +// +// STDSSimulatorChecker.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSSimulatorChecker : NSObject + ++ (BOOL)isRunningOnSimulator; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.m b/Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.m new file mode 100644 index 00000000..47ea89e1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSimulatorChecker.m @@ -0,0 +1,25 @@ +// +// STDSSimulatorChecker.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/8/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSSimulatorChecker.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSSimulatorChecker + ++ (BOOL)isRunningOnSimulator { +#if TARGET_IPHONE_SIMULATOR + return YES; +#else + return NO; +#endif +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSSpacerView.h b/Stripe3DS2/Stripe3DS2/STDSSpacerView.h new file mode 100644 index 00000000..072ad1ed --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSpacerView.h @@ -0,0 +1,20 @@ +// +// STDSSpacerView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSStackView.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSSpacerView : UIView + +- (instancetype)initWithLayoutAxis:(STDSStackViewLayoutAxis)layoutAxis dimension:(CGFloat)dimension; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSSpacerView.m b/Stripe3DS2/Stripe3DS2/STDSSpacerView.m new file mode 100644 index 00000000..a674e68a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSpacerView.m @@ -0,0 +1,34 @@ +// +// STDSSpacerView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSSpacerView.h" + +@implementation STDSSpacerView + +- (instancetype)initWithLayoutAxis:(STDSStackViewLayoutAxis)layoutAxis dimension:(CGFloat)dimension { + self = [super initWithFrame:CGRectZero]; + + if (self) { + NSLayoutConstraint *constraint; + + switch (layoutAxis) { + case STDSStackViewLayoutAxisHorizontal: + constraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1.0 constant:dimension]; + break; + case STDSStackViewLayoutAxisVertical: + constraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1.0 constant:dimension]; + break; + } + + [NSLayoutConstraint activateConstraints:@[constraint]]; + } + + return self; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSStackView.h b/Stripe3DS2/Stripe3DS2/STDSStackView.h new file mode 100644 index 00000000..cfedb01a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSStackView.h @@ -0,0 +1,56 @@ +// +// STDSStackView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +typedef NS_ENUM(NSInteger, STDSStackViewLayoutAxis) { + + /// A horizontal layout for the stack view to use. + STDSStackViewLayoutAxisHorizontal = 0, + + /// A vertical layout for the stack view to use. + STDSStackViewLayoutAxisVertical = 1, +}; + +@interface STDSStackView: UIView + +/** + Initializes an `STDSStackView`. + + @param alignment The alignment for the stack view to use. + @return An initialized `STDSStackView`. + */ +- (instancetype)initWithAlignment:(STDSStackViewLayoutAxis)alignment; + +/** + Adds a subview to the list of arranged subviews. Views will be displayed in the order they are added. + + @param view The view to add to the stack view. + */ +- (void)addArrangedSubview:(UIView *)view; + +/** + Removes a subview from the list of arranged subviews. + + @param view The view to remove. + */ +- (void)removeArrangedSubview:(UIView *)view; + +/** + Adds a spacer that fits the layout axis of the `STDSStackView`. + + @param dimension How wide or tall the spacer should be, depending on the axis of the `STDSStackView`. + @note Spacers added through this function will not be removed or hidden automatically when they no longer fall between two views. For more precise interactions, add an `STDSSpacerView` manually through `addArrangedSubview:`. + */ +- (void)addSpacer:(CGFloat)dimension; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSStackView.m b/Stripe3DS2/Stripe3DS2/STDSStackView.m new file mode 100644 index 00000000..7ec01d1e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSStackView.m @@ -0,0 +1,164 @@ +// +// STDSStackView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSStackView.h" +#import "STDSSpacerView.h" +#import "NSLayoutConstraint+LayoutSupport.h" + +@interface STDSStackView() + +@property (nonatomic) STDSStackViewLayoutAxis layoutAxis; +@property (nonatomic, strong) NSMutableArray *arrangedSubviews; +@property (nonatomic, strong, readonly) NSArray *visibleArrangedSubviews; + +@end + +@implementation STDSStackView + +static NSString *UIViewHiddenKeyPath = @"hidden"; + +- (instancetype)initWithAlignment:(STDSStackViewLayoutAxis)layoutAxis { + self = [super initWithFrame:CGRectZero]; + + if (self) { + _layoutAxis = layoutAxis; + _arrangedSubviews = [NSMutableArray array]; + } + + return self; +} + +- (NSArray *)visibleArrangedSubviews { + return [self.arrangedSubviews filteredArrayUsingPredicate:[NSPredicate predicateWithBlock:^BOOL(UIView *object, NSDictionary *bindings) { + return !object.isHidden; + }]]; +} + +- (void)addArrangedSubview:(UIView *)view { + view.translatesAutoresizingMaskIntoConstraints = false; + + [self _deactivateExistingConstraints]; + + [self.arrangedSubviews addObject:view]; + [self addSubview:view]; + + [self _applyConstraints]; + + [view addObserver:self forKeyPath:UIViewHiddenKeyPath options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL]; +} + +- (void)removeArrangedSubview:(UIView *)view { + if (![self.arrangedSubviews containsObject:view]) { + return; + } + + [self _deactivateExistingConstraints]; + + [view removeObserver:self forKeyPath:UIViewHiddenKeyPath]; + + [self.arrangedSubviews removeObject:view]; + [view removeFromSuperview]; + + [self _applyConstraints]; +} + +- (void)addSpacer:(CGFloat)dimension { + STDSSpacerView *spacerView = [[STDSSpacerView alloc] initWithLayoutAxis:self.layoutAxis dimension:dimension]; + + [self addArrangedSubview:spacerView]; +} + +- (void)dealloc { + for (UIView *view in self.arrangedSubviews) { + [view removeObserver:self forKeyPath:UIViewHiddenKeyPath]; + } +} + +- (void)_applyConstraints { + if (self.layoutAxis == STDSStackViewLayoutAxisHorizontal) { + [self _applyHorizontalConstraints]; + } else { + [self _applyVerticalConstraints]; + } +} + +- (void)_deactivateExistingConstraints { + [NSLayoutConstraint deactivateConstraints:self.constraints]; +} + +- (void)_applyVerticalConstraints { + UIView *previousView; + + for (UIView *view in self.visibleArrangedSubviews) { + NSLayoutConstraint *leftConstraint = [NSLayoutConstraint _stds_leftConstraintWithItem:view toItem:self]; + NSLayoutConstraint *rightConstraint = [NSLayoutConstraint _stds_rightConstraintWithItem:view toItem:self]; + NSLayoutConstraint *topConstraint; + + if (previousView == nil) { + topConstraint = [NSLayoutConstraint _stds_topConstraintWithItem:view toItem:self]; + } else { + topConstraint = [NSLayoutConstraint constraintWithItem:view attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:previousView attribute:NSLayoutAttributeBottom multiplier:1 constant:0]; + } + + [NSLayoutConstraint activateConstraints:@[topConstraint, leftConstraint, rightConstraint]]; + + if (view == self.visibleArrangedSubviews.lastObject) { + NSLayoutConstraint *bottomConstraint = [NSLayoutConstraint _stds_bottomConstraintWithItem:view toItem:self]; + + [NSLayoutConstraint activateConstraints:@[bottomConstraint]]; + } + + previousView = view; + } +} + +- (void)_applyHorizontalConstraints { + UIView *previousView; + NSLayoutConstraint *previousRightConstraint; + + for (UIView *view in self.visibleArrangedSubviews) { + NSLayoutConstraint *topConstraint = [NSLayoutConstraint _stds_topConstraintWithItem:view toItem:self]; + NSLayoutConstraint *bottomConstraint = [NSLayoutConstraint _stds_bottomConstraintWithItem:view toItem:self]; + NSLayoutConstraint *rightConstraint = [NSLayoutConstraint _stds_rightConstraintWithItem:view toItem:self]; + + if (previousView == nil) { + NSLayoutConstraint *leftConstraint = [NSLayoutConstraint _stds_leftConstraintWithItem:view toItem:self]; + + [NSLayoutConstraint activateConstraints:@[topConstraint, leftConstraint, rightConstraint, bottomConstraint]]; + } else { + NSLayoutConstraint *leftConstraint = [NSLayoutConstraint constraintWithItem:view attribute:NSLayoutAttributeLeft relatedBy:NSLayoutRelationEqual toItem:previousView attribute:NSLayoutAttributeRight multiplier:1 constant:0]; + + if (previousRightConstraint != nil) { + [NSLayoutConstraint deactivateConstraints:@[previousRightConstraint]]; + } + + NSLayoutConstraint *previousConstraint = [NSLayoutConstraint constraintWithItem:previousView attribute:NSLayoutAttributeRight relatedBy:NSLayoutRelationEqual toItem:view attribute:NSLayoutAttributeLeft multiplier:1 constant:0]; + + [NSLayoutConstraint activateConstraints:@[topConstraint, leftConstraint, rightConstraint, previousConstraint, bottomConstraint]]; + } + + previousView = view; + previousRightConstraint = rightConstraint; + } +} + +- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { + if ([object isKindOfClass:[UIView class]] && [keyPath isEqualToString:UIViewHiddenKeyPath]) { + BOOL hiddenStatusChanged = [change[NSKeyValueChangeNewKey] boolValue] != [change[NSKeyValueChangeOldKey] boolValue]; + + if (hiddenStatusChanged) { + [self _deactivateExistingConstraints]; + + [self _applyConstraints]; + } + } else { + [super observeValueForKeyPath:keyPath ofObject:object change:change context:context]; + } +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.h b/Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.h new file mode 100644 index 00000000..883c6a35 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.h @@ -0,0 +1,26 @@ +// +// STDSSynchronousLocationManager.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class CLLocation; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSSynchronousLocationManager : NSObject + ++ (instancetype)sharedManager; + ++ (BOOL)hasPermissions; + +// May be long running. Will return nil on failure or if app lacks permissions +- (nullable CLLocation *)deviceLocation; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.m b/Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.m new file mode 100644 index 00000000..c492c6fa --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSSynchronousLocationManager.m @@ -0,0 +1,110 @@ +// +// STDSSynchronousLocationManager.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/23/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSSynchronousLocationManager.h" + +#import + +NS_ASSUME_NONNULL_BEGIN + +static const int64_t kLocationFetchTimeoutSeconds = 15; + +typedef void (^LocationUpdateCompletionBlock)(CLLocation * _Nullable); + +@interface STDSSynchronousLocationManager () + +@end + +@implementation STDSSynchronousLocationManager +{ + CLLocationManager * _Nullable _locationManager; + dispatch_queue_t _Nullable _locationFetchQueue; + NSMutableArray *_pendingLocationUpdateCompletions; +} + ++ (instancetype)sharedManager { + static STDSSynchronousLocationManager *sharedManager = nil; + static dispatch_once_t onceToken; + dispatch_once(&onceToken, ^{ + sharedManager = [[STDSSynchronousLocationManager alloc] init]; + }); + return sharedManager; +} + ++ (BOOL)hasPermissions { + CLAuthorizationStatus authorizationStatus = [CLLocationManager authorizationStatus]; + return [CLLocationManager locationServicesEnabled] && + (authorizationStatus == kCLAuthorizationStatusAuthorizedAlways || authorizationStatus == kCLAuthorizationStatusAuthorizedWhenInUse); +} + +- (instancetype)init { + self = [super init]; + if (self) { + if ([STDSSynchronousLocationManager hasPermissions]) { + _locationManager = [[CLLocationManager alloc] init]; + _locationManager.delegate = self; + _locationFetchQueue = dispatch_queue_create("com.stripe.3ds2locationqueue", DISPATCH_QUEUE_SERIAL); + } + _pendingLocationUpdateCompletions = [NSMutableArray array]; + } + + return self; +} + +- (nullable CLLocation *)deviceLocation { + + __block CLLocation *location = nil; + dispatch_group_t group = dispatch_group_create(); + dispatch_group_enter(group); + [self _fetchDeviceLocation:^(CLLocation * _Nullable latestLocation) { + location = latestLocation; + dispatch_group_leave(group); + }]; + + dispatch_group_wait(group, dispatch_time(DISPATCH_TIME_NOW, NSEC_PER_SEC * kLocationFetchTimeoutSeconds)); + return location; +} + +- (void)_fetchDeviceLocation:(void (^)(CLLocation * _Nullable))completion { + + if (![STDSSynchronousLocationManager hasPermissions] || _locationFetchQueue == nil) { + return completion(nil); + } + + dispatch_async(_locationFetchQueue, ^{ + [self->_pendingLocationUpdateCompletions addObject:completion]; + + if (self->_pendingLocationUpdateCompletions.count == 1) { + [self->_locationManager requestLocation]; + } + }); +} + +- (void)_stopUpdatingLocationAndReportResult:(nullable CLLocation *)location { + [_locationManager stopUpdatingLocation]; + + dispatch_async(_locationFetchQueue, ^{ + for (LocationUpdateCompletionBlock completion in self->_pendingLocationUpdateCompletions) { + completion(location); + } + [self->_pendingLocationUpdateCompletions removeAllObjects]; + }); +} + +#pragma mark - CLLocationManagerDelegate +- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations { + [self _stopUpdatingLocationAndReportResult:locations.firstObject]; +} + +- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error { + [self _stopUpdatingLocationAndReportResult:nil]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSTextChallengeView.h b/Stripe3DS2/Stripe3DS2/STDSTextChallengeView.h new file mode 100644 index 00000000..d44ea54a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSTextChallengeView.h @@ -0,0 +1,26 @@ +// +// STDSTextChallengeView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/5/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSTextFieldCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSTextField: UITextField + +@end + +@interface STDSTextChallengeView : UIView + +@property (nonatomic, strong, nullable) STDSTextFieldCustomization *textFieldCustomization; +@property (nonatomic, copy, readonly, nullable) NSString *inputText; +@property (nonatomic, strong) STDSTextField *textField; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSTextChallengeView.m b/Stripe3DS2/Stripe3DS2/STDSTextChallengeView.m new file mode 100644 index 00000000..f67fc5a8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSTextChallengeView.m @@ -0,0 +1,133 @@ +// +// STDSTextChallengeView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/5/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSTextChallengeView.h" +#import "STDSStackView.h" +#import "UIView+LayoutSupport.h" +#import "NSString+EmptyChecking.h" +#import "UIColor+ThirteenSupport.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSTextField + +static const CGFloat kTextFieldMargin = (CGFloat)8.0; + +- (CGRect)textRectForBounds:(CGRect)bounds { + return CGRectInset(bounds, kTextFieldMargin, 0); +} + +- (CGRect)editingRectForBounds:(CGRect)bounds { + return CGRectInset(bounds, kTextFieldMargin, 0); +} + +- (nullable NSString *)accessibilityIdentifier { + return @"STDSTextField"; +} + +@end + +@interface STDSTextChallengeView() + +@property (nonatomic, strong) STDSStackView *containerView; +@property (nonatomic, strong) NSLayoutConstraint *borderViewHeightConstraint; + +@end + +@implementation STDSTextChallengeView + +static const CGFloat kBorderViewHeight = 1; +static const CGFloat kTextFieldKernSpacing = 3; +static const CGFloat kTextFieldPlaceholderKernSpacing = 14; +static const CGFloat kTextChallengeViewBottomPadding = 11; + +- (instancetype)initWithFrame:(CGRect)frame { + self = [super initWithFrame:frame]; + + if (self) { + [self _setupViewHierarchy]; + } + + return self; +} + +- (void)_setupViewHierarchy { + self.layoutMargins = UIEdgeInsetsMake(0, 0, kTextChallengeViewBottomPadding, 0); + + self.containerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + + self.textField = [[STDSTextField alloc] init]; + self.textField.autocorrectionType = UITextAutocorrectionTypeNo; + self.textField.autocapitalizationType = UITextAutocapitalizationTypeNone; + self.textField.delegate = self; + self.textField.clearButtonMode = UITextFieldViewModeWhileEditing; + if (@available(iOS 12, *)) { + self.textField.textContentType = UITextContentTypeOneTimeCode; + } else { + // no-op + } + [self.textField.defaultTextAttributes setValue:@(kTextFieldKernSpacing) forKey:NSKernAttributeName]; + + UIView *borderView = [UIView new]; + if (@available(iOS 12.0, *)) { + borderView.backgroundColor = [UIColor _stds_colorWithDynamicProvider:^UIColor * _Nonnull(UITraitCollection * _Nonnull traitCollection) { + return [[UIColor _stds_systemGray2Color] colorWithAlphaComponent:(CGFloat)0.6]; + }]; + } else { + borderView.backgroundColor = [[UIColor lightGrayColor] colorWithAlphaComponent:(CGFloat)0.6]; + } + + [self.containerView addArrangedSubview:self.textField]; + [self.containerView addArrangedSubview:borderView]; + [self addSubview:self.containerView]; + [self.containerView _stds_pinToSuperviewBounds]; + + self.borderViewHeightConstraint = [NSLayoutConstraint constraintWithItem:borderView attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1.0 constant:kBorderViewHeight]; + [NSLayoutConstraint activateConstraints:@[self.borderViewHeightConstraint]]; +} + +- (void)setTextFieldCustomization:(STDSTextFieldCustomization * _Nullable)textFieldCustomization { + _textFieldCustomization = textFieldCustomization; + + self.textField.font = textFieldCustomization.font; + self.textField.textColor = textFieldCustomization.textColor; + self.textField.layer.borderColor = textFieldCustomization.borderColor.CGColor; + self.textField.layer.borderWidth = textFieldCustomization.borderWidth; + self.textField.layer.cornerRadius = textFieldCustomization.cornerRadius; + self.textField.keyboardAppearance = textFieldCustomization.keyboardAppearance; + NSDictionary *placeholderTextAttributes = @{ + NSKernAttributeName: @(kTextFieldPlaceholderKernSpacing), + NSBaselineOffsetAttributeName: @(3.0f), + NSForegroundColorAttributeName: textFieldCustomization.placeholderTextColor, + }; + self.textField.attributedPlaceholder = [[NSAttributedString alloc] initWithString:@"••••••" attributes:placeholderTextAttributes]; +} + +- (NSString * _Nullable)inputText { + return self.textField.text; +} + +- (void)didMoveToWindow { + [super didMoveToWindow]; + + if (self.window.screen.nativeScale > 0) { + self.borderViewHeightConstraint.constant = kBorderViewHeight / self.window.screen.nativeScale; + } +} + +#pragma mark - UITextFieldDelegate + +- (BOOL)textFieldShouldReturn:(UITextField *)textField { + [textField resignFirstResponder]; + + return NO; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion+Private.h b/Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion+Private.h new file mode 100644 index 00000000..43257e78 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion+Private.h @@ -0,0 +1,53 @@ +// +// STDSThreeDSProtocolVersion+Private.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSThreeDSProtocolVersion.h" + +NS_ASSUME_NONNULL_BEGIN + + +typedef NS_ENUM(NSInteger, STDSThreeDSProtocolVersion) { + STDSThreeDSProtocolVersion2_1_0, + STDSThreeDSProtocolVersion2_2_0, + STDSThreeDSProtocolVersionUnknown, + STDSThreeDSProtocolVersionFallbackTest, +}; + +static NSString * const kThreeDS2ProtocolVersion2_1_0 = @"2.1.0"; +static NSString * const kThreeDS2ProtocolVersion2_2_0 = @"2.2.0"; +static NSString * const kThreeDSProtocolVersionFallbackTest = @"2.0.0"; + +NS_INLINE STDSThreeDSProtocolVersion STDSThreeDSProtocolVersionForString(NSString *stringValue) { + if ([stringValue isEqualToString:kThreeDS2ProtocolVersion2_1_0]) { + return STDSThreeDSProtocolVersion2_1_0; + } else if ([stringValue isEqualToString:kThreeDS2ProtocolVersion2_2_0]) { + return STDSThreeDSProtocolVersion2_2_0; + } else if ([stringValue isEqualToString:kThreeDSProtocolVersionFallbackTest]) { + return STDSThreeDSProtocolVersionFallbackTest; + } else { + return STDSThreeDSProtocolVersionUnknown; + } +} + +NS_INLINE NSString * _Nullable STDSThreeDSProtocolVersionStringValue(STDSThreeDSProtocolVersion protocolVersion) { + switch (protocolVersion) { + case STDSThreeDSProtocolVersion2_1_0: + return kThreeDS2ProtocolVersion2_1_0; + + case STDSThreeDSProtocolVersion2_2_0: + return kThreeDS2ProtocolVersion2_2_0; + + case STDSThreeDSProtocolVersionFallbackTest: + return kThreeDSProtocolVersionFallbackTest; + + case STDSThreeDSProtocolVersionUnknown: + return nil; + } +} + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion.m b/Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion.m new file mode 100644 index 00000000..3d275b14 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSThreeDSProtocolVersion.m @@ -0,0 +1,15 @@ +// +// STDSThreeDSProtocolVersion.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 6/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSThreeDSProtocolVersion+Private.h" + +NS_ASSUME_NONNULL_BEGIN + +NSString * const Stripe3DS2ProtocolVersion = @"2.1.0"; + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSTransaction+Private.h b/Stripe3DS2/Stripe3DS2/STDSTransaction+Private.h new file mode 100644 index 00000000..2c35e956 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSTransaction+Private.h @@ -0,0 +1,41 @@ +// +// STDSTransaction+Private.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSTransaction.h" + +@class STDSDeviceInformation; +@class STDSDirectoryServerCertificate; + +#import "STDSDirectoryServer.h" +#import "STDSThreeDSProtocolVersion+Private.h" +#import "STDSUICustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSTransaction () + +- (instancetype)initWithDeviceInformation:(STDSDeviceInformation *)deviceInformation + directoryServer:(STDSDirectoryServer)directoryServer + protocolVersion:(STDSThreeDSProtocolVersion)protocolVersion + uiCustomization:(STDSUICustomization *)uiCustomization; + +- (instancetype)initWithDeviceInformation:(STDSDeviceInformation *)deviceInformation + directoryServerID:(NSString *)directoryServerID + serverKeyID:(nullable NSString *)serverKeyID + directoryServerCertificate:(STDSDirectoryServerCertificate *)directoryServerCertificate + rootCertificateStrings:(NSArray *)rootCertificateStrings + protocolVersion:(STDSThreeDSProtocolVersion)protocolVersion + uiCustomization:(STDSUICustomization *)uiCustomization; + +@property (nonatomic, strong) NSTimer *timeoutTimer; +@property (nonatomic) BOOL bypassTestModeVerification; // Should be used during internal testing ONLY +@property (nonatomic) BOOL useULTestLOA; // Should only be used when running tests with the UL reference app + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSWebView.h b/Stripe3DS2/Stripe3DS2/STDSWebView.h new file mode 100644 index 00000000..7e6673fc --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSWebView.h @@ -0,0 +1,22 @@ +// +// STDSWebView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSWebView : WKWebView + +/** + Convenience method that prepends the given HTML string with a CSP meta tag that disables external resource loading, and passes it to `loadHTMLString:baseURL:`. + */ +- (WKNavigation *)loadExternalResourceBlockingHTMLString:(NSString *)html; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSWebView.m b/Stripe3DS2/Stripe3DS2/STDSWebView.m new file mode 100644 index 00000000..d30da6bc --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSWebView.m @@ -0,0 +1,37 @@ +// +// STDSWebView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSWebView.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSWebView + +- (instancetype)init { + WKWebViewConfiguration *configuration = [[WKWebViewConfiguration alloc] init]; + configuration.preferences.javaScriptEnabled = NO; + return [super initWithFrame:CGRectZero configuration:configuration]; +} + +/// Overriden to do nothing per 3DS2 security guidelines. +- (void)evaluateJavaScript:(NSString *)javaScriptString completionHandler:(void (^ _Nullable)(_Nullable id, NSError * _Nullable error))completionHandler { + +} + +- (WKNavigation *)loadExternalResourceBlockingHTMLString:(NSString *)html { + NSString *cspMetaTag = @""; + return [self loadHTMLString:[cspMetaTag stringByAppendingString:html] baseURL:nil]; +} + +- (nullable NSString *)accessibilityIdentifier { + return @"STDSWebView"; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSWhitelistView.h b/Stripe3DS2/Stripe3DS2/STDSWhitelistView.h new file mode 100644 index 00000000..a744f2b8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSWhitelistView.h @@ -0,0 +1,25 @@ +// +// STDSWhitelistView.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponseSelectionInfo.h" +#import "STDSLabelCustomization.h" +#import "STDSSelectionCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSWhitelistView : UIView + +@property (nonatomic, strong, nullable) NSString *whitelistText; +@property (nonatomic, readonly, nullable) id selectedResponse; +@property (nonatomic, strong, nullable) STDSLabelCustomization *labelCustomization; +@property (nonatomic, strong, nullable) STDSSelectionCustomization *selectionCustomization; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/STDSWhitelistView.m b/Stripe3DS2/Stripe3DS2/STDSWhitelistView.m new file mode 100644 index 00000000..a5181fa0 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/STDSWhitelistView.m @@ -0,0 +1,103 @@ +// +// STDSWhitelistView.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSLocalizedString.h" +#import "STDSWhitelistView.h" +#import "STDSStackView.h" +#import "STDSChallengeResponseSelectionInfoObject.h" +#import "NSString+EmptyChecking.h" +#import "UIView+LayoutSupport.h" +#import "STDSSelectionButton.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSWhitelistView() + +@property (nonatomic, strong) UILabel *whitelistLabel; +@property (nonatomic, strong) STDSSelectionButton *selectionButton; + +@end + +@implementation STDSWhitelistView + +- (instancetype)initWithFrame:(CGRect)frame { + self = [super initWithFrame:frame]; + + if (self) { + [self _setupViewHierarchy]; + } + + return self; +} + +- (void)_setupViewHierarchy { + self.layoutMargins = UIEdgeInsetsZero; + + STDSStackView *containerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + [self addSubview:containerView]; + [containerView _stds_pinToSuperviewBounds]; + + self.whitelistLabel = [[UILabel alloc] init]; + self.whitelistLabel.numberOfLines = 0; + + self.selectionButton = [[STDSSelectionButton alloc] initWithCustomization:self.selectionCustomization]; + self.selectionButton.isCheckbox = YES; + [self.selectionButton addTarget:self action:@selector(_selectionButtonWasTapped) forControlEvents:UIControlEventTouchUpInside]; + + UIStackView *stackView = [self _buildStackView]; + [stackView addArrangedSubview:self.selectionButton]; + [stackView addArrangedSubview:self.whitelistLabel]; + + [containerView addArrangedSubview:stackView]; +} + +- (void)setWhitelistText:(NSString * _Nullable)whitelistText { + _whitelistText = whitelistText; + + self.whitelistLabel.text = whitelistText; + self.whitelistLabel.hidden = [NSString _stds_isStringEmpty:whitelistText]; + self.selectionButton.hidden = self.whitelistLabel.hidden; +} + +- (id _Nullable)selectedResponse { + if (self.selectionButton.selected) { + return [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"Y" value:STDSLocalizedString(@"Yes", @"The yes answer to a yes or no question.")];; + } + + return [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"N" value:STDSLocalizedString(@"No", @"The no answer to a yes or no question.")]; +} + +- (void)setLabelCustomization:(STDSLabelCustomization * _Nullable)labelCustomization { + _labelCustomization = labelCustomization; + + self.whitelistLabel.font = labelCustomization.font; + self.whitelistLabel.textColor = labelCustomization.textColor; +} + +- (void)setSelectionCustomization:(STDSSelectionCustomization * _Nullable)selectionCustomization { + _selectionCustomization = selectionCustomization; + self.selectionButton.customization = selectionCustomization; +} + +- (UIStackView *)_buildStackView { + UIStackView *stackView = [[UIStackView alloc] init]; + stackView.axis = UILayoutConstraintAxisHorizontal; + stackView.distribution = UIStackViewDistributionFillProportionally; + stackView.alignment = UIStackViewAlignmentCenter; + stackView.spacing = 20; + stackView.translatesAutoresizingMaskIntoConstraints = NO; + return stackView; +} + +- (void)_selectionButtonWasTapped { + self.selectionButton.selected = !self.selectionButton.selected; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/Stripe3DS2-Bridging-Header.h b/Stripe3DS2/Stripe3DS2/Stripe3DS2-Bridging-Header.h new file mode 100644 index 00000000..cfc1861e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/Stripe3DS2-Bridging-Header.h @@ -0,0 +1,12 @@ +// +// Stripe3DS2-Bridging-Header.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 4/10/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#ifndef Stripe3DS2_Bridging_Header_h +#define Stripe3DS2_Bridging_Header_h + +#endif /* Stripe3DS2_Bridging_Header_h */ diff --git a/Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.h b/Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.h new file mode 100644 index 00000000..badb4d6d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.h @@ -0,0 +1,21 @@ +// +// UIButton+CustomInitialization.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSUICustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface UIButton (CustomInitialization) + ++ (UIButton *)_stds_buttonWithTitle:(NSString * _Nullable)title customization:(STDSButtonCustomization * _Nullable)customization; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.m b/Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.m new file mode 100644 index 00000000..cd289e81 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIButton+CustomInitialization.m @@ -0,0 +1,64 @@ +// +// UIButton+CustomInitialization.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "UIButton+CustomInitialization.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation UIButton (CustomInitialization) + +static const CGFloat kDefaultButtonContentInset = (CGFloat)12.0; + ++ (UIButton *)_stds_buttonWithTitle:(NSString * _Nullable)title customization:(STDSButtonCustomization * _Nullable)customization { + UIButton *button = [UIButton buttonWithType:UIButtonTypeSystem]; + button.clipsToBounds = YES; + button.contentEdgeInsets = UIEdgeInsetsMake(kDefaultButtonContentInset, 0, kDefaultButtonContentInset, 0); + [[self class] _stds_configureButton:button withTitle:title customization:customization]; + + return button; +} + ++ (void)_stds_configureButton:(UIButton *)button withTitle:(NSString * _Nullable)buttonTitle customization:(STDSButtonCustomization * _Nullable)buttonCustomization { + button.backgroundColor = buttonCustomization.backgroundColor; + button.layer.cornerRadius = buttonCustomization.cornerRadius; + + UIFont *font = buttonCustomization.font; + UIColor *textColor = buttonCustomization.textColor; + + if (buttonTitle != nil) { + NSMutableDictionary *attributesDictionary = [NSMutableDictionary dictionary]; + + if (font != nil) { + attributesDictionary[NSFontAttributeName] = font; + } + + if (textColor != nil) { + attributesDictionary[NSForegroundColorAttributeName] = textColor; + } + switch (buttonCustomization.titleStyle) { + case STDSButtonTitleStyleDefault: + break; + case STDSButtonTitleStyleSentenceCapitalized: + buttonTitle = [buttonTitle localizedCapitalizedString]; + break; + case STDSButtonTitleStyleLowercase: + buttonTitle = [buttonTitle localizedLowercaseString]; + break; + case STDSButtonTitleStyleUppercase: + buttonTitle = [buttonTitle localizedUppercaseString]; + break; + } + + NSAttributedString *title = [[NSAttributedString alloc] initWithString:buttonTitle attributes:attributesDictionary]; + [button setAttributedTitle:title forState:UIControlStateNormal]; + } +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.h b/Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.h new file mode 100644 index 00000000..3c45eb9c --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.h @@ -0,0 +1,21 @@ +// +// UIColor+DefaultColors.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface UIColor (DefaultColors) + +/// The challenge view footer background color ++ (UIColor *)_stds_defaultFooterBackgroundColor; ++ (UIColor *)_stds_blueColor; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.m b/Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.m new file mode 100644 index 00000000..0cd88c6e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIColor+DefaultColors.m @@ -0,0 +1,31 @@ +// +// UIColor+DefaultColors.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "UIColor+DefaultColors.h" +#import "UIColor+ThirteenSupport.h" + +@implementation UIColor (DefaultColors) + ++ (UIColor *)_stds_defaultFooterBackgroundColor { + return [UIColor _stds_systemGray5Color]; +} + ++ (UIColor *)_stds_blueColor { + if (@available(iOS 12.0, *)) { + return [UIColor _stds_colorWithDynamicProvider:^UIColor * _Nonnull(UITraitCollection * _Nonnull traitCollection) { + return (traitCollection.userInterfaceStyle == UIUserInterfaceStyleLight) ? [UIColor colorWithRed:(CGFloat)(29.0 / 255.0) green:(CGFloat)(115.0 / 255.0) blue:(CGFloat)(250.0 / 255.0) alpha:1.0] : [UIColor colorWithRed:(CGFloat)(39.0 / 255.0) green:(CGFloat)(125.0 / 255.0) blue:(CGFloat)(255.0 / 255.0) alpha:1.0]; + }]; + } else { + CGFloat redValue = (CGFloat)29.0 / (CGFloat)255.0; + CGFloat greenValue = (CGFloat)115.0 / (CGFloat)255.0; + CGFloat blueValue = (CGFloat)250.0 / (CGFloat)255.0; + return [UIColor colorWithRed:redValue green:greenValue blue:blueValue alpha:1.0]; + } +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.h b/Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.h new file mode 100644 index 00000000..fdccd761 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.h @@ -0,0 +1,24 @@ +// +// UIColor+ThirteenSupport.h +// Stripe3DS2 +// +// Created by David Estes on 8/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface UIColor (STDSThirteenSupport) + ++ (UIColor *)_stds_colorWithDynamicProvider:(UIColor * _Nonnull (^)(UITraitCollection *traitCollection))dynamicProvider; ++ (UIColor *)_stds_systemGray5Color; ++ (UIColor *)_stds_systemGray2Color; ++ (UIColor *)_stds_systemBackgroundColor; ++ (UIColor *)_stds_labelColor; + + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.m b/Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.m new file mode 100644 index 00000000..a48203ec --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIColor+ThirteenSupport.m @@ -0,0 +1,53 @@ +// +// UIColor+ThirteenSupport.m +// Stripe3DS2 +// +// Created by David Estes on 8/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "UIColor+ThirteenSupport.h" + +@implementation UIColor (STDSThirteenSupport) + ++ (UIColor *)_stds_colorWithDynamicProvider:(UIColor * _Nonnull (^)(UITraitCollection *traitCollection))dynamicProvider { + if (@available(iOS 13.0, *)) { + return [UIColor colorWithDynamicProvider:dynamicProvider]; + } else { + return dynamicProvider([[UITraitCollection alloc] init]); + } +} + ++ (UIColor *)_stds_systemGray5Color { + if (@available(iOS 13.0, *)) { + return [UIColor systemGray5Color]; + } else { + return [UIColor colorWithRed:(CGFloat)229.0/(CGFloat)255.0 green:(CGFloat)229.0/(CGFloat)255.0 blue:(CGFloat)234.0/(CGFloat)255.0 alpha:1.0]; + } +} + ++ (UIColor *)_stds_systemGray2Color { + if (@available(iOS 13.0, *)) { + return [UIColor systemGray2Color]; + } else { + return [UIColor colorWithRed:(CGFloat)174.0/(CGFloat)255.0 green:(CGFloat)174.0/(CGFloat)255.0 blue:(CGFloat)178.0/(CGFloat)255.0 alpha:1.0]; + } +} + ++ (UIColor *)_stds_systemBackgroundColor { + if (@available(iOS 13.0, *)) { + return [UIColor systemBackgroundColor]; + } else { + return [UIColor whiteColor]; + } +} + ++ (UIColor *)_stds_labelColor { + if (@available(iOS 13.0, *)) { + return [UIColor labelColor]; + } else { + return [UIColor blackColor]; + } +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.h b/Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.h new file mode 100644 index 00000000..ceb5bfa9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.h @@ -0,0 +1,23 @@ +// +// UIFont+DefaultFonts.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface UIFont (DefaultFonts) + ++ (UIFont *)_stds_defaultHeadingTextFont; + ++ (UIFont *)_stds_defaultLabelTextFontWithScale:(CGFloat)scale; ++ (UIFont *)_stds_defaultButtonTextFontWithScale:(CGFloat)scale; ++ (UIFont *)_stds_defaultBoldLabelTextFontWithScale:(CGFloat)scale; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.m b/Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.m new file mode 100644 index 00000000..26e10ef7 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIFont+DefaultFonts.m @@ -0,0 +1,41 @@ +// +// UIFont+DefaultFonts.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "UIFont+DefaultFonts.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation UIFont (DefaultFonts) + ++ (UIFont *)_stds_defaultHeadingTextFont { + UIFontDescriptor *fontDescriptor = [[UIFontDescriptor preferredFontDescriptorWithTextStyle:UIFontTextStyleHeadline] fontDescriptorWithSymbolicTraits:UIFontDescriptorTraitBold]; + + return [UIFont fontWithDescriptor:fontDescriptor size:fontDescriptor.pointSize * (CGFloat)1.1]; +} + ++ (UIFont *)_stds_defaultLabelTextFontWithScale:(CGFloat)scale { + UIFontDescriptor *fontDescriptor = [UIFontDescriptor preferredFontDescriptorWithTextStyle:UIFontTextStyleBody]; + + return [UIFont fontWithDescriptor:fontDescriptor size:fontDescriptor.pointSize * scale]; +} + ++ (UIFont *)_stds_defaultBoldLabelTextFontWithScale:(CGFloat)scale { + UIFontDescriptor *fontDescriptor = [[UIFontDescriptor preferredFontDescriptorWithTextStyle:UIFontTextStyleBody] fontDescriptorWithSymbolicTraits:UIFontDescriptorTraitBold]; + + return [UIFont fontWithDescriptor:fontDescriptor size:fontDescriptor.pointSize * scale]; +} + ++ (UIFont *)_stds_defaultButtonTextFontWithScale:(CGFloat)scale { + UIFontDescriptor *fontDescriptor = [UIFontDescriptor preferredFontDescriptorWithTextStyle:UIFontTextStyleHeadline]; + + return [UIFont fontWithDescriptor:fontDescriptor size:fontDescriptor.pointSize * scale]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.h b/Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.h new file mode 100644 index 00000000..2c849dee --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.h @@ -0,0 +1,24 @@ +// +// UIView+LayoutSupport.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface UIView (LayoutSupport) + +/** + Pins the view to its superview's bounds. + */ +- (void)_stds_pinToSuperviewBounds; + +- (void)_stds_pinToSuperviewBoundsWithoutMargin; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.m b/Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.m new file mode 100644 index 00000000..374088b0 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIView+LayoutSupport.m @@ -0,0 +1,35 @@ +// +// UIView+LayoutSupport.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "UIView+LayoutSupport.h" + +@implementation UIView (LayoutSupport) + +- (void)_stds_pinToSuperviewBounds { + self.translatesAutoresizingMaskIntoConstraints = false; + + NSLayoutConstraint *topConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeTopMargin multiplier:1 constant:0]; + NSLayoutConstraint *bottomConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeBottomMargin multiplier:1 constant:0]; + NSLayoutConstraint *leftConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeLeft relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeLeftMargin multiplier:1 constant:0]; + NSLayoutConstraint *rightConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeRight relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeRightMargin multiplier:1 constant:0]; + + [NSLayoutConstraint activateConstraints:@[topConstraint, bottomConstraint, leftConstraint, rightConstraint]]; +} + +- (void)_stds_pinToSuperviewBoundsWithoutMargin { + self.translatesAutoresizingMaskIntoConstraints = false; + + NSLayoutConstraint *topConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeTop multiplier:1 constant:0]; + NSLayoutConstraint *bottomConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeBottom multiplier:1 constant:0]; + NSLayoutConstraint *leftConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeLeft relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeLeft multiplier:1 constant:0]; + NSLayoutConstraint *rightConstraint = [NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeRight relatedBy:NSLayoutRelationEqual toItem:self.superview attribute:NSLayoutAttributeRight multiplier:1 constant:0]; + + [NSLayoutConstraint activateConstraints:@[topConstraint, bottomConstraint, leftConstraint, rightConstraint]]; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.h b/Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.h new file mode 100644 index 00000000..130a490f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.h @@ -0,0 +1,21 @@ +// +// UIViewController+Stripe3DS2.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 5/6/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@class STDSUICustomization; + +@interface UIViewController (Stripe3DS2) + +- (void)_stds_setupNavigationBarElementsWithCustomization:(STDSUICustomization *)customization cancelButtonSelector:(SEL)cancelButtonSelector; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.m b/Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.m new file mode 100644 index 00000000..e681ba95 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/UIViewController+Stripe3DS2.m @@ -0,0 +1,49 @@ +// +// UIViewController+Stripe3DS2.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 5/6/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "UIViewController+Stripe3DS2.h" + +#import "UIButton+CustomInitialization.h" +#import "STDSUICustomization.h" + +@implementation UIViewController (Stripe3DS2) + +- (void)_stds_setupNavigationBarElementsWithCustomization:(STDSUICustomization *)customization cancelButtonSelector:(SEL)cancelButtonSelector { + STDSNavigationBarCustomization *navigationBarCustomization = customization.navigationBarCustomization; + + self.navigationController.navigationBar.barStyle = customization.navigationBarCustomization.barStyle; + + // Cancel button + STDSButtonCustomization *cancelButtonCustomization = [customization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeCancel]; + UIButton *cancelButton = [UIButton _stds_buttonWithTitle:navigationBarCustomization.buttonText customization:cancelButtonCustomization]; + // The cancel button's frame has a size of 0 in iOS 8 + cancelButton.frame = CGRectMake(0, 0, cancelButton.intrinsicContentSize.width, cancelButton.intrinsicContentSize.height); + cancelButton.accessibilityIdentifier = @"Cancel"; + [cancelButton addTarget:self action:cancelButtonSelector forControlEvents:UIControlEventTouchUpInside]; + UIBarButtonItem *cancelBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:cancelButton]; + self.navigationItem.rightBarButtonItem = cancelBarButtonItem; + + // Title + self.title = navigationBarCustomization.headerText; + NSMutableDictionary *titleTextAttributes = [NSMutableDictionary dictionary]; + UIFont *headerFont = navigationBarCustomization.font; + if (headerFont) { + titleTextAttributes[NSFontAttributeName] = headerFont; + } + UIColor *headerColor = navigationBarCustomization.textColor; + if (headerColor) { + titleTextAttributes[NSForegroundColorAttributeName] = headerColor; + } + self.navigationController.navigationBar.titleTextAttributes = titleTextAttributes; + + // Color + self.navigationController.navigationBar.barTintColor = navigationBarCustomization.barTintColor; + self.navigationController.navigationBar.translucent = navigationBarCustomization.translucent; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.h b/Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.h new file mode 100644 index 00000000..c39a85d8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.h @@ -0,0 +1,22 @@ +// +// STDSAlreadyInitializedException.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSException.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSAlreadyInitializedException` represents an exception that will be thrown in the `STDSThreeDS2Service` instance has already been initialized. + + @see STDSThreeDS2Service + */ +@interface STDSAlreadyInitializedException : STDSException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.m b/Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.m new file mode 100644 index 00000000..af161983 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSAlreadyInitializedException.m @@ -0,0 +1,17 @@ +// +// STDSAlreadyInitializedException.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSAlreadyInitializedException.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSAlreadyInitializedException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.h b/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.h new file mode 100644 index 00000000..766e4857 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.h @@ -0,0 +1,68 @@ +// +// STDSAuthenticationRequestParameters.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSJSONEncodable.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSAuthenticationRequestParameters : NSObject + +/** + Designated initializer for `STDSAuthenticationRequestParameters`. + + @param sdkTransactionIdentifier The SDK Transaction Identifier, as created by `[STDSTransaction createTransaction]` + @param deviceData Optional device data collected by the SDK. + @param sdkEphemeralPublicKey The SDK ephemeral public key. + @param sdkAppIdentifier The SDK app identifier. + @param sdkReferenceNumber The SDK reference number. + @param messageVersion The protocol version that is supported by the SDK and used for the transaction. + + @exception InvalidInputException Thrown if an input parameter is invalid. @see InvalidInputException + */ +- (instancetype)initWithSDKTransactionIdentifier:(NSString *)sdkTransactionIdentifier + deviceData:(nullable NSString *)deviceData + sdkEphemeralPublicKey:(NSString *)sdkEphemeralPublicKey + sdkAppIdentifier:(NSString *)sdkAppIdentifier + sdkReferenceNumber:(NSString *)sdkReferenceNumber + messageVersion:(NSString *)messageVersion; + +/** + The encrypted device data as a JWE string. + */ +@property (nonatomic, readonly, nullable) NSString *deviceData; + +/** + The SDK Transaction Identifier. + */ +@property (nonatomic, readonly) NSString *sdkTransactionIdentifier; + +/** + The SDK App Identifier. + */ +@property (nonatomic, readonly) NSString *sdkAppIdentifier; + +/** + The SDK reference number. + */ +@property (nonatomic, readonly) NSString *sdkReferenceNumber; + +/** + The SDK ephemeral public key. + */ +@property (nonatomic, readonly) NSString *sdkEphemeralPublicKey; + +/** + The protocol version that is used for the transaction. + */ +@property (nonatomic, readonly) NSString *messageVersion; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.m b/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.m new file mode 100644 index 00000000..dc6286b7 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationRequestParameters.m @@ -0,0 +1,63 @@ +// +// STDSAuthenticationRequestParameters.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSAuthenticationRequestParameters.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSAuthenticationRequestParameters () + +@property (nonatomic, nullable, readonly) NSDictionary *sdkEphemeralPublicKeyJSON; + +@end + +@implementation STDSAuthenticationRequestParameters + +- (instancetype)initWithSDKTransactionIdentifier:(NSString *)sdkTransactionIdentifier + deviceData:(nullable NSString *)deviceData + sdkEphemeralPublicKey:(NSString *)sdkEphemeralPublicKey + sdkAppIdentifier:(NSString *)sdkAppIdentifier + sdkReferenceNumber:(NSString *)sdkReferenceNumber + messageVersion:(NSString *)messageVersion { + self = [super init]; + if (self) { + _sdkTransactionIdentifier = [sdkTransactionIdentifier copy]; + _deviceData = [deviceData copy]; + _sdkEphemeralPublicKey = [sdkEphemeralPublicKey copy]; + _sdkAppIdentifier = [sdkAppIdentifier copy]; + _sdkReferenceNumber = [sdkReferenceNumber copy]; + _messageVersion = [messageVersion copy]; + } + return self; +} + +- (nullable NSDictionary *)sdkEphemeralPublicKeyJSON { + NSData *data = [self.sdkEphemeralPublicKey dataUsingEncoding:NSUTF8StringEncoding]; + if (data == nil) { + return nil; + } + + return [NSJSONSerialization JSONObjectWithData:data options:0 error:NULL]; +} + +#pragma mark - STDSJSONEncodable + ++ (NSDictionary *)propertyNamesToJSONKeysMapping { + return @{ + NSStringFromSelector(@selector(sdkTransactionIdentifier)): @"sdkTransID", + NSStringFromSelector(@selector(deviceData)): @"sdkEncData", + NSStringFromSelector(@selector(sdkEphemeralPublicKeyJSON)): @"sdkEphemPubKey", + NSStringFromSelector(@selector(sdkAppIdentifier)): @"sdkAppID", + NSStringFromSelector(@selector(sdkReferenceNumber)): @"sdkReferenceNumber", + NSStringFromSelector(@selector(messageVersion)): @"messageVersion", + }; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationResponse.h b/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationResponse.h new file mode 100644 index 00000000..d527f38d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSAuthenticationResponse.h @@ -0,0 +1,115 @@ +// +// STDSAuthenticationResponse.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 2/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSJSONDecodable.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + The `STDSACSStatusType` enum defines the status of a transaction, as detailed in + 3DS2 Spec Seq 3.33: + */ +typedef NS_ENUM(NSInteger, STDSACSStatusType) { + /// The status is unknown or invalid + STDSACSStatusTypeUnknown = 0, + + /// Authenticated + STDSACSStatusTypeAuthenticated = 1, + + /// Requires a Cardholder challenge to complete authentication + STDSACSStatusTypeChallengeRequired = 2, + + /// Requires a Cardholder challenge using Decoupled Authentication + STDSACSStatusTypeDecoupledAuthentication = 3, + + /// Not authenticated + STDSACSStatusTypeNotAuthenticated = 4, + + /// Not authenticated, but a proof of authentication attempt (Authentication Value) + /// was generated + STDSACSStatusTypeProofGenerated = 5, + + /// Not authenticated, as authentication could not be performed due to technical or + /// other issue + STDSACSStatusTypeError = 6, + + /// Not authenticated because the Issuer is rejecting authentication and requesting + /// that authorisation not be attempted + STDSACSStatusTypeRejected = 7, + + /// Authentication not requested by the 3DS Server for data sent for informational + /// purposes only + STDSACSStatusTypeInformationalOnly = 8, +}; + +/** + A native protocol representing the response sent by the 3DS Server. + Only parameters relevant to performing 3DS2 authentication in the mobile SDK are exposed. + */ +@protocol STDSAuthenticationResponse + +/// Universally unique transaction identifier assigned by the 3DS Server to identify a single transaction. +@property (nonatomic, readonly) NSString *threeDSServerTransactionID; + +/// Transaction status +@property (nonatomic, readonly) STDSACSStatusType status; + +/// Indication of whether a challenge is required. +@property (nonatomic, readonly, getter=isChallengeRequired) BOOL challengeRequired; + +/// Indicates whether the ACS confirms utilisation of Decoupled Authentication and agrees to utilise Decoupled Authentication to authenticate the Cardholder. +@property (nonatomic, readonly) BOOL willUseDecoupledAuthentication; + +/** + DS assigned ACS identifier. + Each DS can provide a unique ID to each ACS on an individual basis. + */ +@property (nonatomic, readonly, nullable) NSString *acsOperatorID; + +/// Unique identifier assigned by the EMVCo Secretariat upon Testing and Approval. +@property (nonatomic, readonly, nullable) NSString *acsReferenceNumber; + +/// Contains the JWS object (represented as a string) created by the ACS for the ARes message. +@property (nonatomic, readonly, nullable) NSString *acsSignedContent; + +/// Universally Unique transaction identifier assigned by the ACS to identify a single transaction. +@property (nonatomic, readonly) NSString *acsTransactionID; + +/// Fully qualified URL of the ACS to be used for the challenge. +@property (nonatomic, readonly, nullable) NSURL *acsURL; + +/** + Text provided by the ACS/Issuer to Cardholder during a Frictionless or Decoupled transaction. The Issuer can provide information to Cardholder. + For example, “Additional authentication is needed for this transaction, please contact (Issuer Name) at xxx-xxx-xxxx.” + */ +@property (nonatomic, readonly, nullable) NSString *cardholderInfo; + +/// EMVCo-assigned unique identifier to track approved DS. +@property (nonatomic, readonly, nullable) NSString *directoryServerReferenceNumber; + +/// Universally unique transaction identifier assigned by the DS to identify a single transaction. +@property (nonatomic, readonly, nullable) NSString *directoryServerTransactionID; + +/** + Protocol version identifier This shall be the Protocol Version Number of the specification utilised by the system creating this message. + The Message Version Number is set by the 3DS Server which originates the protocol with the AReq message. + The Message Version Number does not change during a 3DS transaction. + */ +@property (nonatomic, readonly) NSString *protocolVersion; + +/// Universally unique transaction identifier assigned by the 3DS SDK to identify a single transaction. +@property (nonatomic, readonly) NSString *sdkTransactionID; + +@end + +/// A utility to parse an STDSAuthenticationResponse from JSON +id _Nullable STDSAuthenticationResponseFromJSON(NSDictionary *json); + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.h new file mode 100644 index 00000000..dd63f982 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.h @@ -0,0 +1,83 @@ +// +// STDSButtonCustomization.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// +#import +#import + +#import "STDSCustomization.h" + +/// An enum that defines the different types of buttons that are able to be customized. +typedef NS_ENUM(NSInteger, STDSUICustomizationButtonType) { + + /// The submit button type. + STDSUICustomizationButtonTypeSubmit = 0, + + /// The continue button type. + STDSUICustomizationButtonTypeContinue = 1, + + /// The next button type. + STDSUICustomizationButtonTypeNext = 2, + + /// The cancel button type. + STDSUICustomizationButtonTypeCancel = 3, + + /// The resend button type. + STDSUICustomizationButtonTypeResend = 4, +}; + +/// An enumeration of the case transformations that can be applied to the button's title +typedef NS_ENUM(NSInteger, STDSButtonTitleStyle) { + /// Default style, doesn't modify the title + STDSButtonTitleStyleDefault, + + /// Applies localizedUppercaseString to the title + STDSButtonTitleStyleUppercase, + + /// Applies localizedLowercaseString to the title + STDSButtonTitleStyleLowercase, + + /// Applies localizedCapitalizedString to the title + STDSButtonTitleStyleSentenceCapitalized, +}; + +NS_ASSUME_NONNULL_BEGIN + +/// A customization object to use to configure the UI of a button. +@interface STDSButtonCustomization: STDSCustomization + +/// The default settings for the provided button type. ++ (instancetype)defaultSettingsForButtonType:(STDSUICustomizationButtonType)type; + +/** + Initializes an instance of STDSButtonCustomization with the given backgroundColor and colorRadius. + */ +- (instancetype)initWithBackgroundColor:(UIColor *)backgroundColor cornerRadius:(CGFloat)cornerRadius; + +/** + This is unavailable because there are no sensible default property values without a button type. + Use `defaultSettingsForButtonType:` or `initWithBackgroundColor:cornerRadius:` instead. + */ +- (instancetype)init NS_UNAVAILABLE; + +/** + The background color of the button. + The default for .resend and .cancel is clear. + The default for .submit, .continue, and .next is blue. + */ +@property (nonatomic) UIColor *backgroundColor; + +/// The corner radius of the button. Defaults to 8. +@property (nonatomic) CGFloat cornerRadius; + +/** + The capitalization style of the button title + */ +@property (nonatomic) STDSButtonTitleStyle titleStyle; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.m new file mode 100644 index 00000000..a82a6cbe --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSButtonCustomization.m @@ -0,0 +1,69 @@ +// +// STDSButtonCustomization.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSButtonCustomization.h" + +#import "STDSUICustomization.h" +#import "UIColor+DefaultColors.h" +#import "UIFont+DefaultFonts.h" + +static const CGFloat kDefaultButtonCornerRadius = 8.0; +static const CGFloat kDefaultButtonFontScale = (CGFloat)0.9; + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSButtonCustomization + ++ (instancetype)defaultSettingsForButtonType:(STDSUICustomizationButtonType)type { + UIColor *backgroundColor = [UIColor _stds_blueColor]; + CGFloat cornerRadius = kDefaultButtonCornerRadius; + UIFont *font = [UIFont _stds_defaultBoldLabelTextFontWithScale:kDefaultButtonFontScale]; + UIColor *textColor = UIColor.whiteColor; + switch (type) { + case STDSUICustomizationButtonTypeContinue: + case STDSUICustomizationButtonTypeSubmit: + case STDSUICustomizationButtonTypeNext: + break; + case STDSUICustomizationButtonTypeResend: + backgroundColor = UIColor.clearColor; + textColor = [UIColor _stds_blueColor]; + font = nil; + break; + case STDSUICustomizationButtonTypeCancel: + backgroundColor = UIColor.clearColor; + textColor = nil; + font = nil; + break; + } + STDSButtonCustomization *buttonCustomization = [[self alloc] initWithBackgroundColor:backgroundColor cornerRadius:cornerRadius]; + buttonCustomization.font = font; + buttonCustomization.textColor = textColor; + return buttonCustomization; +} + +- (instancetype)initWithBackgroundColor:(UIColor *)backgroundColor cornerRadius:(CGFloat)cornerRadius { + self = [super init]; + if (self) { + _backgroundColor = backgroundColor; + _cornerRadius = cornerRadius; + } + return self; +} + +- (id)copyWithZone:(nullable NSZone *)zone { + STDSButtonCustomization *copy = [super copyWithZone:zone]; + copy.backgroundColor = self.backgroundColor; + copy.cornerRadius = self.cornerRadius; + copy.titleStyle = self.titleStyle; + + return copy; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.h b/Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.h new file mode 100644 index 00000000..e1aa151c --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.h @@ -0,0 +1,61 @@ +// +// STDSChallengeParameters.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 2/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@protocol STDSAuthenticationResponse; + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSChallengeParameters` contains information from the 3DS Server's + authentication response that are used by the 3DS2 SDK to initiate + the challenge flow. + */ +@interface STDSChallengeParameters : NSObject + +/** + Convenience intiializer to create an instace of `STDSChallengeParameters` from an + `STDSAuthenticationResponse` + */ +- (instancetype)initWithAuthenticationResponse:(id)authResponse; + +/** + Transaction identifier assigned by the 3DS Server to uniquely identify + a transaction. + */ +@property (nonatomic, copy) NSString *threeDSServerTransactionID; + +/** + Transaction identifier assigned by the Access Control Server (ACS) + to uniquely identify a transaction. + */ +@property (nonatomic, copy) NSString *acsTransactionID; + +/** + The reference number of the relevant Access Control Server. + */ +@property (nonatomic, copy) NSString *acsReferenceNumber; + +/** + The encrypted message sent by the Access Control Server + containing the ACS URL, epthemeral public key, and the + 3DS2 SDK ephemeral public key. + */ +@property (nonatomic, copy) NSString *acsSignedContent; + +/** + The URL for the application that is requesting 3DS2 verification. + This property can be optionally set and will be included with the + messages sent to the Directory Server during the challenge flow. + */ +@property (nonatomic, copy, nullable) NSString *threeDSRequestorAppURL; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.m b/Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.m new file mode 100644 index 00000000..ac166cd7 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSChallengeParameters.m @@ -0,0 +1,31 @@ +// +// STDSChallengeParameters.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 2/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeParameters.h" + +#import "STDSAuthenticationResponse.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSChallengeParameters + +- (instancetype)initWithAuthenticationResponse:(id)authResponse { + self = [self init]; + if (self) { + _threeDSServerTransactionID = [authResponse.threeDSServerTransactionID copy]; + _acsTransactionID = [authResponse.acsTransactionID copy]; + _acsReferenceNumber = [authResponse.acsReferenceNumber copy]; + _acsSignedContent = [authResponse.acsSignedContent copy]; + } + + return self; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSChallengeStatusReceiver.h b/Stripe3DS2/Stripe3DS2/include/STDSChallengeStatusReceiver.h new file mode 100644 index 00000000..a352e0c4 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSChallengeStatusReceiver.h @@ -0,0 +1,67 @@ +// +// STDSChallengeStatusReceiver.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import + +@class STDSTransaction, STDSCompletionEvent, STDSRuntimeErrorEvent, STDSProtocolErrorEvent; + +NS_ASSUME_NONNULL_BEGIN + +/** + Implement the `STDSChallengeStatusReceiver` protocol to receive challenge status notifications at the end of the challenge process. + @see `STDSTransaction.doChallenge` + */ +@protocol STDSChallengeStatusReceiver + +/** + Called when the challenge process is completed. + + @param completionEvent Information about the completion of the challenge process. @see `STDSCompletionEvent` + */ +- (void)transaction:(STDSTransaction *)transaction didCompleteChallengeWithCompletionEvent:(STDSCompletionEvent *)completionEvent; + +/** + Called when the user selects the option to cancel the transaction on the challenge screen. + */ +- (void)transactionDidCancel:(STDSTransaction *)transaction; + +/** + Called when the challenge process reaches or exceeds the timeout interval that was passed to `STDSTransaction.doChallenge` + */ +- (void)transactionDidTimeOut:(STDSTransaction *)transaction; + +/** + Called when the 3DS SDK receives an EMV 3-D Secure protocol-defined error message from the ACS. + + @param protocolErrorEvent The error code and details. @see `STDSProtocolErrorEvent` + */ +- (void)transaction:(STDSTransaction *)transaction didErrorWithProtocolErrorEvent:(STDSProtocolErrorEvent *)protocolErrorEvent; + +/** + Called when the 3DS SDK encounters errors during the challenge process. These errors include all errors except those covered by `didErrorWithProtocolErrorEvent`. + + @param runtimeErrorEvent The error code and details. @see `STDSRuntimeErrorEvent` + */ +- (void)transaction:(STDSTransaction *)transaction didErrorWithRuntimeErrorEvent:(STDSRuntimeErrorEvent *)runtimeErrorEvent; + +@optional + +/** + Optional method that will be called when the transaction displays a new challenge screen. + */ +- (void)transactionDidPresentChallengeScreen:(STDSTransaction *)transaction; + +/** + Optional method for custom dismissal of the challenge view controller. Meant only for internal use by Stripe SDK. + */ +- (void)dismissChallengeViewController:(UIViewController *)challengeViewController forTransaction:(STDSTransaction *)transaction; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.h b/Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.h new file mode 100644 index 00000000..851c71db --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.h @@ -0,0 +1,40 @@ +// +// STDSCompletionEvent.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSCompletionEvent` contains information about completion of the challenge process. + */ +@interface STDSCompletionEvent : NSObject + +/** + Designated initializer for `STDSCompletionEvent`. + */ +- (instancetype)initWithSDKTransactionIdentifier:(NSString *)identifier transactionStatus:(NSString *)transactionStatus NS_DESIGNATED_INITIALIZER; + +/** + `STDSCompletionEvent` should not be directly initialized. + */ +- (instancetype)init NS_UNAVAILABLE; + +/** + The SDK Transaction ID. + */ +@property (nonatomic, readonly) NSString *sdkTransactionIdentifier; + +/** + The transaction status that was received in the final challenge response. + */ +@property (nonatomic, readonly) NSString *transactionStatus; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.m b/Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.m new file mode 100644 index 00000000..6ee3c43d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSCompletionEvent.m @@ -0,0 +1,26 @@ +// +// STDSCompletionEvent.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSCompletionEvent.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSCompletionEvent + +- (instancetype)initWithSDKTransactionIdentifier:(NSString *)identifier transactionStatus:(NSString *)transactionStatus { + self = [super init]; + if (self) { + _sdkTransactionIdentifier = [identifier copy]; + _transactionStatus = [transactionStatus copy]; + } + return self; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.h b/Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.h new file mode 100644 index 00000000..4d77ba5d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.h @@ -0,0 +1,96 @@ +// +// STDSConfigParameters.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/** + The default group name that will be used to group additional + configuration parameters. + */ +extern NSString * const kSTDSConfigDefaultGroupName; + +/** + `STDSConfigParameters` represents additional configuration parameters + that can be passed to the Stripe3DS2 SDK during initialization. + + There are currently no supported additional parameters and apps can + just pass `[STDSConfigParameters alloc] initWithStandardParameters` + to the `STDSThreeDS2Service` instance. + */ +@interface STDSConfigParameters : NSObject + +/** + Convenience initializer to get an `STDSConfigParameters` instance + with the default expected configuration parameters. + */ +- (instancetype)initWithStandardParameters; + +/** + Adds the parameter to this instance. + + @param paramName The name of the parameter to add + @param paramValue The value of the parameter to add + @param paramGroup The group to which this parameter will be added. If `nil` the parameter will be added to `kSTDSConfigDefaultGroupName` + + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `paramName` or `paramValue` are `nil`. @see STDSInvalidInputException + */ +- (void)addParameterNamed:(NSString *)paramName withValue:(NSString *)paramValue toGroup:(nullable NSString *)paramGroup; + +/** + Adds the parameter to the default group in this instance. + + @param paramName The name of the parameter to add + @param paramValue The value of the parameter to add + + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `paramName` or `paramValue` are `nil`. @see STDSInvalidInputException + */ +- (void)addParameterNamed:(NSString *)paramName withValue:(NSString *)paramValue; + +/** + Returns the value for `paramName` in `paramGroup` or `nil` if the parameter value is not set. + + @param paramName The name of the parameter to return + @param paramGroup The group from which to fetch the parameter value. If `nil` will default to `kSTDSConfigDefaultGroupName` + + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `paramName` is `nil`. @see STDSInvalidInputException + */ +- (nullable NSString *)parameterValue:(NSString *)paramName inGroup:(nullable NSString *)paramGroup; + +/** + Returns the value for `paramName` in the default group or `nil` if the parameter value is not set. + + @param paramName The name of the parameter to return + + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `paramName` is `nil`. @see STDSInvalidInputException + */ +- (nullable NSString *)parameterValue:(NSString *)paramName; + +/** + Removes the specified parameter from the group and returns the value or `nil` if the parameter was not found. + + @param paramName The name of the parameter to remove + @param paramGroup The group from which to remove this parameter. If `nil` will default to `kSTDSConfigDefaultGroupName` + + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `paramName` is `nil`. @see STDSInvalidInputException + */ +- (nullable NSString *)removeParameterNamed:(NSString *)paramName fromGroup:(nullable NSString *)paramGroup; + +/** + Removes the specified parameter from the default group and returns the value or `nil` if the parameter was not found. + + @param paramName The name of the parameter to remove + + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `paramName` is `nil`. @see STDSInvalidInputException + */ +- (nullable NSString *)removeParameterNamed:(NSString *)paramName; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.m b/Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.m new file mode 100644 index 00000000..e334a5bb --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSConfigParameters.m @@ -0,0 +1,113 @@ +// +// STDSConfigParameters.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSConfigParameters.h" + +#import "STDSException+Internal.h" +#import "STDSInvalidInputException.h" + +NS_ASSUME_NONNULL_BEGIN + +NSString * const kSTDSConfigDefaultGroupName = @"STDSConfigParameters.group.default"; + +@implementation STDSConfigParameters +{ + NSMutableDictionary *_parameters; +} + +- (instancetype)init { + self = [super init]; + if (self) { + _parameters = [[NSMutableDictionary alloc] init]; + } + + return self; +} + +- (instancetype)initWithStandardParameters { + self = [self init]; + if (self) { + // Nothing for now because we don't have any standard parameters + } + + return self; +} + +- (void)addParameterNamed:(NSString *)paramName withValue:(NSString *)paramValue { + [self _addParameterNamed:paramName withValue:paramValue toGroup:kSTDSConfigDefaultGroupName]; +} + +- (void)addParameterNamed:(NSString *)paramName withValue:(NSString *)paramValue toGroup:(nullable NSString *)paramGroup { + [self _addParameterNamed:paramName withValue:paramValue toGroup:paramGroup ?: kSTDSConfigDefaultGroupName]; +} + +- (void)_addParameterNamed:(NSString *)paramName withValue:(NSString *)paramValue toGroup:(NSString *)paramGroup { + if (paramName == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"nil paramName passed to instance %@", self]; + } else if (paramValue == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"nil paramValue passed to instance %@", self]; + } else if (paramGroup == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"nil paramGroup passed to instance %@", self]; + } + + NSMutableDictionary *groupParameters = _parameters[paramGroup]; + if (groupParameters == nil) { + groupParameters = [[NSMutableDictionary alloc] init]; + _parameters[paramGroup] = groupParameters; + } + + if (groupParameters[paramName] != nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"Cannot override value of %@ for parameter %@ in group %@ with value %@", groupParameters[paramName], paramName, paramGroup, paramValue]; + } + + groupParameters[paramName] = paramValue; +} + +- (nullable NSString *)parameterValue:(NSString *)paramName { + return [self _parameterValue:paramName inGroup:kSTDSConfigDefaultGroupName]; +} + +- (nullable NSString *)parameterValue:(NSString *)paramName inGroup:(nullable NSString *)paramGroup { + return [self _parameterValue:paramName inGroup:paramGroup ?: kSTDSConfigDefaultGroupName]; +} + +- (nullable NSString *)_parameterValue:(NSString *)paramName inGroup:(NSString *)paramGroup { + if (paramName == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"nil paramName passed to instance %@", self]; + } else if (paramGroup == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"nil paramGroup passed to instance %@", self]; + } + + NSMutableDictionary *groupParameters = _parameters[paramGroup]; + return groupParameters[paramName]; +} + +- (nullable NSString *)removeParameterNamed:(NSString *)paramName { + return [self _removeParameterNamed:paramName fromGroup:kSTDSConfigDefaultGroupName]; +} + +- (nullable NSString *)removeParameterNamed:(NSString *)paramName fromGroup:(nullable NSString *)paramGroup { + return [self _removeParameterNamed:paramName fromGroup:paramGroup ?: kSTDSConfigDefaultGroupName]; +} + +- (nullable NSString *)_removeParameterNamed:(NSString *)paramName fromGroup:(NSString *)paramGroup { + if (paramName == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"nil paramName passed to instance %@", self]; + } else if (paramGroup == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"nil paramGroup passed to instance %@", self]; + } + + NSMutableDictionary *groupParameters = _parameters[paramGroup]; + NSString *paramValue = groupParameters[paramName]; + [groupParameters removeObjectForKey:paramName]; + return paramValue; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSCustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSCustomization.h new file mode 100644 index 00000000..516eff9e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSCustomization.h @@ -0,0 +1,25 @@ +// +// STDSCustomization.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import + +NS_ASSUME_NONNULL_BEGIN + +/// This class provides a common set of customization parameters, used to customize elements of the UI. +@interface STDSCustomization : NSObject + +/// The font to use for text. +@property (nonatomic, nullable) UIFont *font; + +/// The color to use for the text. +@property (nonatomic, nullable) UIColor *textColor; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSCustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSCustomization.m new file mode 100644 index 00000000..4544024c --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSCustomization.m @@ -0,0 +1,25 @@ +// +// STDSCustomization.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSCustomization + +- (id)copyWithZone:(nullable NSZone *)zone { + STDSCustomization *copy = [[[self class] allocWithZone:zone] init]; + copy.font = self.font; + copy.textColor = self.textColor; + + return copy; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.h b/Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.h new file mode 100644 index 00000000..29c6c5a6 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.h @@ -0,0 +1,103 @@ +// +// STDSErrorMessage.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSJSONEncodable.h" +#import "STDSJSONDecodable.h" + +NS_ASSUME_NONNULL_BEGIN + +/// Error codes as defined by the 3DS2 spec. +typedef NS_ENUM(NSInteger, STDSErrorMessageCode) { + /// The SDK received a message that is not an ARes, CRes, or ErrorMessage. + STDSErrorMessageCodeInvalidMessage = 101, + + /// A required data element is missing from the network response. + STDSErrorMessageCodeRequiredDataElementMissing = 201, + + // Critical message extension not recognised + STDSErrorMessageCodeUnrecognizedCriticalMessageExtension = 202, + + /// A data element is not in the required format or the value is invalid. + STDSErrorMessageErrorInvalidDataElement = 203, + + // Transaction ID not recognized + STDSErrorMessageErrorTransactionIDNotRecognized = 301, + + /// A network response could not be decrypted or verified. + STDSErrorMessageErrorDataDecryptionFailure = 302, + + /// The SDK timed out + STDSErrorMessageErrorTimeout = 402, +}; + +/** + `STDSErrorMessage` represents an error message that is returned by the ACS or to be sent to the ACS. + */ +@interface STDSErrorMessage : NSObject + +/** + Designated initializer for `STDSErrorMessage`. + + @param errorCode The error code. + @param errorComponent The component that identified the error. + @param errorDescription Text describing the error. + @param errorDetails Additional error details. Optional. + */ +- (instancetype)initWithErrorCode:(NSString *)errorCode + errorComponent:(NSString *)errorComponent + errorDescription:(NSString *)errorDescription + errorDetails:(nullable NSString *)errorDetails + messageVersion:(NSString *)messageVersion + acsTransactionIdentifier:(nullable NSString *)acsTransactionIdentifier + errorMessageType:(NSString *)errorMessageType; + +/** + The error code. + */ +@property (nonatomic, readonly) NSString *errorCode; + +/** + The 3-D Secure component that identified the error. + */ +@property (nonatomic, readonly) NSString *errorComponent; + +/** + Text describing the error. + */ +@property (nonatomic, readonly) NSString *errorDescription; + +/** + Additional error details. + */ +@property (nonatomic, nullable, readonly) NSString *errorDetails; + +/** + The protocol version identifier. + */ +@property (nonatomic, readonly) NSString *messageVersion; + +/** + The ACS transaction identifier. + */ +@property (nonatomic, readonly, nullable) NSString *acsTransactionIdentifier; + +/** + The message type that was identified as erroneous. + */ +@property (nonatomic, readonly, nullable) NSString *errorMessageType; + +/** + A representation of the `STDSErrorMessage` as an `NSError` + */ +- (NSError *)NSErrorValue; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.m b/Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.m new file mode 100644 index 00000000..1e80b645 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSErrorMessage.m @@ -0,0 +1,94 @@ +// +// STDSErrorMessage.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSErrorMessage.h" + +#import "NSDictionary+DecodingHelpers.h" +#import "STDSJSONEncoder.h" +#import "STDSStripe3DS2Error.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSErrorMessage + +- (instancetype)initWithErrorCode:(NSString *)errorCode + errorComponent:(NSString *)errorComponent + errorDescription:(NSString *)errorDescription + errorDetails:(nullable NSString *)errorDetails + messageVersion:(NSString *)messageVersion + acsTransactionIdentifier:(nullable NSString *)acsTransactionIdentifier + errorMessageType:(NSString *)errorMessageType { + self = [super init]; + if (self) { + _errorCode = [errorCode copy]; + _errorComponent = [errorComponent copy]; + _errorDescription = [errorDescription copy]; + _errorDetails = [errorDetails copy]; + _messageVersion = [messageVersion copy]; + _acsTransactionIdentifier = [acsTransactionIdentifier copy]; + _errorMessageType = [errorMessageType copy]; + } + return self; +} + +- (NSString *)messageType { + return @"Erro"; +} + +- (NSError *)NSErrorValue { + return [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:[self.errorCode integerValue] + userInfo: [STDSJSONEncoder dictionaryForObject:self]]; +} + +#pragma mark - STDSJSONEncodable + ++ (NSDictionary *)propertyNamesToJSONKeysMapping { + return @{ + NSStringFromSelector(@selector(errorCode)): @"errorCode", + NSStringFromSelector(@selector(errorComponent)): @"errorComponent", + NSStringFromSelector(@selector(errorDescription)): @"errorDescription", + NSStringFromSelector(@selector(errorDetails)): @"errorDetail", + NSStringFromSelector(@selector(messageType)): @"messageType", + NSStringFromSelector(@selector(messageVersion)): @"messageVersion", + NSStringFromSelector(@selector(acsTransactionIdentifier)): @"acsTransID", + NSStringFromSelector(@selector(errorMessageType)): @"errorMessageType", + }; +} + +#pragma mark - STDSJSONDecodable + ++ (nullable instancetype)decodedObjectFromJSON:(nullable NSDictionary *)json error:(NSError * _Nullable __autoreleasing * _Nullable)outError { + if (json == nil) { + return nil; + } + NSError *error; + + // Required + NSString *errorCode = [json _stds_stringForKey:@"errorCode" required:YES error:&error]; + NSString *errorComponent = [json _stds_stringForKey:@"errorComponent" required:YES error:&error]; + NSString *errorDescription = [json _stds_stringForKey:@"errorDescription" required:YES error:&error]; + NSString *errorDetail = [json _stds_stringForKey:@"errorDetail" required:YES error:&error]; + NSString *messageVersion = [json _stds_stringForKey:@"messageVersion" required:YES error:&error]; + + // Optional + NSString *errorMessageType = [json _stds_stringForKey:@"errorMessageType" required:NO error:&error]; + NSString *acsTransactionIdentifier = [json _stds_stringForKey:@"acsTransID" required:NO error:nil]; + + if (error) { + if (outError) { + *outError = error; + } + return nil; + } + return [[self alloc] initWithErrorCode:errorCode errorComponent:errorComponent errorDescription:errorDescription errorDetails:errorDetail messageVersion:messageVersion acsTransactionIdentifier:acsTransactionIdentifier errorMessageType:errorMessageType]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSException.h b/Stripe3DS2/Stripe3DS2/include/STDSException.h new file mode 100644 index 00000000..1f2c5ecd --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSException.h @@ -0,0 +1,25 @@ +// +// STDSException.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/** + An abstract class to represent 3DS2 SDK custom exceptions + */ +@interface STDSException : NSException + +/** + A description of the exception. + */ +@property (nonatomic, readonly) NSString *message; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSException.m b/Stripe3DS2/Stripe3DS2/include/STDSException.m new file mode 100644 index 00000000..673d5bbd --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSException.m @@ -0,0 +1,27 @@ +// +// STDSException.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSException.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSException + ++ (instancetype)exceptionWithMessage:(NSString *)format, ... { + va_list args; + va_start(args, format); + NSString *message = [[NSString alloc] initWithFormat:format arguments:args]; + va_end(args); + STDSException *exception = [[[self class] alloc] initWithName:NSStringFromClass([self class]) reason:message userInfo:nil]; + exception->_message = [message copy]; + return exception; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.h new file mode 100644 index 00000000..990ffd9b --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.h @@ -0,0 +1,41 @@ +// +// STDSFooterCustomization.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 6/10/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + The Challenge view displays a footer with additional details that + expand when tapped. This object configures the appearance of that view. +*/ +@interface STDSFooterCustomization : STDSCustomization + +/// The default settings. ++ (instancetype)defaultSettings; + +/** + The background color of the footer. + Defaults to gray. + */ +@property (nonatomic) UIColor *backgroundColor; + +/// The color of the chevron. Defaults to a dark gray. +@property (nonatomic) UIColor *chevronColor; + +/// The color of the heading text. Defaults to black. +@property (nonatomic) UIColor *headingTextColor; + +/// The font to use for the heading text. +@property (nonatomic) UIFont *headingFont; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.m new file mode 100644 index 00000000..6c770fda --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSFooterCustomization.m @@ -0,0 +1,45 @@ +// +// STDSFooterCustomization.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 6/10/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSFooterCustomization.h" + +#import "UIFont+DefaultFonts.h" +#import "UIColor+DefaultColors.h" +#import "UIColor+ThirteenSupport.h" + +@implementation STDSFooterCustomization + ++ (instancetype)defaultSettings { + return [self new]; +} + +- (instancetype)init { + self = [super init]; + if (self) { + self.textColor = UIColor._stds_labelColor; + _headingTextColor = UIColor._stds_labelColor; + _backgroundColor = [UIColor _stds_defaultFooterBackgroundColor]; + _chevronColor = [UIColor _stds_systemGray2Color]; + self.font = [UIFont _stds_defaultLabelTextFontWithScale:(CGFloat)0.9]; + _headingFont = [UIFont _stds_defaultLabelTextFontWithScale:(CGFloat)0.9]; + } + return self; +} + +- (instancetype)copyWithZone:(nullable NSZone *)zone { + STDSFooterCustomization *copy = [super copyWithZone:zone]; + copy.headingTextColor = self.headingTextColor; + copy.headingFont = self.headingFont; + copy.backgroundColor = self.backgroundColor; + copy.chevronColor = self.chevronColor; + + return copy; +} + + +@end diff --git a/Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.h b/Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.h new file mode 100644 index 00000000..b30d4e10 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.h @@ -0,0 +1,21 @@ +// +// STDSInvalidInputException.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSException.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSInvalidInputException` represents an exception that will be thrown by + Stripe3DS2 SDK methods that are called with invalid input arguments. + */ +@interface STDSInvalidInputException : STDSException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.m b/Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.m new file mode 100644 index 00000000..7ff4cd3e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSInvalidInputException.m @@ -0,0 +1,17 @@ +// +// STDSInvalidInputException.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSInvalidInputException.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSInvalidInputException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSJSONDecodable.h b/Stripe3DS2/Stripe3DS2/include/STDSJSONDecodable.h new file mode 100644 index 00000000..f93428ac --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSJSONDecodable.h @@ -0,0 +1,33 @@ +// +// STDSJSONDecodable.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@protocol STDSJSONDecodable + +/** + Initializes an instance of the class from its JSON representation. + + This method recognizes two categories of errors: + - a required field is missing. + - a required field value is in valid (e.g. expected 'Y' or 'N' but received 'X'). + + Errors populating optional fields are ignored. + + @param json The JSON dictionary that represents an object of this type + @param outError If there was a missing required field or invalid field value, contains an instance of NSError. + + @return The object represented by the JSON dictionary. If the object could not be decoded, returns nil and populates the outError argument. + */ ++ (nullable instancetype)decodedObjectFromJSON:(nullable NSDictionary *)json error:(NSError **)outError; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSJSONEncodable.h b/Stripe3DS2/Stripe3DS2/include/STDSJSONEncodable.h new file mode 100644 index 00000000..9861a23c --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSJSONEncodable.h @@ -0,0 +1,22 @@ +// +// STDSJSONEncodable.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@protocol STDSJSONEncodable + +/** + Returns a map of property names to their JSON representation's key value. For example, `STDSChallengeParameters` has a property called `acsTransactionID`, but the 3DS2 JSON spec expects a field called `acsTransID`. This dictionary represents a mapping from the former to the latter (in other words, [STDSChallengeParameters propertyNamesToJSONKeysMapping][@"acsTransactionID"] == @"acsTransID".) + */ ++ (NSDictionary *)propertyNamesToJSONKeysMapping; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.h b/Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.h new file mode 100644 index 00000000..7ccc07e6 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.h @@ -0,0 +1,27 @@ +// +// STDSJSONEncoder.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSJSONEncodable.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSJSONEncoder` is a utility class to help with converting API objects into JSON + */ +@interface STDSJSONEncoder : NSObject + +/** + Method to convert an STDSJSONEncodable object into a JSON dictionary. + */ ++ (NSDictionary *)dictionaryForObject:(NSObject *)object; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.m b/Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.m new file mode 100644 index 00000000..4c55b02c --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSJSONEncoder.m @@ -0,0 +1,51 @@ +// +// STDSJSONEncoder.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSJSONEncoder.h" + +@implementation STDSJSONEncoder + ++ (NSDictionary *)dictionaryForObject:(nonnull NSObject *)object { + NSMutableDictionary *keyPairs = [NSMutableDictionary dictionary]; + [[object.class propertyNamesToJSONKeysMapping] enumerateKeysAndObjectsUsingBlock:^(NSString * _Nonnull propertyName, NSString * _Nonnull keyName, __unused BOOL * _Nonnull stop) { + id value = [self jsonEncodableValueForObject:[object valueForKey:propertyName]]; + if (value != [NSNull null]) { + keyPairs[keyName] = value; + } + }]; + return [keyPairs copy]; +} + ++ (id)jsonEncodableValueForObject:(NSObject *)object { + if ([object conformsToProtocol:@protocol(STDSJSONEncodable)]) { + return [self dictionaryForObject:(NSObject*)object]; + } else if ([object isKindOfClass:[NSDictionary class]]) { + NSDictionary *dict = (NSDictionary *)object; + NSMutableDictionary *result = [NSMutableDictionary dictionaryWithCapacity:dict.count]; + + [dict enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull value, __unused BOOL * _Nonnull stop) { + result[key] = [self jsonEncodableValueForObject:value]; + }]; + + return result; + } else if ([object isKindOfClass:[NSArray class]]) { + NSArray *array = (NSArray *)object; + NSMutableArray *result = [NSMutableArray arrayWithCapacity:array.count]; + + for (NSObject *element in array) { + [result addObject:[self jsonEncodableValueForObject:element]]; + } + return result; + } else if ([object isKindOfClass:[NSString class]] || [object isKindOfClass:[NSNumber class]]) { + return object; + } else { + return [NSNull null]; + } +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.h new file mode 100644 index 00000000..af75cf39 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.h @@ -0,0 +1,31 @@ +// +// STDSLabelCustomization.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + A customization object to use to configure the UI of a text label. + + The font and textColor inherited from `STDSCustomization` configure non-heading labels. + */ +@interface STDSLabelCustomization : STDSCustomization + +/// The default settings. ++ (instancetype)defaultSettings; + +/// The color of the heading text. Defaults to black. +@property (nonatomic) UIColor *headingTextColor; + +/// The font to use for the heading text. +@property (nonatomic) UIFont *headingFont; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.m new file mode 100644 index 00000000..bb57c5f8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSLabelCustomization.m @@ -0,0 +1,43 @@ +// +// STDSLabelCustomization.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSLabelCustomization.h" + +#import "UIFont+DefaultFonts.h" +#import "UIColor+ThirteenSupport.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSLabelCustomization + ++ (instancetype)defaultSettings { + return [self new]; +} + +- (instancetype)init { + self = [super init]; + if (self) { + self.textColor = UIColor._stds_labelColor; + _headingTextColor = UIColor._stds_labelColor; + self.font = [UIFont _stds_defaultLabelTextFontWithScale:(CGFloat)0.9]; + _headingFont = [UIFont _stds_defaultHeadingTextFont]; + } + return self; +} + +- (id)copyWithZone:(nullable NSZone *)zone { + STDSLabelCustomization *copy = [super copyWithZone:zone]; + copy.headingTextColor = self.headingTextColor; + copy.headingFont = self.headingFont; + + return copy; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.h new file mode 100644 index 00000000..18affaaf --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.h @@ -0,0 +1,59 @@ +// +// STDSNavigationBarCustomization.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import + +#import "STDSCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + A customization object to use to configure a UINavigationBar. + + The font and textColor inherited from `STDSCustomization` configure the + title of the navigation bar, and default to nil. + */ +@interface STDSNavigationBarCustomization : STDSCustomization + +/// The default settings. ++ (instancetype)defaultSettings; + +/** + The tint color of the navigation bar background. + Defaults to nil. + */ +@property (nonatomic, nullable) UIColor *barTintColor; + +/** + The navigation bar style. + Defaults to UIBarStyleDefault. + */ +@property (nonatomic) UIBarStyle barStyle; + +/** + A Boolean value indicating whether the navigation bar is translucent or not. + Defaults to YES. + */ +@property (nonatomic) BOOL translucent; + +/** + The text to display in the title of the navigation bar. + Defaults to "Secure checkout". + */ +@property (nonatomic, copy) NSString *headerText; + +/** + The text to display for the button in the navigation bar. + Defaults to "Cancel". + */ +@property (nonatomic, copy) NSString *buttonText; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.m new file mode 100644 index 00000000..5e062356 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSNavigationBarCustomization.m @@ -0,0 +1,44 @@ +// +// STDSNavigationBarCustomization.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSLocalizedString.h" +#import "STDSNavigationBarCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSNavigationBarCustomization + ++ (instancetype)defaultSettings { + return [self new]; +} + +- (instancetype)init { + self = [super init]; + if (self) { + _barTintColor = nil; + _headerText = STDSLocalizedString(@"Secure checkout", @"The title for the challenge response step of an authenticated checkout."); + _buttonText = STDSLocalizedString(@"Cancel", "The text for the button that cancels the current challenge process."); + _translucent = YES; + } + return self; +} + +- (id)copyWithZone:(nullable NSZone *)zone { + STDSNavigationBarCustomization *copy = [super copyWithZone:zone]; + copy.barTintColor = self.barTintColor; + copy.headerText = self.headerText; + copy.buttonText = self.buttonText; + copy.barStyle = self.barStyle; + copy.translucent = self.translucent; + + return copy; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.h b/Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.h new file mode 100644 index 00000000..cb836d5e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.h @@ -0,0 +1,23 @@ +// +// STDSNotInitializedException.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 2/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSException.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSNotInitializedException` represents an exception that will be thrown by + the the Stripe3DS2 SDK if methods are called without initializing `STDSThreeDS2Service`. + + @see STDSThreeDS2Service + */ +@interface STDSNotInitializedException : STDSException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.m b/Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.m new file mode 100644 index 00000000..58544cc2 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSNotInitializedException.m @@ -0,0 +1,17 @@ +// +// STDSNotInitializedException.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 2/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSNotInitializedException.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSNotInitializedException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.h b/Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.h new file mode 100644 index 00000000..1e3142a7 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.h @@ -0,0 +1,42 @@ +// +// STDSProtocolErrorEvent.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class STDSErrorMessage; + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSProtocolErrorEvent` contains details about erorrs received from or sent to the ACS. + */ +@interface STDSProtocolErrorEvent : NSObject + +/** + Designated initializer for `STDSProtocolErrorEvent`. + */ +- (instancetype)initWithSDKTransactionIdentifier:(NSString *)identifier errorMessage:(STDSErrorMessage *)errorMessage; + +/** + `STDSProtocolErrorEvent` should not be directly initialized. + */ +- (instancetype)init NS_UNAVAILABLE; + +/** + Details about the error. + */ +@property (nonatomic, readonly) STDSErrorMessage *errorMessage; + +/** + The SDK Transaction Identifier. + */ +@property (nonatomic, readonly) NSString *sdkTransactionIdentifier; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.m b/Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.m new file mode 100644 index 00000000..0dbd75b5 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSProtocolErrorEvent.m @@ -0,0 +1,28 @@ +// +// STDSProtocolErrorEvent.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSProtocolErrorEvent.h" + +#import "STDSErrorMessage.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSProtocolErrorEvent + +- (instancetype)initWithSDKTransactionIdentifier:(NSString *)identifier errorMessage:(STDSErrorMessage *)errorMessage { + self = [super init]; + if (self) { + _sdkTransactionIdentifier = identifier; + _errorMessage = errorMessage; + } + return self; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.h b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.h new file mode 100644 index 00000000..658cdb90 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.h @@ -0,0 +1,53 @@ +// +// STDSRuntimeErrorEvent.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +FOUNDATION_EXTERN NSString * const kSTDSRuntimeErrorCodeParsingError; +FOUNDATION_EXTERN NSString * const kSTDSRuntimeErrorCodeEncryptionError; + +/** + `STDSRuntimeErrorEvent` contains details about run-time errors encountered during authentication. + + The following are examples of run-time errors: + - ACS is unreachable + - Unparseable message + - Network issues + */ +@interface STDSRuntimeErrorEvent : NSObject + +/** + A code corresponding to the type of error this represents. + */ +@property (nonatomic, readonly) NSString *errorCode; + +/** + Details about the error. + */ +@property (nonatomic, readonly) NSString *errorMessage; + +/** + Designated initializer for `STDSRuntimeErrorEvent`. + */ +- (instancetype)initWithErrorCode:(NSString *)errorCode errorMessage:(NSString *)errorMessage NS_DESIGNATED_INITIALIZER; + +/** + A representation of the `STDSRuntimeErrorEvent` as an `NSError` + */ +- (NSError *)NSErrorValue; + +/** + `STDSRuntimeErrorEvent` should not be directly initialized. + */ +- (instancetype)init NS_UNAVAILABLE; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.m b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.m new file mode 100644 index 00000000..8155e06e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeErrorEvent.m @@ -0,0 +1,37 @@ +// +// STDSRuntimeErrorEvent.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSRuntimeErrorEvent.h" + +#import "STDSStripe3DS2Error.h" + +NS_ASSUME_NONNULL_BEGIN + +NSString * const kSTDSRuntimeErrorCodeParsingError = @"STDSRuntimeErrorCodeParsingError"; +NSString * const kSTDSRuntimeErrorCodeEncryptionError = @"STDSRuntimeErrorCodeEncryptionError"; + +@implementation STDSRuntimeErrorEvent + +- (instancetype)initWithErrorCode:(NSString *)errorCode errorMessage:(NSString *)errorMessage { + self = [super init]; + if (self) { + _errorCode = [errorCode copy]; + _errorMessage = [errorMessage copy]; + } + return self; +} + +- (NSError *)NSErrorValue { + return [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:[self.errorCode isEqualToString:kSTDSRuntimeErrorCodeParsingError] ? STDSErrorCodeRuntimeParsing : STDSErrorCodeRuntimeEncryption + userInfo:@{@"errorMessage": self.errorMessage}]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.h b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.h new file mode 100644 index 00000000..5b63a2d3 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.h @@ -0,0 +1,21 @@ +// +// STDSRuntimeException.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSException.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSRuntimeException` represents an exception that will be thrown by the + Stripe3DS2 SDK if it encounters an internal error. + */ +@interface STDSRuntimeException : STDSException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.m b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.m new file mode 100644 index 00000000..3a70d376 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSRuntimeException.m @@ -0,0 +1,17 @@ +// +// STDSRuntimeException.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSRuntimeException.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSRuntimeException + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.h new file mode 100644 index 00000000..a9818bb9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.h @@ -0,0 +1,48 @@ +// +// STDSSelectionCustomization.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 6/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/** + A customization object that configures the appearance of + radio buttons and checkboxes. + */ +@interface STDSSelectionCustomization: NSObject + +/// The default settings. ++ (instancetype)defaultSettings; + +/** + The primary color of the selected state. + Defaults to blue. + */ +@property (nonatomic) UIColor *primarySelectedColor; + +/** + The secondary color of the selected state (e.g. the checkmark color). + Defaults to white. + */ +@property (nonatomic) UIColor *secondarySelectedColor; + +/** + The background color displayed in the unselected state. + Defaults to light blue. + */ +@property (nonatomic) UIColor *unselectedBackgroundColor; + +/** + The color of the border drawn around the view in the unselected state. + Defaults to blue. + */ +@property (nonatomic) UIColor *unselectedBorderColor; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.m new file mode 100644 index 00000000..c5301ef8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSSelectionCustomization.m @@ -0,0 +1,77 @@ +// +// STDSSelectionCustomization.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 6/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSSelectionCustomization.h" + +#import "UIColor+DefaultColors.h" +#import "UIColor+ThirteenSupport.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSSelectionCustomization + ++ (instancetype)defaultSettings { + return [self new]; +} + +- (instancetype)init { + self = [super init]; + if (self) { + if (@available(iOS 12.0, *)) { + _primarySelectedColor = [UIColor _stds_blueColor]; + _secondarySelectedColor = UIColor.whiteColor; + _unselectedBackgroundColor = [UIColor _stds_colorWithDynamicProvider:^UIColor * _Nonnull(UITraitCollection * _Nonnull traitCollection) { + return (traitCollection.userInterfaceStyle == UIUserInterfaceStyleLight) ? + [UIColor colorWithRed:(CGFloat)231.0 / (CGFloat)255.0 + green:(CGFloat)241.0 / (CGFloat)255.0 + blue:(CGFloat)254.0 / (CGFloat)255.0 + alpha:1.0] : + [UIColor colorWithRed:(CGFloat)30.0 / (CGFloat)255.0 + green:(CGFloat)63.0 / (CGFloat)255.0 + blue:(CGFloat)84.0 / (CGFloat)255.0 + alpha:1.0]; + }]; + _unselectedBorderColor = [UIColor _stds_colorWithDynamicProvider:^UIColor * _Nonnull(UITraitCollection * _Nonnull traitCollection) { + return (traitCollection.userInterfaceStyle == UIUserInterfaceStyleLight) ? + [UIColor colorWithRed:(CGFloat)131.0 / (CGFloat)255.0 + green:(CGFloat)191.0 / (CGFloat)255.0 + blue:(CGFloat)250.0 / (CGFloat)255.0 + alpha:1] : + [UIColor colorWithRed:(CGFloat)65.0 / (CGFloat)255.0 + green:(CGFloat)94.0 / (CGFloat)255.0 + blue:(CGFloat)123.0 / (CGFloat)255.0 + alpha:1]; + }]; + } else { + _primarySelectedColor = [UIColor _stds_blueColor]; + _secondarySelectedColor = UIColor.whiteColor; + _unselectedBackgroundColor = [UIColor colorWithRed:(CGFloat)231.0 / (CGFloat)255.0 + green:(CGFloat)241.0 / (CGFloat)255.0 + blue:(CGFloat)254.0 / (CGFloat)255.0 + alpha:1.0]; + _unselectedBorderColor = [UIColor colorWithRed:(CGFloat)131.0 / (CGFloat)255.0 + green:(CGFloat)191.0 / (CGFloat)255.0 + blue:(CGFloat)250.0 / (CGFloat)255.0 + alpha:1]; + } + } + return self; +} + +- (nonnull id)copyWithZone:(nullable NSZone *)zone { + STDSSelectionCustomization *copy = [STDSSelectionCustomization new]; + copy.primarySelectedColor = self.primarySelectedColor; + copy.secondarySelectedColor = self.secondarySelectedColor; + copy.unselectedBackgroundColor = self.unselectedBackgroundColor; + copy.unselectedBorderColor = self.unselectedBorderColor; + return copy; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.h b/Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.h new file mode 100644 index 00000000..2e506fe2 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.h @@ -0,0 +1,68 @@ +// +// STDSStripe3DS2Error.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +FOUNDATION_EXPORT NSString * const STDSStripe3DS2ErrorDomain; + +/** + NSError.userInfo contains this key if we received an ErrorMessage instead of the expected response object. + The value of this key is the ErrorMessage. + */ +FOUNDATION_EXPORT NSString * const STDSStripe3DS2ErrorMessageErrorKey; + +/** + NSError.userInfo contains this key if we errored parsing JSON. + The value of this key is the invalid or missing field. + */ +FOUNDATION_EXPORT NSString * const STDSStripe3DS2ErrorFieldKey; + +/** + NSError.userInfo contains this key if we couldn't recognize critical message extension(s) + The value of this key is an array of identifiers. + */ +FOUNDATION_EXPORT NSString * const STDSStripe3DS2UnrecognizedCriticalMessageExtensionsKey; + + +typedef NS_ENUM(NSInteger, STDSErrorCode) { + + /// Code triggered an assertion + STDSErrorCodeAssertionFailed = 204, + + // JSON Parsing + /// Received invalid or malformed data + STDSErrorCodeJSONFieldInvalid = 203, + /// Expected field missing + STDSErrorCodeJSONFieldMissing = 201, + + /// Critical message extension not recognised + STDSErrorCodeUnrecognizedCriticalMessageExtension = 202, + + /// Decryption or verification error + STDSErrorCodeDecryptionVerification = 302, + + /// Error code corresponding to a `STDSRuntimeErrorEvent` for an unparseable network response + STDSErrorCodeRuntimeParsing = 400, + /// Error code corresponding to a `STDSRuntimeErrorEvent` for an error with decrypting or verifying a network response + STDSErrorCodeRuntimeEncryption = 401, + + // Networking + /// We received an ErrorMessage instead of the expected response object. `userInfo[STDSStripe3DS2ErrorMessageErrorKey]` will contain the ErrorMessage object. + STDSErrorCodeReceivedErrorMessage = 1000, + /// We received an unknown message type. + STDSErrorCodeUnknownMessageType = 1001, + /// Request timed out + STDSErrorCodeTimeout = 1002, + + /// Unknown + STDSErrorCodeUnknownError = 2000, +}; + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.m b/Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.m new file mode 100644 index 00000000..6d2330ed --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSStripe3DS2Error.m @@ -0,0 +1,15 @@ +// +// STDSStripe3DS2Error.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSStripe3DS2Error.h" + +NSString * const STDSStripe3DS2ErrorDomain = @"com.stripe3ds2"; +NSString * const STDSStripe3DS2ErrorMessageErrorKey = @"STDSStripe3DS2ErrorMessageErrorKey"; +NSString * const STDSStripe3DS2ErrorFieldKey = @"STDSStripe3DS2ErrorFieldKey"; +NSString * const STDSStripe3DS2UnrecognizedCriticalMessageExtensionsKey = @"STDSStripe3DS2UnrecognizedCriticalMessageExtensionsKey"; + diff --git a/Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.h b/Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.h new file mode 100644 index 00000000..47443368 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.h @@ -0,0 +1,50 @@ +// +// STDSSwiftTryCatch.h +// +// Created by William Falcon on 10/10/14. +// Copyright (c) 2014 William Falcon. All rights reserved. +// +/* + 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. + */ + +#import + +NS_ASSUME_NONNULL_BEGIN + +/** + Provides try catch functionality for swift by wrapping around Objective-C + */ +@interface STDSSwiftTryCatch : NSObject + +/** + Provides try catch functionality for swift by wrapping around Objective-C + */ ++ (void)tryBlock:(void(^)(void))tryBlock catchBlock:(void(^)(NSException*exception))catchBlock finallyBlock:(void(^)(void))finallyBlock; +/** + Throws Objective-C exception with name and reason set to `s` + */ ++ (void)throwString:(NSString*)s; + +/** + Throws exception `e` + */ ++ (void)throwException:(NSException*)e; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.m b/Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.m new file mode 100644 index 00000000..a9ac8601 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSSwiftTryCatch.m @@ -0,0 +1,59 @@ +// +// STDSSwiftTryCatch.h +// +// Created by William Falcon on 10/10/14. +// Copyright (c) 2014 William Falcon. All rights reserved. +// +/* + 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. + */ + +#import "STDSSwiftTryCatch.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSSwiftTryCatch + ++ (void)tryBlock:(void(^)(void))tryBlock catchBlock:(void(^)(NSException*exception))catchBlock finallyBlock:(void(^)(void))finallyBlock { + @try { + tryBlock ? tryBlock() : nil; + } + + @catch (NSException *exception) { + catchBlock ? catchBlock(exception) : nil; + } + @finally { + finallyBlock ? finallyBlock() : nil; + } +} + ++ (void)throwString:(NSString*)s +{ + @throw [NSException exceptionWithName:s reason:s userInfo:nil]; +} + ++ (void)throwException:(NSException*)e +{ + @throw e; +} + + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.h new file mode 100644 index 00000000..6f2bfede --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.h @@ -0,0 +1,47 @@ +// +// STDSTextFieldCustomization.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + A customization object to use to configure the UI of a text field. + + The font and textColor inherited from `STDSCustomization` configure + the user input text. + */ +@interface STDSTextFieldCustomization : STDSCustomization + +/** + The default settings. + + The default textColor is black. + */ ++ (instancetype)defaultSettings; + +/// The border width of the text field. Defaults to 2. +@property (nonatomic) CGFloat borderWidth; + +/// The color of the border of the text field. Defaults to clear. +@property (nonatomic) UIColor *borderColor; + +/// The corner radius of the edges of the text field. Defaults to 8. +@property (nonatomic) CGFloat cornerRadius; + +/// The appearance of the keyboard. Defaults to UIKeyboardAppearanceDefault. +@property (nonatomic) UIKeyboardAppearance keyboardAppearance; + +/// The color of the placeholder text. Defaults to light gray. +@property (nonatomic) UIColor *placeholderTextColor; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.m new file mode 100644 index 00000000..18a85c8d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSTextFieldCustomization.m @@ -0,0 +1,50 @@ +// +// STDSTextFieldCustomization.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSTextFieldCustomization.h" + +#import "UIFont+DefaultFonts.h" +#import "UIColor+ThirteenSupport.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSTextFieldCustomization + ++ (instancetype)defaultSettings { + return [self new]; +} + +- (instancetype)init { + self = [super init]; + if (self) { + self.font = [UIFont _stds_defaultLabelTextFontWithScale:(CGFloat)1.9]; + _borderWidth = 2; + _cornerRadius = 8; + _keyboardAppearance = UIKeyboardAppearanceDefault; + + self.textColor = UIColor._stds_labelColor; + _borderColor = UIColor.clearColor; + _placeholderTextColor = UIColor._stds_systemGray2Color; + } + return self; +} + +- (id)copyWithZone:(nullable NSZone *)zone { + STDSTextFieldCustomization *copy = [super copyWithZone:zone]; + copy.borderWidth = self.borderWidth; + copy.borderColor = self.borderColor; + copy.cornerRadius = self.cornerRadius; + copy.keyboardAppearance = self.keyboardAppearance; + copy.placeholderTextColor = self.placeholderTextColor; + + return copy; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.h b/Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.h new file mode 100644 index 00000000..c0baa55f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.h @@ -0,0 +1,84 @@ +// +// STDSThreeDS2Service.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@class STDSConfigParameters; +@class STDSTransaction; +@class STDSUICustomization; +@class STDSWarning; + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSThreeDS2Service` is the main 3DS SDK interface and provides methods to process transactions. + */ +@interface STDSThreeDS2Service : NSObject + +/** + A list of warnings that may be populated once the SDK has been initialized. + */ +@property (nonatomic, readonly, nullable) NSArray *warnings; + +/** + Initializes the 3DS SDK instance. + + This method should be called at the start of the payment stage of a transaction. + **Note: Until the `STDSThreeDS2Service instance is initialized, it will be unusable.** + + - Performs security checks + - Collects device information + + @param config Configuration information that will be used during initialization. @see STDSConfigParameters + @param locale Optional override for the locale to use in UI. If `nil`, will default to the current system locale. + @param uiSettings Optional custom UI settings. If `nil`, will default to `[STDSUICustomization defaultSettings]`. + This argument is copied; any further changes to the customization object have no effect. @see STDSUICustomization + + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `config` is `nil` or any of `config`, `locale`, or `uiSettings` are invalid. @see STDSInvalidInputException + @exception STDSAlreadyInitializedException Will throw an `STDSAlreadyInitializedException` if the 3DS SDK instance has already been initialized. @see STDSSDKAlreadyInitializedException + @exception STDSRuntimeException Will throw an `STDSRuntimeException` if there is an internal error in the SDK. @see STDSRuntimeException + */ +- (void)initializeWithConfig:(STDSConfigParameters *)config + locale:(nullable NSLocale *)locale + uiSettings:(nullable STDSUICustomization *)uiSettings; + +/** + Creates and returns an instance of `STDSTransaction`. + + @param directoryServerID The Directory Server identifier returned in the authentication response + @param protocolVersion 3DS protocol version according to which the transaction will be created. Uses the default value of 2.1.0 if nil + + @exception STDSNotInitializedException Will throw an `STDSNotInitializedException` if the the `STDSThreeDS2Service` instance hasn't been initialized with a call to `initializeWithConfig:locale:uiSettings:`. @see STDSNotInitializedException + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if `directoryServerID` is not recognized or if the `protocolVersion` is not supported by this version of the SDK. @see STDSInvalidInputException + @exception STDSRuntimeException Will throw an `STDSRuntimeException` if there is an internal error in the SDK. @see STDSRuntimeException + */ +- (STDSTransaction *)createTransactionForDirectoryServer:(NSString *)directoryServerID + withProtocolVersion:(nullable NSString *)protocolVersion; + +/** + Creates and returns an instance of `STDSTransaction` using a custom directory server certificate. + Will return nil if unable to create a certificate from the provided params. + + @param directoryServerID The Directory Server identifier returned in the authentication response + @param serverKeyID An additional authentication key used by some Directory Servers + @param certificateString A Base64-encoded PEM or DER formatted certificate string containing the directory server's public key + @param rootCertificateStrings An arry of base64-encoded PEM or DER formatted certificate strings containing the DS root certificate used for signature checks + @param protocolVersion 3DS protocol version according to which the transaction will be created. Uses the default value of 2.1.0 if nil + + @exception STDSNotInitializedException Will throw an `STDSNotInitializedException` if the the `STDSThreeDS2Service` instance hasn't been initialized with a call to `initializeWithConfig:locale:uiSettings:`. @see STDSNotInitializedException + @exception STDSInvalidInputException Will throw an `STDSInvalidInputException` if the `protocolVersion` is not supported by this version of the SDK. @see STDSInvalidInputException + */ +- (nullable STDSTransaction *)createTransactionForDirectoryServer:(NSString *)directoryServerID + serverKeyID:(nullable NSString *)serverKeyID + certificateString:(NSString *)certificateString + rootCertificateStrings:(NSArray *)rootCertificateStrings + withProtocolVersion:(nullable NSString *)protocolVersion; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.m b/Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.m new file mode 100644 index 00000000..2c45de66 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSThreeDS2Service.m @@ -0,0 +1,191 @@ +// +// STDSThreeDS2Service.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSThreeDS2Service.h" + +#include + +#import "STDSAlreadyInitializedException.h" +#import "STDSConfigParameters.h" +#import "STDSDebuggerChecker.h" +#import "STDSDeviceInformationManager.h" +#import "STDSDirectoryServerCertificate.h" +#import "STDSException+Internal.h" +#import "STDSInvalidInputException.h" +#import "STDSLocalizedString.h" +#import "STDSJailbreakChecker.h" +#import "STDSIntegrityChecker.h" +#import "STDSNotInitializedException.h" +#import "STDSOSVersionChecker.h" +#import "STDSSecTypeUtilities.h" +#import "STDSSimulatorChecker.h" +#import "STDSThreeDSProtocolVersion.h" +#import "STDSTransaction+Private.h" +#import "STDSWarning.h" + +static const int kServiceNotInitialized = 0; +static const int kServiceInitialized = 1; + +static NSString * const kInternalStripeTestingConfigParam = @"kInternalStripeTestingConfigParam"; +static NSString * const kIgnoreDeviceInformationRestrictionsParam = @"kIgnoreDeviceInformationRestrictionsParam"; +static NSString * const kUseULTestLOAParam = @"kUseULTestLOAParam"; + +@implementation STDSThreeDS2Service +{ + atomic_int _initialized; + + STDSDeviceInformation *_deviceInformation; + STDSUICustomization *_uiSettings; + STDSConfigParameters *_configuration; +} + +@synthesize warnings = _warnings; + +- (void)initializeWithConfig:(STDSConfigParameters *)config + locale:(nullable NSLocale *)locale + uiSettings:(nullable STDSUICustomization *)uiSettings { + if (config == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:[NSString stringWithFormat:@"%@ config parameter must be non-nil.", NSStringFromSelector(_cmd)]]; + } + + int notInitialized = kServiceNotInitialized; // Can't pass a const to atomic_compare_exchange_strong_explicit so copy here + if (!atomic_compare_exchange_strong_explicit(&_initialized, ¬Initialized, kServiceInitialized, memory_order_release, memory_order_relaxed)) { + @throw [STDSAlreadyInitializedException exceptionWithMessage:[NSString stringWithFormat:@"STDSThreeDS2Service instance %p has already been initialized.", self]]; + } + + _configuration = config; + _uiSettings = uiSettings ? [uiSettings copy] : [STDSUICustomization defaultSettings]; + + NSMutableArray *warnings = [NSMutableArray array]; + if ([STDSJailbreakChecker isJailbroken]) { + STDSWarning *jailbrokenWarning = [[STDSWarning alloc] initWithIdentifier:@"SW01" message:STDSLocalizedString(@"The device is jailbroken.", @"The text for warning when a device is jailbroken") severity:STDSWarningSeverityHigh]; + [warnings addObject:jailbrokenWarning]; + } + + if (![STDSIntegrityChecker SDKIntegrityIsValid]) { + STDSWarning *integrityWarning = [[STDSWarning alloc] initWithIdentifier:@"SW02" message:STDSLocalizedString(@"The integrity of the SDK has been tampered.", @"The text for warning when the integrity of the SDK has been tampered with") severity:STDSWarningSeverityHigh]; + [warnings addObject:integrityWarning]; + } + + if ([STDSSimulatorChecker isRunningOnSimulator]) { + STDSWarning *simulatorWarning = [[STDSWarning alloc] initWithIdentifier:@"SW03" message:STDSLocalizedString(@"An emulator is being used to run the App.", @"The text for warning when an emulator is being used to run the application.") severity:STDSWarningSeverityHigh]; + [warnings addObject:simulatorWarning]; + } + + if ([STDSDebuggerChecker processIsCurrentlyAttachedToDebugger]) { + STDSWarning *debuggerWarning = [[STDSWarning alloc] initWithIdentifier:@"SW04" message:STDSLocalizedString(@"A debugger is attached to the App.", @"The text for warning when a debugger is currently attached to the process.") severity:STDSWarningSeverityMedium]; + [warnings addObject:debuggerWarning]; + } + + if (![STDSOSVersionChecker isSupportedOSVersion]) { + STDSWarning *versionWarning = [[STDSWarning alloc] initWithIdentifier:@"SW05" message:STDSLocalizedString(@"The OS or the OS Version is not supported.", "The text for warning when the SDK is running on an unsupported OS or OS version.") severity:STDSWarningSeverityHigh]; + [warnings addObject:versionWarning]; + } + + _warnings = [warnings copy]; + + _deviceInformation = [STDSDeviceInformationManager deviceInformationWithWarnings:_warnings + ignoringRestrictions:[[_configuration parameterValue:kIgnoreDeviceInformationRestrictionsParam] isEqualToString:@"Y"]]; + +} + +- (STDSTransaction *)createTransactionForDirectoryServer:(NSString *)directoryServerID + withProtocolVersion:(nullable NSString *)protocolVersion { + if (_initialized != kServiceInitialized) { + @throw [STDSNotInitializedException exceptionWithMessage:@"STDSThreeDS2Service instance %p has not been initialized before call to %@", self, NSStringFromSelector(_cmd)]; + } + + if (directoryServerID == nil) { + @throw [STDSInvalidInputException exceptionWithMessage:@"%@ directoryServerID parameter must be non-nil.", NSStringFromSelector(_cmd)]; + } + + STDSDirectoryServer directoryServer = STDSDirectoryServerForID(directoryServerID); + if (directoryServer == STDSDirectoryServerUnknown) { + if ([[_configuration parameterValue:kInternalStripeTestingConfigParam] isEqualToString:@"Y"]) { + directoryServer = STDSDirectoryServerULTestRSA; + } else { + @throw [STDSInvalidInputException exceptionWithMessage:@"%@ is an invalid directoryServerID value", directoryServerID]; + } + } + + if (protocolVersion != nil && ![self _supportsProtocolVersion:protocolVersion]) { + @throw [STDSInvalidInputException exceptionWithMessage:@"3DS2 Protocol Version %@ is not supported by this SDK", protocolVersion]; + } + + + + STDSTransaction *transaction = [[STDSTransaction alloc] initWithDeviceInformation:_deviceInformation + directoryServer:directoryServer + protocolVersion:(protocolVersion != nil) ? STDSThreeDSProtocolVersionForString(protocolVersion) : STDSThreeDSProtocolVersion2_1_0 + uiCustomization:_uiSettings]; + transaction.bypassTestModeVerification = [[_configuration parameterValue:kInternalStripeTestingConfigParam] isEqualToString:@"Y"]; + transaction.useULTestLOA = [[_configuration parameterValue:kUseULTestLOAParam] isEqualToString:@"Y"]; + return transaction; + +} + +- (nullable STDSTransaction *)createTransactionForDirectoryServer:(NSString *)directoryServerID + serverKeyID:(nullable NSString *)serverKeyID + certificateString:(NSString *)certificateString + rootCertificateStrings:(NSArray *)rootCertificateStrings + withProtocolVersion:(nullable NSString *)protocolVersion { + if (_initialized != kServiceInitialized) { + @throw [STDSNotInitializedException exceptionWithMessage:@"STDSThreeDS2Service instance %p has not been initialized before call to %@", self, NSStringFromSelector(_cmd)]; + } + + if (protocolVersion != nil && ![self _supportsProtocolVersion:protocolVersion]) { + @throw [STDSInvalidInputException exceptionWithMessage:@"3DS2 Protocol Version %@ is not supported by this SDK", protocolVersion]; + } + + STDSTransaction *transaction = nil; + + STDSDirectoryServerCertificate *certificate = [STDSDirectoryServerCertificate customCertificateWithString:certificateString]; + + if (certificate != nil) { + transaction = [[STDSTransaction alloc] initWithDeviceInformation:_deviceInformation + directoryServerID:directoryServerID + serverKeyID:serverKeyID + directoryServerCertificate:certificate + rootCertificateStrings:rootCertificateStrings + protocolVersion:(protocolVersion != nil) ? STDSThreeDSProtocolVersionForString(protocolVersion) : STDSThreeDSProtocolVersion2_1_0 + uiCustomization:_uiSettings]; + transaction.bypassTestModeVerification = [_configuration parameterValue:kInternalStripeTestingConfigParam] != nil; + } + + return transaction; +} + +- (nullable NSArray *)warnings { + if (_initialized != kServiceInitialized) { + @throw [STDSNotInitializedException exceptionWithMessage:@"STDSThreeDS2Service instance %p has not been initialized before call to %@", self, NSStringFromSelector(_cmd)]; + } + + return _warnings; +} + +#pragma mark - Internal + +- (BOOL)_supportsProtocolVersion:(NSString *)protocolVersion { + STDSThreeDSProtocolVersion version = STDSThreeDSProtocolVersionForString(protocolVersion); + switch (version) { + case STDSThreeDSProtocolVersion2_1_0: + return YES; + + case STDSThreeDSProtocolVersion2_2_0: + return YES; + + case STDSThreeDSProtocolVersionFallbackTest: + // only support fallback test if we have the internal testing config param + return [[_configuration parameterValue:kInternalStripeTestingConfigParam] isEqualToString:@"Y"]; + + case STDSThreeDSProtocolVersionUnknown: + return NO; + } +} + +@end diff --git a/Stripe3DS2/Stripe3DS2/include/STDSThreeDSProtocolVersion.h b/Stripe3DS2/Stripe3DS2/include/STDSThreeDSProtocolVersion.h new file mode 100644 index 00000000..b4161a6e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSThreeDSProtocolVersion.h @@ -0,0 +1,15 @@ +// +// STDSThreeDSProtocolVersion.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 6/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +FOUNDATION_EXPORT NSString * const Stripe3DS2ProtocolVersion; + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSTransaction.h b/Stripe3DS2/Stripe3DS2/include/STDSTransaction.h new file mode 100644 index 00000000..02a1f1af --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSTransaction.h @@ -0,0 +1,94 @@ +// +// STDSTransaction.h +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import + +typedef void (^STDSTransactionVoidBlock)(void); + +@class STDSAuthenticationRequestParameters, STDSChallengeParameters; +@protocol STDSChallengeStatusReceiver; + +NS_ASSUME_NONNULL_BEGIN + +/** + `STDSTransaction` holds parameters that the 3DS Server requires to create AReq messages and to perform the Challenge Flow. + */ +@interface STDSTransaction : NSObject + +/** + The UI type of the presented challenge for this transaction if applicable. Will be one of + "none" + "text" + "single_select" + "multi_select" + "oob" + "html" + */ +@property (nonatomic, readonly, copy) NSString *presentedChallengeUIType; + +/** + Encrypts device information collected during initialization and returns it along with SDK details. + + @return Encrypted device information and details about this SDK. @see STDSAuthenticationRequestParameters + + @exception SDKRuntimeException Thrown if an internal error is encountered. + */ +- (STDSAuthenticationRequestParameters *)createAuthenticationRequestParameters; + +/** + Returns a UIViewController instance displaying the Directory Server logo and a spinner. Present this during the Authentication Request/Response. + */ +- (UIViewController *)createProgressViewControllerWithDidCancel:(STDSTransactionVoidBlock)didCancel; + +/** + Initiates the challenge process, displaying challenge UI as needed. + + @param presentingViewController The UIViewController used to present the challenge response UIViewController + @param challengeParameters Details required to conduct the challenge process. @see STDSChallengeParameters + @param challengeStatusReceiver A callback object to receive the status of the challenge. See @STDSChallengeStatusReceiver + @param timeout An interval in seconds within which the challenge process will finish. Must be at least 5 minutes. + + @exception STDSInvalidInputException Thrown if an argument is invalid (e.g. timeout less than 5 minutes). @see STDSInvalidInputException + @exception STDSSDKRuntimeException Thrown if an internal error is encountered, and if you call this method after calling `close`. @see SDKRuntimeException + + @note challengeStatusReceiver must conform to . This is a workaround: When the static Stripe3DS2 is compiled into Stripe.framework, the resulting swiftinterface and generated .h files reference this protocol. To allow users to build without including Stripe3DS2 directly, we'll take an `id` here instead. + */ +- (void)doChallengeWithViewController:(UIViewController *)presentingViewController + challengeParameters:(STDSChallengeParameters *)challengeParameters + challengeStatusReceiver:(id)challengeStatusReceiver + timeout:(NSTimeInterval)timeout; + +/** + Returns the version of the Stripe3DS2 SDK, e.g. @"1.0" + */ +- (NSString *)sdkVersion; + +/** +Cleans up resources held by `STDSTransaction`. Call this when the transaction is completed, if `doChallengeWithChallengeParameters:challengeStatusReceiver:timeout` is not called. + + @note Don't use this object after calling this method. Calling `doChallengeWithViewController:challengeParameters:challengeStatusReceiver:timeout` after calling this method will throw an `STDSSDKRuntimeException` + */ +- (void)close; + +/** + Alternate challenge initiation method meant only for internal use by Stripe SDK. + */ +- (void)doChallengeWithChallengeParameters:(STDSChallengeParameters *)challengeParameters + challengeStatusReceiver:(id)challengeStatusReceiver + timeout:(NSTimeInterval)timeout + presentationBlock:(void (^)(UIViewController *, void(^)(void)))presentationBlock; + +/** + Function to manually cancel the challenge flow. + */ +- (void)cancelChallengeFlow; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSTransaction.m b/Stripe3DS2/Stripe3DS2/include/STDSTransaction.m new file mode 100644 index 00000000..654efa64 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSTransaction.m @@ -0,0 +1,531 @@ +// +// STDSTransaction.m +// Stripe3DS2 +// +// Created by Yuki Tokuhiro on 3/21/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSTransaction.h" +#import "STDSTransaction+Private.h" + +#import "STDSBundleLocator.h" +#import "NSDictionary+DecodingHelpers.h" +#import "NSError+Stripe3DS2.h" +#import "NSString+JWEHelpers.h" +#import "STDSACSNetworkingManager.h" +#import "STDSAuthenticationRequestParameters.h" +#import "STDSChallengeRequestParameters.h" +#import "STDSCompletionEvent.h" +#import "STDSChallengeParameters.h" +#import "STDSChallengeResponseObject.h" +#import "STDSChallengeResponseViewController.h" +#import "STDSChallengeStatusReceiver.h" +#import "STDSDeviceInformation.h" +#import "STDSEllipticCurvePoint.h" +#import "STDSEphemeralKeyPair.h" +#import "STDSErrorMessage+Internal.h" +#import "STDSException+Internal.h" +#import "STDSInvalidInputException.h" +#import "STDSJSONWebEncryption.h" +#import "STDSJSONWebSignature.h" +#import "STDSProgressViewController.h" +#import "STDSProtocolErrorEvent.h" +#import "STDSRuntimeErrorEvent.h" +#import "STDSRuntimeException.h" +#import "STDSSecTypeUtilities.h" +#import "STDSStripe3DS2Error.h" +#import "STDSDeviceInformationParameter.h" + +static const NSTimeInterval kMinimumTimeout = 5 * 60; +static NSString * const kStripeLOA = @"3DS_LOA_SDK_STIN_020100_00162"; +static NSString * const kULTestLOA = @"3DS_LOA_SDK_PPFU_020100_00007"; + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSTransaction() + +@property (nonatomic, weak) id challengeStatusReceiver; +@property (nonatomic, strong, nullable) STDSChallengeResponseViewController *challengeResponseViewController; +/// Stores the most recent parameters used to make a CReq +@property (nonatomic, nullable) STDSChallengeRequestParameters *challengeRequestParameters; +/// YES if `close` was called or the challenge flow finished. +@property (nonatomic, getter=isCompleted) BOOL completed; +@end + +@implementation STDSTransaction +{ + STDSDeviceInformation *_deviceInformation; + STDSDirectoryServer _directoryServer; + STDSEphemeralKeyPair *_ephemeralKeyPair; + STDSThreeDSProtocolVersion _protocolVersion; + NSString *_identifier; + + STDSDirectoryServerCertificate *_customDirectoryServerCertificate; + NSArray *_rootCertificateStrings; + NSString *_customDirectoryServerID; + NSString *_serverKeyID; + + STDSACSNetworkingManager *_networkingManager; + + STDSUICustomization *_uiCustomization; +} + +- (instancetype)initWithDeviceInformation:(STDSDeviceInformation *)deviceInformation + directoryServer:(STDSDirectoryServer)directoryServer + protocolVersion:(STDSThreeDSProtocolVersion)protocolVersion + uiCustomization:(nonnull STDSUICustomization *)uiCustomization { + self = [super init]; + if (self) { + _deviceInformation = deviceInformation; + _directoryServer = directoryServer; + _protocolVersion = protocolVersion; + _completed = NO; + _identifier = [NSUUID UUID].UUIDString.lowercaseString; + _ephemeralKeyPair = [STDSEphemeralKeyPair ephemeralKeyPair]; + _uiCustomization = uiCustomization; + if (_ephemeralKeyPair == nil) { + return nil; + } + } + + return self; +} + +- (instancetype)initWithDeviceInformation:(STDSDeviceInformation *)deviceInformation + directoryServerID:(NSString *)directoryServerID + serverKeyID:(nullable NSString *)serverKeyID + directoryServerCertificate:(STDSDirectoryServerCertificate *)directoryServerCertificate + rootCertificateStrings:(NSArray *)rootCertificateStrings + protocolVersion:(STDSThreeDSProtocolVersion)protocolVersion + uiCustomization:(STDSUICustomization *)uiCustomization { + self = [super init]; + if (self) { + _deviceInformation = deviceInformation; + _directoryServer = STDSDirectoryServerCustom; + _customDirectoryServerCertificate = directoryServerCertificate; + _rootCertificateStrings = rootCertificateStrings; + _customDirectoryServerID = [directoryServerID copy]; + _serverKeyID = [serverKeyID copy]; + _protocolVersion = protocolVersion; + _completed = NO; + _identifier = [NSUUID UUID].UUIDString.lowercaseString; + _ephemeralKeyPair = [STDSEphemeralKeyPair ephemeralKeyPair]; + _uiCustomization = uiCustomization; + if (_ephemeralKeyPair == nil) { + return nil; + } + } + + return self; +} + +- (NSString *)sdkVersion { + return [[STDSBundleLocator stdsResourcesBundle] objectForInfoDictionaryKey:@"CFBundleVersion"]; +} + +- (NSString *)presentedChallengeUIType { + switch (self.challengeResponseViewController.response.acsUIType) { + + case STDSACSUITypeNone: + return @"none"; + + case STDSACSUITypeText: + return @"text"; + + case STDSACSUITypeSingleSelect: + return @"single_select"; + + case STDSACSUITypeMultiSelect: + return @"multi_select"; + + case STDSACSUITypeOOB: + return @"oob"; + + case STDSACSUITypeHTML: + return @"html"; + } +} + +- (STDSAuthenticationRequestParameters *)createAuthenticationRequestParameters { + NSError *error = nil; + NSString *encryptedDeviceData = nil; + + if (_directoryServer == STDSDirectoryServerCustom) { + encryptedDeviceData = [STDSJSONWebEncryption encryptJSON:_deviceInformation.dictionaryValue + withCertificate:_customDirectoryServerCertificate + directoryServerID:_customDirectoryServerID + serverKeyID:_serverKeyID + error:&error]; + } else { + encryptedDeviceData = [STDSJSONWebEncryption encryptJSON:_deviceInformation.dictionaryValue + forDirectoryServer:_directoryServer + error:&error]; + } + if (encryptedDeviceData == nil) { + @throw [STDSRuntimeException exceptionWithMessage:@"Error encrypting device information %@", error]; + } + + return [[STDSAuthenticationRequestParameters alloc] initWithSDKTransactionIdentifier:_identifier + deviceData:encryptedDeviceData + sdkEphemeralPublicKey:_ephemeralKeyPair.publicKeyJWK + sdkAppIdentifier:[STDSDeviceInformationParameter sdkAppIdentifier] + sdkReferenceNumber:self.useULTestLOA ? kULTestLOA : kStripeLOA + messageVersion:[self _messageVersion]]; +} + +- (UIViewController *)createProgressViewControllerWithDidCancel:(void (^)(void))didCancel { + return [[STDSProgressViewController alloc] initWithDirectoryServer:[self _directoryServerForUI] uiCustomization:_uiCustomization didCancel:didCancel]; +} + +- (void)doChallengeWithViewController:(UIViewController *)presentingViewController + challengeParameters:(STDSChallengeParameters *)challengeParameters + challengeStatusReceiver:(id)challengeStatusReceiver + timeout:(NSTimeInterval)timeout { + + [self doChallengeWithChallengeParameters:challengeParameters + challengeStatusReceiver:challengeStatusReceiver + timeout:timeout + presentationBlock:^(UIViewController * _Nonnull challengeVC, void (^completion)(void)) { + UINavigationController *navigationController = [[UINavigationController alloc] initWithRootViewController:challengeVC]; + + // Disable "swipe to dismiss" behavior in iOS 13 + if ([navigationController respondsToSelector:NSSelectorFromString(@"isModalInPresentation")]) { + [navigationController setValue:@YES forKey:@"modalInPresentation"]; + } + + [presentingViewController presentViewController:navigationController animated:YES completion:^{ + completion(); + }]; + }]; +} + +- (void)doChallengeWithChallengeParameters:(STDSChallengeParameters *)challengeParameters + challengeStatusReceiver:(id)challengeStatusReceiver + timeout:(NSTimeInterval)timeout + presentationBlock:(void (^)(UIViewController *, void(^)(void)))presentationBlock { + if (self.isCompleted) { + @throw [STDSRuntimeException exceptionWithMessage:@"The transaction has already completed."]; + } else if (timeout < kMinimumTimeout) { + @throw [STDSInvalidInputException exceptionWithMessage:@"Timeout value of %lf seconds is less than 5 minutes", timeout]; + } + self.challengeStatusReceiver = challengeStatusReceiver; + self.timeoutTimer = [NSTimer scheduledTimerWithTimeInterval:(timeout) target:self selector:@selector(_didTimeout) userInfo:nil repeats:NO]; + + self.challengeRequestParameters = [[STDSChallengeRequestParameters alloc] initWithChallengeParameters:challengeParameters + transactionIdentifier:_identifier + messageVersion:[self _messageVersion]]; + + STDSJSONWebSignature *jws = [[STDSJSONWebSignature alloc] initWithString:challengeParameters.acsSignedContent allowNilKey:self.bypassTestModeVerification]; + BOOL validJWS = jws != nil; + if (validJWS && !self.bypassTestModeVerification) { + if (_customDirectoryServerCertificate != nil) { + if (_rootCertificateStrings.count == 0) { + validJWS = NO; + } else { + validJWS = [STDSJSONWebEncryption verifyJSONWebSignature:jws withCertificate:_customDirectoryServerCertificate rootCertificates:_rootCertificateStrings]; + } + } else { + validJWS = [STDSJSONWebEncryption verifyJSONWebSignature:jws forDirectoryServer:_directoryServer]; + } + } + if (!validJWS) { + dispatch_async(dispatch_get_main_queue(), ^{ + [challengeStatusReceiver transaction:self + didErrorWithRuntimeErrorEvent:[[STDSRuntimeErrorEvent alloc] initWithErrorCode:kSTDSRuntimeErrorCodeEncryptionError errorMessage:@"Error verifying JWS response."]]; + [self _cleanUp]; + }); + return; + } + + NSError *jsonError = nil; + NSDictionary *json = [NSJSONSerialization JSONObjectWithData:jws.payload options:0 error:&jsonError]; + NSDictionary *acsEphmeralKeyJWK = [json _stds_dictionaryForKey:@"acsEphemPubKey" required:NO error:NULL]; + STDSEllipticCurvePoint *acsEphemeralKey = [[STDSEllipticCurvePoint alloc] initWithJWK:acsEphmeralKeyJWK]; + NSString *acsURLString = [json _stds_stringForKey:@"acsURL" required:NO error:NULL]; + NSURL *acsURL = [NSURL URLWithString:acsURLString ?: @""]; + if (acsEphemeralKey == nil || acsURL == nil) { + dispatch_async(dispatch_get_main_queue(), ^{ + [challengeStatusReceiver transaction:self + didErrorWithRuntimeErrorEvent:[[STDSRuntimeErrorEvent alloc] initWithErrorCode:kSTDSRuntimeErrorCodeParsingError errorMessage:[NSString stringWithFormat:@"Unable to create key or url from ACS json: %@\n\n jsonError: %@", json, jsonError]]]; + [self _cleanUp]; + }); + return; + } + + NSData *ecdhSecret = [_ephemeralKeyPair createSharedSecretWithEllipticCurveKey:acsEphemeralKey]; + if (ecdhSecret == nil) { + dispatch_async(dispatch_get_main_queue(), ^{ + [challengeStatusReceiver transaction:self + didErrorWithRuntimeErrorEvent:[[STDSRuntimeErrorEvent alloc] initWithErrorCode:kSTDSRuntimeErrorCodeEncryptionError errorMessage:@"Error during Diffie-Helman key exchange"]]; + [self _cleanUp]; + }); + return; + } + + NSData *contentEncryptionKeySDKtoACS = STDSCreateConcatKDFWithSHA256(ecdhSecret, 32, self.useULTestLOA ? kULTestLOA : kStripeLOA); + // These two keys are intentionally identical + // ref. Protocol and Core Functions Specification Version 2.2.0 Section 6.2.3.2 & 6.2.3.3 + // "In this version of the specification [contentEncryptionKeyACStoSDK] and [contentEncryptionKeySDKtoACS] + // are extracted with the same value." + NSData *contentEncryptionKeyACStoSDK = [contentEncryptionKeySDKtoACS copy]; + + _networkingManager = [[STDSACSNetworkingManager alloc] initWithURL:acsURL + sdkContentEncryptionKey:contentEncryptionKeySDKtoACS + acsContentEncryptionKey:contentEncryptionKeyACStoSDK + acsTransactionIdentifier:self.challengeRequestParameters.acsTransactionIdentifier]; + // Start the Challenge flow + STDSImageLoader *imageLoader = [[STDSImageLoader alloc] initWithURLSession:NSURLSession.sharedSession]; + self.challengeResponseViewController = [[STDSChallengeResponseViewController alloc] initWithUICustomization:_uiCustomization imageLoader:imageLoader directoryServer:[self _directoryServerForUI]]; + self.challengeResponseViewController.delegate = self; + + presentationBlock(self.challengeResponseViewController, ^{ [self _makeChallengeRequest:self.challengeRequestParameters didCancel:NO]; }); +} + +- (void)close { + [self _cleanUp]; +} + +- (void)cancelChallengeFlow { + [self challengeResponseViewControllerDidCancel:self.challengeResponseViewController]; +} + +- (void)dealloc { + [self _cleanUp]; +} + +#pragma mark - Private + +// When we get a directory certificate and ID from the server, we mark it as Custom for encryption, +// but may have a correct mapping to a DS logo for the UI +- (STDSDirectoryServer)_directoryServerForUI { + return (_customDirectoryServerID != nil) ? STDSDirectoryServerForID(_customDirectoryServerID) : _directoryServer; +} + +- (void)_makeChallengeRequest:(STDSChallengeRequestParameters *)challengeRequestParameters didCancel:(BOOL)didCancel { + [self.challengeResponseViewController setLoading]; + __weak STDSTransaction *weakSelf = self; + [_networkingManager submitChallengeRequest:self.challengeRequestParameters + withCompletion:^(id _Nullable response, NSError * _Nullable error) { + STDSTransaction *strongSelf = weakSelf; + if (strongSelf == nil || strongSelf.isCompleted) { + return; + } + // Parsing or network errors + if (response == nil || error) { + if (!error) { + error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain code:STDSErrorCodeUnknownError userInfo:nil]; + } + [strongSelf _handleError:error]; + return; + } + // Consistency errors (e.g. acsTransID changes) + NSError *validationError; + if (![strongSelf _validateChallengeResponse:response error:&validationError]) { + [strongSelf _handleError:validationError]; + return; + } + [strongSelf _handleChallengeResponse:response didCancel:didCancel]; + }]; +} + +- (BOOL)_validateChallengeResponse:(id)challengeResponse error:(NSError **)outError { + NSError *error; + if (![challengeResponse.acsTransactionID isEqualToString:self.challengeRequestParameters.acsTransactionIdentifier] || + ![challengeResponse.threeDSServerTransactionID isEqualToString:self.challengeRequestParameters.threeDSServerTransactionIdentifier] || + ![challengeResponse.sdkTransactionID isEqualToString:self.challengeRequestParameters.sdkTransactionIdentifier]) { + error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain code:STDSErrorMessageErrorTransactionIDNotRecognized userInfo:nil]; + } else if (![challengeResponse.messageVersion isEqualToString:self.challengeRequestParameters.messageVersion]) { + error = [NSError _stds_invalidJSONFieldError:@"messageVersion"]; + } else if (!self.bypassTestModeVerification && ![challengeResponse.acsCounterACStoSDK isEqualToString:self.challengeRequestParameters.sdkCounterStoA]) { + error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain code:STDSErrorCodeDecryptionVerification userInfo:nil]; + } else if (challengeResponse.acsUIType == STDSACSUITypeHTML && !challengeResponse.acsHTML) { + error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain code:STDSErrorCodeDecryptionVerification userInfo:nil]; + } + + if (error && outError) { + *outError = error; + } + return error == nil; +} + +- (void) _handleError:(NSError *)error { + // All the codes corresponding to errors that we treat as protocol errors (ie send to the ACS and report as an STDSProtocolErrorEvent) + NSSet *protocolErrorCodes = [NSSet setWithArray:@[@(STDSErrorCodeUnknownMessageType), + @(STDSErrorCodeJSONFieldInvalid), + @(STDSErrorCodeJSONFieldMissing), + @(STDSErrorCodeReceivedErrorMessage), + @(STDSErrorMessageErrorTransactionIDNotRecognized), + @(STDSErrorCodeUnrecognizedCriticalMessageExtension), + @(STDSErrorCodeDecryptionVerification)]]; + if (error.domain == STDSStripe3DS2ErrorDomain) { + NSString *sdkTransactionIdentifier = _identifier; + NSString *acsTransactionIdentifier = self.challengeRequestParameters.acsTransactionIdentifier; + NSString *messageVersion = [self _messageVersion]; + STDSErrorMessage *errorMessage; + switch (error.code) { + case STDSErrorCodeReceivedErrorMessage: + errorMessage = error.userInfo[STDSStripe3DS2ErrorMessageErrorKey]; + break; + case STDSErrorCodeUnknownMessageType: + errorMessage = [STDSErrorMessage errorForInvalidMessageWithACSTransactionID:acsTransactionIdentifier messageVersion:messageVersion]; + break; + case STDSErrorCodeJSONFieldInvalid: + errorMessage = [STDSErrorMessage errorForJSONFieldInvalidWithACSTransactionID:acsTransactionIdentifier messageVersion:messageVersion error:error]; + break; + case STDSErrorCodeJSONFieldMissing: + errorMessage = [STDSErrorMessage errorForJSONFieldMissingWithACSTransactionID:acsTransactionIdentifier messageVersion:messageVersion error:error]; + break; + case STDSErrorCodeTimeout: + errorMessage = [STDSErrorMessage errorForTimeoutWithACSTransactionID:acsTransactionIdentifier messageVersion:messageVersion]; + break; + case STDSErrorMessageErrorTransactionIDNotRecognized: + errorMessage = [STDSErrorMessage errorForUnrecognizedIDWithACSTransactionID:acsTransactionIdentifier messageVersion:messageVersion]; + break; + case STDSErrorCodeUnrecognizedCriticalMessageExtension: + errorMessage = [STDSErrorMessage errorForUnrecognizedCriticalMessageExtensionsWithACSTransactionID:acsTransactionIdentifier messageVersion:messageVersion error:error]; + break; + case STDSErrorCodeDecryptionVerification: + errorMessage = [STDSErrorMessage errorForDecryptionErrorWithACSTransactionID:acsTransactionIdentifier messageVersion:messageVersion]; + break; + default: + break; + } + + // Send the ErrorMessage (unless we received one) + if (error.code != STDSErrorCodeReceivedErrorMessage && errorMessage != nil) { + [_networkingManager sendErrorMessage:errorMessage]; + } + + // If it's a protocol error, call back to the challengeStatusReceiver + if ([protocolErrorCodes containsObject:@(error.code)] && errorMessage != nil) { + STDSProtocolErrorEvent *protocolErrorEvent = [[STDSProtocolErrorEvent alloc] initWithSDKTransactionIdentifier:sdkTransactionIdentifier + errorMessage:errorMessage]; + [self.challengeStatusReceiver transaction:self didErrorWithProtocolErrorEvent:protocolErrorEvent]; + } + + } + + if (error.domain != STDSStripe3DS2ErrorDomain || ![protocolErrorCodes containsObject:@(error.code)]) { + // This error is not a protocol error, and therefore a runtime error. + NSString *errorCode = [NSString stringWithFormat:@"%ld", (long)error.code]; + STDSRuntimeErrorEvent *runtimeErrorEvent = [[STDSRuntimeErrorEvent alloc] initWithErrorCode:errorCode errorMessage:error.localizedDescription]; + [self.challengeStatusReceiver transaction:self didErrorWithRuntimeErrorEvent:runtimeErrorEvent]; + } + + [self _dismissChallengeResponseViewController]; + [self _cleanUp]; +} + +- (void)_handleChallengeResponse:(id)challengeResponse didCancel:(BOOL)didCancel { + if (challengeResponse.challengeCompletionIndicator) { + // Final CRes + // We need to pass didCancel to here because we can't distinguish between cancellation and auth failure from the CRes + // (they both result in a transactionStatus of "N") + if (didCancel) { + // We already dismissed the view controller + [self.challengeStatusReceiver transactionDidCancel:self]; + [self _cleanUp]; + } else { + [self _dismissChallengeResponseViewController]; + STDSCompletionEvent *completionEvent = [[STDSCompletionEvent alloc] initWithSDKTransactionIdentifier:_identifier + transactionStatus:challengeResponse.transactionStatus]; + [self.challengeStatusReceiver transaction:self didCompleteChallengeWithCompletionEvent:completionEvent]; + [self _cleanUp]; + } + } else { + [self.challengeResponseViewController setChallengeResponse:challengeResponse animated:YES]; + + if ([self.challengeStatusReceiver respondsToSelector:@selector(transactionDidPresentChallengeScreen:)]) { + [self.challengeStatusReceiver transactionDidPresentChallengeScreen:self]; + } + } +} + +- (void)_cleanUp { + [self.timeoutTimer invalidate]; + self.completed = YES; + self.challengeResponseViewController = nil; + self.challengeStatusReceiver = nil; + _networkingManager = nil; +} + +- (void)_didTimeout { + [self _dismissChallengeResponseViewController]; + [_networkingManager sendErrorMessage:[STDSErrorMessage errorForTimeoutWithACSTransactionID:self.challengeRequestParameters.acsTransactionIdentifier messageVersion:[self _messageVersion]]]; + [self.challengeStatusReceiver transactionDidTimeOut:self]; + [self _cleanUp]; +} + +- (void)_dismissChallengeResponseViewController { + if ([self.challengeStatusReceiver respondsToSelector:@selector(dismissChallengeViewController:forTransaction:)]) { + [self.challengeStatusReceiver dismissChallengeViewController:self.challengeResponseViewController forTransaction:self]; + } else { + [self.challengeResponseViewController dismissViewControllerAnimated:YES completion:nil]; + } +} + +#pragma mark Helpers + +- (nullable NSString *)_messageVersion { + NSString *messageVersion = STDSThreeDSProtocolVersionStringValue(_protocolVersion); + if (messageVersion == nil) { + @throw [STDSRuntimeException exceptionWithMessage:@"Error determining message version."]; + } + return messageVersion; +} + +/// Convenience method to construct a CSV from the names of each STDSChallengeResponseSelectionInfo in the given array +- (NSString *)_csvForChallengeResponseSelectionInfo:(NSArray> *)selectionInfoArray { + NSMutableArray *selectionInfoNames = [NSMutableArray new]; + for (id selectionInfo in selectionInfoArray) { + [selectionInfoNames addObject:selectionInfo.name]; + } + return [selectionInfoNames componentsJoinedByString:@","]; +} + +#pragma mark - STDSChallengeResponseViewController + +- (void)challengeResponseViewController:(nonnull STDSChallengeResponseViewController *)viewController didSubmitInput:(nonnull NSString *)userInput whitelistSelection:(nonnull id)whitelistSelection { + self.challengeRequestParameters = [self.challengeRequestParameters nextChallengeRequestParametersByIncrementCounter]; + self.challengeRequestParameters.challengeDataEntry = userInput; + self.challengeRequestParameters.whitelistingDataEntry = whitelistSelection.name; + [self _makeChallengeRequest:self.challengeRequestParameters didCancel:NO]; +} + +- (void)challengeResponseViewController:(nonnull STDSChallengeResponseViewController *)viewController didSubmitSelection:(nonnull NSArray> *)selection whitelistSelection:(nonnull id)whitelistSelection { + self.challengeRequestParameters = [self.challengeRequestParameters nextChallengeRequestParametersByIncrementCounter]; + self.challengeRequestParameters.challengeDataEntry = [self _csvForChallengeResponseSelectionInfo:selection]; + self.challengeRequestParameters.whitelistingDataEntry = whitelistSelection.name; + [self _makeChallengeRequest:self.challengeRequestParameters didCancel:NO]; +} + +- (void)challengeResponseViewControllerDidOOBContinue:(nonnull STDSChallengeResponseViewController *)viewController whitelistSelection:(nonnull id)whitelistSelection { + self.challengeRequestParameters = [self.challengeRequestParameters nextChallengeRequestParametersByIncrementCounter]; + self.challengeRequestParameters.oobContinue = @(YES); + self.challengeRequestParameters.whitelistingDataEntry = whitelistSelection.name; + [self _makeChallengeRequest:self.challengeRequestParameters didCancel:NO]; +} + +- (void)challengeResponseViewControllerDidCancel:(STDSChallengeResponseViewController *)viewController { + self.challengeRequestParameters = [self.challengeRequestParameters nextChallengeRequestParametersByIncrementCounter]; + self.challengeRequestParameters.challengeCancel = @(STDSChallengeCancelTypeCardholderSelectedCancel); + [self _dismissChallengeResponseViewController]; + [self _makeChallengeRequest:self.challengeRequestParameters didCancel:YES]; +} + +- (void)challengeResponseViewControllerDidRequestResend:(STDSChallengeResponseViewController *)viewController { + self.challengeRequestParameters = [self.challengeRequestParameters nextChallengeRequestParametersByIncrementCounter]; + self.challengeRequestParameters.resendChallenge = @"Y"; + [self _makeChallengeRequest:self.challengeRequestParameters didCancel:NO]; +} + +- (void)challengeResponseViewController:(nonnull STDSChallengeResponseViewController *)viewController didSubmitHTMLForm:(nonnull NSString *)form { + self.challengeRequestParameters = [self.challengeRequestParameters nextChallengeRequestParametersByIncrementCounter]; + self.challengeRequestParameters.challengeHTMLDataEntry = form; + [self _makeChallengeRequest:self.challengeRequestParameters didCancel:NO]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSUICustomization.h b/Stripe3DS2/Stripe3DS2/include/STDSUICustomization.h new file mode 100644 index 00000000..651b22f1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSUICustomization.h @@ -0,0 +1,109 @@ +// +// STDSUICustomization.h +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSCustomization.h" +#import "STDSButtonCustomization.h" +#import "STDSNavigationBarCustomization.h" +#import "STDSLabelCustomization.h" +#import "STDSTextFieldCustomization.h" +#import "STDSFooterCustomization.h" +#import "STDSSelectionCustomization.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + The `STDSUICustomization` provides configuration for UI elements. + + It's important to configure this object appropriately before using it to initialize a + `STDSThreeDS2Service` object. `STDSThreeDS2Service` makes a copy of the customization + settings you provide; it ignores any subsequent changes you make to your `STDSUICustomization` instance. +*/ +@interface STDSUICustomization: NSObject + +/// The default settings. See individual properties for their default values. ++ (instancetype)defaultSettings; + +/** + Provides custom settings for the UINavigationBar of all UIViewControllers the SDK display. + The default is `[STDSNavigationBarCustomization defaultSettings]`. + */ +@property (nonatomic) STDSNavigationBarCustomization *navigationBarCustomization; + +/** + Provides custom settings for labels. + The default is `[STDSLabelCustomization defaultSettings]`. + */ +@property (nonatomic) STDSLabelCustomization *labelCustomization; + +/** + Provides custom settings for text fields. + The default is `[STDSTextFieldCustomization defaultSettings]`. + */ +@property (nonatomic) STDSTextFieldCustomization *textFieldCustomization; + +/** + The primary background color of all UIViewControllers the SDK display. + Defaults to white. + */ +@property (nonatomic) UIColor *backgroundColor; + +/** + The Challenge view displays a footer with additional details. This controls the background color of that view. + Defaults to gray. + */ +@property (nonatomic) STDSFooterCustomization *footerCustomization; + +/** + Sets a given button customization for the specified type. + + @param buttonCustomization The buttom customization to use. + @param buttonType The type of button to use the customization for. + */ +- (void)setButtonCustomization:(STDSButtonCustomization *)buttonCustomization forType:(STDSUICustomizationButtonType)buttonType; + +/** + Retrieves a button customization object for the given button type. + + @param buttonType The button type to retrieve a customization object for. + @return A button customization object, or the default if none was set. + @see STDSButtonCustomization + */ +- (STDSButtonCustomization *)buttonCustomizationForButtonType:(STDSUICustomizationButtonType)buttonType; + +/** + Provides custom settings for radio buttons and checkboxes. + The default is `[STDSSelectionCustomization defaultSettings]`. + */ +@property (nonatomic) STDSSelectionCustomization *selectionCustomization; + + +/** + The preferred status bar style for all UIViewControllers the SDK display. + Defaults to UIStatusBarStyleDefault. + */ +@property (nonatomic) UIStatusBarStyle preferredStatusBarStyle; + +#pragma mark - Progress View + +/** + The style of UIActivityIndicatorViews displayed. + This should contrast with `backgroundColor`. Defaults to regular on iOS 13+, + gray on iOS 10-12. + */ +@property (nonatomic) UIActivityIndicatorViewStyle activityIndicatorViewStyle; + +/** + The style of the UIBlurEffect displayed underneath the UIActivityIndicatorView. + Defaults to UIBlurEffectStyleDefault. + */ +@property (nonatomic) UIBlurEffectStyle blurStyle; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSUICustomization.m b/Stripe3DS2/Stripe3DS2/include/STDSUICustomization.m new file mode 100644 index 00000000..8d1d6812 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSUICustomization.m @@ -0,0 +1,83 @@ +// +// STDSUICustomization.m +// Stripe3DS2 +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSUICustomization.h" +#import "UIColor+ThirteenSupport.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSUICustomization() + +@property (nonatomic, strong) NSMutableDictionary *buttonCustomizationDictionary; + +@end + +@implementation STDSUICustomization + ++ (instancetype)defaultSettings { + return [[STDSUICustomization alloc] init]; +} + +- (instancetype)init { + self = [super init]; + + if (self) { + _buttonCustomizationDictionary = [@{ + @(STDSUICustomizationButtonTypeNext): [STDSButtonCustomization defaultSettingsForButtonType:STDSUICustomizationButtonTypeNext + ], + @(STDSUICustomizationButtonTypeCancel): [STDSButtonCustomization defaultSettingsForButtonType:STDSUICustomizationButtonTypeCancel], + @(STDSUICustomizationButtonTypeResend): [STDSButtonCustomization defaultSettingsForButtonType:STDSUICustomizationButtonTypeResend], + @(STDSUICustomizationButtonTypeSubmit): [STDSButtonCustomization defaultSettingsForButtonType:STDSUICustomizationButtonTypeSubmit], + @(STDSUICustomizationButtonTypeContinue): [STDSButtonCustomization defaultSettingsForButtonType:STDSUICustomizationButtonTypeContinue], + } mutableCopy]; + _navigationBarCustomization = [STDSNavigationBarCustomization defaultSettings]; + _labelCustomization = [STDSLabelCustomization defaultSettings]; + _textFieldCustomization = [STDSTextFieldCustomization defaultSettings]; + _footerCustomization = [STDSFooterCustomization defaultSettings]; + _selectionCustomization = [STDSSelectionCustomization defaultSettings]; + _backgroundColor = UIColor._stds_systemBackgroundColor; + _activityIndicatorViewStyle = UIActivityIndicatorViewStyleMedium; + _blurStyle = UIBlurEffectStyleRegular; + _preferredStatusBarStyle = UIStatusBarStyleDefault; + } + + return self; +} + +- (void)setButtonCustomization:(STDSButtonCustomization *)buttonCustomization forType:(STDSUICustomizationButtonType)buttonType { + self.buttonCustomizationDictionary[@(buttonType)] = buttonCustomization; +} + +- (STDSButtonCustomization *)buttonCustomizationForButtonType:(STDSUICustomizationButtonType)buttonType { + return self.buttonCustomizationDictionary[@(buttonType)]; +} + +#pragma mark - NSCopying + +- (instancetype)copyWithZone:(nullable NSZone *)zone { + STDSUICustomization *copy = [[[self class] allocWithZone:zone] init]; + copy.navigationBarCustomization = [self.navigationBarCustomization copy]; + copy.labelCustomization = [self.labelCustomization copy]; + copy.textFieldCustomization = [self.textFieldCustomization copy]; + NSMutableDictionary *buttonCustomizationDictionary = [NSMutableDictionary new]; + for (NSNumber *buttonCustomization in self.buttonCustomizationDictionary) { + buttonCustomizationDictionary[buttonCustomization] = [self.buttonCustomizationDictionary[buttonCustomization] copy]; + } + copy.buttonCustomizationDictionary = buttonCustomizationDictionary; + copy.footerCustomization = [self.footerCustomization copy]; + copy.selectionCustomization = [self.selectionCustomization copy]; + copy.backgroundColor = self.backgroundColor; + copy.activityIndicatorViewStyle = self.activityIndicatorViewStyle; + copy.blurStyle = self.blurStyle; + copy.preferredStatusBarStyle = self.preferredStatusBarStyle; + return copy; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSWarning.h b/Stripe3DS2/Stripe3DS2/include/STDSWarning.h new file mode 100644 index 00000000..5f09fe70 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSWarning.h @@ -0,0 +1,69 @@ +// +// STDSWarning.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 2/12/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +/** + The `STDSWarningSeverity` enum defines the severity levels of warnings generated + during SDK initialization. @see STDSThreeDS2Service + */ +typedef NS_ENUM(NSInteger, STDSWarningSeverity) { + /** + Low severity + */ + STDSWarningSeverityLow = 0, + + /** + Medium severity + */ + STDSWarningSeverityMedium, + + /** + High severity + */ + STDSWarningSeverityHigh, +}; + +/** + The `STDSWarning` class represents warnings generated by `STDSThreeDS2Service` during + security checks run during initialization. @see STDSThreeDS2Service + */ +@interface STDSWarning : NSObject + +/** + Designated initializer for `STDSWarning`. + */ +- (instancetype)initWithIdentifier:(NSString *)identifier + message:(NSString *)message + severity:(STDSWarningSeverity)severity NS_DESIGNATED_INITIALIZER; + +/** + `STDSWarning` should not be directly initialized. + */ +- (instancetype)init NS_UNAVAILABLE; + +/** + The identifier for this warning instance. + */ +@property (nonatomic, readonly) NSString *identifier; + +/** + The descriptive message for this warning. + */ +@property (nonatomic, readonly) NSString *message; + +/** + The severity of this warning. + */ +@property (nonatomic, readonly) STDSWarningSeverity severity; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/STDSWarning.m b/Stripe3DS2/Stripe3DS2/include/STDSWarning.m new file mode 100644 index 00000000..ad0d6301 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/STDSWarning.m @@ -0,0 +1,30 @@ +// +// STDSWarning.m +// Stripe3DS2 +// +// Created by Cameron Sabol on 2/12/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSWarning.h" + +NS_ASSUME_NONNULL_BEGIN + +@implementation STDSWarning + +- (instancetype)initWithIdentifier:(NSString *)identifier + message:(NSString *)message + severity:(STDSWarningSeverity)severity { + self = [super init]; + if (self) { + _identifier = [identifier copy]; + _message = [message copy]; + _severity = severity; + } + + return self; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2/include/Stripe3DS2-Prefix.pch b/Stripe3DS2/Stripe3DS2/include/Stripe3DS2-Prefix.pch new file mode 100644 index 00000000..888096ff --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/Stripe3DS2-Prefix.pch @@ -0,0 +1,14 @@ +// +// Stripe3DS2-Prefix.pch +// Stripe3DS2 +// +// Created by Cameron Sabol on 4/16/20. +// Copyright © 2020 Stripe. All rights reserved. +// + +#ifndef Stripe3DS2_pch +#define Stripe3DS2_pch + +#import "STDSLocalizedString.h" + +#endif /* Stripe3DS2_pch */ diff --git a/Stripe3DS2/Stripe3DS2/include/Stripe3DS2.h b/Stripe3DS2/Stripe3DS2/include/Stripe3DS2.h new file mode 100644 index 00000000..9e9f3eba --- /dev/null +++ b/Stripe3DS2/Stripe3DS2/include/Stripe3DS2.h @@ -0,0 +1,52 @@ +// +// Stripe3DS2.h +// Stripe3DS2 +// +// Created by Cameron Sabol on 1/16/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +//! Project version number for Stripe3DS2. +FOUNDATION_EXPORT double Stripe3DS2VersionNumber; + +//! Project version string for Stripe3DS2. +FOUNDATION_EXPORT const unsigned char Stripe3DS2VersionString[]; + +#import "STDSConfigParameters.h" +#import "STDSThreeDS2Service.h" +#import "STDSUICustomization.h" +#import "STDSWarning.h" + +#import "STDSAlreadyInitializedException.h" +#import "STDSInvalidInputException.h" +#import "STDSNotInitializedException.h" +#import "STDSRuntimeException.h" + +#import "STDSErrorMessage.h" +#import "STDSProtocolErrorEvent.h" +#import "STDSRuntimeErrorEvent.h" +#import "STDSStripe3DS2Error.h" +#import "STDSThreeDSProtocolVersion.h" + +#import "STDSAuthenticationRequestParameters.h" +#import "STDSAuthenticationResponse.h" +#import "STDSChallengeParameters.h" +#import "STDSChallengeStatusReceiver.h" +#import "STDSCompletionEvent.h" +#import "STDSJSONDecodable.h" +#import "STDSJSONEncoder.h" +#import "STDSTransaction.h" + +#import "STDSButtonCustomization.h" +#import "STDSCustomization.h" +#import "STDSException.h" +#import "STDSFooterCustomization.h" +#import "STDSJSONEncodable.h" +#import "STDSLabelCustomization.h" +#import "STDSNavigationBarCustomization.h" +#import "STDSSelectionCustomization.h" +#import "STDSTextFieldCustomization.h" + +#import "STDSSwiftTryCatch.h" diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Info.plist b/Stripe3DS2/Stripe3DS2DemoUI/Info.plist new file mode 100644 index 00000000..80cd6052 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Info.plist @@ -0,0 +1,45 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + APPL + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + LSRequiresIPhoneOS + + UILaunchStoryboardName + LaunchScreen + UIRequiredDeviceCapabilities + + armv7 + + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UIViewControllerBasedStatusBarAppearance + + + diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Resources/acs_challenge.html b/Stripe3DS2/Stripe3DS2DemoUI/Resources/acs_challenge.html new file mode 100644 index 00000000..4b576e2b --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Resources/acs_challenge.html @@ -0,0 +1,178 @@ + + + + + 3DS - One-Time Passcode - PA + + + + + + +
+ + + + +
+
+

Purchase Authentication

+

We have send you a text message with a code to your registered mobile number ending in ***.

+

You are paying Merchant ABC the amount of $xxx.xx on mm/dd/yy.

+
+ +
+

Enter your code below:

+
+ + +
+
+ + +
+
+
+ + +
+
+
+ + Need some help? +

Help content will be displayed here.

+ +
+
+
+
+ + Learn more about authentication +

Authentication information will be displayed here.

+ +
+
+
+ + +
+ +
+
+ diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.h b/Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.h new file mode 100644 index 00000000..0483838d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.h @@ -0,0 +1,15 @@ +// +// AppDelegate.h +// Stripe3DS2DemoUI +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +@interface AppDelegate : UIResponder + +@property (strong, nonatomic) UIWindow *window; + +@end diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.m b/Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.m new file mode 100644 index 00000000..0cf5fc19 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Sources/AppDelegate.m @@ -0,0 +1,36 @@ +// +// AppDelegate.m +// Stripe3DS2DemoUI +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +@import Stripe3DS2; + +#import "AppDelegate.h" +#import "STDSDemoViewController.h" +#import "STDSImageLoader.h" + +@interface AppDelegate () + +@end + +@implementation AppDelegate + +- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { + + self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds]; + + STDSImageLoader *imageLoader = [[STDSImageLoader alloc] initWithURLSession:NSURLSession.sharedSession]; + STDSDemoViewController *demoViewController = [[STDSDemoViewController alloc] initWithImageLoader:imageLoader]; + UINavigationController *navigationController = [[UINavigationController alloc] initWithRootViewController:demoViewController]; + + self.window.rootViewController = navigationController; + + [self.window makeKeyAndVisible]; + + return YES; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.h b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.h new file mode 100644 index 00000000..3a1ab053 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.h @@ -0,0 +1,23 @@ +// +// STDSChallengeResponseObject+TestObjects.h +// Stripe3DS2DemoUI +// +// Created by Andrew Harrison on 3/7/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeResponseObject.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSChallengeResponseObject (TestObjects) + ++ (id)textChallengeResponseWithWhitelist:(BOOL)whitelist resendCode:(BOOL)resendCode; ++ (id)singleSelectChallengeResponse; ++ (id)multiSelectChallengeResponse; ++ (id)OOBChallengeResponse; ++ (id)HTMLChallengeResponse; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.m b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.m new file mode 100644 index 00000000..934c313d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSChallengeResponseObject+TestObjects.m @@ -0,0 +1,197 @@ +// +// STDSChallengeResponseObject+TestObjects.m +// Stripe3DS2DemoUI +// +// Created by Andrew Harrison on 3/7/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSChallengeResponseObject+TestObjects.h" +#import "STDSChallengeResponseSelectionInfoObject.h" +#import "STDSChallengeResponseImageObject.h" + +@implementation STDSChallengeResponseObject (TestObjects) + ++ (id)textChallengeResponseWithWhitelist:(BOOL)whitelist resendCode:(BOOL)resendCode { + return [[STDSChallengeResponseObject alloc] initWithThreeDSServerTransactionID:@"" + acsCounterACStoSDK:@"" + acsTransactionID:@"" + acsHTML:nil + acsHTMLRefresh:nil + acsUIType:STDSACSUITypeText + challengeCompletionIndicator:NO + challengeInfoHeader:@"Verify by phone" + challengeInfoLabel:@"Enter your 6 digit code:" + challengeInfoText:@"Great! We have sent you a text message with secure code to your registered mobile phone number.\n\nSent to a number ending in •••• •••• 4729." + challengeAdditionalInfoText:nil + showChallengeInfoTextIndicator:NO + challengeSelectInfo:nil + expandInfoLabel:@"Expand Info Label" + expandInfoText:@"This field displays expandable information text provided by the ACS." + issuerImage:[self issuerImage] + messageExtensions:nil + messageVersion:@"" + oobAppURL:nil + oobAppLabel:nil + oobContinueLabel:nil + paymentSystemImage:[self paymentImage] + resendInformationLabel:resendCode ? @"Resend code" : nil + sdkTransactionID:@"" + submitAuthenticationLabel:@"Submit" + whitelistingInfoText:whitelist ? @"Would you like to add this Merchant to your whitelist?" : nil + whyInfoLabel:@"Learn more about authentication" + whyInfoText:@"This is additional information about authentication. You are being provided extra information you wouldn't normally see, because you've tapped on the above label." + transactionStatus:nil]; +} + ++ (id)singleSelectChallengeResponse { + STDSChallengeResponseSelectionInfoObject *infoObject1 = [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"Mobile" value:@"***-***-*321"]; + STDSChallengeResponseSelectionInfoObject *infoObject2 = [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"Email" value:@"a******3@g****.com"]; + + return [[STDSChallengeResponseObject alloc] initWithThreeDSServerTransactionID:@"" + acsCounterACStoSDK:@"" + acsTransactionID:@"" + acsHTML:nil + acsHTMLRefresh:nil + acsUIType:STDSACSUITypeSingleSelect + challengeCompletionIndicator:NO + challengeInfoHeader:@"Payment Security" + challengeInfoLabel:nil + challengeInfoText:@"Hi Steve, your online payment is being secured using Card Network. Please select the location you would like to receive the code from YourBank." + challengeAdditionalInfoText:nil + showChallengeInfoTextIndicator:NO + challengeSelectInfo:@[infoObject1, infoObject2] + expandInfoLabel:@"Need some help?" + expandInfoText:@"You've indicated that you need help! We'd be happy to assist with that, by providing helpful text here that makes sense in context." + issuerImage:nil + messageExtensions:nil + messageVersion:@"" + oobAppURL:nil + oobAppLabel:nil + oobContinueLabel:nil + paymentSystemImage:nil + resendInformationLabel:nil + sdkTransactionID:@"" + submitAuthenticationLabel:@"Next" + whitelistingInfoText:nil + whyInfoLabel:@"Learn more about authentication" + whyInfoText:@"This is additional information about authentication. You are being provided extra information you wouldn't normally see, because you've tapped on the above label." + transactionStatus:nil]; +} + ++ (id)multiSelectChallengeResponse { + STDSChallengeResponseSelectionInfoObject *infoObject1 = [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"Option1" value:@"Chicago, Illinois"]; + STDSChallengeResponseSelectionInfoObject *infoObject2 = [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"Option2" value:@"Portland, Oregon"]; + STDSChallengeResponseSelectionInfoObject *infoObject3 = [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"Option3" value:@"Dallas, Texas"]; + STDSChallengeResponseSelectionInfoObject *infoObject4 = [[STDSChallengeResponseSelectionInfoObject alloc] initWithName:@"Option4" value:@"St Louis, Missouri"]; + + return [[STDSChallengeResponseObject alloc] initWithThreeDSServerTransactionID:@"" + acsCounterACStoSDK:@"" + acsTransactionID:@"" + acsHTML:nil + acsHTMLRefresh:nil + acsUIType:STDSACSUITypeMultiSelect + challengeCompletionIndicator:NO + challengeInfoHeader:@"Payment Security" + challengeInfoLabel:@"Question 2: What cities have you lived in?" + challengeInfoText:@"Please answer 3 security questions from YourBank to complete your payment.\n\nSelect all that apply." + challengeAdditionalInfoText:nil + showChallengeInfoTextIndicator:NO + challengeSelectInfo:@[infoObject1, infoObject2, infoObject3, infoObject4] + expandInfoLabel:nil + expandInfoText:nil + issuerImage:nil + messageExtensions:nil + messageVersion:@"" + oobAppURL:nil + oobAppLabel:nil + oobContinueLabel:nil + paymentSystemImage:nil + resendInformationLabel:nil + sdkTransactionID:@"" + submitAuthenticationLabel:@"Next" + whitelistingInfoText:nil + whyInfoLabel:@"Learn more about authentication" + whyInfoText:@"This is additional information about authentication. You are being provided extra information you wouldn't normally see, because you've tapped on the above label." + transactionStatus:nil]; +} + ++ (id)OOBChallengeResponse { + return [[STDSChallengeResponseObject alloc] initWithThreeDSServerTransactionID:@"" + acsCounterACStoSDK:@"" + acsTransactionID:@"" + acsHTML:nil + acsHTMLRefresh:nil + acsUIType:STDSACSUITypeOOB + challengeCompletionIndicator:NO + challengeInfoHeader:@"Payment Security" + challengeInfoLabel:nil + challengeInfoText:@"For added security, you will be authenticated with YourBank application.\n\nStep 1 - Open your YourBank application directly from your phone and verify this payment.\n\nStep 2 - Tap continue after you have completed authentication with your YourBank application." + challengeAdditionalInfoText:nil + showChallengeInfoTextIndicator:YES + challengeSelectInfo:nil + expandInfoLabel:@"Need some help?" + expandInfoText:@"You've indicated that you need help! We'd be happy to assist with that, by providing helpful text here that makes sense in context." + issuerImage:[self issuerImage] + messageExtensions:nil + messageVersion:@"" + oobAppURL:nil + oobAppLabel:nil + oobContinueLabel:@"Continue" + paymentSystemImage:[self paymentImage] + resendInformationLabel:nil + sdkTransactionID:@"" + submitAuthenticationLabel:nil + whitelistingInfoText:nil + whyInfoLabel:@"Learn more about authentication" + whyInfoText:@"This is additional information about authentication. You are being provided extra information you wouldn't normally see, because you've tapped on the above label." + transactionStatus:nil]; +} + ++ (id)HTMLChallengeResponse { + NSString *htmlFilePath = [[NSBundle mainBundle] pathForResource:@"acs_challenge" ofType:@"html"]; + NSString *html = [NSString stringWithContentsOfFile:htmlFilePath encoding:NSUTF8StringEncoding error:nil]; + return [[STDSChallengeResponseObject alloc] initWithThreeDSServerTransactionID:@"" + acsCounterACStoSDK:@"" + acsTransactionID:@"" + acsHTML:html + acsHTMLRefresh:nil + acsUIType:STDSACSUITypeHTML + challengeCompletionIndicator:NO + challengeInfoHeader:nil + challengeInfoLabel:nil + challengeInfoText:nil + challengeAdditionalInfoText:nil + showChallengeInfoTextIndicator:NO + challengeSelectInfo:nil + expandInfoLabel:nil + expandInfoText:nil + issuerImage:nil + messageExtensions:nil + messageVersion:@"" + oobAppURL:nil + oobAppLabel:nil + oobContinueLabel:nil + paymentSystemImage:nil + resendInformationLabel:nil + sdkTransactionID:@"" + submitAuthenticationLabel:nil + whitelistingInfoText:nil + whyInfoLabel:nil + whyInfoText:nil + transactionStatus:nil]; +} + ++ (id)issuerImage { + return [[STDSChallengeResponseImageObject alloc] initWithMediumDensityURL:[NSURL URLWithString:@"https://via.placeholder.com/150.png?text=150+ISSUER"] + highDensityURL:[NSURL URLWithString:@"https://via.placeholder.com/300.png?text=300+ISSUER"] + extraHighDensityURL:[NSURL URLWithString:@"https://via.placeholder.com/450.png?text=450+ISSUER"]]; +} + ++ (id)paymentImage { + return [[STDSChallengeResponseImageObject alloc] initWithMediumDensityURL:[NSURL URLWithString:@"https://via.placeholder.com/150.png?text=150+PAYMENT"] + highDensityURL:[NSURL URLWithString:@"https://via.placeholder.com/300.png?text=300+PAYMENT"] + extraHighDensityURL:[NSURL URLWithString:@"https://via.placeholder.com/450.png?text=450+PAYMENT"]]; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.h b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.h new file mode 100644 index 00000000..db53f304 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.h @@ -0,0 +1,20 @@ +// +// STDSDemoViewController.h +// Stripe3DS2DemoUI +// +// Created by Andrew Harrison on 3/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSImageLoader.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDemoViewController: UIViewController + +- (instancetype)initWithImageLoader:(STDSImageLoader *)imageLoader; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.m b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.m new file mode 100644 index 00000000..fa4ddeee --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Sources/STDSDemoViewController.m @@ -0,0 +1,225 @@ +// +// STDSDemoViewController.m +// Stripe3DS2DemoUI +// +// Created by Andrew Harrison on 3/11/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSDemoViewController.h" +#import "STDSChallengeResponseViewController.h" +#import "STDSChallengeResponseObject+TestObjects.h" +#import "STDSProgressViewController.h" +#import "STDSStackView.h" +#import "UIView+LayoutSupport.h" +#import "UIColor+ThirteenSupport.h" + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSDemoViewController () + +@property (nonatomic, strong) STDSImageLoader *imageLoader; +@property (nonatomic) BOOL shouldLoadSlowly; +@property (nonatomic) STDSUICustomization *customization; +@property (nonatomic) BOOL isDarkMode; + +@end + +@implementation STDSDemoViewController + +- (instancetype)initWithImageLoader:(STDSImageLoader *)imageLoader { + self = [super initWithNibName:nil bundle:nil]; + + if (self) { + _imageLoader = imageLoader; + _customization = [STDSUICustomization defaultSettings]; + } + + return self; +} + +- (void)viewDidLoad { + [super viewDidLoad]; + + self.view.backgroundColor = [UIColor _stds_systemBackgroundColor]; + + STDSStackView *containerView = [[STDSStackView alloc] initWithAlignment:STDSStackViewLayoutAxisVertical]; + [self.view addSubview:containerView]; + [containerView _stds_pinToSuperviewBounds]; + + NSDictionary *buttonTitleToSelectorMapping = @{ + @"Toggle Dark Mode": NSStringFromSelector(@selector(toggleDarkMode)), + @"Present Text Challenge": NSStringFromSelector(@selector(presentTextChallenge)), + @"Present Text Challenge With Whitelist": NSStringFromSelector(@selector(presentTextChallengeWithWhitelist)), + @"Present Text Challenge With Resend": NSStringFromSelector(@selector(presentTextChallengeWithResendCode)), + @"Present Text Challenge With Whitelist and Resend": NSStringFromSelector(@selector(presentTextChallengeWithResendCodeAndWhitelist)), + @"Present Text Challenge (loads slowly w/ initial progressView)": NSStringFromSelector(@selector(presentTextChallengeLoadsSlowly)), + @"Present Single Select Challenge": NSStringFromSelector(@selector(presentSingleSelectChallenge)), + @"Present Multi Select Challenge": NSStringFromSelector(@selector(presentMultiSelectChallenge)), + @"Present OOB Challenge": NSStringFromSelector(@selector(presentOOBChallenge)), + @"Present HTML Challenge": NSStringFromSelector(@selector(presentHTMLChallenge)), + @"Present Progress View": NSStringFromSelector(@selector(presentProgressView)), + }; + for (NSString *key in [buttonTitleToSelectorMapping keysSortedByValueUsingComparator:^(NSString *a, NSString *b) { + return [a compare:b]; + }]) { + UIButton *button = [UIButton buttonWithType:UIButtonTypeSystem]; + [button addTarget:self action:NSSelectorFromString(buttonTitleToSelectorMapping[key]) forControlEvents:UIControlEventTouchUpInside]; + button.titleLabel.numberOfLines = 0; + button.titleLabel.textAlignment = NSTextAlignmentCenter; + [button setTitle:key forState:UIControlStateNormal]; + [containerView addArrangedSubview:button]; + } + [containerView addArrangedSubview:[UIView new]]; +} + +- (void)toggleDarkMode { + if (self.isDarkMode) { + self.customization = [STDSUICustomization defaultSettings]; + self.isDarkMode = false; + } else { + self.customization = [STDSUICustomization defaultSettings]; + // Navigation bar + self.customization.navigationBarCustomization = [STDSNavigationBarCustomization new]; + self.customization.navigationBarCustomization.headerText = @"Authentication"; + self.customization.navigationBarCustomization.buttonText = @"Nope"; + self.customization.navigationBarCustomization.textColor = UIColor.whiteColor; + self.customization.navigationBarCustomization.barStyle = UIBarStyleBlack; + + // General + self.customization.backgroundColor = UIColor.blackColor; + self.customization.activityIndicatorViewStyle = UIActivityIndicatorViewStyleMedium; + self.customization.preferredStatusBarStyle = UIStatusBarStyleLightContent; + self.customization.footerCustomization = [STDSFooterCustomization new]; + self.customization.footerCustomization.backgroundColor = [UIColor colorWithRed:.08 green:.08 blue:.08 alpha:1]; + self.customization.footerCustomization.headingFont = [UIFont boldSystemFontOfSize:15]; + self.customization.footerCustomization.headingTextColor = [UIColor colorWithRed:.95 green:.95 blue:.95 alpha:1]; + self.customization.footerCustomization.textColor = [UIColor colorWithRed:.90 green:.90 blue:.90 alpha:1]; + + // Cancel button + STDSButtonCustomization *cancelButtonCustomization = [STDSButtonCustomization defaultSettingsForButtonType:STDSUICustomizationButtonTypeCancel]; + cancelButtonCustomization.textColor = UIColor.grayColor; + cancelButtonCustomization.titleStyle = STDSButtonTitleStyleUppercase; + [self.customization setButtonCustomization:cancelButtonCustomization forType:STDSUICustomizationButtonTypeCancel]; + + // Text + self.customization.labelCustomization.headingTextColor = UIColor.whiteColor; + self.customization.labelCustomization.textColor = UIColor.whiteColor; + + // Text field + self.customization.textFieldCustomization.keyboardAppearance = UIKeyboardAppearanceDark; + self.customization.textFieldCustomization.textColor = UIColor.whiteColor; + self.customization.textFieldCustomization.borderColor = UIColor.whiteColor; + + // Radio/Checkbox + self.customization.selectionCustomization.secondarySelectedColor = UIColor.lightGrayColor; + self.customization.selectionCustomization.unselectedBorderColor = UIColor.blackColor; + self.customization.selectionCustomization.unselectedBackgroundColor = UIColor.darkGrayColor; + + self.isDarkMode = true; + } +} + +- (void)presentTextChallenge { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject textChallengeResponseWithWhitelist:NO resendCode:NO]]; +} + +- (void)presentTextChallengeWithWhitelist { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject textChallengeResponseWithWhitelist:YES resendCode:NO]]; +} + +- (void)presentTextChallengeWithResendCode { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject textChallengeResponseWithWhitelist:NO resendCode:YES]]; +} + +- (void)presentTextChallengeWithResendCodeAndWhitelist { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject textChallengeResponseWithWhitelist:YES resendCode:YES]]; +} + +- (void)presentTextChallengeLoadsSlowly { + self.shouldLoadSlowly = YES; + STDSProgressViewController *progressVC = [[STDSProgressViewController alloc] initWithDirectoryServer:STDSDirectoryServerULTestEC uiCustomization:self.customization didCancel:^{}]; + UINavigationController *navigationController = [[UINavigationController alloc] initWithRootViewController:progressVC]; + [self.navigationController presentViewController:navigationController animated:YES completion:nil]; + dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ + [self.navigationController dismissViewControllerAnimated:YES completion:nil]; + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject textChallengeResponseWithWhitelist:NO resendCode:NO]]; + }); + +} + +- (void)presentSingleSelectChallenge { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject singleSelectChallengeResponse]]; +} + +- (void)presentMultiSelectChallenge { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject multiSelectChallengeResponse]]; +} + +- (void)presentOOBChallenge { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject OOBChallengeResponse]]; +} + +- (void)presentHTMLChallenge { + [self presentChallengeForChallengeResponse:[STDSChallengeResponseObject HTMLChallengeResponse]]; +} + +- (void)presentProgressView { + __weak typeof(self) weakSelf = self; + UIViewController *vc = [[STDSProgressViewController alloc] initWithDirectoryServer:STDSDirectoryServerULTestEC uiCustomization:self.customization didCancel:^{ + [weakSelf dismissViewControllerAnimated:YES completion:nil]; + }]; + UINavigationController *navigationController = [[UINavigationController alloc] initWithRootViewController:vc]; + [self.navigationController presentViewController:navigationController animated:YES completion:nil]; +} + +- (void)presentChallengeForChallengeResponse:(id)challengeResponse { + STDSChallengeResponseViewController *challengeResponseViewController = [[STDSChallengeResponseViewController alloc] initWithUICustomization:self.customization imageLoader:self.imageLoader directoryServer:STDSDirectoryServerULTestEC]; + challengeResponseViewController.delegate = self; + + UINavigationController *navigationController = [[UINavigationController alloc] initWithRootViewController:challengeResponseViewController]; + [self.navigationController presentViewController:navigationController animated:YES completion:nil]; + // Simulate what `STDSTransaction` does + [challengeResponseViewController setLoading]; + NSUInteger delay = self.shouldLoadSlowly ? 5 : 0; + dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delay * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ + [challengeResponseViewController setChallengeResponse:challengeResponse animated:YES]; + }); +} + +#pragma mark - STDSChallengeResponseViewControllerDelegate + +- (void)challengeResponseViewController:(nonnull STDSChallengeResponseViewController *)viewController didSubmitHTMLForm:(nonnull NSString *)form { + [self.navigationController dismissViewControllerAnimated:YES completion:nil]; +} + +- (void)challengeResponseViewController:(nonnull STDSChallengeResponseViewController *)viewController didSubmitInput:(nonnull NSString *)userInput whitelistSelection:(nonnull id)whitelistSelection { + [viewController setLoading]; + NSUInteger delay = self.shouldLoadSlowly ? 5 : 0; + self.shouldLoadSlowly = NO; + dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delay * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ + [viewController setChallengeResponse:[STDSChallengeResponseObject OOBChallengeResponse] animated:YES]; + }); +} + +- (void)challengeResponseViewController:(nonnull STDSChallengeResponseViewController *)viewController didSubmitSelection:(nonnull NSArray> *)selection whitelistSelection:(nonnull id)whitelistSelection { + [viewController setLoading]; + [viewController setChallengeResponse:[STDSChallengeResponseObject textChallengeResponseWithWhitelist:YES resendCode:YES] animated:YES]; +} + +- (void)challengeResponseViewControllerDidCancel:(nonnull STDSChallengeResponseViewController *)viewController { + self.shouldLoadSlowly = NO; + [self.navigationController dismissViewControllerAnimated:YES completion:nil]; +} + +- (void)challengeResponseViewControllerDidOOBContinue:(nonnull STDSChallengeResponseViewController *)viewController whitelistSelection:(nonnull id)whitelistSelection { + [viewController setLoading]; + [viewController setChallengeResponse:[STDSChallengeResponseObject singleSelectChallengeResponse] animated:YES]; +} + +- (void)challengeResponseViewControllerDidRequestResend:(nonnull STDSChallengeResponseViewController *)viewController { +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2DemoUI/Sources/main.m b/Stripe3DS2/Stripe3DS2DemoUI/Sources/main.m new file mode 100644 index 00000000..ca445d07 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUI/Sources/main.m @@ -0,0 +1,16 @@ +// +// main.m +// Stripe3DS2DemoUI +// +// Created by Andrew Harrison on 2/27/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "AppDelegate.h" + +int main(int argc, char * argv[]) { + @autoreleasepool { + return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class])); + } +} diff --git a/Stripe3DS2/Stripe3DS2DemoUITests/Info.plist b/Stripe3DS2/Stripe3DS2DemoUITests/Info.plist new file mode 100644 index 00000000..afb93bf7 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUITests/Info.plist @@ -0,0 +1,24 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + BNDL + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + UILaunchStoryboardName + LaunchScreen + + diff --git a/Stripe3DS2/Stripe3DS2DemoUITests/STDSChallengeResponseViewControllerSnapshotTests.m b/Stripe3DS2/Stripe3DS2DemoUITests/STDSChallengeResponseViewControllerSnapshotTests.m new file mode 100644 index 00000000..94904843 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2DemoUITests/STDSChallengeResponseViewControllerSnapshotTests.m @@ -0,0 +1,117 @@ +// +// STDSChallengeResponseViewControllerSnapshotTests.m +// Stripe3DS2DemoUITests +// +// Created by Andrew Harrison on 3/28/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +@import iOSSnapshotTestCaseCore; + +#import + +#import "STDSChallengeResponseViewController.h" +#import "STDSChallengeResponseObject+TestObjects.h" + +/** + Calls FBSnapshotVerifyView with a default 2% per-pixel color differentiation, as M1 and Intel machines render shadows differently. + @param view The view to snapshot. + @param identifier An optional identifier, used if there are multiple snapshot tests in a given -test method. + */ +#define STPSnapshotVerifyView(view__, identifier__) \ +FBSnapshotVerifyViewWithPixelOptions(view__, identifier__, FBSnapshotTestCaseDefaultSuffixes(), 0.02, 0) + +@interface STDSChallengeResponseViewControllerSnapshotTests: FBSnapshotTestCase + +@end + +@implementation STDSChallengeResponseViewControllerSnapshotTests + +- (void)setUp { + [super setUp]; + + /// Recorded on an iPhone 12 Mini running iOS 15.4 +// self.recordMode = YES; +} + +- (void)testVerifyTextChallengeDesign { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:[STDSChallengeResponseObject textChallengeResponseWithWhitelist:NO resendCode:NO] directoryServer:STDSDirectoryServerCustom]; + [challengeResponseViewController view]; + + [self waitForChallengeResponseTimer]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"TextChallengeResponse"); +} + +- (void)testVerifySingleSelectDesign { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:[STDSChallengeResponseObject singleSelectChallengeResponse] directoryServer:STDSDirectoryServerCustom]; + [challengeResponseViewController view]; + + [self waitForChallengeResponseTimer]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"SingleSelectResponse"); +} + +- (void)testVerifyMultiSelectDesign { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:[STDSChallengeResponseObject multiSelectChallengeResponse] directoryServer:STDSDirectoryServerCustom]; + [challengeResponseViewController view]; + + [self waitForChallengeResponseTimer]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"MultiSelectResponse"); +} + +- (void)testVerifyOOBDesign { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:[STDSChallengeResponseObject OOBChallengeResponse] directoryServer:STDSDirectoryServerCustom]; + [challengeResponseViewController view]; + + [self waitForChallengeResponseTimer]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"OOBResponse"); +} + +- (void)testLoadingAmex { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:nil directoryServer:STDSDirectoryServerAmex]; + [challengeResponseViewController view]; + [challengeResponseViewController setLoading]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"LoadingAmex"); +} + +- (void)testLoadingDiscover { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:nil directoryServer:STDSDirectoryServerDiscover]; + [challengeResponseViewController view]; + [challengeResponseViewController setLoading]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"LoadingDiscover"); +} + +- (void)testLoadingMastercard { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:nil directoryServer:STDSDirectoryServerMastercard]; + [challengeResponseViewController view]; + [challengeResponseViewController setLoading]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"LoadingMastercard"); +} + +- (void)testLoadingVisa { + STDSChallengeResponseViewController *challengeResponseViewController = [self challengeResponseViewControllerForResponse:nil directoryServer:STDSDirectoryServerVisa]; + [challengeResponseViewController view]; + [challengeResponseViewController setLoading]; + + STPSnapshotVerifyView(challengeResponseViewController.view, @"LoadingVisa"); +} + +- (STDSChallengeResponseViewController *)challengeResponseViewControllerForResponse:(id)response directoryServer:(STDSDirectoryServer)directoryServer { + STDSImageLoader *imageLoader = [[STDSImageLoader alloc] initWithURLSession:NSURLSession.sharedSession]; + + STDSChallengeResponseViewController *vc = [[STDSChallengeResponseViewController alloc] initWithUICustomization:[STDSUICustomization defaultSettings] imageLoader:imageLoader directoryServer:directoryServer]; + [vc setChallengeResponse:response animated:NO]; + return vc; +} + +- (void)waitForChallengeResponseTimer { + (void)[XCTWaiter waitForExpectations:@[[self expectationWithDescription:@""]] timeout:2.5]; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Resources/Info.plist b/Stripe3DS2/Stripe3DS2Resources/Info.plist new file mode 100644 index 00000000..7607d027 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Resources/Info.plist @@ -0,0 +1,26 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + NSHumanReadableCopyright + Copyright © 2020 Stripe. All rights reserved. + NSPrincipalClass + + + diff --git a/Stripe3DS2/Stripe3DS2Tests/Info.plist b/Stripe3DS2/Stripe3DS2Tests/Info.plist new file mode 100644 index 00000000..6c40a6cd --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + BNDL + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + + diff --git a/Stripe3DS2/Stripe3DS2Tests/JSON/ARes.json b/Stripe3DS2/Stripe3DS2Tests/JSON/ARes.json new file mode 100644 index 00000000..a1634721 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/JSON/ARes.json @@ -0,0 +1,16 @@ +{ + "dsTransID": "4e4750e7-6ab5-45a4-accf-9c668ed3b5a7", + "acsTransID": "fa695a82-a48c-455d-9566-a652058dda27", + "p_messageVersion": "1.0.5", + "acsOperatorID": "acsOperatorUL", + "sdkTransID": "D77EB83F-F317-4E29-9852-EBAAB55515B7", + "eci": "00", + "dsReferenceNumber": "3DS_LOA_DIS_PPFU_020100_00010", + "acsReferenceNumber": "3DS_LOA_ACS_PPFU_020100_00009", + "threeDSServerTransID": "fc7a39de-dc41-4b65-ba76-a322769b2efc", + "messageVersion": "2.1.0", + "authenticationValue": "AABBCCDDEEFFAABBCCDDEEFFAAA=", + "messageType": "pArs", + "transStatus": "C", + "acsChallengeMandated": "NO" +} diff --git a/Stripe3DS2/Stripe3DS2Tests/JSON/CRes.json b/Stripe3DS2/Stripe3DS2Tests/JSON/CRes.json new file mode 100644 index 00000000..98895661 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/JSON/CRes.json @@ -0,0 +1,31 @@ +{ + "threeDSServerTransID": "8a880dc0-d2d2-4067-bcb1-b08d1690b26e", + "acsTransID": "d7c1ee99-9478-44a6-b1f2-391e29c6b340", + "acsUiType": "01", + "challengeAddInfo": "Additional information to be shown.", + "challengeCompletionInd": "N", + "challengeInfoHeader": "Header information", + "challengeInfoLabel": "One-time-password", + "challengeInfoText": "Please enter the received one-time-password", + "challengeInfoTextIndicator": "N", + "expandInfoLabel": "Additional instructions", + "expandInfoText": "The issuer will send you via SMS a one-time password. Please enter the value in the designated input field above and press continue to complete the 3-D Secure authentication process.", + "issuerImage": { + "medium": "https://acs.com/medium_image.svg", + "high": "https://acs.com/high_image.svg", + "extraHigh": "https://acs.com/extraHigh_image.svg" + }, + "messageType": "CRes", + "messageVersion": "2.1.0", + "psImage": { + "medium": "https://ds.com/medium_image.svg", + "high": "https://ds.com/high_image.svg", + "extraHigh": "https://ds.com/extraHigh_image.svg" + }, + "resendInformationLabel": "Send new One-time-password", + "sdkTransID": "b2385523-a66c-4907-ac3c-91848e8c0067", + "submitAuthenticationLabel": "Continue", + "whyInfoLabel": "Why using 3-D Secure?", + "whyInfoText": "Some explanation about why using 3-D Secure is an excellent idea as part of an online payment transaction", + "acsCounterAtoS": "001" +} diff --git a/Stripe3DS2/Stripe3DS2Tests/JSON/ErrorMessage.json b/Stripe3DS2/Stripe3DS2Tests/JSON/ErrorMessage.json new file mode 100644 index 00000000..6acd63f1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/JSON/ErrorMessage.json @@ -0,0 +1,13 @@ +{ + "threeDSServerTransID": "6afa6072-9412-446b-9673-2f98b3ee98a2", + "acsTransID": "375d90ad-3873-498b-9133-380cbbc8d99d", + "dsTransID": "0b470d4f-fdf8-429f-9147-505b1a589883", + "errorCode": "203", + "errorComponent": "A", + "errorDescription": "Data element not in the required format. Not numeric or wrong length.", + "errorDetail": "billAddrCountry,billAddrPostCode,dsURL", + "errorMessageType": "AReq", + "messageType": "Erro", + "messageVersion": "2.1.0", + "sdkTransID": "b2385523-a66c-4907-ac3c-91848e8c0067" +} diff --git a/Stripe3DS2/Stripe3DS2Tests/NSDictionary+DecodingHelpersTest.m b/Stripe3DS2/Stripe3DS2Tests/NSDictionary+DecodingHelpersTest.m new file mode 100644 index 00000000..846c5268 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/NSDictionary+DecodingHelpersTest.m @@ -0,0 +1,312 @@ +// +// NSDictionary+DecodingHelpersTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/28/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "NSDictionary+DecodingHelpers.h" +#import "STDSStripe3DS2Error.h" +#import "NSError+Stripe3DS2.h" + +@interface JSONDecodableTestObject : NSObject +@property (nonatomic, copy) NSString *value; +@end + +@implementation JSONDecodableTestObject + ++ (instancetype)decodedObjectFromJSON:(NSDictionary *)json error:(NSError * _Nullable __autoreleasing *)outError { + NSString *value = [json _stds_stringForKey:@"key" required:YES error:outError]; + if (outError && *outError) { + return nil; + } + JSONDecodableTestObject *obj = [[self alloc] init]; + obj.value = value; + return obj; +} + +@end + +@interface NSDictionary_DecodingHelpersTest : XCTestCase + +@end + +@implementation NSDictionary_DecodingHelpersTest + +- (void)testMissingRequiredKey { + // Every getter should fail the same way if the key is not present + NSDictionary *json = @{}; + NSError *expectedError = [NSError _stds_missingJSONFieldError:@"key"]; + NSError *error; + id value; + + value = [json _stds_stringForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_stringForKey:@"key" validator:^BOOL (NSString *value) { + return NO; + } required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_boolForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_arrayForKey:@"key" arrayElementType:[JSONDecodableTestObject class] required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_urlForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_dictionaryForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); +} + +- (void)testInvalidType { + // Every getter should fail the same way if the value is not the expected type + NSDictionary *json = @{@"key": [NSObject new]}; + NSError *expectedError = [NSError _stds_invalidJSONFieldError:@"key"]; + NSError *error; + id value; + + value = [json _stds_stringForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_stringForKey:@"key" validator:^BOOL (NSString *value) { + return NO; + } required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_boolForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_arrayForKey:@"key" arrayElementType:[JSONDecodableTestObject class] required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_urlForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + value = [json _stds_dictionaryForKey:@"key" required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); +} + +#pragma mark NSString + +- (void)testString { + NSDictionary *json = [self _basicJSONDictionary]; + NSError *error; + NSString *value = [json _stds_stringForKey:@"key" required:YES error:&error]; + XCTAssertNil(error); + XCTAssertNotNil(value); + XCTAssertEqualObjects(value, @"value"); +} + +- (void)testEmptyString { + NSDictionary *json = @{@"key": @""}; + NSError *expectedError = [NSError _stds_missingJSONFieldError:@"key"]; + NSError *error; + id value; + + // Required empty string should produce a missing error + value = [json _stds_stringForKey:@"key" required:YES error:&error]; + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); + + // Not required empty string should produce an invalid error + expectedError = [NSError _stds_invalidJSONFieldError:@"key"]; + value = [json _stds_stringForKey:@"key" required:NO error:&error]; + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); +} + +- (void)testInvalidValueString { + NSDictionary *json = [self _basicJSONDictionary]; + NSError *expectedError = [NSError _stds_invalidJSONFieldError:@"key"]; + NSError *error; + NSString *value = [json _stds_stringForKey:@"key" validator:^BOOL (NSString *value) { + return NO; + } required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); +} + +#pragma mark NSArray + +- (void)testArray { + NSDictionary *json = @{ + @"key": @[@{@"key": @"value"}] + }; + NSError *error; + NSArray *array = [json _stds_arrayForKey:@"key" arrayElementType:[JSONDecodableTestObject class] required:YES error:&error]; + XCTAssertNil(error); + XCTAssertNotNil(array); + if (array.count == 1) { + XCTAssertEqualObjects([array[0] class], [JSONDecodableTestObject class]); + XCTAssertEqualObjects(array[0].value, @"value"); + } else { + XCTFail(@"Array was not populated"); + } +} + +- (void)testInvalidElementTypeArray { + NSDictionary *json = @{ + @"key": @[@"value1", @"value2"] + }; + NSError *expectedError = [NSError _stds_invalidJSONFieldError:@"key"]; + NSError *error; + NSArray *value = [json _stds_arrayForKey:@"key" arrayElementType:[JSONDecodableTestObject class] required:YES error:&error]; + XCTAssertNotNil(error); + if (error) { + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } else { + XCTFail(@"Error should have a value"); + } + XCTAssertNil(value); +} + +#pragma mark NSDictionary + +- (void)testDictionary { + NSDictionary *nestedJSON = [self _basicJSONDictionary]; + NSDictionary *json = @{ + @"key": nestedJSON + }; + NSError *error; + NSDictionary *value = [json _stds_dictionaryForKey:@"key" required:YES error:&error]; + XCTAssertNil(error); + XCTAssertNotNil(value); + XCTAssertEqualObjects(value, nestedJSON); +} + +#pragma mark NSURL + +- (void)testURL { + NSDictionary *json = @{@"key": @"www.stripe.com"}; + NSError *error; + NSURL *value = [json _stds_urlForKey:@"key" required:YES error:&error]; + XCTAssertNil(error); + XCTAssertNotNil(value); + XCTAssertEqualObjects(value, [NSURL URLWithString:@"www.stripe.com"]); +} + +#pragma mark BOOL + +- (void)testBOOL { + NSDictionary *json = @{@"key": @(YES)}; + NSError *error; + BOOL value = [json _stds_boolForKey:@"key" required:YES error:&error]; + XCTAssertNil(error); + XCTAssertEqual(value, YES); +} + +#pragma mark Helpers + +- (NSDictionary *)_basicJSONDictionary { + return @{@"key": @"value"}; +} + + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/NSString+EmptyCheckingTests.m b/Stripe3DS2/Stripe3DS2Tests/NSString+EmptyCheckingTests.m new file mode 100644 index 00000000..49024547 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/NSString+EmptyCheckingTests.m @@ -0,0 +1,32 @@ +// +// NSString+EmptyCheckingTests.m +// Stripe3DS2Tests +// +// Created by Andrew Harrison on 3/4/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "NSString+EmptyChecking.h" + +@interface NSString_EmptyCheckingTests : XCTestCase + +@end + +@implementation NSString_EmptyCheckingTests + +- (void)testStringIsEmpty { + XCTAssertTrue([NSString _stds_isStringEmpty:@""]); + XCTAssertTrue([NSString _stds_isStringEmpty:@" "]); + XCTAssertTrue([NSString _stds_isStringEmpty:@"\n"]); + XCTAssertTrue([NSString _stds_isStringEmpty:@"\t"]); +} + +- (void)testStringIsNotEmpty { + XCTAssertFalse([NSString _stds_isStringEmpty:@"Hello"]); + XCTAssertFalse([NSString _stds_isStringEmpty:@","]); + XCTAssertFalse([NSString _stds_isStringEmpty:@"\\n"]); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSACSNetworkingManagerTest.m b/Stripe3DS2/Stripe3DS2Tests/STDSACSNetworkingManagerTest.m new file mode 100644 index 00000000..33fca13f --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSACSNetworkingManagerTest.m @@ -0,0 +1,54 @@ +// +// STDSACSNetworkingManagerTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 4/18/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSStripe3DS2Error.h" +#import "STDSACSNetworkingManager.h" +#import "STDSTestJSONUtils.h" +#import "STDSErrorMessage.h" +#import "STDSChallengeResponseObject.h" + +@interface STDSACSNetworkingManager (Private) +- (nullable id)decodeJSON:(NSDictionary *)dict error:(NSError * _Nullable *)outError; +@end + +@interface STDSACSNetworkingManagerTest : XCTestCase + +@end + +@implementation STDSACSNetworkingManagerTest + +- (void)testDecodeJSON { + STDSACSNetworkingManager *manager = [[STDSACSNetworkingManager alloc] init]; + NSError *error; + id decoded; + + // Unknown message type + NSDictionary *unknownMessageDict = @{@"messageType": @"foo"}; + decoded = [manager decodeJSON:unknownMessageDict error:&error]; + XCTAssertEqual(error.code, STDSErrorCodeUnknownMessageType); + XCTAssertNil(decoded); + error = nil; + + // Error Message type + NSDictionary *errorMessageDict = [STDSTestJSONUtils jsonNamed:@"ErrorMessage"]; + decoded = [manager decodeJSON:errorMessageDict error:&error]; + XCTAssertEqual(error.code, STDSErrorCodeReceivedErrorMessage); + XCTAssertTrue([error.userInfo[STDSStripe3DS2ErrorMessageErrorKey] isKindOfClass:[STDSErrorMessage class]]); + XCTAssertNil(decoded); + error = nil; + + // ChallengeResponse message type + NSDictionary *challengeResponseDict = [STDSTestJSONUtils jsonNamed:@"CRes"]; + decoded = [manager decodeJSON:challengeResponseDict error:&error]; + XCTAssertNil(error); + XCTAssertTrue([decoded isKindOfClass:[STDSChallengeResponseObject class]]); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationRequestParametersTest.m b/Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationRequestParametersTest.m new file mode 100644 index 00000000..7e701da8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationRequestParametersTest.m @@ -0,0 +1,41 @@ +// +// STDSAuthenticationRequestParametersTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSAuthenticationRequestParameters.h" +#import "STDSJSONEncoder.h" + +@interface STDSAuthenticationRequestParametersTest : XCTestCase + +@end + +@implementation STDSAuthenticationRequestParametersTest + +#pragma mark - STDSJSONEncodable + +- (void)testPropertyNamesToJSONKeysMapping { + STDSAuthenticationRequestParameters *params = [STDSAuthenticationRequestParameters new]; + + NSDictionary *mapping = [STDSAuthenticationRequestParameters propertyNamesToJSONKeysMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([params respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:[STDSJSONEncoder dictionaryForObject:params]]); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationResponseTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationResponseTests.m new file mode 100644 index 00000000..7dac3fd1 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSAuthenticationResponseTests.m @@ -0,0 +1,32 @@ +// +// STDSAuthenticationResponseTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 5/20/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSAuthenticationResponseObject.h" +#import "STDSTestJSONUtils.h" + +@interface STDSAuthenticationResponseTests : XCTestCase + +@end + +@implementation STDSAuthenticationResponseTests + +- (void)testInitWithJSON { + NSError *error = nil; + STDSAuthenticationResponseObject *ares = [STDSAuthenticationResponseObject decodedObjectFromJSON:[STDSTestJSONUtils jsonNamed:@"ARes"] error:&error]; + + XCTAssertNil(error); + XCTAssertNotNil(ares, @"Failed to create an ares parsed from JSON"); + + id authResponse = STDSAuthenticationResponseFromJSON([STDSTestJSONUtils jsonNamed:@"ARes"]); + XCTAssertNotNil(authResponse, @"Failed to create an ares parsed from JSON"); + XCTAssert(authResponse.isChallengeRequired, @"ares did not indicate that a challenge was required"); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSBase64URLEncodingTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSBase64URLEncodingTests.m new file mode 100644 index 00000000..25642a5d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSBase64URLEncodingTests.m @@ -0,0 +1,59 @@ +// +// STDSBase64URLEncodingTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "NSData+JWEHelpers.h" +#import "NSString+JWEHelpers.h" + +@interface STDSBase64URLEncodingTests : XCTestCase + +@end + +@implementation STDSBase64URLEncodingTests + +// test cases from https://tools.ietf.org/html/draft-ietf-jose-json-web-signature-41 + +- (void)testEncodingDataToString { + { + Byte bytes[5] = {3, 236, 255, 224, 193}; + NSData *data = [NSData dataWithBytes:bytes length:5]; + XCTAssertEqualObjects([data _stds_base64URLEncodedString], @"A-z_4ME"); + } + + { + Byte bytes[30] = {123, 34, 116, 121, 112, 34, 58, 34, 74, 87, 84, 34, 44, 13, 10, 32, 34, 97, 108, 103, 34, 58, 34, 72, 83, 50, 53, 54, 34, 125}; + NSData *data = [NSData dataWithBytes:bytes length:30]; + XCTAssertEqualObjects([data _stds_base64URLEncodedString], @"eyJ0eXAiOiJKV1QiLA0KICJhbGciOiJIUzI1NiJ9"); + } + + { + Byte bytes[70] = {123, 34, 105, 115, 115, 34, 58, 34, 106, 111, 101, 34, 44, 13, 10, + 32, 34, 101, 120, 112, 34, 58, 49, 51, 48, 48, 56, 49, 57, 51, 56, + 48, 44, 13, 10, 32, 34, 104, 116, 116, 112, 58, 47, 47, 101, 120, 97, + 109, 112, 108, 101, 46, 99, 111, 109, 47, 105, 115, 95, 114, 111, + 111, 116, 34, 58, 116, 114, 117, 101, 125}; + NSData *data = [NSData dataWithBytes:bytes length:70]; + XCTAssertEqualObjects([data _stds_base64URLEncodedString], @"eyJpc3MiOiJqb2UiLA0KICJleHAiOjEzMDA4MTkzODAsDQogImh0dHA6Ly9leGFtcGxlLmNvbS9pc19yb290Ijp0cnVlfQ"); + } + +} + +- (void)testEncodingString { + XCTAssertEqualObjects([@"{\"typ\":\"JWT\",\r\n \"alg\":\"HS256\"}" _stds_base64URLEncodedString], @"eyJ0eXAiOiJKV1QiLA0KICJhbGciOiJIUzI1NiJ9"); + + XCTAssertEqualObjects([@"{\"iss\":\"joe\",\r\n \"exp\":1300819380,\r\n \"http://example.com/is_root\":true}" _stds_base64URLEncodedString], @"eyJpc3MiOiJqb2UiLA0KICJleHAiOjEzMDA4MTkzODAsDQogImh0dHA6Ly9leGFtcGxlLmNvbS9pc19yb290Ijp0cnVlfQ"); +} + +- (void)testDecodingString { + XCTAssertEqualObjects([@"eyJ0eXAiOiJKV1QiLA0KICJhbGciOiJIUzI1NiJ9" _stds_base64URLDecodedString], @"{\"typ\":\"JWT\",\r\n \"alg\":\"HS256\"}"); + + XCTAssertEqualObjects([ @"eyJpc3MiOiJqb2UiLA0KICJleHAiOjEzMDA4MTkzODAsDQogImh0dHA6Ly9leGFtcGxlLmNvbS9pc19yb290Ijp0cnVlfQ" _stds_base64URLDecodedString], @"{\"iss\":\"joe\",\r\n \"exp\":1300819380,\r\n \"http://example.com/is_root\":true}"); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSChallengeParametersTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSChallengeParametersTests.m new file mode 100644 index 00000000..8b430d10 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSChallengeParametersTests.m @@ -0,0 +1,56 @@ +// +// STDSChallengeParametersTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 2/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSAuthenticationResponseObject.h" +#import "STDSChallengeParameters.h" + +@interface TestAuthResponse: STDSAuthenticationResponseObject + +@end + +@interface STDSChallengeParametersTests : XCTestCase + +@end + +@implementation STDSChallengeParametersTests + +- (void)testInitWithAuthResponse { + STDSChallengeParameters *params = [[STDSChallengeParameters alloc] initWithAuthenticationResponse:[[TestAuthResponse alloc] init]]; + + XCTAssertEqual(params.threeDSServerTransactionID, @"test_threeDSServerTransactionID", @"Failed to set test_threeDSServerTransactionID"); + XCTAssertEqual(params.acsTransactionID, @"test_acsTransactionID", @"Failed to set test_acsTransactionID"); + XCTAssertEqual(params.acsReferenceNumber, @"test_acsReferenceNumber", @"Failed to set test_acsReferenceNumber"); + XCTAssertEqual(params.acsSignedContent, @"test_acsSignedContent", @"Failed to set test_acsSignedContent"); + XCTAssertNil(params.threeDSRequestorAppURL, @"Should not have set threeDSRequestorAppURL"); +} + +@end + +#pragma mark - TestAuthResponse + +@implementation TestAuthResponse + +- (NSString *)threeDSServerTransactionID { + return @"test_threeDSServerTransactionID"; +} + +- (NSString *)acsTransactionID { + return @"test_acsTransactionID"; +} + +- (NSString *)acsReferenceNumber { + return @"test_acsReferenceNumber"; +} + +- (NSString *)acsSignedContent { + return @"test_acsSignedContent"; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSChallengeRequestParametersTest.m b/Stripe3DS2/Stripe3DS2Tests/STDSChallengeRequestParametersTest.m new file mode 100644 index 00000000..e97b030e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSChallengeRequestParametersTest.m @@ -0,0 +1,71 @@ +// +// STDSChallengeRequestParametersTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 4/1/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSChallengeRequestParameters.h" +#import "STDSJSONEncoder.h" + +@interface STDSChallengeRequestParametersTest : XCTestCase + +@end + +@implementation STDSChallengeRequestParametersTest + +#pragma mark - STDSJSONEncodable + +- (void)testPropertyNamesToJSONKeysMapping { + STDSChallengeRequestParameters *params = [[STDSChallengeRequestParameters alloc] initWithThreeDSServerTransactionIdentifier:@"server id" + acsTransactionIdentifier:@"acs id" + messageVersion:@"message version" + sdkTransactionIdentifier:@"sdk id" + requestorAppUrl:@"requestor app url" + sdkCounterStoA:0]; + + NSDictionary *mapping = [STDSChallengeRequestParameters propertyNamesToJSONKeysMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([params respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:[STDSJSONEncoder dictionaryForObject:params]]); +} + +- (void)testNextChallengeRequestParametersIncrementsCounter { + STDSChallengeRequestParameters *params = [[STDSChallengeRequestParameters alloc] initWithThreeDSServerTransactionIdentifier:@"server id" + acsTransactionIdentifier:@"acs id" + messageVersion:@"message version" + sdkTransactionIdentifier:@"sdk id" + requestorAppUrl:@"requestor app url" + sdkCounterStoA:0]; + for (NSInteger i = 0; i < 1000; i++) { + XCTAssertEqual(params.sdkCounterStoA.length, 3); + XCTAssertEqual(params.sdkCounterStoA.integerValue, i); + params = [params nextChallengeRequestParametersByIncrementCounter]; + } +} + +- (void)testEmptyChallengeDataEntryField { + STDSChallengeRequestParameters *params = [[STDSChallengeRequestParameters alloc] initWithThreeDSServerTransactionIdentifier:@"server id" + acsTransactionIdentifier:@"acs id" + messageVersion:@"message version" + sdkTransactionIdentifier:@"sdk id" + requestorAppUrl:@"requestor app url" + sdkCounterStoA:0]; + params.challengeDataEntry = @""; + XCTAssertNil(params.challengeDataEntry); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSChallengeResponseObjectTest.m b/Stripe3DS2/Stripe3DS2Tests/STDSChallengeResponseObjectTest.m new file mode 100644 index 00000000..7ffdd95a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSChallengeResponseObjectTest.m @@ -0,0 +1,79 @@ +// +// STDSChallengeResponseObjectTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSChallengeResponseObject.h" +#import "STDSTestJSONUtils.h" +#import "NSError+Stripe3DS2.h" + +@interface STDSChallengeResponseObjectTest : XCTestCase + +@end + +@implementation STDSChallengeResponseObjectTest + +- (void)testSuccessfulDecode { + NSDictionary *json = [STDSTestJSONUtils jsonNamed:@"CRes"]; + NSError *error; + STDSChallengeResponseObject *cr = [STDSChallengeResponseObject decodedObjectFromJSON:json error:&error]; + + XCTAssertNil(error); + XCTAssertNotNil(cr); + + XCTAssertEqualObjects(cr.threeDSServerTransactionID, @"8a880dc0-d2d2-4067-bcb1-b08d1690b26e"); + XCTAssertEqualObjects(cr.acsTransactionID, @"d7c1ee99-9478-44a6-b1f2-391e29c6b340"); + XCTAssertEqual(cr.acsUIType, STDSACSUITypeText); + XCTAssertEqual(cr.challengeCompletionIndicator, NO); + XCTAssertEqualObjects(cr.challengeInfoHeader, @"Header information"); + XCTAssertEqualObjects(cr.challengeInfoLabel, @"One-time-password"); + XCTAssertEqualObjects(cr.challengeInfoText, @"Please enter the received one-time-password"); + XCTAssertEqual(cr.showChallengeInfoTextIndicator, NO); + XCTAssertEqualObjects(cr.expandInfoLabel, @"Additional instructions"); + XCTAssertEqualObjects(cr.expandInfoText, @"The issuer will send you via SMS a one-time password. Please enter the value in the designated input field above and press continue to complete the 3-D Secure authentication process."); + XCTAssertEqualObjects(cr.issuerImage.mediumDensityURL, [NSURL URLWithString:@"https://acs.com/medium_image.svg"]); + XCTAssertEqualObjects(cr.issuerImage.highDensityURL, [NSURL URLWithString:@"https://acs.com/high_image.svg"]); + XCTAssertEqualObjects(cr.issuerImage.extraHighDensityURL, [NSURL URLWithString:@"https://acs.com/extraHigh_image.svg"]); + XCTAssertEqualObjects(cr.messageType, @"CRes"); + XCTAssertEqualObjects(cr.messageVersion, @"2.1.0"); + XCTAssertEqualObjects(cr.paymentSystemImage.mediumDensityURL, [NSURL URLWithString:@"https://ds.com/medium_image.svg"]); + XCTAssertEqualObjects(cr.paymentSystemImage.highDensityURL, [NSURL URLWithString:@"https://ds.com/high_image.svg"]); + XCTAssertEqualObjects(cr.paymentSystemImage.extraHighDensityURL, [NSURL URLWithString:@"https://ds.com/extraHigh_image.svg"]); + XCTAssertEqualObjects(cr.resendInformationLabel, @"Send new One-time-password"); + XCTAssertEqualObjects(cr.sdkTransactionID, @"b2385523-a66c-4907-ac3c-91848e8c0067"); + XCTAssertEqualObjects(cr.submitAuthenticationLabel, @"Continue"); + XCTAssertEqualObjects(cr.whyInfoLabel, @"Why using 3-D Secure?"); + XCTAssertEqualObjects(cr.whyInfoText, @"Some explanation about why using 3-D Secure is an excellent idea as part of an online payment transaction"); + XCTAssertEqualObjects(cr.acsCounterACStoSDK, @"001"); +} + +- (void)testMissingFields { + NSArray *requiredFields = @[ + @"threeDSServerTransID", + @"acsCounterAtoS", + @"acsTransID", + @"acsUiType", + @"challengeCompletionInd", + @"messageType", + @"messageVersion", + @"sdkTransID", + ]; + + for (NSString *field in requiredFields) { + NSMutableDictionary *response = [[STDSTestJSONUtils jsonNamed:@"CRes"] mutableCopy]; + [response removeObjectForKey:field]; + + NSError *error; + NSError *expectedError = [NSError _stds_missingJSONFieldError:field]; + XCTAssertNil([STDSChallengeResponseObject decodedObjectFromJSON:response error:&error]); + XCTAssertNotNil(error); + XCTAssertEqual(error.code, expectedError.code); + XCTAssertEqualObjects(error.userInfo, expectedError.userInfo); + } +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSConfigParametersTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSConfigParametersTests.m new file mode 100644 index 00000000..28114985 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSConfigParametersTests.m @@ -0,0 +1,67 @@ +// +// STDSConfigParametersTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 2/13/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSConfigParameters.h" +#import "STDSInvalidInputException.h" + +@interface STDSConfigParametersTests : XCTestCase + +@end + +@implementation STDSConfigParametersTests + +- (void)testStandardParameters { + STDSConfigParameters *defaultParameters = [[STDSConfigParameters alloc] initWithStandardParameters]; + XCTAssertNotNil(defaultParameters, @"Should return a non-nil instance"); +} + +- (void)testAddRead { + STDSConfigParameters *parameters = [[STDSConfigParameters alloc] init]; + XCTAssertNoThrow([parameters addParameterNamed:@"testName" withValue:@"testValue"], @"Should not throw with non-nil name and value."); + NSString *paramValue = nil; + XCTAssertNoThrow(paramValue = [parameters parameterValue:@"testName"], @"Should not throw with non-nil name."); + XCTAssertEqual(paramValue, @"testValue", @"Returned value does not match expectation."); +} + +- (void)testDefaultGroup { + STDSConfigParameters *parameters = [[STDSConfigParameters alloc] init]; + XCTAssertNoThrow([parameters addParameterNamed:@"testName" withValue:@"testValue"], @"Should not throw with non-nil name and value."); + XCTAssertNoThrow([parameters addParameterNamed:@"testName" withValue:@"testValue2" toGroup:@"otherGroup"], @"Should not throw with non-nil name, value, and group."); + NSString *paramValue = nil; + XCTAssertNoThrow(paramValue = [parameters parameterValue:@"testName"], @"Should not throw with non-nil name."); + XCTAssertEqual(paramValue, @"testValue", @"Returned value does not match expectation. Should default to default group's value."); + XCTAssertNoThrow(paramValue = [parameters parameterValue:@"testName" inGroup:@"otherGroup"], @"Should not throw with non-nil name and group name."); + XCTAssertEqual(paramValue, @"testValue2", @"Returned value does not match expectation. Should read from custom group."); +} + +- (void)testExceptions { + STDSConfigParameters *parameters = [[STDSConfigParameters alloc] init]; + XCTAssertNoThrow([parameters addParameterNamed:@"testParam" withValue:@"testValue" toGroup:nil], @"Should not throw with nil group."); + + XCTAssertThrowsSpecific([parameters addParameterNamed:@"testParam" withValue:@"value2"], STDSInvalidInputException, @"Should throw STDSInvalidInputException if trying to override testParam value."); + [parameters addParameterNamed:@"testParam" withValue:@"testValue" toGroup:@"otherGroup"]; + XCTAssertThrowsSpecific([parameters addParameterNamed:@"testParam" withValue:@"value2" toGroup:@"otherGroup"], STDSInvalidInputException, @"Should throw STDSInvalidInputException if trying to override testParam value in non-default group."); +} + +- (void)testRemove { + STDSConfigParameters *parameters = [[STDSConfigParameters alloc] init]; + + [parameters addParameterNamed:@"testParam" withValue:@"testValue"]; + [parameters addParameterNamed:@"testParam" withValue:@"testValue2" toGroup:@"otherGroup"]; + XCTAssertEqual([parameters removeParameterNamed:@"testParam"], @"testValue", @"Should return testValue when removing."); + XCTAssertNil([parameters parameterValue:@"testParam"]); + XCTAssertNotNil([parameters parameterValue:@"testParam" inGroup:@"otherGroup"], @"Should only remove param in specified group."); + + XCTAssertNil([parameters removeParameterNamed:@"testParam"], @"Should return nil if removing a non-existant parameter."); + + XCTAssertEqual([parameters removeParameterNamed:@"testParam" fromGroup:@"otherGroup"], @"testValue2", @"Should return group-specific value when removing."); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationManagerTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationManagerTests.m new file mode 100644 index 00000000..d86c2b9e --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationManagerTests.m @@ -0,0 +1,33 @@ +// +// STDSDeviceInformationManagerTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 1/24/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDeviceInformation.h" +#import "STDSDeviceInformationManager.h" +#import "STDSWarning.h" + +@interface STDSDeviceInformationManagerTests : XCTestCase + +@end + +@implementation STDSDeviceInformationManagerTests + +- (void)testDeviceInformation { + STDSDeviceInformation *deviceInformation = [STDSDeviceInformationManager deviceInformationWithWarnings:@[] ignoringRestrictions:NO]; + XCTAssertEqualObjects(deviceInformation.dictionaryValue[@"DV"], @"1.4", @"Device data version check."); + XCTAssertNotNil(deviceInformation.dictionaryValue[@"DD"], @"Device data should be non-nil"); + XCTAssertNotNil(deviceInformation.dictionaryValue[@"DPNA"], @"Param not available should be non-nil in simulator"); + XCTAssertNil(deviceInformation.dictionaryValue[@"SW"]); + + deviceInformation = [STDSDeviceInformationManager deviceInformationWithWarnings:@[[[STDSWarning alloc] initWithIdentifier:@"WARNING_1" message:@"" severity:STDSWarningSeverityMedium], [[STDSWarning alloc] initWithIdentifier:@"WARNING_2" message:@"" severity:STDSWarningSeverityMedium], ] ignoringRestrictions:NO]; + NSArray *warningIDs = @[@"WARNING_1", @"WARNING_2"]; + XCTAssertEqualObjects(deviceInformation.dictionaryValue[@"SW"], warningIDs, @"Failed to set warning identifiers correctly"); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationParameterTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationParameterTests.m new file mode 100644 index 00000000..c550a7eb --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSDeviceInformationParameterTests.m @@ -0,0 +1,214 @@ +// +// STDSDeviceInformationParameterTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 1/24/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDeviceInformationParameter+Private.h" + +@interface STDSDeviceInformationParameterTests : XCTestCase + +@end + +@implementation STDSDeviceInformationParameterTests + +- (void)testNoPermissions { + STDSDeviceInformationParameter *noPermissionParam = [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"NoPermissionID" + permissionCheck:^BOOL{ + return NO; + } + valueCheck:^id _Nullable{ + XCTFail(@"Should not try to collect value if we don't have permission for it"); + return @"fail"; + }]; + [noPermissionParam collectIgnoringRestrictions:YES withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + XCTAssertFalse(collected, @"Should not have collected a param we don't have permission for."); + XCTAssertTrue([value isKindOfClass:[NSString class]], @"No permission value should be a string."); + XCTAssertEqualObjects(value, @"RE03", @"Returned value should be 'RE03' for param with missing permissions."); + }]; +} + +- (void)testNoValue { + STDSDeviceInformationParameter *noValueParam = [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"NoValueID" + permissionCheck:^BOOL{ + return YES; + } + valueCheck:^id _Nullable{ + return nil; + }]; + [noValueParam collectIgnoringRestrictions:YES withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + XCTAssertFalse(collected, @"Should not have collected a param we don't have a value for."); + XCTAssertTrue([value isKindOfClass:[NSString class]], @"No value value should be a string."); + XCTAssertEqualObjects(value, @"RE02", @"Returned value should be 'RE02' for param with unavailable value."); + }]; +} + +- (void)testCollect { + __block BOOL permissionCheckCalled = NO; + __block BOOL valueCheckCalled = NO; + __block BOOL collectedHandlerCalled = NO; + + STDSDeviceInformationParameter *param = [[STDSDeviceInformationParameter alloc] initWithIdentifier:@"ParamID" + permissionCheck:^BOOL{ + XCTAssertFalse(valueCheckCalled); + permissionCheckCalled = YES; + return YES; + } + valueCheck:^id _Nullable{ + XCTAssertTrue(permissionCheckCalled); + valueCheckCalled = YES; + return @"param_val"; + }]; + [param collectIgnoringRestrictions:YES withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + XCTAssertTrue(collected, @"Should have marked value as collected."); + XCTAssertEqualObjects(value, @"param_val", @"Inaccurate returned value."); + XCTAssertTrue(permissionCheckCalled); + XCTAssertTrue(valueCheckCalled); + collectedHandlerCalled = YES; + }]; + + // This check tests that collect is synchronous for now + XCTAssertTrue(collectedHandlerCalled); + + // reset so the permission before value check doesn't fail on the second call + permissionCheckCalled = NO; + valueCheckCalled = NO; + + // make sure the ignoreRestrictions param is respected + [param collectIgnoringRestrictions:NO withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + XCTAssertFalse(collected, @"Should not have marked value as collected."); + XCTAssertFalse(permissionCheckCalled, @"Restrictions shouldn't even check the runtime permission."); + XCTAssertFalse(valueCheckCalled, @"Should not have tried to get the value."); + XCTAssertEqualObjects(value, @"RE01", @"Should return market restricted code as the value."); + }]; +} + +- (void)testAllParameters { + NSArray *allParams = [STDSDeviceInformationParameter allParameters]; + XCTAssertEqual(allParams.count, 29, @"iOS should collect 29 separate parameters."); + NSMutableSet *allParamIdentifiers = [[NSMutableSet alloc] init]; + for (STDSDeviceInformationParameter *param in allParams) { + [param collectIgnoringRestrictions:YES withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + [allParamIdentifiers addObject:identifier]; + }]; + } + XCTAssertEqual(allParamIdentifiers.count, allParams.count, @"Sanity check that there are not duplicate identifiers."); + NSArray *expectedIdentifiers = @[ + @"C001", + @"C002", + @"C003", + @"C004", + @"C005", + @"C006", + @"C007", + @"C008", + @"C009", + @"C010", + @"C011", + @"C012", + @"C013", + @"C014", + @"C015", + @"I001", + @"I002", + @"I003", + @"I004", + @"I005", + @"I006", + @"I007", + @"I008", + @"I009", + @"I010", + @"I011", + @"I012", + @"I013", + @"I014", + ]; + for (NSString *identifier in expectedIdentifiers) { + XCTAssertTrue([allParamIdentifiers containsObject:identifier], @"Missing identifier %@", identifier); + } +} + +- (void)testOnlyApprovedIdentifiers { + NSArray *allParams = [STDSDeviceInformationParameter allParameters]; + NSMutableSet *collectedParameterIdentifiers = [[NSMutableSet alloc] init]; + for (STDSDeviceInformationParameter *param in allParams) { + [param collectIgnoringRestrictions:NO withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + + if (collected) { + [collectedParameterIdentifiers addObject:identifier]; + } + }]; + } + NSArray *expectedIdentifiers = @[ + @"C001", + @"C002", + @"C003", + @"C004", + @"C005", + @"C006", + @"C007", + @"C008", + ]; + XCTAssertEqual(collectedParameterIdentifiers.count, expectedIdentifiers.count, @"Should only have collected the expected amount."); + + for (NSString *identifier in expectedIdentifiers) { + XCTAssertTrue([collectedParameterIdentifiers containsObject:identifier], @"Missing identifier %@", identifier); + } +} + +- (void)testIdentifiersAccurate { + NSDictionary *expectedIdentifiers = @{ + @"C001": [STDSDeviceInformationParameter platform], + @"C002": [STDSDeviceInformationParameter deviceModel], + @"C003": [STDSDeviceInformationParameter OSName], + @"C004": [STDSDeviceInformationParameter OSVersion], + @"C005": [STDSDeviceInformationParameter locale], + @"C006": [STDSDeviceInformationParameter timeZone], + @"C007": [STDSDeviceInformationParameter advertisingID], + @"C008": [STDSDeviceInformationParameter screenResolution], + @"C009": [STDSDeviceInformationParameter deviceName], + @"C010": [STDSDeviceInformationParameter IPAddress], + @"C011": [STDSDeviceInformationParameter latitude], + @"C012": [STDSDeviceInformationParameter longitude], + @"I001": [STDSDeviceInformationParameter identiferForVendor], + @"I002": [STDSDeviceInformationParameter userInterfaceIdiom], + @"I003": [STDSDeviceInformationParameter familyNames], + @"I004": [STDSDeviceInformationParameter fontNamesForFamilyName], + @"I005": [STDSDeviceInformationParameter systemFont], + @"I006": [STDSDeviceInformationParameter labelFontSize], + @"I007": [STDSDeviceInformationParameter buttonFontSize], + @"I008": [STDSDeviceInformationParameter smallSystemFontSize], + @"I009": [STDSDeviceInformationParameter systemFontSize], + @"I010": [STDSDeviceInformationParameter systemLocale], + @"I011": [STDSDeviceInformationParameter availableLocaleIdentifiers], + @"I012": [STDSDeviceInformationParameter preferredLanguages], + @"I013": [STDSDeviceInformationParameter defaultTimeZone], + }; + + [expectedIdentifiers enumerateKeysAndObjectsUsingBlock:^(NSString * _Nonnull key, STDSDeviceInformationParameter * _Nonnull obj, BOOL * _Nonnull stop) { + [obj collectIgnoringRestrictions:YES withHandler:^(BOOL collected, NSString * _Nonnull identifier, id _Nonnull value) { + XCTAssertEqualObjects(key, identifier); + }]; + }]; +} + +#pragma mark - App ID + +- (void)testSDKAppIdentifier { + // xctest in Xcode 13+ uses the Xcode version for the current app id string, previous versions are empty + NSString *appIdentifierKeyPrefix = @"STDSStripe3DS2AppIdentifierKey"; + NSString *appVersion = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleShortVersionString"] ?: @""; + NSString *appIdentifierUserDefaultsKey = [appIdentifierKeyPrefix stringByAppendingString:appVersion]; + + [[NSUserDefaults standardUserDefaults] removeObjectForKey:appIdentifierUserDefaultsKey]; + NSString *appId = [STDSDeviceInformationParameter sdkAppIdentifier]; + XCTAssertNotNil(appId); + XCTAssertEqualObjects(appId, [[NSUserDefaults standardUserDefaults] stringForKey:appIdentifierUserDefaultsKey]); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSDirectoryServerCertificateTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSDirectoryServerCertificateTests.m new file mode 100644 index 00000000..43e7bab5 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSDirectoryServerCertificateTests.m @@ -0,0 +1,811 @@ +// +// STDSDirectoryServerCertificateTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 3/28/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDirectoryServerCertificate+Internal.h" +#import "STDSJSONWebSignature.h" +#import "NSString+JWEHelpers.h" + +@interface STDSDirectoryServerCertificateTests : XCTestCase + +@end + +@implementation STDSDirectoryServerCertificateTests + +- (void)testCertificateForDirectoryServer { + NSArray *directoryServers = @[ + @(STDSDirectoryServerULTestRSA), + @(STDSDirectoryServerULTestEC), + @(STDSDirectoryServerSTPTestRSA), + @(STDSDirectoryServerSTPTestEC), + @(STDSDirectoryServerAmex), + @(STDSDirectoryServerDiscover), + @(STDSDirectoryServerMastercard), + @(STDSDirectoryServerVisa), + @(STDSDirectoryServerCustom), + @(STDSDirectoryServerUnknown), + ]; + + for (NSNumber *directoryServerNum in directoryServers) { + STDSDirectoryServer directoryServer = (STDSDirectoryServer)[directoryServerNum integerValue]; + switch (directoryServer) { + case STDSDirectoryServerULTestRSA: + case STDSDirectoryServerULTestEC: + case STDSDirectoryServerSTPTestRSA: + case STDSDirectoryServerSTPTestEC: + case STDSDirectoryServerAmex: + case STDSDirectoryServerCartesBancaires: + case STDSDirectoryServerDiscover: + case STDSDirectoryServerMastercard: + case STDSDirectoryServerVisa: + XCTAssertNotNil([STDSDirectoryServerCertificate certificateForDirectoryServer:directoryServer], @"Failed creating certificate for type %@", directoryServerNum); + break; + + case STDSDirectoryServerCustom: + case STDSDirectoryServerUnknown: + XCTAssertNil([STDSDirectoryServerCertificate certificateForDirectoryServer:directoryServer], @"Should return nil for STDSDirectoryServerUnknown"); + break; + } + } +} + +- (void)testKeyType { + NSArray *directoryServers = @[ + @(STDSDirectoryServerULTestRSA), + @(STDSDirectoryServerULTestEC), + @(STDSDirectoryServerSTPTestRSA), + @(STDSDirectoryServerSTPTestEC), + @(STDSDirectoryServerAmex), + @(STDSDirectoryServerCartesBancaires), + @(STDSDirectoryServerDiscover), + @(STDSDirectoryServerMastercard), + @(STDSDirectoryServerVisa), + @(STDSDirectoryServerCustom), + @(STDSDirectoryServerUnknown), + ]; + + for (NSNumber *directoryServerNum in directoryServers) { + STDSDirectoryServer directoryServer = (STDSDirectoryServer)[directoryServerNum integerValue]; + STDSDirectoryServerCertificate *certificate = [STDSDirectoryServerCertificate certificateForDirectoryServer:directoryServer]; + switch (directoryServer) { + case STDSDirectoryServerULTestRSA: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Incorrect key type for STDSDirectoryServerULTestRSA"); + break; + + case STDSDirectoryServerULTestEC: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeEC, @"Incorrect key type for STDSDirectoryServerULTestEC"); + break; + + case STDSDirectoryServerSTPTestRSA: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Incorrect key type for STDSDirectoryServerSTPTestRSA"); + break; + + case STDSDirectoryServerSTPTestEC: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeEC, @"Incorrect key type for STDSDirectoryServerSTPTestEC"); + break; + + case STDSDirectoryServerAmex: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Incorrect key type for STDSDirectoryServerAmex"); + break; + + case STDSDirectoryServerCartesBancaires: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Incorrect key type for STDSDirectoryServerCartesBancaires"); + break; + + case STDSDirectoryServerDiscover: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Incorrect key type for STDSDirectoryServerDiscover"); + break; + + case STDSDirectoryServerMastercard: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Incorrect key type for STDSDirectoryServerMastercard"); + break; + + case STDSDirectoryServerVisa: + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Incorrect key type for STDSDirectoryServerVisa"); + break; + + case STDSDirectoryServerCustom: + case STDSDirectoryServerUnknown: + // asserts + break; + } + } +} + +- (void)testRSA_OAEP_SHA256Encryption { + STDSDirectoryServerCertificate *certificate = [STDSDirectoryServerCertificate certificateForDirectoryServer:STDSDirectoryServerSTPTestRSA]; + if (certificate != nil) { + + XCTAssertNotNil([certificate encryptDataUsingRSA_OAEP_SHA256:[@"In nature's infinite book of secrecy a little I can read." dataUsingEncoding:NSUTF8StringEncoding]]); + } else { + XCTFail(@"Failed loading certificate for %@", NSStringFromSelector(_cmd)); + } +} + +- (void)testCustomCertificate { + NSString *certificateString = @"MIIDXTCCAkWgAwIBAgIQbS4C4BSig7uuJ5uDpeT4VjANBgkqhkiG9w0BAQsFADBH" + "MRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYHZXhhbXBsZTEX" + "MBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE0ODQ5WhcNMjcxMjMx" + "MTQwMDAwWjBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYH" + "ZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwggEiMA0GCSqGSIb3DQEB" + "AQUAA4IBDwAwggEKAoIBAQCfgQ+0A4Jz0CWR5Ac/MdK2ABuCzttNkvBQFl1Hz8q4" + "o8Qct3isdVN5P475dXaNGiN02HElZMO813uepDRUSJlAfP8AmZIKkxokxEFIUqsp" + "vbCpXAZT82xg5gv5C2JY3aVvNwR7pcLR0CmvnJ1AuseqQceKDdEGit1pnoCP6gEe" + "oUQdik97tOl7459V8d3UTpxLozUVlwPU00tgPmUUek8j1tPAmWx17e6EaoLRkK4Q" + "eDyWHPA4eu0hBtLQVVtv2Tf61VNTh+D/cv++eJQUArC4IuoqdLYFjB2r+bNKdstj" + "uH+qLGhHuOKDf/+RGG5rHBSRHPmJqJCSqBzmAd2s0/nPAgMBAAGjRTBDMBIGA1Ud" + "EwEB/wQIMAYBAf8CAQAwDgYDVR0PAQH/BAQDAgEGMB0GA1UdDgQWBBTDgwKdvAPq" + "bbCmehDaw0PwavI83jANBgkqhkiG9w0BAQsFAAOCAQEAOUcKqpzNQ6lr0PbDSsns" + "D6onfi+8j3TD0xG0zBSf+8G4zs8Zb6vzzQ5qHKgfr4aeen8Pw0cw2KKUJ2dFaBqj" + "n3/6/MIZbgaBvXKUbmY8xCxKQ+tOFc3KWIu4pSaO50tMPJjU/lP35bv19AA9vs9M" + "TKY2qLf88bmoNYT3W8VSDcB58KBHa7HVIPx7BUUtSyb2N2Jqx5AOiYy4NarhB3hV" + "ftkZBmCzi2Qw50KWIgTFYcIVeRTx3Js/F0IuEdgZHBK2gmO7fdM7+QKYm83401vl" + "YRNCXfIZ0H9E1V3NddqJuqIutdUajckSzMhXdNCJqfI4FAQAymTWGL3/lZyr/30x" + "Fg=="; + + STDSDirectoryServerCertificate *certificate = [STDSDirectoryServerCertificate customCertificateWithData:[[NSData alloc] initWithBase64EncodedString:certificateString options:0]]; + XCTAssertNotNil(certificate, @"Failed to create certificate from string."); + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Parsed incorrect key type from custom certificate"); +} + +- (void)testCustomCertificateWithString { + // Using ds-amex.pm.PEM + NSString *certificateString = @"MIIE0TCCA7mgAwIBAgIUXbeqM1duFcHk4dDBwT8o7Ln5wX8wDQYJKoZIhvcNAQEL" + "BQAwXjELMAkGA1UEBhMCVVMxITAfBgNVBAoTGEFtZXJpY2FuIEV4cHJlc3MgQ29t" + "cGFueTEsMCoGA1UEAxMjQW1lcmljYW4gRXhwcmVzcyBTYWZla2V5IElzc3Vpbmcg" + "Q0EwHhcNMTgwMjIxMjM0OTMxWhcNMjAwMjIxMjM0OTMwWjCB0DELMAkGA1UEBhMC" + "VVMxETAPBgNVBAgTCE5ldyBZb3JrMREwDwYDVQQHEwhOZXcgWW9yazE/MD0GA1UE" + "ChM2QW1lcmljYW4gRXhwcmVzcyBUcmF2ZWwgUmVsYXRlZCBTZXJ2aWNlcyBDb21w" + "YW55LCBJbmMuMTkwNwYDVQQLEzBHbG9iYWwgTmV0d29yayBUZWNobm9sb2d5IC0g" + "TmV0d29yayBBUEkgUGxhdGZvcm0xHzAdBgNVBAMTFlNESy5TYWZlS2V5LkVuY3J5" + "cHRLZXkwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQDSFF9kTYbwRrxX" + "C6WcJJYio5TZDM62+CnjQRfggV3GMI+xIDtMIN8LL/jbWBTycu97vrNjNNv+UPhI" + "WzhFDdUqyRfrY337A39uE8k1xhdDI3dNeZz6xgq8r9hn2NBou78YPBKidpN5oiHn" + "TxcFq1zudut2fmaldaa9a4ZKgIQo+02heiJfJ8XNWkoWJ17GcjJ59UU8C1KF/y1G" + "ymYO5ha2QRsVZYI17+ZFsqnpcXwK4Mr6RQKV6UimmO0nr5++CgvXfekcWAlLV6Xq" + "juACWi3kw0haepaX/9qHRu1OSyjzWNcSVZ0On6plB5Lq6Y9ylgmxDrv+zltz3MrT" + "K7txIAFFAgMBAAGjggESMIIBDjAMBgNVHRMBAf8EAjAAMCEGA1UdEQQaMBiCFlNE" + "Sy5TYWZlS2V5LkVuY3J5cHRLZXkwRQYJKwYBBAGCNxQCBDgeNgBBAE0ARQBYAF8A" + "UwBBAEYARQBLAEUAWQAyAF8ARABTAF8ARQBOAEMAUgBZAFAAVABJAE8ATjAOBgNV" + "HQ8BAf8EBAMCBJAwHwYDVR0jBBgwFoAU7k/rXuVMhTBxB1zSftPgmLFuDIgwRAYD" + "VR0fBD0wOzA5oDegNYYzaHR0cDovL2FtZXhzay5jcmwuY29tLXN0cm9uZy1pZC5u" + "ZXQvYW1leHNhZmVrZXkuY3JsMB0GA1UdDgQWBBQHclVTo5nwZGH8labJ2F2P45xi" + "fDANBgkqhkiG9w0BAQsFAAOCAQEAWY6b77VBoGLs3k5vOqSU7QRqT+4v6y77T8LA" + "BKrSZ58DiVZWVyDSxyftQUiRRgFHt2gTN0yfJTP50Fyp84nCEWC0tugZ4iIhgPss" + "HzL+4/u4eG/MTzK2ESxvPgr6YHajyuU+GXA89u8+bsFrFmojOjhTgFKli7YUeV/0" + "xoiYZf2utlns800ofJrcrfiFoqE6PvK4Od0jpeMgfSKv71nK5ihA1+wTk76ge1fs" + "PxL23hEdRpWW11ofaLfJGkLFXMM3/LHSXWy7HhsBgDELdzLSHU4VkSv8yTOZxsRO" + "ByxdC5v3tXGcK56iQdtKVPhFGOOEBugw7AcuRzv3f1GhvzAQZg=="; + STDSDirectoryServerCertificate *certificate = certificate = [STDSDirectoryServerCertificate customCertificateWithString:certificateString]; + XCTAssertNotNil(certificate, @"Failed to create certificate from string."); + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Parsed incorrect key type from custom certificate"); + + // 3ds2.rsa.encryption.crt + certificateString = @"-----BEGIN CERTIFICATE-----" + "MIIFrjCCBJagAwIBAgIQB2rJmsHVwbONd36WP9QPrTANBgkqhkiG9w0BAQsFADBx" + "MQswCQYDVQQGEwJVUzENMAsGA1UEChMEVklTQTEvMC0GA1UECxMmVmlzYSBJbnRl" + "cm5hdGlvbmFsIFNlcnZpY2UgQXNzb2NpYXRpb24xIjAgBgNVBAMTGVZpc2EgZUNv" + "bW1lcmNlIElzc3VpbmcgQ0EwHhcNMTcxMTAyMjIyMzEwWhcNMjAxMTAzMDAyMzEw" + "WjCBoTEYMBYGA1UEBxMPSGlnaGxhbmRzIFJhbmNoMREwDwYDVQQIEwhDb2xvcmFk" + "bzELMAkGA1UEBhMCVVMxDTALBgNVBAoTBFZJU0ExLzAtBgNVBAsTJlZpc2EgSW50" + "ZXJuYXRpb25hbCBTZXJ2aWNlIEFzc29jaWF0aW9uMSUwIwYDVQQDExwzZHMyLnJz" + "YS5lbmNyeXB0aW9uLnZpc2EuY29tMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIB" + "CgKCAQEAst+HGfPPsX3p6HHEQ9YzourlQj16Nscmm13Cp7cZe4dZB2oWnJqZ7oh/" + "pEoEoOAxBw1x4NFgXKTKdHAeu3VBNVw8SwMTdIC+X16VV+3VIyPbUvJXFp3QoR8W" + "UwPB3F1Lb9SMFNS95boYDZKIOdPW0cP1dRi7pFugsBUZDCP/H3nFfBFHMCBoga+P" + "3AHGj5y8RVpv0hS9jaIsYjX+i58B61OGCB7D0AiADNZJuFzw2+xpNkt6NJJF66FP" + "O8qIh8xR2xGVDf7TtCbss/CugLRgSqKab9YRB8/TBTcy5bxj6O8HD6aL2zGLcMY9" + "dCobXxCodLEtMjJdVL8N+iZrsI2gtwIDAQABo4ICDzCCAgswEwYDVR0lBAwwCgYI" + "KwYBBQUHAwEwZQYIKwYBBQUHAQEEWTBXMCUGCCsGAQUFBzABhhlodHRwOi8vb2Nz" + "cC52aXNhLmNvbS9vY3NwMC4GCCsGAQUFBzAChiJodHRwOi8vZW5yb2xsLnZpc2Fj" + "YS5jb20vZWNvbW0uY2VyMB8GA1UdIwQYMBaAFN/DKlUuL0I6ekCdkqD3R3nXj4eK" + "MAwGA1UdEwEB/wQCMAAwgcoGA1UdHwSBwjCBvzAooCagJIYiaHR0cDovL0Vucm9s" + "bC52aXNhY2EuY29tL2VDb21tLmNybDCBkqCBj6CBjIaBiWxkYXA6Ly9FbnJvbGwu" + "dmlzYWNhLmNvbTozODkvY249VmlzYSBlQ29tbWVyY2UgSXNzdWluZyBDQSxjPVVT" + "LG91PVZpc2EgSW50ZXJuYXRpb25hbCBTZXJ2aWNlIEFzc29jaWF0aW9uLG89VklT" + "QT9jZXJ0aWZpY2F0ZVJldm9jYXRpb25MaXN0MA4GA1UdDwEB/wQEAwIFoDAnBgNV" + "HREEIDAeghwzZHMyLnJzYS5lbmNyeXB0aW9uLnZpc2EuY29tMB0GA1UdDgQWBBT8" + "m2pDtUtY13f/3NCOmexHavP5zDA5BgNVHSAEMjAwMC4GBWeBAwEBMCUwIwYIKwYB" + "BQUHAgEWF2h0dHA6Ly93d3cudmlzYS5jb20vcGtpMA0GCSqGSIb3DQEBCwUAA4IB" + "AQCcCUhU7KnHUDuLXqwSuzC8lWWCcEqPRgPPzY3mgBUg9ya0p+v7QF2BG77tpygK" + "E2yDPkOE8trzYeMi7TCuvKgZvUXDSOka8SId9QleMBlo2pzNi0vKKBG8+E7qmGaf" + "etQHVaoFvhg24/e7y8q89VYNKfLXn8TWMUOJdTQoNP+4bHcCnBvWWUcI2LlyEog1" + "2FDSG8hgP3cpw+0B2Hace9BQGR7ZgTIJAANEHZ54QGOYdxZEcDS5IEpKZlN8INs/" + "NKJyCkqP09VA4NO/WHaGFAXtgoLmjlA9Kal+4ieJPKijVDxcHVv/uPSfVQJ0/vCa" + "udJGOXV9q4VteupwLxfOGW8w" + "-----END CERTIFICATE-----"; + + certificateString = @"-----BEGIN CERTIFICATE-----\n" + "MIIFrjCCBJagAwIBAgIQB2rJmsHVwbONd36WP9QPrTANBgkqhkiG9w0BAQsFADBx\n" + "MQswCQYDVQQGEwJVUzENMAsGA1UEChMEVklTQTEvMC0GA1UECxMmVmlzYSBJbnRl\n" + "cm5hdGlvbmFsIFNlcnZpY2UgQXNzb2NpYXRpb24xIjAgBgNVBAMTGVZpc2EgZUNv\n" + "bW1lcmNlIElzc3VpbmcgQ0EwHhcNMTcxMTAyMjIyMzEwWhcNMjAxMTAzMDAyMzEw\n" + "WjCBoTEYMBYGA1UEBxMPSGlnaGxhbmRzIFJhbmNoMREwDwYDVQQIEwhDb2xvcmFk\n" + "bzELMAkGA1UEBhMCVVMxDTALBgNVBAoTBFZJU0ExLzAtBgNVBAsTJlZpc2EgSW50\n" + "ZXJuYXRpb25hbCBTZXJ2aWNlIEFzc29jaWF0aW9uMSUwIwYDVQQDExwzZHMyLnJz\n" + "YS5lbmNyeXB0aW9uLnZpc2EuY29tMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIB\n" + "CgKCAQEAst+HGfPPsX3p6HHEQ9YzourlQj16Nscmm13Cp7cZe4dZB2oWnJqZ7oh/\n" + "pEoEoOAxBw1x4NFgXKTKdHAeu3VBNVw8SwMTdIC+X16VV+3VIyPbUvJXFp3QoR8W\n" + "UwPB3F1Lb9SMFNS95boYDZKIOdPW0cP1dRi7pFugsBUZDCP/H3nFfBFHMCBoga+P\n" + "3AHGj5y8RVpv0hS9jaIsYjX+i58B61OGCB7D0AiADNZJuFzw2+xpNkt6NJJF66FP\n" + "O8qIh8xR2xGVDf7TtCbss/CugLRgSqKab9YRB8/TBTcy5bxj6O8HD6aL2zGLcMY9\n" + "dCobXxCodLEtMjJdVL8N+iZrsI2gtwIDAQABo4ICDzCCAgswEwYDVR0lBAwwCgYI\n" + "KwYBBQUHAwEwZQYIKwYBBQUHAQEEWTBXMCUGCCsGAQUFBzABhhlodHRwOi8vb2Nz\n" + "cC52aXNhLmNvbS9vY3NwMC4GCCsGAQUFBzAChiJodHRwOi8vZW5yb2xsLnZpc2Fj\n" + "YS5jb20vZWNvbW0uY2VyMB8GA1UdIwQYMBaAFN/DKlUuL0I6ekCdkqD3R3nXj4eK\n" + "MAwGA1UdEwEB/wQCMAAwgcoGA1UdHwSBwjCBvzAooCagJIYiaHR0cDovL0Vucm9s\n" + "bC52aXNhY2EuY29tL2VDb21tLmNybDCBkqCBj6CBjIaBiWxkYXA6Ly9FbnJvbGwu\n" + "dmlzYWNhLmNvbTozODkvY249VmlzYSBlQ29tbWVyY2UgSXNzdWluZyBDQSxjPVVT\n" + "LG91PVZpc2EgSW50ZXJuYXRpb25hbCBTZXJ2aWNlIEFzc29jaWF0aW9uLG89VklT\n" + "QT9jZXJ0aWZpY2F0ZVJldm9jYXRpb25MaXN0MA4GA1UdDwEB/wQEAwIFoDAnBgNV\n" + "HREEIDAeghwzZHMyLnJzYS5lbmNyeXB0aW9uLnZpc2EuY29tMB0GA1UdDgQWBBT8\n" + "m2pDtUtY13f/3NCOmexHavP5zDA5BgNVHSAEMjAwMC4GBWeBAwEBMCUwIwYIKwYB\n" + "BQUHAgEWF2h0dHA6Ly93d3cudmlzYS5jb20vcGtpMA0GCSqGSIb3DQEBCwUAA4IB\n" + "AQCcCUhU7KnHUDuLXqwSuzC8lWWCcEqPRgPPzY3mgBUg9ya0p+v7QF2BG77tpygK\n" + "E2yDPkOE8trzYeMi7TCuvKgZvUXDSOka8SId9QleMBlo2pzNi0vKKBG8+E7qmGaf\n" + "etQHVaoFvhg24/e7y8q89VYNKfLXn8TWMUOJdTQoNP+4bHcCnBvWWUcI2LlyEog1\n" + "2FDSG8hgP3cpw+0B2Hace9BQGR7ZgTIJAANEHZ54QGOYdxZEcDS5IEpKZlN8INs/\n" + "NKJyCkqP09VA4NO/WHaGFAXtgoLmjlA9Kal+4ieJPKijVDxcHVv/uPSfVQJ0/vCa\n" + "udJGOXV9q4VteupwLxfOGW8w\n" + "-----END CERTIFICATE-----\n"; + certificate = [STDSDirectoryServerCertificate customCertificateWithString:certificateString]; + XCTAssertNotNil(certificate, @"Failed to create certificate from string."); + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Parsed incorrect key type from custom certificate"); + + certificateString = @"-----BEGIN CERTIFICATE-----" + "-----END CERTIFICATE-----"; + certificate = [STDSDirectoryServerCertificate customCertificateWithString:certificateString]; + XCTAssertNil(certificate, @"Should not return a valid value with only anchors"); + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Parsed incorrect key type from custom certificate"); + certificateString = @"-----END CERTIFICATE-----"; + + certificate = [STDSDirectoryServerCertificate customCertificateWithString:certificateString]; + XCTAssertNil(certificate, @"Should not return a valid value with only suffix anchor"); + XCTAssertEqual(certificate.keyType, STDSDirectoryServerKeyTypeRSA, @"Parsed incorrect key type from custom certificate"); +} + +- (void)testCertificateChainValidation { + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *certificateString = @"MIIDXTCCAkWgAwIBAgIQbS4C4BSig7uuJ5uDpeT4VjANBgkqhkiG9w0BAQsFADBH" + "MRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYHZXhhbXBsZTEX" + "MBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE0ODQ5WhcNMjcxMjMx" + "MTQwMDAwWjBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYH" + "ZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwggEiMA0GCSqGSIb3DQEB" + "AQUAA4IBDwAwggEKAoIBAQCfgQ+0A4Jz0CWR5Ac/MdK2ABuCzttNkvBQFl1Hz8q4" + "o8Qct3isdVN5P475dXaNGiN02HElZMO813uepDRUSJlAfP8AmZIKkxokxEFIUqsp" + "vbCpXAZT82xg5gv5C2JY3aVvNwR7pcLR0CmvnJ1AuseqQceKDdEGit1pnoCP6gEe" + "oUQdik97tOl7459V8d3UTpxLozUVlwPU00tgPmUUek8j1tPAmWx17e6EaoLRkK4Q" + "eDyWHPA4eu0hBtLQVVtv2Tf61VNTh+D/cv++eJQUArC4IuoqdLYFjB2r+bNKdstj" + "uH+qLGhHuOKDf/+RGG5rHBSRHPmJqJCSqBzmAd2s0/nPAgMBAAGjRTBDMBIGA1Ud" + "EwEB/wQIMAYBAf8CAQAwDgYDVR0PAQH/BAQDAgEGMB0GA1UdDgQWBBTDgwKdvAPq" + "bbCmehDaw0PwavI83jANBgkqhkiG9w0BAQsFAAOCAQEAOUcKqpzNQ6lr0PbDSsns" + "D6onfi+8j3TD0xG0zBSf+8G4zs8Zb6vzzQ5qHKgfr4aeen8Pw0cw2KKUJ2dFaBqj" + "n3/6/MIZbgaBvXKUbmY8xCxKQ+tOFc3KWIu4pSaO50tMPJjU/lP35bv19AA9vs9M" + "TKY2qLf88bmoNYT3W8VSDcB58KBHa7HVIPx7BUUtSyb2N2Jqx5AOiYy4NarhB3hV" + "ftkZBmCzi2Qw50KWIgTFYcIVeRTx3Js/F0IuEdgZHBK2gmO7fdM7+QKYm83401vl" + "YRNCXfIZ0H9E1V3NddqJuqIutdUajckSzMhXdNCJqfI4FAQAymTWGL3/lZyr/30x" + "Fg=="; + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wobjc-string-concatenation" + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSArray *certChain = @[@"MIIDeTCCAmGgAwIBAgIQbS4C4BSig7uuJ5uDpeT4WDANB" + "gkqhkiG9w0BAQsFADBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBG" + "RYHZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE1NDAyW" + "hcNMjcxMjMxMTMzMDAwWjBIMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyL" + "GQBGRYHZXhhbXBsZTEYMBYGA1UEAwwPUlNBIEV4YW1wbGUgQUNTMIIBIjANBgkqhkiG9" + "w0BAQEFAAOCAQ8AMIIBCgKCAQEAkNrPIBDXMU6fcyv5i+QHQAQ+K8gsC3HJb7FYhYaw8" + "hXbNJa+t8q0lDKwLZgQXYV+ffWxXJv5GGrlZE4GU52lfMEegTDzYTrRQ3tepgKFjMGg6" + "Iy6fkl1ZNsx2gEonsnlShfzA9GJwRTmtKPbk1s+hwx1IU5AT+AIelNqBgcF2vE5W25/S" + "GGBoaROVdUYxqETDggM1z5cKV4ZjDZ8+lh4oVB07bkac6LQdHpJUUySH/Er20DXx30Ky" + "i97PciXKTS+QKXnmm8ivyRCmux22ZoPUind2BKC5OiG4MwALhaL2Z2k8CsRdfy+7dg7z" + "41Rp6D0ZeEvtaUp4bX4aKraL4rTfwIDAQABo2AwXjAMBgNVHRMBAf8EAjAAMA4GA1UdD" + "wEB/wQEAwIHgDAdBgNVHQ4EFgQUktwf6ZpTCxjYKw/BLW6PeiNX4swwHwYDVR0jBBgwF" + "oAUw4MCnbwD6m2wpnoQ2sND8GryPN4wDQYJKoZIhvcNAQELBQADggEBAGuNHxv/BR6j7" + "lCPysm1uhrbjBOqdrhJMR/Id4dB2GtdEScl3irGPmXyQ2SncTWhNfsgsKDZWp5Bk7+Ot" + "nty0eNUMk3hZEqgYjxhzau048XHbsfGvoJaMGZZNTwUvTUz2hkkhgpx9yQAKIA2LzFKc" + "gYhelPu4GW5rtEuxu3IS6WYy3D1GtF3naEWkjUra8hQOhOl2S+CYHmRd6lGkXykVDajM" + "gd2AJFzXdKLxTt0OYrWDGlUSzGACRBCd5xbRmATIldtccaGqDN1cNWv0I/bPN8EpKS6B" + "0WaZcPasItKWpDC85Jw1GrDxdhwoKHoxtSG+odiTwB5zLbrn2OsRE5bV7E="]; +#pragma clang diagnostic pop + + XCTAssertTrue([STDSDirectoryServerCertificate _verifyCertificateChain:certChain withRootCertificates:@[certificateString]], @"Failed to verify certificate chain."); + certChain = @[@"junk_data", @"more_junk"]; + XCTAssertFalse([STDSDirectoryServerCertificate _verifyCertificateChain:certChain withRootCertificates:@[certificateString]], @"Verified certificate chain with invalid certificates."); +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wobjc-string-concatenation" + // Test with valid certificates in the chain, but that are not related to the root certificate + // ref. https://tools.ietf.org/html/rfc7515 + certChain = @[ + @"MIIE3jCCA8agAwIBAgICAwEwDQYJKoZIhvcNAQEFBQAwYzELMAkGA1UEBhMCVVM" + "xITAfBgNVBAoTGFRoZSBHbyBEYWRkeSBHcm91cCwgSW5jLjExMC8GA1UECxMoR2" + "8gRGFkZHkgQ2xhc3MgMiBDZXJ0aWZpY2F0aW9uIEF1dGhvcml0eTAeFw0wNjExM" + "TYwMTU0MzdaFw0yNjExMTYwMTU0MzdaMIHKMQswCQYDVQQGEwJVUzEQMA4GA1UE" + "CBMHQXJpem9uYTETMBEGA1UEBxMKU2NvdHRzZGFsZTEaMBgGA1UEChMRR29EYWR" + "keS5jb20sIEluYy4xMzAxBgNVBAsTKmh0dHA6Ly9jZXJ0aWZpY2F0ZXMuZ29kYW" + "RkeS5jb20vcmVwb3NpdG9yeTEwMC4GA1UEAxMnR28gRGFkZHkgU2VjdXJlIENlc" + "nRpZmljYXRpb24gQXV0aG9yaXR5MREwDwYDVQQFEwgwNzk2OTI4NzCCASIwDQYJ" + "KoZIhvcNAQEBBQADggEPADCCAQoCggEBAMQt1RWMnCZM7DI161+4WQFapmGBWTt" + "wY6vj3D3HKrjJM9N55DrtPDAjhI6zMBS2sofDPZVUBJ7fmd0LJR4h3mUpfjWoqV" + "Tr9vcyOdQmVZWt7/v+WIbXnvQAjYwqDL1CBM6nPwT27oDyqu9SoWlm2r4arV3aL" + "GbqGmu75RpRSgAvSMeYddi5Kcju+GZtCpyz8/x4fKL4o/K1w/O5epHBp+YlLpyo" + "7RJlbmr2EkRTcDCVw5wrWCs9CHRK8r5RsL+H0EwnWGu1NcWdrxcx+AuP7q2BNgW" + "JCJjPOq8lh8BJ6qf9Z/dFjpfMFDniNoW1fho3/Rb2cRGadDAW/hOUoz+EDU8CAw" + "EAAaOCATIwggEuMB0GA1UdDgQWBBT9rGEyk2xF1uLuhV+auud2mWjM5zAfBgNVH" + "SMEGDAWgBTSxLDSkdRMEXGzYcs9of7dqGrU4zASBgNVHRMBAf8ECDAGAQH/AgEA" + "MDMGCCsGAQUFBwEBBCcwJTAjBggrBgEFBQcwAYYXaHR0cDovL29jc3AuZ29kYWR" + "keS5jb20wRgYDVR0fBD8wPTA7oDmgN4Y1aHR0cDovL2NlcnRpZmljYXRlcy5nb2" + "RhZGR5LmNvbS9yZXBvc2l0b3J5L2dkcm9vdC5jcmwwSwYDVR0gBEQwQjBABgRVH" + "SAAMDgwNgYIKwYBBQUHAgEWKmh0dHA6Ly9jZXJ0aWZpY2F0ZXMuZ29kYWRkeS5j" + "b20vcmVwb3NpdG9yeTAOBgNVHQ8BAf8EBAMCAQYwDQYJKoZIhvcNAQEFBQADggE" + "BANKGwOy9+aG2Z+5mC6IGOgRQjhVyrEp0lVPLN8tESe8HkGsz2ZbwlFalEzAFPI" + "UyIXvJxwqoJKSQ3kbTJSMUA2fCENZvD117esyfxVgqwcSeIaha86ykRvOe5GPLL" + "5CkKSkB2XIsKd83ASe8T+5o0yGPwLPk9Qnt0hCqU7S+8MxZC9Y7lhyVJEnfzuz9" + "p0iRFEUOOjZv2kWzRaJBydTXRE4+uXR21aITVSzGh6O1mawGhId/dQb8vxRMDsx" + "uxN89txJx9OjxUUAiKEngHUuHqDTMBqLdElrRhjZkAzVvb3du6/KFUJheqwNTrZ" + "EjYx8WnM25sgVjOuH0aBsXBTWVU+4=", + @"MIIE+zCCBGSgAwIBAgICAQ0wDQYJKoZIhvcNAQEFBQAwgbsxJDAiBgNVBAcTG1Z" + "hbGlDZXJ0IFZhbGlkYXRpb24gTmV0d29yazEXMBUGA1UEChMOVmFsaUNlcnQsIE" + "luYy4xNTAzBgNVBAsTLFZhbGlDZXJ0IENsYXNzIDIgUG9saWN5IFZhbGlkYXRpb" + "24gQXV0aG9yaXR5MSEwHwYDVQQDExhodHRwOi8vd3d3LnZhbGljZXJ0LmNvbS8x" + "IDAeBgkqhkiG9w0BCQEWEWluZm9AdmFsaWNlcnQuY29tMB4XDTA0MDYyOTE3MDY" + "yMFoXDTI0MDYyOTE3MDYyMFowYzELMAkGA1UEBhMCVVMxITAfBgNVBAoTGFRoZS" + "BHbyBEYWRkeSBHcm91cCwgSW5jLjExMC8GA1UECxMoR28gRGFkZHkgQ2xhc3MgM" + "iBDZXJ0aWZpY2F0aW9uIEF1dGhvcml0eTCCASAwDQYJKoZIhvcNAQEBBQADggEN" + "ADCCAQgCggEBAN6d1+pXGEmhW+vXX0iG6r7d/+TvZxz0ZWizV3GgXne77ZtJ6XC" + "APVYYYwhv2vLM0D9/AlQiVBDYsoHUwHU9S3/Hd8M+eKsaA7Ugay9qK7HFiH7Eux" + "6wwdhFJ2+qN1j3hybX2C32qRe3H3I2TqYXP2WYktsqbl2i/ojgC95/5Y0V4evLO" + "tXiEqITLdiOr18SPaAIBQi2XKVlOARFmR6jYGB0xUGlcmIbYsUfb18aQr4CUWWo" + "riMYavx4A6lNf4DD+qta/KFApMoZFv6yyO9ecw3ud72a9nmYvLEHZ6IVDd2gWMZ" + "Eewo+YihfukEHU1jPEX44dMX4/7VpkI+EdOqXG68CAQOjggHhMIIB3TAdBgNVHQ" + "4EFgQU0sSw0pHUTBFxs2HLPaH+3ahq1OMwgdIGA1UdIwSByjCBx6GBwaSBvjCBu" + "zEkMCIGA1UEBxMbVmFsaUNlcnQgVmFsaWRhdGlvbiBOZXR3b3JrMRcwFQYDVQQK" + "Ew5WYWxpQ2VydCwgSW5jLjE1MDMGA1UECxMsVmFsaUNlcnQgQ2xhc3MgMiBQb2x" + "pY3kgVmFsaWRhdGlvbiBBdXRob3JpdHkxITAfBgNVBAMTGGh0dHA6Ly93d3cudm" + "FsaWNlcnQuY29tLzEgMB4GCSqGSIb3DQEJARYRaW5mb0B2YWxpY2VydC5jb22CA" + "QEwDwYDVR0TAQH/BAUwAwEB/zAzBggrBgEFBQcBAQQnMCUwIwYIKwYBBQUHMAGG" + "F2h0dHA6Ly9vY3NwLmdvZGFkZHkuY29tMEQGA1UdHwQ9MDswOaA3oDWGM2h0dHA" + "6Ly9jZXJ0aWZpY2F0ZXMuZ29kYWRkeS5jb20vcmVwb3NpdG9yeS9yb290LmNybD" + "BLBgNVHSAERDBCMEAGBFUdIAAwODA2BggrBgEFBQcCARYqaHR0cDovL2NlcnRpZ" + "mljYXRlcy5nb2RhZGR5LmNvbS9yZXBvc2l0b3J5MA4GA1UdDwEB/wQEAwIBBjAN" + "BgkqhkiG9w0BAQUFAAOBgQC1QPmnHfbq/qQaQlpE9xXUhUaJwL6e4+PrxeNYiY+" + "Sn1eocSxI0YGyeR+sBjUZsE4OWBsUs5iB0QQeyAfJg594RAoYC5jcdnplDQ1tgM" + "QLARzLrUc+cb53S8wGd9D0VmsfSxOaFIqII6hR8INMqzW/Rn453HWkrugp++85j" + "09VZw==", + @"MIIC5zCCAlACAQEwDQYJKoZIhvcNAQEFBQAwgbsxJDAiBgNVBAcTG1ZhbGlDZXJ" + "0IFZhbGlkYXRpb24gTmV0d29yazEXMBUGA1UEChMOVmFsaUNlcnQsIEluYy4xNT" + "AzBgNVBAsTLFZhbGlDZXJ0IENsYXNzIDIgUG9saWN5IFZhbGlkYXRpb24gQXV0a" + "G9yaXR5MSEwHwYDVQQDExhodHRwOi8vd3d3LnZhbGljZXJ0LmNvbS8xIDAeBgkq" + "hkiG9w0BCQEWEWluZm9AdmFsaWNlcnQuY29tMB4XDTk5MDYyNjAwMTk1NFoXDTE" + "5MDYyNjAwMTk1NFowgbsxJDAiBgNVBAcTG1ZhbGlDZXJ0IFZhbGlkYXRpb24gTm" + "V0d29yazEXMBUGA1UEChMOVmFsaUNlcnQsIEluYy4xNTAzBgNVBAsTLFZhbGlDZ" + "XJ0IENsYXNzIDIgUG9saWN5IFZhbGlkYXRpb24gQXV0aG9yaXR5MSEwHwYDVQQD" + "ExhodHRwOi8vd3d3LnZhbGljZXJ0LmNvbS8xIDAeBgkqhkiG9w0BCQEWEWluZm9" + "AdmFsaWNlcnQuY29tMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDOOnHK5a" + "vIWZJV16vYdA757tn2VUdZZUcOBVXc65g2PFxTXdMwzzjsvUGJ7SVCCSRrCl6zf" + "N1SLUzm1NZ9WlmpZdRJEy0kTRxQb7XBhVQ7/nHk01xC+YDgkRoKWzk2Z/M/VXwb" + "P7RfZHM047QSv4dk+NoS/zcnwbNDu+97bi5p9wIDAQABMA0GCSqGSIb3DQEBBQU" + "AA4GBADt/UG9vUJSZSWI4OB9L+KXIPqeCgfYrx+jFzug6EILLGACOTb2oWH+heQ" + "C1u+mNr0HZDzTuIYEZoDJJKPTEjlbVUjP9UNV+mWwD5MlM/Mtsq2azSiGM5bUMM" + "j4QssxsodyamEwCW/POuZ6lcg5Ktz885hZo+L7tdEy8W9ViH0Pd", + @"MIIDeTCCAmGgAwIBAgIQbS4C4BSig7uuJ5uDpeT4WDANB" + "gkqhkiG9w0BAQsFADBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBG" + "RYHZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE1NDAyW" + "hcNMjcxMjMxMTMzMDAwWjBIMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyL" + "GQBGRYHZXhhbXBsZTEYMBYGA1UEAwwPUlNBIEV4YW1wbGUgQUNTMIIBIjANBgkqhkiG9" + "w0BAQEFAAOCAQ8AMIIBCgKCAQEAkNrPIBDXMU6fcyv5i+QHQAQ+K8gsC3HJb7FYhYaw8" + "hXbNJa+t8q0lDKwLZgQXYV+ffWxXJv5GGrlZE4GU52lfMEegTDzYTrRQ3tepgKFjMGg6" + "Iy6fkl1ZNsx2gEonsnlShfzA9GJwRTmtKPbk1s+hwx1IU5AT+AIelNqBgcF2vE5W25/S" + "GGBoaROVdUYxqETDggM1z5cKV4ZjDZ8+lh4oVB07bkac6LQdHpJUUySH/Er20DXx30Ky" + "i97PciXKTS+QKXnmm8ivyRCmux22ZoPUind2BKC5OiG4MwALhaL2Z2k8CsRdfy+7dg7z" + "41Rp6D0ZeEvtaUp4bX4aKraL4rTfwIDAQABo2AwXjAMBgNVHRMBAf8EAjAAMA4GA1UdD" + "wEB/wQEAwIHgDAdBgNVHQ4EFgQUktwf6ZpTCxjYKw/BLW6PeiNX4swwHwYDVR0jBBgwF" + "oAUw4MCnbwD6m2wpnoQ2sND8GryPN4wDQYJKoZIhvcNAQELBQADggEBAGuNHxv/BR6j7" + "lCPysm1uhrbjBOqdrhJMR/Id4dB2GtdEScl3irGPmXyQ2SncTWhNfsgsKDZWp5Bk7+Ot" + "nty0eNUMk3hZEqgYjxhzau048XHbsfGvoJaMGZZNTwUvTUz2hkkhgpx9yQAKIA2LzFKc" + "gYhelPu4GW5rtEuxu3IS6WYy3D1GtF3naEWkjUra8hQOhOl2S+CYHmRd6lGkXykVDajM" + "gd2AJFzXdKLxTt0OYrWDGlUSzGACRBCd5xbRmATIldtccaGqDN1cNWv0I/bPN8EpKS6B" + "0WaZcPasItKWpDC85Jw1GrDxdhwoKHoxtSG+odiTwB5zLbrn2OsRE5bV7E=",]; + XCTAssertFalse([STDSDirectoryServerCertificate _verifyCertificateChain:certChain withRootCertificates:@[certificateString]], @"Verified invalid certificate chain."); + + // stripe.com's cert chain retrieved by https://whatsmychaincert.com/ + // If below test cases start failing, the certs may have expired and should be replaced with newer ones + // from whatsmychaincert.com, select Generate Cert Chain, include root for stripe.com + // print the entire certificate chain with `openssl crl2pkcs7 -nocrl -certfile | openssl pkcs7 -print_certs -text` + // The root is the last printed certificate and the cert chain should be ordered with the top most at the end of certChain + + certChain = @[ + @"MIIEtjCCA56gAwIBAgIQDHmpRLCMEZUgkmFf4msdgzANBgkqhkiG9w0BAQsFADBs" + "MQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3" + "d3cuZGlnaWNlcnQuY29tMSswKQYDVQQDEyJEaWdpQ2VydCBIaWdoIEFzc3VyYW5j" + "ZSBFViBSb290IENBMB4XDTEzMTAyMjEyMDAwMFoXDTI4MTAyMjEyMDAwMFowdTEL" + "MAkGA1UEBhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3" + "LmRpZ2ljZXJ0LmNvbTE0MDIGA1UEAxMrRGlnaUNlcnQgU0hBMiBFeHRlbmRlZCBW" + "YWxpZGF0aW9uIFNlcnZlciBDQTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoC" + "ggEBANdTpARR+JmmFkhLZyeqk0nQOe0MsLAAh/FnKIaFjI5j2ryxQDji0/XspQUY" + "uD0+xZkXMuwYjPrxDKZkIYXLBxA0sFKIKx9om9KxjxKws9LniB8f7zh3VFNfgHk/" + "LhqqqB5LKw2rt2O5Nbd9FLxZS99RStKh4gzikIKHaq7q12TWmFXo/a8aUGxUvBHy" + "/Urynbt/DvTVvo4WiRJV2MBxNO723C3sxIclho3YIeSwTQyJ3DkmF93215SF2AQh" + "cJ1vb/9cuhnhRctWVyh+HA1BV6q3uCe7seT6Ku8hI3UarS2bhjWMnHe1c63YlC3k" + "8wyd7sFOYn4XwHGeLN7x+RAoGTMCAwEAAaOCAUkwggFFMBIGA1UdEwEB/wQIMAYB" + "Af8CAQAwDgYDVR0PAQH/BAQDAgGGMB0GA1UdJQQWMBQGCCsGAQUFBwMBBggrBgEF" + "BQcDAjA0BggrBgEFBQcBAQQoMCYwJAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRp" + "Z2ljZXJ0LmNvbTBLBgNVHR8ERDBCMECgPqA8hjpodHRwOi8vY3JsNC5kaWdpY2Vy" + "dC5jb20vRGlnaUNlcnRIaWdoQXNzdXJhbmNlRVZSb290Q0EuY3JsMD0GA1UdIAQ2" + "MDQwMgYEVR0gADAqMCgGCCsGAQUFBwIBFhxodHRwczovL3d3dy5kaWdpY2VydC5j" + "b20vQ1BTMB0GA1UdDgQWBBQ901Cl1qCt7vNKYApl0yHU+PjWDzAfBgNVHSMEGDAW" + "gBSxPsNpA/i/RwHUmCYaCALvY2QrwzANBgkqhkiG9w0BAQsFAAOCAQEAnbbQkIbh" + "hgLtxaDwNBx0wY12zIYKqPBKikLWP8ipTa18CK3mtlC4ohpNiAexKSHc59rGPCHg" + "4xFJcKx6HQGkyhE6V6t9VypAdP3THYUYUN9XR3WhfVUgLkc3UHKMf4Ib0mKPLQNa" + "2sPIoc4sUqIAY+tzunHISScjl2SFnjgOrWNoPLpSgVh5oywM395t6zHyuqB8bPEs" + "1OG9d4Q3A84ytciagRpKkk47RpqF/oOi+Z6Mo8wNXrM9zwR4jxQUezKcxwCmXMS1" + "oVWNWlZopCJwqjyBcdmdqEU79OX2olHdx3ti6G8MdOu42vi/hw15UJGQmxg7kVkn" + "8TUoE6smftX3eg==", + + @"MIIHQDCCBiigAwIBAgIQD2ygPziYarLZojJGDizjjzANBgkqhkiG9w0BAQsFADB1" + "MQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3" + "d3cuZGlnaWNlcnQuY29tMTQwMgYDVQQDEytEaWdpQ2VydCBTSEEyIEV4dGVuZGVk" + "IFZhbGlkYXRpb24gU2VydmVyIENBMB4XDTIxMTAyMDAwMDAwMFoXDTIyMDIwMjIz" + "NTk1OVowgcYxHTAbBgNVBA8MFFByaXZhdGUgT3JnYW5pemF0aW9uMRMwEQYLKwYB" + "BAGCNzwCAQMTAlVTMRkwFwYLKwYBBAGCNzwCAQITCERlbGF3YXJlMRAwDgYDVQQF" + "Ewc0Njc1NTA2MQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQG" + "A1UEBxMNU2FuIEZyYW5jaXNjbzEUMBIGA1UEChMLU3RyaXBlLCBJbmMxEzARBgNV" + "BAMTCnN0cmlwZS5jb20wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC1" + "ZyRTERjtA28e16WX4AUBBbXE8YTA6F5RNp5NEy2px6GF2LbjR8TobVfJoj63+CXR" + "W4uox0TQV526ZnPKbvYpAilYxEc9/fJTPS3lDJ4EgKRZlZ97UrGY8dzOR2aebpw4" + "xYnN8gYXFHeISG1ieQnnyZUck5HUF1FX3rcWh8y7doNLL9cvIt5+R6yLqeTZkCX3" + "eIAvANuhdN++nqTM7JoFSViZa8VNQ18wviB5Hw+VWdpZXF9SQmWpP4M7pgDUYTVa" + "xIsywBiisQBExh5NXEhSAboTtqxmt5IADSQx9E2tp5u3oY2Ql3YRGmYnfe1y7QKD" + "PdEVMRa4aFod+w9qe2OfAgMBAAGjggN4MIIDdDAfBgNVHSMEGDAWgBQ901Cl1qCt" + "7vNKYApl0yHU+PjWDzAdBgNVHQ4EFgQUzjnGrO9r9NbeN6cRkzoAtaKNZLowJQYD" + "VR0RBB4wHIIKc3RyaXBlLmNvbYIOd3d3LnN0cmlwZS5jb20wDgYDVR0PAQH/BAQD" + "AgWgMB0GA1UdJQQWMBQGCCsGAQUFBwMBBggrBgEFBQcDAjB1BgNVHR8EbjBsMDSg" + "MqAwhi5odHRwOi8vY3JsMy5kaWdpY2VydC5jb20vc2hhMi1ldi1zZXJ2ZXItZzMu" + "Y3JsMDSgMqAwhi5odHRwOi8vY3JsNC5kaWdpY2VydC5jb20vc2hhMi1ldi1zZXJ2" + "ZXItZzMuY3JsMEoGA1UdIARDMEEwCwYJYIZIAYb9bAIBMDIGBWeBDAEBMCkwJwYI" + "KwYBBQUHAgEWG2h0dHA6Ly93d3cuZGlnaWNlcnQuY29tL0NQUzCBiAYIKwYBBQUH" + "AQEEfDB6MCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wUgYI" + "KwYBBQUHMAKGRmh0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFNI" + "QTJFeHRlbmRlZFZhbGlkYXRpb25TZXJ2ZXJDQS5jcnQwDAYDVR0TAQH/BAIwADCC" + "AX4GCisGAQQB1nkCBAIEggFuBIIBagFoAHYAKXm+8J45OSHwVnOfY6V35b5XfZxg" + "Cvj5TV0mXCVdx4QAAAF8n1z3KwAABAMARzBFAiAWda1cPqRI3YujJxhh3ahOlUoN" + "L+bpZFA30lL+L3UB1AIhAIAX2FUC4uyshmrtgnnq7OcmIzsOEVW7LDlFUlfokAbW" + "AHUAUaOw9f0BeZxWbbg3eI8MpHrMGyfL956IQpoN/tSLBeUAAAF8n1z3JAAABAMA" + "RjBEAiAPAv+vPLXXB1hB5S+MIrWXIIIsa0u+cPIARVuMEQPz5gIgLbrsj2bNmxYF" + "mfvwsORC61rCPouzzFkvQGy81lz0L38AdwBByMqx3yJGShDGoToJQodeTjGLGwPr" + "60vHaPCQYpYG9gAAAXyfXPahAAAEAwBIMEYCIQCdieTsguvk9YFGdpMPsSp4CJYY" + "1cn1lTiYQQBwzKSqogIhAOyyQKx4fhNqtWOiXUYhPaQtYLS7Ey4TeOlmm7U3OktO" + "MA0GCSqGSIb3DQEBCwUAA4IBAQBmZNhqh9q/u4hNPX7fPhIXTkHJD+B30kefLTOK" + "OA/ZlMzGdvrGaJAJNzQzZL0gTEf3x+D9yjOv8shBIfblugTTa/+ulDbSzjM7qrn6" + "nfojxtLdJ90UvkDCeJ30+Mn+FRahAeLN1jVAgXAUKTgU8QoAM0URZuLh/yWSoMzh" + "kx9PiDtEJR6DMoVVOPTf9Yk3iUBr//KDzVMbz5aJFDO+2fUjGqVmW+3rapYfealY" + "LNz/SmaO8S5U4z9A3xkwXSWxlEKkboKvngSdhNgYyPeHTmNKhcmFnnA9q2LvAl9u" + "UUhp5kZA9EKInRZiVl6i7z1RNhxe2RISbq1MlZi/raZYACt8", + ]; +#pragma clang diagnostic pop + + // root + NSString *rootCertificateString = @"MIIDxTCCAq2gAwIBAgIQAqxcJmoLQJuPC3nyrkYldzANBgkqhkiG9w0BAQUFADBs" + "MQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3" + "d3cuZGlnaWNlcnQuY29tMSswKQYDVQQDEyJEaWdpQ2VydCBIaWdoIEFzc3VyYW5j" + "ZSBFViBSb290IENBMB4XDTA2MTExMDAwMDAwMFoXDTMxMTExMDAwMDAwMFowbDEL" + "MAkGA1UEBhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3" + "LmRpZ2ljZXJ0LmNvbTErMCkGA1UEAxMiRGlnaUNlcnQgSGlnaCBBc3N1cmFuY2Ug" + "RVYgUm9vdCBDQTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBAMbM5XPm" + "+9S75S0tMqbf5YE/yc0lSbZxKsPVlDRnogocsF9ppkCxxLeyj9CYpKlBWTrT3JTW" + "PNt0OKRKzE0lgvdKpVMSOO7zSW1xkX5jtqumX8OkhPhPYlG++MXs2ziS4wblCJEM" + "xChBVfvLWokVfnHoNb9Ncgk9vjo4UFt3MRuNs8ckRZqnrG0AFFoEt7oT61EKmEFB" + "Ik5lYYeBQVCmeVyJ3hlKV9Uu5l0cUyx+mM0aBhakaHPQNAQTXKFx01p8VdteZOE3" + "hzBWBOURtCmAEvF5OYiiAhF8J2a3iLd48soKqDirCmTCv2ZdlYTBoSUeh10aUAsg" + "EsxBu24LUTi4S8sCAwEAAaNjMGEwDgYDVR0PAQH/BAQDAgGGMA8GA1UdEwEB/wQF" + "MAMBAf8wHQYDVR0OBBYEFLE+w2kD+L9HAdSYJhoIAu9jZCvDMB8GA1UdIwQYMBaA" + "FLE+w2kD+L9HAdSYJhoIAu9jZCvDMA0GCSqGSIb3DQEBBQUAA4IBAQAcGgaX3Nec" + "nzyIZgYIVyHbIUf4KmeqvxgydkAQV8GK83rZEWWONfqe/EW1ntlMMUu4kehDLI6z" + "eM7b41N5cdblIZQB2lWHmiRk9opmzN6cN82oNLFpmyPInngiK3BD41VHMWEZ71jF" + "hS9OMPagMRYjyOfiZRYzy78aG6A9+MpeizGLYAiJLQwGXFK3xPkKmNEVX58Svnw2" + "Yzi9RKR/5CYrCsSXaQ3pjOLAEFe4yHYSkVXySGnYvCoCWw9E1CAx2/S6cCZdkGCe" + "vEsXCS+0yx5DaMkHJ8HSXPfqIbloEpw8nL+e/IBcm2PN7EeqJSdnoDfzAIJ9VNep" + "+OkuE6N36B9K"; + + // mastercard DS (from mastercard.der) + NSString *dsCertificateString = @"MIIFtTCCA52gAwIBAgIQJqSRaPua/6cpablmVDHWUDANBgkqhkiG9w0BAQsFADB6" + "MQswCQYDVQQGEwJVUzETMBEGA1UEChMKTWFzdGVyQ2FyZDEoMCYGA1UECxMfTWFz" + "dGVyQ2FyZCBJZGVudGl0eSBDaGVjayBHZW4gMzEsMCoGA1UEAxMjUFJEIE1hc3Rl" + "ckNhcmQgM0RTMiBBY3F1aXJlciBTdWIgQ0EwHhcNMTgxMTIwMTQ1MzIzWhcNMjEx" + "MTIwMTQ1MzIzWjBxMQswCQYDVQQGEwJVUzEdMBsGA1UEChMUTWFzdGVyQ2FyZCBX" + "b3JsZHdpZGUxGzAZBgNVBAsTEmdhdGV3YXktZW5jcnlwdGlvbjEmMCQGA1UEAxMd" + "M2RzMi5kaXJlY3RvcnkubWFzdGVyY2FyZC5jb20wggEiMA0GCSqGSIb3DQEBAQUA" + "A4IBDwAwggEKAoIBAQCFlZjqbbL9bDKOzZFawdbyfQcezVEUSDCWWsYKw/V6co9A" + "GaPBUsGgzxF6+EDgVj3vYytgSl8xFvVPsb4ZJ6BJGvimda8QiIyrX7WUxQMB3hyS" + "BOPf4OB72CP+UkaFNR6hdlO5ofzTmB2oj1FdLGZmTN/sj6ZoHkn2Zzums8QAHFjv" + "FjspKUYCmms91gpNpJPUUztn0N1YMWVFpFMytahHIlpiGqTDt4314F7sFABLxzFr" + "Dmcqhf623SPV3kwQiLVWOvewO62ItYUFgHwle2dq76YiKrUv1C7vADSk2Am4gqwv" + "7dcCnFeM2AHbBFBa1ZBRQXosuXVw8ZcQqfY8m4iNAgMBAAGjggE+MIIBOjAOBgNV" + "HQ8BAf8EBAMCAygwCQYDVR0TBAIwADAfBgNVHSMEGDAWgBSakqJUx4CN/s5W4wMU" + "/17uSLhFuzBIBggrBgEFBQcBAQQ8MDowOAYIKwYBBQUHMAGGLGh0dHA6Ly9vY3Nw" + "LnBraS5pZGVudGl0eWNoZWNrLm1hc3RlcmNhcmQuY29tMCgGA1UdEQQhMB+CHTNk" + "czIuZGlyZWN0b3J5Lm1hc3RlcmNhcmQuY29tMGkGA1UdHwRiMGAwXqBcoFqGWGh0" + "dHA6Ly9jcmwucGtpLmlkZW50aXR5Y2hlY2subWFzdGVyY2FyZC5jb20vOWE5MmEy" + "NTRjNzgwOGRmZWNlNTZlMzAzMTRmZjVlZWU0OGI4NDViYi5jcmwwHQYDVR0OBBYE" + "FHxN6+P0r3+dFWmi/+pDQ8JWaCbuMA0GCSqGSIb3DQEBCwUAA4ICAQAtwW8siyCi" + "mhon1WUAUmufZ7bbegf3cTOafQh77NvA0xgVeloELUNCwsSSZgcOIa4Zgpsa0xi5" + "fYxXsPLgVPLM0mBhTOD1DnPu1AAm32QVelHe6oB98XxbkQlHGXeOLs62PLtDZd94" + "7pm08QMVb+MoCnHLaBLV6eKhKK+SNrfcxr33m0h3v2EMoiJ6zCvp8HgIHEhVpleU" + "8H2Uo5YObatb/KUHgtp2z0vEfyGhZR7hrr48vUQpfVGBABsCV0aqUkPxtAXWfQo9" + "1N9B7H3EIcSjbiUz5vkj9YeDSyJIi0Y/IZbzuNMsz2cRi1CWLl37w2fe128qWxYq" + "Y/k+Y4HX7uYchB8xPaZR4JczCvg1FV2JrkOcFvElVXWSMpBbe2PS6OMr3XxrHjzp" + "DyM9qvzge0Ai9+rq8AyGoG1dP2Ay83Ndlgi42X3yl1uEUW2feGojCQQCFFArazEj" + "LUkSlrB2kA12SWAhsqqQwnBLGSTp7PqPZeWkluQVXS0sbj0878kTra6TjG3U+KqO" + "JCj8v6G380qIkAXe1xMHHNQ6GS59HZMeBPYkK2y5hmh/JVo4bRfK7Ya3blBSBfB8" + "AVWQ5GqVWklvXZsQLN7FH/fMIT3y8iE1W19Ua4whlhvn7o/aYWOkHr1G2xyh8BHj" + "7H63A2hjcPlW/ZAJSTuBZUClAhsNohH2Jg=="; + + XCTAssertTrue([STDSDirectoryServerCertificate _verifyCertificateChain:certChain withRootCertificates:@[rootCertificateString]], @"Failed to verify w/ root certificate."); + XCTAssertFalse([STDSDirectoryServerCertificate _verifyCertificateChain:certChain withRootCertificates:@[dsCertificateString]], @"Should not verify w/ DS certificate."); + NSArray *bothCertificateStrings = @[dsCertificateString, rootCertificateString]; + XCTAssertTrue([STDSDirectoryServerCertificate _verifyCertificateChain:certChain withRootCertificates:bothCertificateStrings], @"Failed to verify w/ root certificate and ds certificate."); +} + +- (void)testVerifyPS256Signature { + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *jwsString = @"eyJhbGciOiJQUzI1NiIsIng1YyI6WyJNSUlEZVRDQ0FtR2dBd0lCQWdJUWJTNEM0QlNp" + "Zzd1dUo1dURwZVQ0V0RBTkJna3Foa2lHOXcwQkFRc0ZBREJITVJNd0VRWUtDWkltaVpQ" + "eUxHUUJHUllEWTI5dE1SY3dGUVlLQ1pJbWlaUHlMR1FCR1JZSFpYaGhiWEJzWlRFWE1C" + "VUdBMVVFQXd3T1VsTkJJRVY0WVcxd2JHVWdSRk13SGhjTk1UY3hNVEl4TVRFMU5EQXlX" + "aGNOTWpjeE1qTXhNVE16TURBd1dqQklNUk13RVFZS0NaSW1pWlB5TEdRQkdSWURZMjl0" + "TVJjd0ZRWUtDWkltaVpQeUxHUUJHUllIWlhoaGJYQnNaVEVZTUJZR0ExVUVBd3dQVWxO" + "QklFVjRZVzF3YkdVZ1FVTlRNSUlCSWpBTkJna3Foa2lHOXcwQkFRRUZBQU9DQVE4QU1J" + "SUJDZ0tDQVFFQWtOclBJQkRYTVU2ZmN5djVpK1FIUUFRK0s4Z3NDM0hKYjdGWWhZYXc4" + "aFhiTkphK3Q4cTBsREt3TFpnUVhZVitmZld4WEp2NUdHcmxaRTRHVTUybGZNRWVnVER6" + "WVRyUlEzdGVwZ0tGak1HZzZJeTZma2wxWk5zeDJnRW9uc25sU2hmekE5R0p3UlRtdEtQ" + "Ymsxcytod3gxSVU1QVQrQUllbE5xQmdjRjJ2RTVXMjUvU0dHQm9hUk9WZFVZeHFFVERn" + "Z00xejVjS1Y0WmpEWjgrbGg0b1ZCMDdia2FjNkxRZEhwSlVVeVNIL0VyMjBEWHgzMEt5" + "aTk3UGNpWEtUUytRS1hubW04aXZ5UkNtdXgyMlpvUFVpbmQyQktDNU9pRzRNd0FMaGFM" + "MloyazhDc1JkZnkrN2RnN3o0MVJwNkQwWmVFdnRhVXA0Ylg0YUtyYUw0clRmd0lEQVFB" + "Qm8yQXdYakFNQmdOVkhSTUJBZjhFQWpBQU1BNEdBMVVkRHdFQi93UUVBd0lIZ0RBZEJn" + "TlZIUTRFRmdRVWt0d2Y2WnBUQ3hqWUt3L0JMVzZQZWlOWDRzd3dId1lEVlIwakJCZ3dG" + "b0FVdzRNQ25id0Q2bTJ3cG5vUTJzTkQ4R3J5UE40d0RRWUpLb1pJaHZjTkFRRUxCUUFE" + "Z2dFQkFHdU5IeHYvQlI2ajdsQ1B5c20xdWhyYmpCT3FkcmhKTVIvSWQ0ZEIyR3RkRVNj" + "bDNpckdQbVh5UTJTbmNUV2hOZnNnc0tEWldwNUJrNytPdG50eTBlTlVNazNoWkVxZ1lq" + "eGh6YXUwNDhYSGJzZkd2b0phTUdaWk5Ud1V2VFV6Mmhra2hncHg5eVFBS0lBMkx6Rktj" + "Z1loZWxQdTRHVzVydEV1eHUzSVM2V1l5M0QxR3RGM25hRVdralVyYThoUU9oT2wyUytD" + "WUhtUmQ2bEdrWHlrVkRhak1nZDJBSkZ6WGRLTHhUdDBPWXJXREdsVVN6R0FDUkJDZDV4" + "YlJtQVRJbGR0Y2NhR3FETjFjTld2MEkvYlBOOEVwS1M2QjBXYVpjUGFzSXRLV3BEQzg1" + "SncxR3JEeGRod29LSG94dFNHK29kaVR3QjV6TGJybjJPc1JFNWJWN0U9Il19" + "." + "eyJBQ1MgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFUKSI6eyJrdHkiOiJFQyIsImNydiI6" + "IlAtMjU2IiwieCI6Im1QVUtUX2JBV0dISWhnMFRwampxVnNQMXJYV1F1X3Z3Vk9ISHRO" + "a2RZb0EiLCJ5IjoiOEJRQXNJbUdlQVM0NmZ5V3c1TWhmR1RUMElqQnBGdzJTUzM0RHY0" + "SXJzIix9LCJTREsgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFDKSI6eyJrdHkiOiJFQyIs" + "ImNydiI6IlAtMjU2IiwieCI6IlplMmxvU1Yzd3Jyb0tVTl80emh3R2hDcW8zWGh1MXRk" + "NFFqZVE1d0lWUjAiLCJ5IjoiSGxMdGRYQVJZX2Y1NUEzZm56UWJQY202aGdyMzRNcDhw" + "LW51elFDRTBadyIsfSwiQUNTIFVSTCI6Imh0dHA6Ly9hY3NzZXJ2ZXIuZG9tYWlubmFt" + "ZS5jb20ifQ" + "." + "OiiD5pbwe_0wrG_j61LmUxidBzTbUHyZXrKE9efRylEgMJy0axYJLOUshIloiKMexPdo" + "yDdpZV5pMQR588q-" + "uuUfssKpDXj3dCrIlUCEwgs4yfIBAUwd2NHoDg20w25ek0NPH9RV_T867DAXIjDWyd2I" + "y0ZFvJeHSrRskyPY75PA_mAUIpahaW20rxJHHMyGuWx2byKxnIx6G14vXCOPp1xEv49K" + "xKWrgFLS3_GtVYuNDqQC5pleHd4drLKSbq1bjwqEM1osYEZvw9y-" + "f1vQYxAQ8GJEti9F_309GVCZSWe1oyNDY51mo-" + "BwiyojoCDPxQDOTp6g4lp656tUQNW16g"; + STDSJSONWebSignature *jws = [[STDSJSONWebSignature alloc] initWithString:jwsString]; + + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *certificateString = @"MIIDXTCCAkWgAwIBAgIQbS4C4BSig7uuJ5uDpeT4VjANBgkqhkiG9w0BAQsFADBH" + "MRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYHZXhhbXBsZTEX" + "MBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE0ODQ5WhcNMjcxMjMx" + "MTQwMDAwWjBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYH" + "ZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwggEiMA0GCSqGSIb3DQEB" + "AQUAA4IBDwAwggEKAoIBAQCfgQ+0A4Jz0CWR5Ac/MdK2ABuCzttNkvBQFl1Hz8q4" + "o8Qct3isdVN5P475dXaNGiN02HElZMO813uepDRUSJlAfP8AmZIKkxokxEFIUqsp" + "vbCpXAZT82xg5gv5C2JY3aVvNwR7pcLR0CmvnJ1AuseqQceKDdEGit1pnoCP6gEe" + "oUQdik97tOl7459V8d3UTpxLozUVlwPU00tgPmUUek8j1tPAmWx17e6EaoLRkK4Q" + "eDyWHPA4eu0hBtLQVVtv2Tf61VNTh+D/cv++eJQUArC4IuoqdLYFjB2r+bNKdstj" + "uH+qLGhHuOKDf/+RGG5rHBSRHPmJqJCSqBzmAd2s0/nPAgMBAAGjRTBDMBIGA1Ud" + "EwEB/wQIMAYBAf8CAQAwDgYDVR0PAQH/BAQDAgEGMB0GA1UdDgQWBBTDgwKdvAPq" + "bbCmehDaw0PwavI83jANBgkqhkiG9w0BAQsFAAOCAQEAOUcKqpzNQ6lr0PbDSsns" + "D6onfi+8j3TD0xG0zBSf+8G4zs8Zb6vzzQ5qHKgfr4aeen8Pw0cw2KKUJ2dFaBqj" + "n3/6/MIZbgaBvXKUbmY8xCxKQ+tOFc3KWIu4pSaO50tMPJjU/lP35bv19AA9vs9M" + "TKY2qLf88bmoNYT3W8VSDcB58KBHa7HVIPx7BUUtSyb2N2Jqx5AOiYy4NarhB3hV" + "ftkZBmCzi2Qw50KWIgTFYcIVeRTx3Js/F0IuEdgZHBK2gmO7fdM7+QKYm83401vl" + "YRNCXfIZ0H9E1V3NddqJuqIutdUajckSzMhXdNCJqfI4FAQAymTWGL3/lZyr/30x" + "Fg=="; + + XCTAssertTrue([STDSDirectoryServerCertificate verifyJSONWebSignature:jws withRootCertificates:@[certificateString]], @"Failed to validate correct PS256 signature."); +} + +- (void)testVerifyPS256SignatureFail { + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *jwsString = @"eyJhbGciOiJQUzI1NiIsIng1YyI6WyJNSUlEZVRDQ0FtR2dBd0lCQWdJUWJTNEM0QlNp" + "Zzd1dUo1dURwZVQ0V0RBTkJna3Foa2lHOXcwQkFRc0ZBREJITVJNd0VRWUtDWkltaVpQ" + "eUxHUUJHUllEWTI5dE1SY3dGUVlLQ1pJbWlaUHlMR1FCR1JZSFpYaGhiWEJzWlRFWE1C" + "VUdBMVVFQXd3T1VsTkJJRVY0WVcxd2JHVWdSRk13SGhjTk1UY3hNVEl4TVRFMU5EQXlX" + "aGNOTWpjeE1qTXhNVE16TURBd1dqQklNUk13RVFZS0NaSW1pWlB5TEdRQkdSWURZMjl0" + "TVJjd0ZRWUtDWkltaVpQeUxHUUJHUllIWlhoaGJYQnNaVEVZTUJZR0ExVUVBd3dQVWxO" + "QklFVjRZVzF3YkdVZ1FVTlRNSUlCSWpBTkJna3Foa2lHOXcwQkFRRUZBQU9DQVE4QU1J" + "SUJDZ0tDQVFFQWtOclBJQkRYTVU2ZmN5djVpK1FIUUFRK0s4Z3NDM0hKYjdGWWhZYXc4" + "aFhiTkphK3Q4cTBsREt3TFpnUVhZVitmZld4WEp2NUdHcmxaRTRHVTUybGZNRWVnVER6" + "WVRyUlEzdGVwZ0tGak1HZzZJeTZma2wxWk5zeDJnRW9uc25sU2hmekE5R0p3UlRtdEtQ" + "Ymsxcytod3gxSVU1QVQrQUllbE5xQmdjRjJ2RTVXMjUvU0dHQm9hUk9WZFVZeHFFVERn" + "Z00xejVjS1Y0WmpEWjgrbGg0b1ZCMDdia2FjNkxRZEhwSlVVeVNIL0VyMjBEWHgzMEt5" + "aTk3UGNpWEtUUytRS1hubW04aXZ5UkNtdXgyMlpvUFVpbmQyQktDNU9pRzRNd0FMaGFM" + "MloyazhDc1JkZnkrN2RnN3o0MVJwNkQwWmVFdnRhVXA0Ylg0YUtyYUw0clRmd0lEQVFB" + "Qm8yQXdYakFNQmdOVkhSTUJBZjhFQWpBQU1BNEdBMVVkRHdFQi93UUVBd0lIZ0RBZEJn" + "TlZIUTRFRmdRVWt0d2Y2WnBUQ3hqWUt3L0JMVzZQZWlOWDRzd3dId1lEVlIwakJCZ3dG" + "b0FVdzRNQ25id0Q2bTJ3cG5vUTJzTkQ4R3J5UE40d0RRWUpLb1pJaHZjTkFRRUxCUUFE" + "Z2dFQkFHdU5IeHYvQlI2ajdsQ1B5c20xdWhyYmpCT3FkcmhKTVIvSWQ0ZEIyR3RkRVNj" + "bDNpckdQbVh5UTJTbmNUV2hOZnNnc0tEWldwNUJrNytPdG50eTBlTlVNazNoWkVxZ1lq" + "eGh6YXUwNDhYSGJzZkd2b0phTUdaWk5Ud1V2VFV6Mmhra2hncHg5eVFBS0lBMkx6Rktj" + "Z1loZWxQdTRHVzVydEV1eHUzSVM2V1l5M0QxR3RGM25hRVdralVyYThoUU9oT2wyUytD" + "WUhtUmQ2bEdrWHlrVkRhak1nZDJBSkZ6WGRLTHhUdDBPWXJXREdsVVN6R0FDUkJDZDV4" + "YlJtQVRJbGR0Y2NhR3FETjFjTld2MEkvYlBOOEVwS1M2QjBXYVpjUGFzSXRLV3BEQzg1" + "SncxR3JEeGRod29LSG94dFNHK29kaVR3QjV6TGJybjJPc1JFNWJWN0U9Il19" + "." + "eyJBQ1MgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFUKSI6eyJrdHkiOiJFQyIsImNydiI6" + "IlAtMjU2IiwieCI6Im1QVUtUX2JBV0dISWhnMFRwampxVnNQMXJYV1F1X3Z3Vk9ISHRO" + "a2RZb0EiLCJ5IjoiOEJRQXNJbUdlQVM0NmZ5V3c1TWhmR1RUMElqQnBGdzJTUzM0RHY0" + "SXJzIix9LCJTREsgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFDKSI6eyJrdHkiOiJFQyIs" + "ImNydiI6IlAtMjU2IiwieCI6IlplMmxvU1Yzd3Jyb0tVTl80emh3R2hDcW8zWGh1MXRk" + "NFFqZVE1d0lWUjAiLCJ5IjoiSGxMdGRYQVJZX2Y1NUEzZm56UWJQY202aGdyMzRNcDhw" + "LW51elFDRTBadyIsfSwiQUNTIFVSTCI6Imh0dHA6Ly9hY3NzZXJ2ZXIuZG9tYWlubmFt" + "ZS5jb20ifQ" + "." + "OiiD5pbwe_0wrG_j61LmUxidBzTbUHyZXrKE9efRylEgMJy0axYJLOUshIloiKMexPdo" + "yDdpZV5pMQR588q-" + "uuUfssKpDXj3dCrIlUCEwgs4yfIBAUwd2NHoDg20w25ek0NPH9RV_T867DAXIjDWyd2I" + "y0ZFvJeHSrRskyPY75PA_mAUIpahaW20rxJHHMyGuWx2byKxnIx6G14vXCOPp1xEv49K" + "xKWrgFLS3_GtVYuNDqQC5pleHd4drLKSbq1bjwqEM1osYEZvw9y-" + "f1vQYxAQ8GJEti9F_309GVCZSWe1oyNEY51mo-" // This is all the same as testVerifyPS256Signature, except this line where "NDY51mo-" became "NEY51mo-" + "BwiyojoCDPxQDOTp6g4lp656tUQNW16g"; + STDSJSONWebSignature *jws = [[STDSJSONWebSignature alloc] initWithString:jwsString]; + + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *certificateString = @"MIIDXTCCAkWgAwIBAgIQbS4C4BSig7uuJ5uDpeT4VjANBgkqhkiG9w0BAQsFADBH" + "MRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYHZXhhbXBsZTEX" + "MBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE0ODQ5WhcNMjcxMjMx" + "MTQwMDAwWjBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYH" + "ZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwggEiMA0GCSqGSIb3DQEB" + "AQUAA4IBDwAwggEKAoIBAQCfgQ+0A4Jz0CWR5Ac/MdK2ABuCzttNkvBQFl1Hz8q4" + "o8Qct3isdVN5P475dXaNGiN02HElZMO813uepDRUSJlAfP8AmZIKkxokxEFIUqsp" + "vbCpXAZT82xg5gv5C2JY3aVvNwR7pcLR0CmvnJ1AuseqQceKDdEGit1pnoCP6gEe" + "oUQdik97tOl7459V8d3UTpxLozUVlwPU00tgPmUUek8j1tPAmWx17e6EaoLRkK4Q" + "eDyWHPA4eu0hBtLQVVtv2Tf61VNTh+D/cv++eJQUArC4IuoqdLYFjB2r+bNKdstj" + "uH+qLGhHuOKDf/+RGG5rHBSRHPmJqJCSqBzmAd2s0/nPAgMBAAGjRTBDMBIGA1Ud" + "EwEB/wQIMAYBAf8CAQAwDgYDVR0PAQH/BAQDAgEGMB0GA1UdDgQWBBTDgwKdvAPq" + "bbCmehDaw0PwavI83jANBgkqhkiG9w0BAQsFAAOCAQEAOUcKqpzNQ6lr0PbDSsns" + "D6onfi+8j3TD0xG0zBSf+8G4zs8Zb6vzzQ5qHKgfr4aeen8Pw0cw2KKUJ2dFaBqj" + "n3/6/MIZbgaBvXKUbmY8xCxKQ+tOFc3KWIu4pSaO50tMPJjU/lP35bv19AA9vs9M" + "TKY2qLf88bmoNYT3W8VSDcB58KBHa7HVIPx7BUUtSyb2N2Jqx5AOiYy4NarhB3hV" + "ftkZBmCzi2Qw50KWIgTFYcIVeRTx3Js/F0IuEdgZHBK2gmO7fdM7+QKYm83401vl" + "YRNCXfIZ0H9E1V3NddqJuqIutdUajckSzMhXdNCJqfI4FAQAymTWGL3/lZyr/30x" + "Fg=="; + + XCTAssertFalse([STDSDirectoryServerCertificate verifyJSONWebSignature:jws withRootCertificates:@[certificateString]], @"Incorrectly validated PS256 signature."); +} + +- (void)testVerifyES256Signature { + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *jwsString = @"eyJhbGciOiJFUzI1NiIsIng1YyI6WyJNSUlDclRDQ0FaV2dBd0lCQWdJUWJTNEM0QlNp" + "Zzd1dUo1dURwZVQ0V1RBTkJna3Foa2lHOXcwQkFRc0ZBREJITVJNd0VRWUtDWkltaVpQ" + "eUxHUUJHUllEWTI5dE1SY3dGUVlLQ1pJbWlaUHlMR1FCR1JZSFpYaGhiWEJzWlRFWE1C" + "VUdBMVVFQXd3T1VsTkJJRVY0WVcxd2JHVWdSRk13SGhjTk1UY3hNVEl4TVRVME16STNX" + "aGNOTWpjeE1qTXhNVE16TURBd1dqQkhNUk13RVFZS0NaSW1pWlB5TEdRQkdSWURZMjl0" + "TVJjd0ZRWUtDWkltaVpQeUxHUUJHUllIWlhoaGJYQnNaVEVYTUJVR0ExVUVBd3dPUlVN" + "Z1JYaGhiWEJzWlNCQlExTXdXVEFUQmdjcWhrak9QUUlCQmdncWhrak9QUU1CQndOQ0FB" + "VGZvZml3YzZBaXUxWWc1dkc5ZUtYSGVEQ1ZoOWgzVk1xTjIveUoxQ1dHVWlwOEJqOHEr" + "ZXJPbzc0dHQ2akVUTXpBYVRwekxaNW9HRFpjZVpmR21NN2RvMkF3WGpBTUJnTlZIUk1C" + "QWY4RUFqQUFNQTRHQTFVZER3RUIvd1FFQXdJSGdEQWRCZ05WSFE0RUZnUVUwQVd0REhS" + "L3ZsUXJSQXo0YUtnSkJsbkZqRXN3SHdZRFZSMGpCQmd3Rm9BVXc0TUNuYndENm0yd3Bu" + "b1Eyc05EOEdyeVBONHdEUVlKS29aSWh2Y05BUUVMQlFBRGdnRUJBRXFsRVJld1VDZUV0" + "dEFrQzBGMTZIamp4ZnYxV2E4bmFEbWFSTDk5UTAvcXFVTjh3MHF3cEFQRjd3bjJhZkxm" + "YUdkKzV1WkViMVROWXdWOUF3OUwvczNCY1NURVJJbDZPRVduK3g3Y3RPbUh5MnZ2N21p" + "dGFVcmlsZUdvZGVubS9mYURkeTVWZ0tZaitLc01WTTJzTlZhZWtYK1Qwc3dBQ1g5Qjkw" + "dW5aeGE2MjU2dDJPSjJRVjV6dTNzWU8xTjBqOXY3K3lGK0ZneDAxNE5ydzcvWHQ4SUxH" + "RjU4TnhiUWhraGtmV1NmSHRhRTVtb0JBYldSdUZURmJrQmY0NVNLZTBVTWlVNUxhYzl4" + "STBPN1hDRCt6TkI1bXdzNE5PMkFZdnl4SHE5WCthNjRJaFhjbFhuZ1BRTXJVcU1vTFdJ" + "MTY2Z1JKU3ZRRVdzSUxJVXR4MndzaVlzPSJdfQ" + "." + "eyJBQ1MgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFUKSI6eyJrdHkiOiJFQyIsImNydiI6" + "IlAtMjU2IiwieCI6Im1QVUtUX2JBV0dISWhnMFRwampxVnNQMXJYV1F1X3Z3Vk9ISHRO" + "a2RZb0EiLCJ5IjoiOEJRQXNJbUdlQVM0NmZ5V3c1TWhmR1RUMElqQnBGdzJTUzM0RHY0" + "SXJzIix9LCJTREsgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFDKSI6eyJrdHkiOiJFQyIs" + "ImNydiI6IlAtMjU2IiwieCI6IlplMmxvU1Yzd3Jyb0tVTl80emh3R2hDcW8zWGh1MXRk" + "NFFqZVE1d0lWUjAiLCJ5IjoiSGxMdGRYQVJZX2Y1NUEzZm56UWJQY202aGdyMzRNcDhw" + "LW51elFDRTBadyIsfSwiQUNTIFVSTCI6Imh0dHA6Ly9hY3NzZXJ2ZXIuZG9tYWlubmFt" + "ZS5jb20ifQ" + "." + "KMNtxy3eZMDnVRK_UZvFtRY8fDuAAHJXHCaKncoViB3dy56u5VOT0XnB8K-" + "0nWwFhwWmwU1xGVwimBcfxNplCA"; + STDSJSONWebSignature *jws = [[STDSJSONWebSignature alloc] initWithString:jwsString]; + + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *certificateString = @"MIIDXTCCAkWgAwIBAgIQbS4C4BSig7uuJ5uDpeT4VjANBgkqhkiG9w0BAQsFADBH" + "MRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYHZXhhbXBsZTEX" + "MBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE0ODQ5WhcNMjcxMjMx" + "MTQwMDAwWjBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYH" + "ZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwggEiMA0GCSqGSIb3DQEB" + "AQUAA4IBDwAwggEKAoIBAQCfgQ+0A4Jz0CWR5Ac/MdK2ABuCzttNkvBQFl1Hz8q4" + "o8Qct3isdVN5P475dXaNGiN02HElZMO813uepDRUSJlAfP8AmZIKkxokxEFIUqsp" + "vbCpXAZT82xg5gv5C2JY3aVvNwR7pcLR0CmvnJ1AuseqQceKDdEGit1pnoCP6gEe" + "oUQdik97tOl7459V8d3UTpxLozUVlwPU00tgPmUUek8j1tPAmWx17e6EaoLRkK4Q" + "eDyWHPA4eu0hBtLQVVtv2Tf61VNTh+D/cv++eJQUArC4IuoqdLYFjB2r+bNKdstj" + "uH+qLGhHuOKDf/+RGG5rHBSRHPmJqJCSqBzmAd2s0/nPAgMBAAGjRTBDMBIGA1Ud" + "EwEB/wQIMAYBAf8CAQAwDgYDVR0PAQH/BAQDAgEGMB0GA1UdDgQWBBTDgwKdvAPq" + "bbCmehDaw0PwavI83jANBgkqhkiG9w0BAQsFAAOCAQEAOUcKqpzNQ6lr0PbDSsns" + "D6onfi+8j3TD0xG0zBSf+8G4zs8Zb6vzzQ5qHKgfr4aeen8Pw0cw2KKUJ2dFaBqj" + "n3/6/MIZbgaBvXKUbmY8xCxKQ+tOFc3KWIu4pSaO50tMPJjU/lP35bv19AA9vs9M" + "TKY2qLf88bmoNYT3W8VSDcB58KBHa7HVIPx7BUUtSyb2N2Jqx5AOiYy4NarhB3hV" + "ftkZBmCzi2Qw50KWIgTFYcIVeRTx3Js/F0IuEdgZHBK2gmO7fdM7+QKYm83401vl" + "YRNCXfIZ0H9E1V3NddqJuqIutdUajckSzMhXdNCJqfI4FAQAymTWGL3/lZyr/30x" + "Fg=="; + + XCTAssertTrue([STDSDirectoryServerCertificate verifyJSONWebSignature:jws withRootCertificates:@[certificateString]], @"Failed to verify valid ES256 signature."); +} + +- (void)testVerifyES256SignatureFail { + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *jwsString = @"eyJhbGciOiJFUzI1NiIsIng1YyI6WyJNSUlDclRDQ0FaV2dBd0lCQWdJUWJTNEM0QlNp" + "Zzd1dUo1dURwZVQ0V1RBTkJna3Foa2lHOXcwQkFRc0ZBREJITVJNd0VRWUtDWkltaVpQ" + "eUxHUUJHUllEWTI5dE1SY3dGUVlLQ1pJbWlaUHlMR1FCR1JZSFpYaGhiWEJzWlRFWE1C" + "VUdBMVVFQXd3T1VsTkJJRVY0WVcxd2JHVWdSRk13SGhjTk1UY3hNVEl4TVRVME16STNX" + "aGNOTWpjeE1qTXhNVE16TURBd1dqQkhNUk13RVFZS0NaSW1pWlB5TEdRQkdSWURZMjl0" + "TVJjd0ZRWUtDWkltaVpQeUxHUUJHUllIWlhoaGJYQnNaVEVYTUJVR0ExVUVBd3dPUlVN" + "Z1JYaGhiWEJzWlNCQlExTXdXVEFUQmdjcWhrak9QUUlCQmdncWhrak9QUU1CQndOQ0FB" + "VGZvZml3YzZBaXUxWWc1dkc5ZUtYSGVEQ1ZoOWgzVk1xTjIveUoxQ1dHVWlwOEJqOHEr" + "ZXJPbzc0dHQ2akVUTXpBYVRwekxaNW9HRFpjZVpmR21NN2RvMkF3WGpBTUJnTlZIUk1C" + "QWY4RUFqQUFNQTRHQTFVZER3RUIvd1FFQXdJSGdEQWRCZ05WSFE0RUZnUVUwQVd0REhS" + "L3ZsUXJSQXo0YUtnSkJsbkZqRXN3SHdZRFZSMGpCQmd3Rm9BVXc0TUNuYndENm0yd3Bu" + "b1Eyc05EOEdyeVBONHdEUVlKS29aSWh2Y05BUUVMQlFBRGdnRUJBRXFsRVJld1VDZUV0" + "dEFrQzBGMTZIamp4ZnYxV2E4bmFEbWFSTDk5UTAvcXFVTjh3MHF3cEFQRjd3bjJhZkxm" + "YUdkKzV1WkViMVROWXdWOUF3OUwvczNCY1NURVJJbDZPRVduK3g3Y3RPbUh5MnZ2N21p" + "dGFVcmlsZUdvZGVubS9mYURkeTVWZ0tZaitLc01WTTJzTlZhZWtYK1Qwc3dBQ1g5Qjkw" + "dW5aeGE2MjU2dDJPSjJRVjV6dTNzWU8xTjBqOXY3K3lGK0ZneDAxNE5ydzcvWHQ4SUxH" + "RjU4TnhiUWhraGtmV1NmSHRhRTVtb0JBYldSdUZURmJrQmY0NVNLZTBVTWlVNUxhYzl4" + "STBPN1hDRCt6TkI1bXdzNE5PMkFZdnl4SHE5WCthNjRJaFhjbFhuZ1BRTXJVcU1vTFdJ" + "MTY2Z1JKU3ZRRVdzSUxJVXR4MndzaVlzPSJdfQ" + "." + "eyJBQ1MgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFUKSI6eyJrdHkiOiJFQyIsImNydiI6" + "IlAtMjU2IiwieCI6Im1QVUtUX2JBV0dISWhnMFRwampxVnNQMXJYV1F1X3Z3Vk9ISHRO" + "a2RZb0EiLCJ5IjoiOEJRQXNJbUdlQVM0NmZ5V3c1TWhmR1RUMElqQnBGdzJTUzM0RHY0" + "SXJzIix9LCJTREsgRXBoZW1lcmFsIFB1YmxpYyBLZXkgKFFDKSI6eyJrdHkiOiJFQyIs" + "ImNydiI6IlAtMjU2IiwieCI6IlplMmxvU1Yzd3Jyb0tVTl80emh3R2hDcW8zWGh1MXRk" + "NFFqZVE1d0lWUjAiLCJ5IjoiSGxMdGRYQVJZX2Y1NUEzZm56UWJQY202aGdyMzRNcDhw" + "LW51elFDRTBadyIsfSwiQUNTIFVSTCI6Imh0dHA6Ly9hY3NzZXJ2ZXIuZG9tYWlubMFt" // This is all the same as testVerifyPS256Signature, except this line where "bmFt" became "bMFt" + "ZS5jb20ifQ" + "." + "KMNtxy3eZMDnVRK_UZvFtRY8fDuAAHJXHCaKncoViB3dy56u5VOT0XnB8K-" + "0nWwFhwWmwU1xGVwimBcfxNplCA"; + STDSJSONWebSignature *jws = [[STDSJSONWebSignature alloc] initWithString:jwsString]; + + // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSString *certificateString = @"MIIDXTCCAkWgAwIBAgIQbS4C4BSig7uuJ5uDpeT4VjANBgkqhkiG9w0BAQsFADBH" + "MRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYHZXhhbXBsZTEX" + "MBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwHhcNMTcxMTIxMTE0ODQ5WhcNMjcxMjMx" + "MTQwMDAwWjBHMRMwEQYKCZImiZPyLGQBGRYDY29tMRcwFQYKCZImiZPyLGQBGRYH" + "ZXhhbXBsZTEXMBUGA1UEAwwOUlNBIEV4YW1wbGUgRFMwggEiMA0GCSqGSIb3DQEB" + "AQUAA4IBDwAwggEKAoIBAQCfgQ+0A4Jz0CWR5Ac/MdK2ABuCzttNkvBQFl1Hz8q4" + "o8Qct3isdVN5P475dXaNGiN02HElZMO813uepDRUSJlAfP8AmZIKkxokxEFIUqsp" + "vbCpXAZT82xg5gv5C2JY3aVvNwR7pcLR0CmvnJ1AuseqQceKDdEGit1pnoCP6gEe" + "oUQdik97tOl7459V8d3UTpxLozUVlwPU00tgPmUUek8j1tPAmWx17e6EaoLRkK4Q" + "eDyWHPA4eu0hBtLQVVtv2Tf61VNTh+D/cv++eJQUArC4IuoqdLYFjB2r+bNKdstj" + "uH+qLGhHuOKDf/+RGG5rHBSRHPmJqJCSqBzmAd2s0/nPAgMBAAGjRTBDMBIGA1Ud" + "EwEB/wQIMAYBAf8CAQAwDgYDVR0PAQH/BAQDAgEGMB0GA1UdDgQWBBTDgwKdvAPq" + "bbCmehDaw0PwavI83jANBgkqhkiG9w0BAQsFAAOCAQEAOUcKqpzNQ6lr0PbDSsns" + "D6onfi+8j3TD0xG0zBSf+8G4zs8Zb6vzzQ5qHKgfr4aeen8Pw0cw2KKUJ2dFaBqj" + "n3/6/MIZbgaBvXKUbmY8xCxKQ+tOFc3KWIu4pSaO50tMPJjU/lP35bv19AA9vs9M" + "TKY2qLf88bmoNYT3W8VSDcB58KBHa7HVIPx7BUUtSyb2N2Jqx5AOiYy4NarhB3hV" + "ftkZBmCzi2Qw50KWIgTFYcIVeRTx3Js/F0IuEdgZHBK2gmO7fdM7+QKYm83401vl" + "YRNCXfIZ0H9E1V3NddqJuqIutdUajckSzMhXdNCJqfI4FAQAymTWGL3/lZyr/30x" + "Fg=="; + + XCTAssertFalse([STDSDirectoryServerCertificate verifyJSONWebSignature:jws withRootCertificates:@[certificateString]], @"Verified invalid ES256 signature."); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSEllipticCurvePointTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSEllipticCurvePointTests.m new file mode 100644 index 00000000..a315dcd8 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSEllipticCurvePointTests.m @@ -0,0 +1,42 @@ +// +// STDSEllipticCurvePointTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 4/5/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "NSString+JWEHelpers.h" +#import "STDSEllipticCurvePoint.h" + +@interface STDSEllipticCurvePointTests : XCTestCase + +@end + +@implementation STDSEllipticCurvePointTests + +- (void)testInitWithJWK { + + STDSEllipticCurvePoint *ecPoint = [[STDSEllipticCurvePoint alloc] initWithJWK:@{ // ref. EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + @"kty":@"EC", + @"crv":@"P-256", + @"x":@"mPUKT_bAWGHIhg0TpjjqVsP1rXWQu_vwVOHHtNkdYoA", + @"y":@"8BQAsImGeAS46fyWw5MhYfGTT0IjBpFw2SS34Dv4Irs", + }]; + + XCTAssertNotNil(ecPoint, @"Failed to create point with valid jwk"); + XCTAssertEqualObjects(ecPoint.x, [@"mPUKT_bAWGHIhg0TpjjqVsP1rXWQu_vwVOHHtNkdYoA" _stds_base64URLDecodedData], @"Parsed incorrect x-coordinate"); + XCTAssertEqualObjects(ecPoint.y, [@"8BQAsImGeAS46fyWw5MhYfGTT0IjBpFw2SS34Dv4Irs" _stds_base64URLDecodedData], @"Parsed incorrect y-coordinate"); + + ecPoint = [[STDSEllipticCurvePoint alloc] initWithJWK:@{ + @"kty":@"EC", + @"crv":@"P-128", + @"x":@"mPUKT_bAWGHIhg0TpjjqVsP1rXWQu_vwVOHHtNkdYoA", + @"y":@"8BQAsImGeAS46fyWw5MhYfGTT0IjBpFw2SS34Dv4Irs", + }]; + XCTAssertNil(ecPoint, @"Shoud return nil for non P-256 curve."); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSEphemeralKeyPairTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSEphemeralKeyPairTests.m new file mode 100644 index 00000000..5d8cb194 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSEphemeralKeyPairTests.m @@ -0,0 +1,41 @@ +// +// STDSEphemeralKeyPairTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 3/26/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSDirectoryServerCertificate.h" +#import "STDSEllipticCurvePoint.h" +#import "STDSEphemeralKeyPair+Testing.h" +#import "STDSSecTypeUtilities.h" + +@interface STDSEphemeralKeyPairTests : XCTestCase + +@end + +@implementation STDSEphemeralKeyPairTests + +- (void)testCreateEpehemeralKeyPair { + STDSEphemeralKeyPair *keyPair = [STDSEphemeralKeyPair ephemeralKeyPair]; + XCTAssertNotNil(keyPair.publicKeyJWK, @"Failed to create a valid public key JWK"); + STDSEphemeralKeyPair *keyPair2 = [STDSEphemeralKeyPair ephemeralKeyPair]; + XCTAssertNotEqual(keyPair.publicKeyJWK, keyPair2.publicKeyJWK, @"Failed sanity check that two different ephemeral key pairs don't have the same public key JWK."); +} + +- (void)testDiffieHellmanSharedSecret { + // values from EMVCo_3DS_-AppBased_CryptoExamples_082018.pdf + NSDictionary *jwk = [NSJSONSerialization JSONObjectWithData:[@"{\"kty\":\"EC\",\"crv\":\"P-256\",\"kid\":\"UUIDkeyidentifierforDS-EC\", \"x\":\"2_v-MuNZccqwM7PXlakW9oHLP5XyrjMG1UVS8OxYrgA\", \"y\":\"rm1ktLmFIsP2R0YyJGXtsCbaTUesUK31Xc04tHJRolc\"}" dataUsingEncoding:NSUTF8StringEncoding] options:0 error:NULL]; + + STDSEllipticCurvePoint *publicKey = [[STDSEllipticCurvePoint alloc] initWithJWK:jwk]; + STDSEphemeralKeyPair *keyPair = [STDSEphemeralKeyPair testKeyPair]; + NSData *secret = [keyPair createSharedSecretWithEllipticCurveKey:publicKey]; + const unsigned char expectedSecretBytes[] = {0x5C, 0x32, 0xBC, 0x13, 0xF8, 0xEC, 0xEB, 0x14, 0x8A, 0xBA, 0xF2, 0xA6, 0xB9, 0xDD, 0x1F, 0x68, 0x91, 0xBB, 0x2A, 0x80, 0xAB, 0x09, 0x34, 0x7C, 0x64, 0x06, 0x82, 0x31, 0xA5, 0x9E, 0x8C, 0xA2}; + NSData *expectedSecret = [[NSData alloc] initWithBytes:expectedSecretBytes length:32]; + XCTAssertEqualObjects(secret, expectedSecret, @"Generated incorrect shared secret value"); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSErrorMessageTest.m b/Stripe3DS2/Stripe3DS2Tests/STDSErrorMessageTest.m new file mode 100644 index 00000000..b55af754 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSErrorMessageTest.m @@ -0,0 +1,61 @@ +// +// STDSErrorMessageTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSErrorMessage.h" +#import "STDSJSONEncoder.h" +#import "STDSTestJSONUtils.h" + +@interface STDSErrorMessageTest : XCTestCase + +@end + +@implementation STDSErrorMessageTest + +#pragma mark - STDSJSONDecodable + +- (void)testSuccessfulDecode { + NSDictionary *json = [STDSTestJSONUtils jsonNamed:@"ErrorMessage"]; + NSError *error; + STDSErrorMessage *errorMessage = [STDSErrorMessage decodedObjectFromJSON:json error:&error]; + + XCTAssertNil(error); + XCTAssertNotNil(errorMessage); + XCTAssertEqualObjects(errorMessage.errorCode, @"203"); + XCTAssertEqualObjects(errorMessage.errorComponent, @"A"); + XCTAssertEqualObjects(errorMessage.errorDescription, @"Data element not in the required format. Not numeric or wrong length."); + XCTAssertEqualObjects(errorMessage.errorDetails, @"billAddrCountry,billAddrPostCode,dsURL"); + XCTAssertEqualObjects(errorMessage.errorMessageType, @"AReq"); + XCTAssertEqualObjects(errorMessage.messageVersion, @"2.1.0"); + +} + +#pragma mark - STDSJSONEncodable + +- (void)testPropertyNamesToJSONKeysMapping { + STDSErrorMessage *params = [STDSErrorMessage new]; + + NSDictionary *mapping = [STDSErrorMessage propertyNamesToJSONKeysMapping]; + + for (NSString *propertyName in [mapping allKeys]) { + XCTAssertFalse([propertyName containsString:@":"]); + XCTAssert([params respondsToSelector:NSSelectorFromString(propertyName)]); + } + + for (NSString *formFieldName in [mapping allValues]) { + XCTAssert([formFieldName isKindOfClass:[NSString class]]); + XCTAssert([formFieldName length] > 0); + } + + XCTAssertEqual([[mapping allValues] count], [[NSSet setWithArray:[mapping allValues]] count]); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:[STDSJSONEncoder dictionaryForObject:params]]); +} + + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSJSONEncoderTest.m b/Stripe3DS2/Stripe3DS2Tests/STDSJSONEncoderTest.m new file mode 100644 index 00000000..a42c70d9 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSJSONEncoderTest.m @@ -0,0 +1,201 @@ +// +// STDSJSONEncoderTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/25/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSJSONEncoder.h" + +#pragma mark - STDSJSONEncodableObject + +@interface STDSJSONEncodableObject : NSObject +@property (nonatomic, copy) NSString *testProperty; +@property (nonatomic, copy) NSArray *testArrayProperty; +@property (nonatomic, copy) NSDictionary *testDictionaryProperty; +@property (nonatomic) STDSJSONEncodableObject *testNestedObjectProperty; +@end + +@implementation STDSJSONEncodableObject + ++ (NSDictionary *)propertyNamesToJSONKeysMapping { + return @{ + NSStringFromSelector(@selector(testProperty)): @"test_property", + NSStringFromSelector(@selector(testArrayProperty)): @"test_array_property", + NSStringFromSelector(@selector(testDictionaryProperty)): @"test_dictionary_property", + NSStringFromSelector(@selector(testNestedObjectProperty)): @"test_nested_property", + }; +} + +@end + +#pragma mark - STDSJSONEncoderTest + +@interface STDSJSONEncoderTest : XCTestCase +@end + +@implementation STDSJSONEncoderTest + +- (void)testEmptyEncodableObject { + STDSJSONEncodableObject *object = [STDSJSONEncodableObject new]; + XCTAssertEqualObjects([STDSJSONEncoder dictionaryForObject:object], @{}); +} + +- (void)testNestedObject { + STDSJSONEncodableObject *object = [STDSJSONEncodableObject new]; + STDSJSONEncodableObject *nestedObject = [STDSJSONEncodableObject new]; + nestedObject.testProperty = @"nested_object_property"; + object.testProperty = @"object_property"; + object.testNestedObjectProperty = nestedObject; + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:object]; + NSDictionary *expected = @{ + @"test_property": @"object_property", + @"test_nested_property": @{ + @"test_property": @"nested_object_property", + } + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); +} + +- (void)testSerializeDeserialize { + STDSJSONEncodableObject *object = [STDSJSONEncodableObject new]; + object.testProperty = @"test"; + NSDictionary *expected = @{@"test_property": @"test"}; + NSData *data = [NSJSONSerialization dataWithJSONObject:[STDSJSONEncoder dictionaryForObject:object] options:0 error:nil]; + id jsonObject = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil]; + XCTAssertEqualObjects(expected, jsonObject); +} + +- (void)testBoolAndNumbers { + STDSJSONEncodableObject *testObject = [STDSJSONEncodableObject new]; + testObject.testArrayProperty = @[@0, + @1, + [NSNumber numberWithBool:NO], + [[NSNumber alloc] initWithBool:YES], + @YES]; + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:testObject]; + NSDictionary *expected = @{ + @"test_array_property": @[ + @0, + @1, + [NSNumber numberWithBool:NO], + [[NSNumber alloc] initWithBool:YES], + @YES], + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); + +} + +#pragma mark NSArray + +- (void)testArrayValueEmpty { + STDSJSONEncodableObject *testObject = [STDSJSONEncodableObject new]; + testObject.testProperty = @"success"; + testObject.testArrayProperty = @[]; + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:testObject]; + NSDictionary *expected = @{ + @"test_property": @"success", + @"test_array_property": @[] + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); +} + +- (void)testArrayValue { + STDSJSONEncodableObject *testObject = [STDSJSONEncodableObject new]; + testObject.testProperty = @"success"; + testObject.testArrayProperty = @[@1, @2, @3]; + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:testObject]; + NSDictionary *expected = @{ + @"test_property": @"success", + @"test_array_property": @[@1, @2, @3] + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); + +} + +- (void)testArrayOfEncodable { + STDSJSONEncodableObject *testObject = [STDSJSONEncodableObject new]; + + STDSJSONEncodableObject *inner1 = [STDSJSONEncodableObject new]; + inner1.testProperty = @"inner1"; + STDSJSONEncodableObject *inner2 = [STDSJSONEncodableObject new]; + inner2.testArrayProperty = @[@"inner2"]; + + testObject.testArrayProperty = @[inner1, inner2]; + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:testObject]; + NSDictionary *expected = @{ + @"test_array_property": @[ + @{ + @"test_property": @"inner1" + }, + @{ + @"test_array_property": @[@"inner2"] + } + ] + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); +} + +#pragma mark NSDictionary + +- (void)testDictionaryValueEmpty { + STDSJSONEncodableObject *testObject = [STDSJSONEncodableObject new]; + testObject.testProperty = @"success"; + testObject.testDictionaryProperty = @{}; + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:testObject]; + NSDictionary *expected = @{ + @"test_property": @"success", + @"test_dictionary_property": @{} + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); +} + +- (void)testDictionaryValue { + STDSJSONEncodableObject *testObject = [STDSJSONEncodableObject new]; + testObject.testProperty = @"success"; + testObject.testDictionaryProperty = @{@"foo": @"bar"}; + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:testObject]; + NSDictionary *expected = @{ + @"test_property": @"success", + @"test_dictionary_property": @{@"foo": @"bar"} + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); + +} + +- (void)testDictionaryOfEncodable { + STDSJSONEncodableObject *testObject = [STDSJSONEncodableObject new]; + + STDSJSONEncodableObject *inner1 = [STDSJSONEncodableObject new]; + inner1.testProperty = @"inner1"; + STDSJSONEncodableObject *inner2 = [STDSJSONEncodableObject new]; + inner2.testArrayProperty = @[@"inner2"]; + + testObject.testDictionaryProperty = @{@"one": inner1, @"two": inner2}; + + NSDictionary *jsonDictionary = [STDSJSONEncoder dictionaryForObject:testObject]; + NSDictionary *expected = @{ + @"test_dictionary_property": @{ + @"one": @{ + @"test_property": @"inner1" + }, + @"two": @{ + @"test_array_property": @[@"inner2"] + } + } + }; + XCTAssertEqualObjects(jsonDictionary, expected); + XCTAssertTrue([NSJSONSerialization isValidJSONObject:jsonDictionary]); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSJSONWebEncryptionTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSJSONWebEncryptionTests.m new file mode 100644 index 00000000..9e7cf928 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSJSONWebEncryptionTests.m @@ -0,0 +1,119 @@ +// +// STDSJSONWebEncryptionTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 1/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSStripe3DS2Error.h" +#import "NSString+JWEHelpers.h" +#import "STDSDirectoryServer.h" +#import "STDSDirectoryServerCertificate.h" +#import "STDSJSONWebEncryption.h" +#import "STDSSecTypeUtilities.h" + +@interface STDSJSONWebEncryptionTests : XCTestCase + +@end + +@implementation STDSJSONWebEncryptionTests + +- (void)testEncryption { + NSDictionary *json = @{@"a": @(0), @"b": @[@(0), @(1), @(2)], @"c": @{@"d": @"val"}}; + NSError *error = nil; + + NSString *encrypted = [STDSJSONWebEncryption encryptJSON:json forDirectoryServer:STDSDirectoryServerSTPTestRSA error:&error]; + XCTAssertNotNil(encrypted, @"Should successfully encrypt with ul_test cert."); + XCTAssertNil(error, @"Successful encryption should not populate error."); + + encrypted = [STDSJSONWebEncryption encryptJSON:json forDirectoryServer:STDSDirectoryServerSTPTestEC error:&error]; + XCTAssertNotNil(encrypted, @"Should successfully encrypt with ec_test cert."); + XCTAssertNil(error, @"Successful encryption should not populate error."); + + encrypted = [STDSJSONWebEncryption encryptJSON:json forDirectoryServer:STDSDirectoryServerULTestRSA error:&error]; + XCTAssertNotNil(encrypted, @"Should successfully encrypt with STDSDirectoryServerULTestRSA."); + XCTAssertNil(error, @"Successful encryption should not populate error."); + + encrypted = [STDSJSONWebEncryption encryptJSON:json forDirectoryServer:STDSDirectoryServerULTestEC error:&error]; + XCTAssertNotNil(encrypted, @"Should successfully encrypt with STDSDirectoryServerULTestEC."); + XCTAssertNil(error, @"Successful encryption should not populate error."); + + encrypted = [STDSJSONWebEncryption encryptJSON:json forDirectoryServer:STDSDirectoryServerUnknown error:&error]; + XCTAssertNil(encrypted, @"Invalid server ID should return nil."); + XCTAssertEqual(error.code, STDSErrorCodeDecryptionVerification, @"Failed encryption should provide a STDSErrorCodeDecryptionVerification error."); +} + +- (void)testEncryptionWithCustomCertificate { + NSDictionary *json = @{@"a": @(0), @"b": @[@(0), @(1), @(2)], @"c": @{@"d": @"val"}}; + NSError *error = nil; + // Using ds-amex.pm.PEM + NSString *certificateString = @"MIIE0TCCA7mgAwIBAgIUXbeqM1duFcHk4dDBwT8o7Ln5wX8wDQYJKoZIhvcNAQEL" + "BQAwXjELMAkGA1UEBhMCVVMxITAfBgNVBAoTGEFtZXJpY2FuIEV4cHJlc3MgQ29t" + "cGFueTEsMCoGA1UEAxMjQW1lcmljYW4gRXhwcmVzcyBTYWZla2V5IElzc3Vpbmcg" + "Q0EwHhcNMTgwMjIxMjM0OTMxWhcNMjAwMjIxMjM0OTMwWjCB0DELMAkGA1UEBhMC" + "VVMxETAPBgNVBAgTCE5ldyBZb3JrMREwDwYDVQQHEwhOZXcgWW9yazE/MD0GA1UE" + "ChM2QW1lcmljYW4gRXhwcmVzcyBUcmF2ZWwgUmVsYXRlZCBTZXJ2aWNlcyBDb21w" + "YW55LCBJbmMuMTkwNwYDVQQLEzBHbG9iYWwgTmV0d29yayBUZWNobm9sb2d5IC0g" + "TmV0d29yayBBUEkgUGxhdGZvcm0xHzAdBgNVBAMTFlNESy5TYWZlS2V5LkVuY3J5" + "cHRLZXkwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQDSFF9kTYbwRrxX" + "C6WcJJYio5TZDM62+CnjQRfggV3GMI+xIDtMIN8LL/jbWBTycu97vrNjNNv+UPhI" + "WzhFDdUqyRfrY337A39uE8k1xhdDI3dNeZz6xgq8r9hn2NBou78YPBKidpN5oiHn" + "TxcFq1zudut2fmaldaa9a4ZKgIQo+02heiJfJ8XNWkoWJ17GcjJ59UU8C1KF/y1G" + "ymYO5ha2QRsVZYI17+ZFsqnpcXwK4Mr6RQKV6UimmO0nr5++CgvXfekcWAlLV6Xq" + "juACWi3kw0haepaX/9qHRu1OSyjzWNcSVZ0On6plB5Lq6Y9ylgmxDrv+zltz3MrT" + "K7txIAFFAgMBAAGjggESMIIBDjAMBgNVHRMBAf8EAjAAMCEGA1UdEQQaMBiCFlNE" + "Sy5TYWZlS2V5LkVuY3J5cHRLZXkwRQYJKwYBBAGCNxQCBDgeNgBBAE0ARQBYAF8A" + "UwBBAEYARQBLAEUAWQAyAF8ARABTAF8ARQBOAEMAUgBZAFAAVABJAE8ATjAOBgNV" + "HQ8BAf8EBAMCBJAwHwYDVR0jBBgwFoAU7k/rXuVMhTBxB1zSftPgmLFuDIgwRAYD" + "VR0fBD0wOzA5oDegNYYzaHR0cDovL2FtZXhzay5jcmwuY29tLXN0cm9uZy1pZC5u" + "ZXQvYW1leHNhZmVrZXkuY3JsMB0GA1UdDgQWBBQHclVTo5nwZGH8labJ2F2P45xi" + "fDANBgkqhkiG9w0BAQsFAAOCAQEAWY6b77VBoGLs3k5vOqSU7QRqT+4v6y77T8LA" + "BKrSZ58DiVZWVyDSxyftQUiRRgFHt2gTN0yfJTP50Fyp84nCEWC0tugZ4iIhgPss" + "HzL+4/u4eG/MTzK2ESxvPgr6YHajyuU+GXA89u8+bsFrFmojOjhTgFKli7YUeV/0" + "xoiYZf2utlns800ofJrcrfiFoqE6PvK4Od0jpeMgfSKv71nK5ihA1+wTk76ge1fs" + "PxL23hEdRpWW11ofaLfJGkLFXMM3/LHSXWy7HhsBgDELdzLSHU4VkSv8yTOZxsRO" + "ByxdC5v3tXGcK56iQdtKVPhFGOOEBugw7AcuRzv3f1GhvzAQZg=="; + STDSDirectoryServerCertificate *certificate = [STDSDirectoryServerCertificate customCertificateWithData:[[NSData alloc] initWithBase64EncodedString:certificateString options:0]]; + + NSString *encrypted = [STDSJSONWebEncryption encryptJSON:json + withCertificate:certificate + directoryServerID:@"custom_test" + serverKeyID:nil + error:&error]; + XCTAssertNotNil(encrypted, @"Should successfully encrypt with custom cert."); + XCTAssertNil(error, @"Successful encryption should not populate error."); +} + +- (void)testDirectEncryptionWithKey { + NSDictionary *json = @{@"a": @(0), @"b": @[@(0), @(1), @(2)], @"c": @{@"d": @"val"}}; + NSError *error = nil; + NSData *contentEncryptionKey = STDSCryptoRandomData(32); + NSString *encrypted = [STDSJSONWebEncryption directEncryptJSON:json + withContentEncryptionKey:contentEncryptionKey + forACSTransactionID:@"acs_id" + error:&error]; + XCTAssertNotNil(encrypted, @"Should successfully encrypt with direct encryption key."); + XCTAssertNil(error, @"Successful encryption should not populate error."); +} + +- (void)testDecryption { + NSDictionary *json = @{@"a": @(0), @"b": @[@(0), @(1), @(2)], @"c": @{@"d": @"val"}}; + NSError *error = nil; + NSData *contentEncryptionKey = STDSCryptoRandomData(32); + NSString *encrypted = [STDSJSONWebEncryption directEncryptJSON:json + withContentEncryptionKey:contentEncryptionKey + forACSTransactionID:@"acs_id" + error:&error]; + XCTAssertNotNil(encrypted, @"Should successfully encrypt with direct encryption key."); + XCTAssertNil(error, @"Successful encryption should not populate error."); + + NSDictionary *decrypted = [STDSJSONWebEncryption decryptData:[encrypted dataUsingEncoding:NSUTF8StringEncoding] + withContentEncryptionKey:contentEncryptionKey + error:&error]; + XCTAssertEqualObjects(decrypted, json, @"Decrypted is not equal to the encrypted json"); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSJSONWebSignatureTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSJSONWebSignatureTests.m new file mode 100644 index 00000000..c4587102 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSJSONWebSignatureTests.m @@ -0,0 +1,78 @@ +// +// STDSJSONWebSignatureTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 4/2/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "NSString+JWEHelpers.h" +#import "STDSEllipticCurvePoint.h" +#import "STDSJSONWebSignature.h" + +@interface STDSJSONWebSignatureTests : XCTestCase + +@end + +@implementation STDSJSONWebSignatureTests + +- (void)testInitES256 { + + // generated a private ec key and certificate, plugged into jwt.io with default sample payload. + // This certificate will expire in 2030 but as this test doesn't cover certificate validity + // it shouldn't start failing + STDSJSONWebSignature *jws = [[STDSJSONWebSignature alloc] initWithString:@"eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsIng1YyI6WyJNSUh3TUlHV0Fna0ErdEM1LzJxV1RqRXdDZ1lJS29aSXpqMEVBd0l3QURBZUZ3MHlNVEF4TURReU1UQTBNamxhRncwek1UQXhNREl5TVRBME1qbGFNQUF3V1RBVEJnY3Foa2pPUFFJQkJnZ3Foa2pPUFFNQkJ3TkNBQVFSV3oram42NUJ0T012ZHlIS2N2akJlQlNEWkgycjFSVHdqbVlTaTlSL3pwQm51UTRFaU1uQ3FmTVBXaVpxQjRRZGJBZDBFN29INTBWcHVaMVAwODdHTUFvR0NDcUdTTTQ5QkFNQ0Ewa0FNRVlDSVFETTVRbHRDTFhEeEpvTG1EVXRqREgxZEJQVHBUVG1jS2pjOHlodVp1VHU2UUloQVBEU0cvN3plV09NdkhxNUpaWk8zd3JQeVBhTFlVNHBCcGpWTS95YzQ5MDciXX0.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWUsImlhdCI6MTUxNjIzOTAyMn0.71MhQ7FJavv1nQ7Boujfp7K0iBEYFGSGLZ3osnL9KAY9scF95Hf7ZMQ8I1JSgnGl227UY96is80MlbTijOOxsg"]; + + XCTAssertNotNil(jws, @"Failed to create jws object"); + + XCTAssertEqual(jws.algorithm, STDSJSONWebSignatureAlgorithmES256, @"Parsed incorrect algorithm"); + + NSData *digest = [@"eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsIng1YyI6WyJNSUh3TUlHV0Fna0ErdEM1LzJxV1RqRXdDZ1lJS29aSXpqMEVBd0l3QURBZUZ3MHlNVEF4TURReU1UQTBNamxhRncwek1UQXhNREl5TVRBME1qbGFNQUF3V1RBVEJnY3Foa2pPUFFJQkJnZ3Foa2pPUFFNQkJ3TkNBQVFSV3oram42NUJ0T012ZHlIS2N2akJlQlNEWkgycjFSVHdqbVlTaTlSL3pwQm51UTRFaU1uQ3FmTVBXaVpxQjRRZGJBZDBFN29INTBWcHVaMVAwODdHTUFvR0NDcUdTTTQ5QkFNQ0Ewa0FNRVlDSVFETTVRbHRDTFhEeEpvTG1EVXRqREgxZEJQVHBUVG1jS2pjOHlodVp1VHU2UUloQVBEU0cvN3plV09NdkhxNUpaWk8zd3JQeVBhTFlVNHBCcGpWTS95YzQ5MDciXX0.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWUsImlhdCI6MTUxNjIzOTAyMn0" dataUsingEncoding:NSUTF8StringEncoding]; + XCTAssertEqualObjects(jws.digest, digest, @"Parsed payload incorrectly."); + NSData *signature = [@"71MhQ7FJavv1nQ7Boujfp7K0iBEYFGSGLZ3osnL9KAY9scF95Hf7ZMQ8I1JSgnGl227UY96is80MlbTijOOxsg" _stds_base64URLDecodedData]; + XCTAssertEqualObjects(jws.signature, signature, @"Parsed signature incorrectly."); + NSData *payload = [@"eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWUsImlhdCI6MTUxNjIzOTAyMn0" _stds_base64URLDecodedData]; + XCTAssertEqualObjects(jws.payload, payload, @"Parsed payload incorrectly."); + + XCTAssertNotNil(jws.ellipticCurvePoint, @"Failed to parse elliptic curve point."); + XCTAssertNotNil(jws.certificateChain, @"Must have certificate chain."); + + const unsigned char keyBytes[] = {0x11, 0x5b, 0x3f, 0xa3, 0x9f, 0xae, 0x41, 0xb4, 0xe3, 0x2f, 0x77, 0x21, 0xca, 0x72, + 0xf8, 0xc1, 0x78, 0x14, 0x83, 0x64, 0x7d, 0xab, 0xd5, 0x14, 0xf0, 0x8e, 0x66, 0x12, 0x8b, + 0xd4, 0x7f, 0xce, 0x90, 0x67, 0xb9, 0x0e, 0x04, 0x88, 0xc9, 0xc2, 0xa9, 0xf3, 0x0f, 0x5a, + 0x26, 0x6a, 0x07, 0x84, 0x1d, 0x6c, 0x07, 0x74, 0x13, 0xba, 0x07, 0xe7, 0x45, 0x69, 0xb9, + 0x9d, 0x4f, 0xd3, 0xce, 0xc6}; + size_t keyLength = sizeof(keyBytes)/2; + NSData *coordinateX = [NSData dataWithBytes:keyBytes length:keyLength]; + NSData *coordinateY = [NSData dataWithBytes:keyBytes + keyLength length:keyLength]; + + XCTAssertEqualObjects(jws.ellipticCurvePoint.x, coordinateX, @"Incorrect x-point."); + XCTAssertEqualObjects(jws.ellipticCurvePoint.y, coordinateY, @"Incorrect y-point."); +} + +- (void)testInitPS256 { + + // test jws strings generated from jwt.io + STDSJSONWebSignature *jws = [[STDSJSONWebSignature alloc] initWithString:@"eyJhbGciOiJQUzI1NiIsIng1YyI6WyJNSUkiLCJNSUkyIl19.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWUsImlhdCI6MTUxNjIzOTAyMn0.CDB63_lCSBlmrIfZBvn6w4rDKJgkmKhFe4mfR-xUnfxf9N0g4vZa0R9lFG5pjVkThq9CX-p-vM_64wG4bAC53VlXOk6DhjzN0LTCo1nB81rd8DgqMH4SkLFy3wP-Xe0akRmXE8iHmv63ip7d2LGQVCD38xwXOnoBUVANCrcsC0Iur1TTEXaEfT6ACwg3V1YTu-vygNdbhYZOC_Q9ESbaoPxOQfumXnD44m1EN_FV3d-uQJx1Rn6w3AkDw34P3KunLrwOMJt1mbkWzb66VDVsIxegc4N8TjJTzvxmCk841wUae3kZ97_HPIEfil3ewv80hZstEE2hcEXJbdBfsxsSqg"]; + + XCTAssertNotNil(jws, @"Failed to create jws object"); + + XCTAssertEqual(jws.algorithm, STDSJSONWebSignatureAlgorithmPS256, @"Parsed incorrect algorithm"); + + NSData *digest = [@"eyJhbGciOiJQUzI1NiIsIng1YyI6WyJNSUkiLCJNSUkyIl19.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWUsImlhdCI6MTUxNjIzOTAyMn0" dataUsingEncoding:NSUTF8StringEncoding]; + XCTAssertEqualObjects(jws.digest, digest, @"Parsed payload incorrectly."); + NSData *signature = [@"CDB63_lCSBlmrIfZBvn6w4rDKJgkmKhFe4mfR-xUnfxf9N0g4vZa0R9lFG5pjVkThq9CX-p-vM_64wG4bAC53VlXOk6DhjzN0LTCo1nB81rd8DgqMH4SkLFy3wP-Xe0akRmXE8iHmv63ip7d2LGQVCD38xwXOnoBUVANCrcsC0Iur1TTEXaEfT6ACwg3V1YTu-vygNdbhYZOC_Q9ESbaoPxOQfumXnD44m1EN_FV3d-uQJx1Rn6w3AkDw34P3KunLrwOMJt1mbkWzb66VDVsIxegc4N8TjJTzvxmCk841wUae3kZ97_HPIEfil3ewv80hZstEE2hcEXJbdBfsxsSqg" _stds_base64URLDecodedData]; + XCTAssertEqualObjects(jws.signature, signature, @"Parsed signature incorrectly."); + NSData *payload = [@"eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWUsImlhdCI6MTUxNjIzOTAyMn0" _stds_base64URLDecodedData]; + XCTAssertEqualObjects(jws.payload, payload, @"Parsed payload incorrectly."); + + XCTAssertNil(jws.ellipticCurvePoint, @"Should not create elliptic curve point."); + + NSArray *certChain = @[@"MII", @"MII2"]; + XCTAssertEqualObjects(jws.certificateChain, certChain, @"Failed to parse x5c correctly."); + + +} +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSSecTypeUtilitiesTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSSecTypeUtilitiesTests.m new file mode 100644 index 00000000..9c4c3b2d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSSecTypeUtilitiesTests.m @@ -0,0 +1,142 @@ +// +// STDSSecTypeUtilitiesTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 1/28/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "NSString+JWEHelpers.h" +#import "STDSSecTypeUtilities.h" + +@interface STDSSecTypeUtilitiesTests : XCTestCase + +@end + +@implementation STDSSecTypeUtilitiesTests + +- (void)testDirectoryServerForID { + XCTAssertEqual(STDSDirectoryServerForID(@"ul_test"), STDSDirectoryServerSTPTestRSA, @"ul_test should map to STDSDirectoryServerSTPTestRSA."); + XCTAssertEqual(STDSDirectoryServerForID(@"ec_test"), STDSDirectoryServerSTPTestEC, @"ec_test should map to STDSDirectoryServerSTPTestEC."); + XCTAssertEqual(STDSDirectoryServerForID(@"F055545342"), STDSDirectoryServerULTestRSA, @"F055545342 should map to STDSDirectoryServerULTestRSA."); + XCTAssertEqual(STDSDirectoryServerForID(@"F155545342"), STDSDirectoryServerULTestEC, @"F155545342 should map to STDSDirectoryServerULTestEC."); + + XCTAssertEqual(STDSDirectoryServerForID(@"junk"), STDSDirectoryServerUnknown, @"junk server ID should map to STDSDirectoryServerUnknown."); +} + +- (void)testCertificateForServer { + SecCertificateRef certificate = NULL; + + certificate = STDSCertificateForServer(STDSDirectoryServerSTPTestRSA); + XCTAssertTrue(certificate != NULL, @"Unable to load STDSDirectoryServerSTPTestRSA certificate."); + if (certificate != NULL) { + CFRelease(certificate); + } + certificate = STDSCertificateForServer(STDSDirectoryServerSTPTestEC); + XCTAssertTrue(certificate != NULL, @"Unable to load STDSDirectoryServerSTPTestEC certificate."); + if (certificate != NULL) { + CFRelease(certificate); + } + certificate = STDSCertificateForServer(STDSDirectoryServerUnknown); + if (certificate != NULL) { + XCTFail(@"Should not have an unknown certificate."); + CFRelease(certificate); + } +} + +- (void)testCopyPublicRSAKey { + SecCertificateRef certificate = STDSCertificateForServer(STDSDirectoryServerSTPTestRSA); + if (certificate != NULL) { + SecKeyRef publicKey = SecCertificateCopyKey(certificate); + if (publicKey != NULL) { + CFRelease(publicKey); + } else { + XCTFail(@"Unable to load public key from certificate"); + } + + CFRelease(certificate); + } else { + XCTFail(@"Failed loading certificate for %@", NSStringFromSelector(_cmd)); + } +} + +- (void)testCopyPublicECKey { + SecCertificateRef certificate = STDSCertificateForServer(STDSDirectoryServerSTPTestEC); + if (certificate != NULL) { + SecKeyRef publicKey = SecCertificateCopyKey(certificate); + if (publicKey != NULL) { + CFRelease(publicKey); + } else { + XCTFail(@"Unable to load public key from certificate"); + } + + CFRelease(certificate); + } else { + XCTFail(@"Failed loading certificate for %@", NSStringFromSelector(_cmd)); + } +} + +- (void)testCopyKeyTypeRSA { + SecCertificateRef certificate = STDSCertificateForServer(STDSDirectoryServerSTPTestRSA); + if (certificate != NULL) { + CFStringRef keyType = STDSSecCertificateCopyPublicKeyType(certificate); + if (keyType != NULL) { + XCTAssertTrue(CFStringCompare(keyType, kSecAttrKeyTypeRSA, 0) == kCFCompareEqualTo, @"Key type is incorrect"); + CFRelease(keyType); + } else { + XCTFail(@"Failed to copy key type."); + } + CFRelease(certificate); + } else { + XCTFail(@"Failed loading certificate for %@", NSStringFromSelector(_cmd)); + } +} + +- (void)testCopyKeyTypeEC { + SecCertificateRef certificate = STDSCertificateForServer(STDSDirectoryServerSTPTestEC); + if (certificate != NULL) { + CFStringRef keyType = STDSSecCertificateCopyPublicKeyType(certificate); + if (keyType != NULL) { + XCTAssertTrue(CFStringCompare(keyType, kSecAttrKeyTypeECSECPrimeRandom, 0) == kCFCompareEqualTo, @"Key type is incorrect"); + CFRelease(keyType); + } else { + XCTFail(@"Failed to copy key type."); + } + CFRelease(certificate); + } else { + XCTFail(@"Failed loading certificate for %@", NSStringFromSelector(_cmd)); + } +} + +- (void)testRandomData { + // We're not actually going to implement randomness tests... just sanity + NSData *data1 = STDSCryptoRandomData(32); + NSData *data2 = STDSCryptoRandomData(32); + + XCTAssertNotNil(data1); + XCTAssertTrue(data1.length == 32, @"Random data is not correct length."); + XCTAssertNotEqualObjects(data1, data2, @"Sanity check: two random data's should not equate to equal (unless you get reeeeallly unlucky."); + XCTAssertTrue(STDSCryptoRandomData(12).length == 12, @"Random data is not correct length."); + XCTAssertNotNil(STDSCryptoRandomData(0), @"Empty random data should return empty data."); + XCTAssertTrue(STDSCryptoRandomData(0).length == 0, @"Empty random data should have length 0"); +} + +- (void)testConcatKDFWithSHA256 { + NSData *data = STDSCreateConcatKDFWithSHA256(STDSCryptoRandomData(32), 256, @"acs_identifier"); + XCTAssertNotNil(data, @"Failed to concat KDF and hash."); + XCTAssertEqual(data.length, 256, @"Concat returned data of incorrect length"); +} + + +- (void)testVerifyEllipticCurveP256 { + NSData *payload = [@"eyJhbGciOiJFUzI1NiJ9.eyJpc3MiOiJqb2UiLA0KICJleHAiOjEzMDA4MTkzODAsDQogImh0dHA6Ly9leGFtcGxlLmNvbS9pc19yb290Ijp0cnVlfQ" dataUsingEncoding:NSUTF8StringEncoding]; + NSData *signature = [@"DtEhU3ljbEg8L38VWAfUAqOyKAM6-Xx-F4GawxaepmXFCgfTjDxw5djxLa8ISlSApmWQxfKTUJqPP3-Kg6NU1Q" _stds_base64URLDecodedData]; + + NSData *coordinateX = [@"f83OJ3D2xF1Bg8vub9tLe1gHMzV76e8Tus9uPHvRVEU" _stds_base64URLDecodedData]; + NSData *coordinateY = [@"x_FEzRu9m36HLN_tue659LNpXW6pCyStikYjKIWI5a0" _stds_base64URLDecodedData]; + + XCTAssertTrue(STDSVerifyEllipticCurveP256Signature(coordinateX, coordinateY, payload, signature)); +} +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSSynchronousLocationManagerTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSSynchronousLocationManagerTests.m new file mode 100644 index 00000000..65e2f9a2 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSSynchronousLocationManagerTests.m @@ -0,0 +1,28 @@ +// +// STDSSynchronousLocationManagerTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 1/24/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSSynchronousLocationManager.h" + +@interface STDSSynchronousLocationManagerTests : XCTestCase + +@end + +@implementation STDSSynchronousLocationManagerTests + +- (void)testLocationFetchIsSynchronous { + id originalLocation = [[NSObject alloc] init]; + id location = originalLocation; + + location = [[STDSSynchronousLocationManager sharedManager] deviceLocation]; + // tests that location gets synchronously updated (even if it's to nil due to permissions while running tests) + XCTAssertNotEqual(originalLocation, location); +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.h b/Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.h new file mode 100644 index 00000000..c05e729a --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.h @@ -0,0 +1,19 @@ +// +// STDSTestJSONUtils.h +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +NS_ASSUME_NONNULL_BEGIN + +@interface STDSTestJSONUtils : NSObject + ++ (NSDictionary *)jsonNamed:(NSString *)name; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.m b/Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.m new file mode 100644 index 00000000..f742e1fc --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSTestJSONUtils.m @@ -0,0 +1,54 @@ +// +// STDSTestJSONUtils.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/29/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import "STDSTestJSONUtils.h" + +@implementation STDSTestJSONUtils + ++ (NSDictionary *)jsonNamed:(NSString *)name { + NSData *data = [self dataFromJSONFile:name]; + if (data != nil) { + return [NSJSONSerialization JSONObjectWithData:data options:(NSJSONReadingOptions)kNilOptions error:nil]; + } + return nil; +} + ++ (NSBundle *)testBundle { + return [NSBundle bundleForClass:[STDSTestJSONUtils class]]; +} + ++ (NSData *)dataFromJSONFile:(NSString *)name { + NSBundle *bundle = [self testBundle]; + NSString *path = [bundle pathForResource:name ofType:@"json"]; + + if (!path) { + // Missing JSON file + return nil; + } + + NSError *error = nil; + NSString *jsonString = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:&error]; + + if (!jsonString) { + // File read error + return nil; + } + + // Strip all lines that begin with `//` + NSMutableArray *jsonLines = [[NSMutableArray alloc] init]; + + for (NSString *line in [jsonString componentsSeparatedByString:@"\n"]) { + if (![line hasPrefix:@"//"]) { + [jsonLines addObject:line]; + } + } + + return [[jsonLines componentsJoinedByString:@"\n"] dataUsingEncoding:NSUTF8StringEncoding]; +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSThreeDS2ServiceTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSThreeDS2ServiceTests.m new file mode 100644 index 00000000..237b5127 --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSThreeDS2ServiceTests.m @@ -0,0 +1,62 @@ +// +// STDSThreeDS2ServiceTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 1/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSAlreadyInitializedException.h" +#import "STDSConfigParameters.h" +#import "STDSInvalidInputException.h" +#import "STDSThreeDS2Service.h" +#import "STDSNotInitializedException.h" + +@interface STDSThreeDS2ServiceTests : XCTestCase + +@end + +@implementation STDSThreeDS2ServiceTests + +- (void)testInitialize { + STDSThreeDS2Service *service = [[STDSThreeDS2Service alloc] init]; + XCTAssertNoThrow([service initializeWithConfig:[[STDSConfigParameters alloc] init] + locale:nil + uiSettings:nil], + @"Should not throw with valid input and first call to initialize"); + + XCTAssertThrowsSpecific([service initializeWithConfig:[[STDSConfigParameters alloc] init] + locale:nil + uiSettings:nil], + STDSAlreadyInitializedException, + @"Should throw STDSAlreadyInitializedException if called again with valid input."); + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wnonnull" + XCTAssertThrowsSpecific([service initializeWithConfig:nil + locale:nil + uiSettings:nil], + STDSInvalidInputException, + @"Should throw STDSInvalidInputException for nil config even if already initialized."); + + service = [[STDSThreeDS2Service alloc] init]; + XCTAssertThrowsSpecific([service initializeWithConfig:nil + locale:nil + uiSettings:nil], + STDSInvalidInputException, + @"Should throw STDSInvalidInputException for nil config on first initialize."); +#pragma clang diagnostic pop + + XCTAssertNoThrow([service initializeWithConfig:[[STDSConfigParameters alloc] init] + locale:nil + uiSettings:nil], + @"Should not throw with valid input and first call to initialize even after invalid input"); + + service = [[STDSThreeDS2Service alloc] init]; + XCTAssertThrowsSpecific(service.warnings, STDSNotInitializedException); + +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSTransactionTest.m b/Stripe3DS2/Stripe3DS2Tests/STDSTransactionTest.m new file mode 100644 index 00000000..3dd1187c --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSTransactionTest.m @@ -0,0 +1,157 @@ +// +// STDSTransactionTest.m +// Stripe3DS2Tests +// +// Created by Yuki Tokuhiro on 3/22/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSTransaction+Private.h" +#import "STDSInvalidInputException.h" +#import "STDSChallengeParameters.h" +#import "STDSChallengeStatusReceiver.h" +#import "STDSRuntimeErrorEvent.h" +#import "STDSProtocolErrorEvent.h" +#import "STDSStripe3DS2Error.h" +#import "STDSTransaction.h" +#import "STDSErrorMessage.h" +#import "NSError+Stripe3DS2.h" + +@interface STDSTransaction (Private) +@property (nonatomic, weak) id challengeStatusReceiver; + +- (void)_handleError:(NSError *)error; +- (NSString *)_sdkAppIdentifier; +@end + +@interface STDSTransactionTest : XCTestCase +@property (nonatomic, copy) void (^didErrorWithProtocolErrorEvent)(STDSProtocolErrorEvent *); +@property (nonatomic, copy) void (^didErrorWithRuntimeErrorEvent)(STDSRuntimeErrorEvent *); +@end + +@implementation STDSTransactionTest + +- (void)tearDown { + self.didErrorWithRuntimeErrorEvent = nil; + self.didErrorWithProtocolErrorEvent = nil; + [super tearDown]; +} + +#pragma mark - Timeout + +- (void)testTimeoutBelow5Throws { + STDSTransaction *transaction = [STDSTransaction new]; + STDSChallengeParameters *challengeParameters = [[STDSChallengeParameters alloc] init]; + XCTAssertThrowsSpecific([transaction doChallengeWithViewController:[UIViewController new] + challengeParameters:challengeParameters + challengeStatusReceiver:self + timeout:4 * 60], STDSInvalidInputException); +} + +- (void)testTimeoutFires { + STDSTransaction *transaction = [STDSTransaction new]; + STDSChallengeParameters *challengeParameters = [[STDSChallengeParameters alloc] init]; + + // Assert timer is scheduled to fire 5 minutes from now, give or take 1 second + NSInteger timeout = 300; + [transaction doChallengeWithViewController:[UIViewController new] + challengeParameters:challengeParameters + challengeStatusReceiver:self + timeout:timeout]; + XCTAssertTrue(transaction.timeoutTimer.isValid); + NSTimeInterval secondsIntoTheFutureTimerWillFire = [transaction.timeoutTimer.fireDate timeIntervalSinceNow]; + XCTAssertLessThanOrEqual(fabs(secondsIntoTheFutureTimerWillFire - (timeout)), 1); +} + +#pragma mark - Error Handling + +- (void)testHandleUnknownMessageTypeError { + NSError *error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain code:STDSErrorCodeUnknownMessageType userInfo:nil]; + [self _expectProtocolErrorEventForError:error validator:^(STDSProtocolErrorEvent *protocolErrorEvent) { + XCTAssertEqualObjects(protocolErrorEvent.errorMessage.errorCode, @"101"); + }]; +} + +- (void)testHandleInvalidJSONError { + NSError *error = [NSError _stds_invalidJSONFieldError:@"invalid field"]; + [self _expectProtocolErrorEventForError:error validator:^(STDSProtocolErrorEvent *protocolErrorEvent) { + XCTAssertEqualObjects(protocolErrorEvent.errorMessage.errorCode, @"203"); + XCTAssertEqualObjects(protocolErrorEvent.errorMessage.errorDetails, @"invalid field"); + }]; +} + +- (void)testHandleMissingJSONError { + NSError *error = [NSError _stds_missingJSONFieldError:@"missing field"]; + [self _expectProtocolErrorEventForError:error validator:^(STDSProtocolErrorEvent *protocolErrorEvent) { + XCTAssertEqualObjects(protocolErrorEvent.errorMessage.errorCode, @"201"); + XCTAssertEqualObjects(protocolErrorEvent.errorMessage.errorDetails, @"missing field"); + }]; +} + +- (void)testHandleReceivedErrorMessage { + STDSErrorMessage *receivedErrorMessage = [[STDSErrorMessage alloc] initWithErrorCode:@"" errorComponent:@"" errorDescription:@"" errorDetails:nil messageVersion:@"" acsTransactionIdentifier:@"" errorMessageType:@""]; + NSError *error = [NSError errorWithDomain:STDSStripe3DS2ErrorDomain + code:STDSErrorCodeReceivedErrorMessage + userInfo:@{STDSStripe3DS2ErrorMessageErrorKey: receivedErrorMessage}]; + + [self _expectProtocolErrorEventForError:error validator:^(STDSProtocolErrorEvent *protocolErrorEvent) { + XCTAssertEqualObjects(protocolErrorEvent.errorMessage, receivedErrorMessage); + }]; +} + +- (void)testHandleNetworkConnectionLostError { + NSError *error = [NSError errorWithDomain:NSURLErrorDomain + code:NSURLErrorNetworkConnectionLost + userInfo:nil]; + [self _expectRuntimeErrorEventForError:error validator:^(STDSRuntimeErrorEvent *runtimeErrorEvent) { + XCTAssertEqualObjects(runtimeErrorEvent.errorCode, [@(NSURLErrorNetworkConnectionLost) stringValue]); + }]; +} + +- (void)_expectProtocolErrorEventForError:(NSError *)error validator:(void (^)(STDSProtocolErrorEvent *))protocolErrorEventChecker { + STDSTransaction *transaction = [STDSTransaction new]; + transaction.challengeStatusReceiver = self; + XCTestExpectation *expectation = [self expectationWithDescription:@"Call didErrorWithProtocolErrorEvent"]; + self.didErrorWithProtocolErrorEvent = ^(STDSProtocolErrorEvent *protocolErrorEvent) { + protocolErrorEventChecker(protocolErrorEvent); + [expectation fulfill]; + }; + [transaction _handleError:error]; + [self waitForExpectationsWithTimeout:1 handler:nil]; +} + +- (void)_expectRuntimeErrorEventForError:(NSError *)error validator:(void (^)(STDSRuntimeErrorEvent *))runtimeErrorEventChecker { + STDSTransaction *transaction = [STDSTransaction new]; + transaction.challengeStatusReceiver = self; + XCTestExpectation *expectation = [self expectationWithDescription:@"Call didErrorWithRuntimeErrorEvent"]; + self.didErrorWithRuntimeErrorEvent = ^(STDSRuntimeErrorEvent *runtimeErrorEvent) { + runtimeErrorEventChecker(runtimeErrorEvent); + [expectation fulfill]; + }; + [transaction _handleError:error]; + [self waitForExpectationsWithTimeout:1 handler:nil]; +} + +#pragma mark - STDSChallengeStatusReceiver + +- (void)transaction:(nonnull STDSTransaction *)transaction didCompleteChallengeWithCompletionEvent:(nonnull STDSCompletionEvent *)completionEvent {} + +- (void)transaction:(nonnull STDSTransaction *)transaction didErrorWithProtocolErrorEvent:(nonnull STDSProtocolErrorEvent *)protocolErrorEvent { + if (self.didErrorWithProtocolErrorEvent) { + self.didErrorWithProtocolErrorEvent(protocolErrorEvent); + } +} + +- (void)transaction:(nonnull STDSTransaction *)transaction didErrorWithRuntimeErrorEvent:(nonnull STDSRuntimeErrorEvent *)runtimeErrorEvent { + if (self.didErrorWithRuntimeErrorEvent) { + self.didErrorWithRuntimeErrorEvent(runtimeErrorEvent); + } +} + +- (void)transactionDidCancel:(nonnull STDSTransaction *)transaction {} + +- (void)transactionDidTimeOut:(nonnull STDSTransaction *)transaction {} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSUICustomizationTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSUICustomizationTests.m new file mode 100644 index 00000000..9d231acf --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSUICustomizationTests.m @@ -0,0 +1,249 @@ +// +// STDSUICustomizationTests.m +// Stripe3DS2Tests +// +// Created by Andrew Harrison on 3/14/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import +#import "STDSUICustomization.h" + +@interface STDSUICustomizationTests : XCTestCase + +@end + +@implementation STDSUICustomizationTests + +// The following helper methods return customization objects with properties different than the default. + +- (STDSNavigationBarCustomization *)_customNavigationBar { + STDSNavigationBarCustomization *custom = [STDSNavigationBarCustomization new]; + custom.font = [UIFont italicSystemFontOfSize:1]; + custom.textColor = UIColor.blueColor; + custom.barTintColor = UIColor.redColor; + custom.barStyle = UIBarStyleBlack; + custom.translucent = NO; + custom.headerText = @"foo"; + custom.buttonText = @"bar"; + return custom; +} + +- (STDSLabelCustomization *)_customLabel { + STDSLabelCustomization *custom = [STDSLabelCustomization new]; + custom.font = [UIFont italicSystemFontOfSize:1]; + custom.textColor = UIColor.blueColor; + custom.headingTextColor = UIColor.redColor; + custom.headingFont = [UIFont italicSystemFontOfSize:2]; + return custom; +} + +- (STDSTextFieldCustomization *)_customTextField { + STDSTextFieldCustomization *custom = [STDSTextFieldCustomization new]; + custom.font = [UIFont italicSystemFontOfSize:1]; + custom.textColor = UIColor.blueColor; + custom.borderWidth = -1; + custom.borderColor = UIColor.redColor; + custom.cornerRadius = -8; + custom.keyboardAppearance = UIKeyboardAppearanceAlert; + custom.placeholderTextColor = UIColor.greenColor; + return custom; +} + +- (STDSButtonCustomization *)_customButton { + STDSButtonCustomization *custom = [[STDSButtonCustomization alloc] initWithBackgroundColor:UIColor.redColor cornerRadius:-1]; + custom.font = [UIFont italicSystemFontOfSize:1]; + custom.textColor = UIColor.blueColor; + custom.titleStyle = STDSButtonTitleStyleLowercase; + return custom; +} + +- (STDSFooterCustomization *)_customFooter { + STDSFooterCustomization *custom = [STDSFooterCustomization new]; + custom.font = [UIFont italicSystemFontOfSize:1]; + custom.textColor = UIColor.blueColor; + custom.backgroundColor = UIColor.redColor; + custom.chevronColor = UIColor.greenColor; + custom.headingTextColor = UIColor.grayColor; + custom.headingFont = [UIFont italicSystemFontOfSize:2]; + return custom; +} + +- (STDSSelectionCustomization *)_customSelection { + STDSSelectionCustomization *custom = [STDSSelectionCustomization new]; + custom.primarySelectedColor = UIColor.redColor; + custom.secondarySelectedColor = UIColor.blueColor; + custom.unselectedBorderColor = UIColor.brownColor; + custom.unselectedBackgroundColor = UIColor.cyanColor; + return custom; +} + +#pragma mark - Copying + +- (void)testUICustomizationDeepCopy { + // Make a STDSUICustomization instance with all non-default properties + STDSButtonCustomization *submitButtonCustomization = [self _customButton]; + STDSButtonCustomization *continueButtonCustomization = [self _customButton]; + continueButtonCustomization.cornerRadius = -2; + STDSButtonCustomization *nextButtonCustomization = [self _customButton]; + nextButtonCustomization.cornerRadius = -3; + STDSButtonCustomization *cancelButtonCustomization = [self _customButton]; + cancelButtonCustomization.cornerRadius = -4; + STDSButtonCustomization *resendButtonCustomization = [self _customButton]; + resendButtonCustomization.cornerRadius = -5; + + STDSNavigationBarCustomization *navigationBarCustomization = [self _customNavigationBar]; + STDSLabelCustomization *labelCustomization = [self _customLabel]; + STDSTextFieldCustomization *textFieldCustomization = [self _customTextField]; + STDSFooterCustomization *footerCustomization = [self _customFooter]; + STDSSelectionCustomization *selectionCustomization = [self _customSelection]; + + STDSUICustomization *uiCustomization = [[STDSUICustomization alloc] init]; + uiCustomization.footerCustomization = footerCustomization; + uiCustomization.selectionCustomization = selectionCustomization; + [uiCustomization setButtonCustomization:submitButtonCustomization forType:STDSUICustomizationButtonTypeSubmit]; + [uiCustomization setButtonCustomization:continueButtonCustomization forType:STDSUICustomizationButtonTypeContinue]; + [uiCustomization setButtonCustomization:nextButtonCustomization forType:STDSUICustomizationButtonTypeNext]; + [uiCustomization setButtonCustomization:cancelButtonCustomization forType:STDSUICustomizationButtonTypeCancel]; + [uiCustomization setButtonCustomization:resendButtonCustomization forType:STDSUICustomizationButtonTypeResend]; + uiCustomization.navigationBarCustomization = navigationBarCustomization; + uiCustomization.labelCustomization = labelCustomization; + uiCustomization.textFieldCustomization = textFieldCustomization; + uiCustomization.backgroundColor = UIColor.redColor; + uiCustomization.activityIndicatorViewStyle = UIActivityIndicatorViewStyleLarge; + uiCustomization.blurStyle = UIBlurEffectStyleDark; + uiCustomization.preferredStatusBarStyle = UIStatusBarStyleLightContent; + + STDSUICustomization *copy = [uiCustomization copy]; + XCTAssertNotNil([copy buttonCustomizationForButtonType:STDSUICustomizationButtonTypeNext]); + XCTAssertNotNil(copy.navigationBarCustomization); + XCTAssertNotNil(copy.labelCustomization); + XCTAssertNotNil(copy.textFieldCustomization); + XCTAssertNotNil(copy.footerCustomization); + XCTAssertNotNil(copy.selectionCustomization); + + /// The pointers do not reference the same objects. + XCTAssertNotEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeSubmit], [copy buttonCustomizationForButtonType:STDSUICustomizationButtonTypeSubmit]); + XCTAssertNotEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeContinue], [copy buttonCustomizationForButtonType:STDSUICustomizationButtonTypeContinue]); + XCTAssertNotEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeNext], [copy buttonCustomizationForButtonType:STDSUICustomizationButtonTypeNext]); + XCTAssertNotEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeCancel], [copy buttonCustomizationForButtonType:STDSUICustomizationButtonTypeCancel]); + XCTAssertNotEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeResend], [copy buttonCustomizationForButtonType:STDSUICustomizationButtonTypeResend]); + XCTAssertNotEqual(uiCustomization.navigationBarCustomization, copy.navigationBarCustomization); + XCTAssertNotEqual(uiCustomization.labelCustomization, copy.labelCustomization); + XCTAssertNotEqual(uiCustomization.textFieldCustomization, copy.textFieldCustomization); + XCTAssertNotEqual(uiCustomization.footerCustomization, copy.footerCustomization); + XCTAssertNotEqual(uiCustomization.selectionCustomization, copy.selectionCustomization); + + /// The properties have been successfully copied. + XCTAssertEqualObjects(uiCustomization.backgroundColor, copy.backgroundColor); + XCTAssertEqual(uiCustomization.activityIndicatorViewStyle, copy.activityIndicatorViewStyle); + XCTAssertEqual(uiCustomization.blurStyle, copy.blurStyle); + // A different test case will cover that our custom classes implemented copy correctly; just sanity check one property here + XCTAssertEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeSubmit].cornerRadius, submitButtonCustomization.cornerRadius); + XCTAssertEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeContinue].cornerRadius, continueButtonCustomization.cornerRadius); + XCTAssertEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeNext].cornerRadius, nextButtonCustomization.cornerRadius); + XCTAssertEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeCancel].cornerRadius, cancelButtonCustomization.cornerRadius); + XCTAssertEqual([uiCustomization buttonCustomizationForButtonType:STDSUICustomizationButtonTypeResend].cornerRadius, resendButtonCustomization.cornerRadius); + XCTAssertEqualObjects(uiCustomization.navigationBarCustomization.font, copy.navigationBarCustomization.font); + XCTAssertEqualObjects(uiCustomization.labelCustomization.font, copy.labelCustomization.font); + XCTAssertEqualObjects(uiCustomization.textFieldCustomization.font, copy.textFieldCustomization.font); + XCTAssertEqualObjects(uiCustomization.footerCustomization.font, copy.footerCustomization.font); + XCTAssertEqualObjects(uiCustomization.selectionCustomization.primarySelectedColor, copy.selectionCustomization.primarySelectedColor); + XCTAssertEqual(uiCustomization.preferredStatusBarStyle, copy.preferredStatusBarStyle); +} + +- (void)testButtonCustomizationIsCopied { + STDSButtonCustomization *buttonCustomization = [self _customButton]; + + /// The pointers do not reference the same objects. + STDSButtonCustomization *copy = [buttonCustomization copy]; + XCTAssertNotEqual(buttonCustomization, copy); + + /// The properties have been successfully copied. + XCTAssertEqual(buttonCustomization.cornerRadius, copy.cornerRadius); + XCTAssertEqual(buttonCustomization.backgroundColor, copy.backgroundColor); + XCTAssertEqual(buttonCustomization.font, copy.font); + XCTAssertEqual(buttonCustomization.textColor, copy.textColor); + XCTAssertEqual(buttonCustomization.titleStyle, buttonCustomization.titleStyle); +} + +- (void)testNavigationBarCustomizationIsCopied { + STDSNavigationBarCustomization *navigationBarCustomization = [self _customNavigationBar]; + + /// The pointers do not reference the same objects. + STDSNavigationBarCustomization *copy = [navigationBarCustomization copy]; + XCTAssertNotEqual(navigationBarCustomization, copy); + + /// The properties have been successfully copied. + XCTAssertEqualObjects(navigationBarCustomization.headerText, copy.headerText); + XCTAssertEqualObjects(navigationBarCustomization.buttonText, copy.buttonText); + XCTAssertEqualObjects(navigationBarCustomization.barTintColor, copy.barTintColor); + XCTAssertEqualObjects(navigationBarCustomization.font, copy.font); + XCTAssertEqualObjects(navigationBarCustomization.textColor, copy.textColor); + XCTAssertEqual(navigationBarCustomization.barStyle, copy.barStyle); + XCTAssertEqual(navigationBarCustomization.translucent, copy.translucent); +} + +- (void)testLabelCustomizationIsCopied { + STDSLabelCustomization *labelCustomization = [self _customLabel]; + + /// The pointers do not reference the same objects. + STDSLabelCustomization *copy = [labelCustomization copy]; + XCTAssertNotEqual(labelCustomization, copy); + + /// The properties have been successfully copied. + XCTAssertEqualObjects(labelCustomization.headingTextColor, copy.headingTextColor); + XCTAssertEqualObjects(labelCustomization.headingFont, copy.headingFont); + XCTAssertEqualObjects(labelCustomization.font, copy.font); + XCTAssertEqualObjects(labelCustomization.textColor, copy.textColor); +} + +- (void)testTextFieldCustomizationIsCopied { + STDSTextFieldCustomization *textFieldCustomization = [self _customTextField]; + + /// The pointers do not reference the same objects. + STDSTextFieldCustomization *copy = [textFieldCustomization copy]; + XCTAssertNotEqual(textFieldCustomization, copy); + + /// The properties have been successfully copied. + XCTAssertEqual(textFieldCustomization.borderWidth, copy.borderWidth); + XCTAssertEqualObjects(textFieldCustomization.borderColor, copy.borderColor); + XCTAssertEqual(textFieldCustomization.cornerRadius, copy.cornerRadius); + XCTAssertEqualObjects(textFieldCustomization.font, copy.font); + XCTAssertEqualObjects(textFieldCustomization.textColor, copy.textColor); + XCTAssertEqual(textFieldCustomization.keyboardAppearance, copy.keyboardAppearance); + XCTAssertEqualObjects(textFieldCustomization.placeholderTextColor, copy.placeholderTextColor); +} + +- (void)testFooterCustomizationIsCopied { + STDSFooterCustomization *footerCustomization = [self _customFooter]; + + /// The pointers do not reference the same objects. + STDSFooterCustomization *copy = [footerCustomization copy]; + XCTAssertNotEqual(footerCustomization, copy); + + /// The properties have been successfully copied. + XCTAssertEqualObjects(footerCustomization.textColor, copy.textColor); + XCTAssertEqualObjects(footerCustomization.font, copy.font); + XCTAssertEqualObjects(footerCustomization.backgroundColor, copy.backgroundColor); + XCTAssertEqualObjects(footerCustomization.chevronColor, copy.chevronColor); + XCTAssertEqualObjects(footerCustomization.headingTextColor, copy.headingTextColor); + XCTAssertEqualObjects(footerCustomization.headingFont, copy.headingFont); +} + +- (void)testSelectionCustomizationIsCopied { + STDSSelectionCustomization *customization = [self _customSelection]; + + /// The pointers do not reference the same objects. + STDSSelectionCustomization *copy = [customization copy]; + XCTAssertNotEqual(customization, copy); + + /// The properties have been successfully copied. + XCTAssertEqualObjects(customization.primarySelectedColor, copy.primarySelectedColor); + XCTAssertEqualObjects(customization.secondarySelectedColor, copy.secondarySelectedColor); + XCTAssertEqualObjects(customization.unselectedBorderColor, copy.unselectedBorderColor); + XCTAssertEqualObjects(customization.unselectedBackgroundColor, copy.unselectedBackgroundColor); + +} + +@end diff --git a/Stripe3DS2/Stripe3DS2Tests/STDSWarningTests.m b/Stripe3DS2/Stripe3DS2Tests/STDSWarningTests.m new file mode 100644 index 00000000..ec559c5d --- /dev/null +++ b/Stripe3DS2/Stripe3DS2Tests/STDSWarningTests.m @@ -0,0 +1,26 @@ +// +// STDSWarningTests.m +// Stripe3DS2Tests +// +// Created by Cameron Sabol on 2/12/19. +// Copyright © 2019 Stripe. All rights reserved. +// + +#import + +#import "STDSWarning.h" + +@interface STDSWarningTests : XCTestCase + +@end + +@implementation STDSWarningTests + +- (void)testWarning { + STDSWarning *warning = [[STDSWarning alloc] initWithIdentifier:@"test_id" message:@"test_message" severity:STDSWarningSeverityMedium]; + XCTAssertEqual(warning.identifier, @"test_id", @"Identifier was not set correctly."); + XCTAssertEqual(warning.message, @"test_message", @"Message was not set correctly."); + XCTAssertEqual(warning.severity, STDSWarningSeverityMedium, @"Severity was not set correctly."); +} + +@end diff --git a/Stripe3DS2/exported_symbols.txt b/Stripe3DS2/exported_symbols.txt new file mode 100644 index 00000000..981c8756 --- /dev/null +++ b/Stripe3DS2/exported_symbols.txt @@ -0,0 +1,13 @@ +_OBJC_CLASS_$_STDSButtonCustomization +_OBJC_CLASS_$_STDSChallengeParameters +_OBJC_CLASS_$_STDSConfigParameters +_OBJC_CLASS_$_STDSFooterCustomization +_OBJC_CLASS_$_STDSJSONEncoder +_OBJC_CLASS_$_STDSLabelCustomization +_OBJC_CLASS_$_STDSNavigationBarCustomization +_OBJC_CLASS_$_STDSSelectionCustomization +_OBJC_CLASS_$_STDSTextFieldCustomization +_OBJC_CLASS_$_STDSThreeDS2Service +_OBJC_CLASS_$_STDSUICustomization +_OBJC_CLASS_$_STDSSwiftTryCatch +_STDSAuthenticationResponseFromJSON diff --git a/StripeApplePay/Project.swift b/StripeApplePay/Project.swift new file mode 100644 index 00000000..020478d5 --- /dev/null +++ b/StripeApplePay/Project.swift @@ -0,0 +1,15 @@ +import ProjectDescription +import ProjectDescriptionHelpers + +let project = Project.stripeFramework( + name: "StripeApplePay", + dependencies: [ + .project(target: "StripeCore", path: "//StripeCore"), + ], + unitTestOptions: .testOptions( + dependencies: [ + .project(target: "StripeCoreTestUtils", path: "//StripeCore"), + ], + usesStubs: true + ) +) diff --git a/StripeApplePay/README.md b/StripeApplePay/README.md new file mode 100644 index 00000000..9c225745 --- /dev/null +++ b/StripeApplePay/README.md @@ -0,0 +1,36 @@ +# Stripe Apple Pay iOS SDK + +StripeApplePay is a lightweight Apple Pay SDK intended for building App Clips or other size-constrained apps. + +## Table of contents + + +- [Stripe Apple Pay iOS SDK](#stripe-apple-pay-ios-sdk) +- [Table of contents](#table-of-contents) + - [Requirements](#requirements) + - [Getting started](#getting-started) + - [Integration](#integration) + - [Example](#example) + - [Manual linking](#manual-linking) + + + +## Requirements + +The Stripe Apple Pay SDK is compatible with apps targeting iOS 13.0 or above. + +## Getting started + +### Integration + +Get started with our [📚 Apple Pay integration guide](https://stripe.com/docs/apple-pay?platform=ios) and [example project](../Example/AppClipExample), or [📘 browse the SDK reference](https://stripe.dev/stripe-ios/stripe-applepay/index.html) for fine-grained documentation of all the classes and methods in the SDK. + +### Example + +[AppClipExample](../Example/AppClipExample) – This example demonstrates how to offer Apple Pay in an App Clip. + +## Manual linking + +If you link this library manually, use a version from our [releases](https://github.com/stripe/stripe-ios/releases) page and make sure to embed all of the following frameworks: +- `StripeCore.xcframework` +- `StripeApplePay.xcframework` diff --git a/StripeApplePay/StripeApplePay.xcodeproj/project.pbxproj b/StripeApplePay/StripeApplePay.xcodeproj/project.pbxproj new file mode 100644 index 00000000..4c4ed77b --- /dev/null +++ b/StripeApplePay/StripeApplePay.xcodeproj/project.pbxproj @@ -0,0 +1,610 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 52; + objects = { + +/* Begin PBXBuildFile section */ + 01AA289840E0AC9229A8CF63 /* StripeApplePay.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 29BFCC7B0FCEA743A857B51F /* StripeApplePay.framework */; }; + 09EB0F7E346CB22144515E67 /* SetupIntent+API.swift in Sources */ = {isa = PBXBuildFile; fileRef = 30DDE851AD7450BE70381337 /* SetupIntent+API.swift */; }; + 1990346BA0B39ADD47210E18 /* StripeApplePay.h in Headers */ = {isa = PBXBuildFile; fileRef = 0C1D3421B1B2BB91FAA66620 /* StripeApplePay.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 22E1F50D066A294A316052ED /* PaymentIntent.swift in Sources */ = {isa = PBXBuildFile; fileRef = D2398CA91B601B5FC7C8FB48 /* PaymentIntent.swift */; }; + 234FA38DC46927B23871A75D /* SetupIntentParams.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3D7CF2B75A797D6B2B5A04B4 /* SetupIntentParams.swift */; }; + 2D6CD6872A00B6FE0243C3F5 /* PKPayment+Stripe.swift in Sources */ = {isa = PBXBuildFile; fileRef = 906F7217DBF694BF42F23458 /* PKPayment+Stripe.swift */; }; + 2EBB230815383A8402D71146 /* STPPaymentMethodFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 905182496A07DE4F056A3EAA /* STPPaymentMethodFunctionalTest.swift */; }; + 43682CEAB00A93868FA3188A /* SetupIntent.swift in Sources */ = {isa = PBXBuildFile; fileRef = CC2A354F50C5D563436A0068 /* SetupIntent.swift */; }; + 463680AADB8CED6E962CD45A /* PaymentIntentParams.swift in Sources */ = {isa = PBXBuildFile; fileRef = 478361D29DF10F2B43F7A1D2 /* PaymentIntentParams.swift */; }; + 4AA6A66246DD30798E5CC5F7 /* PaymentIntent+API.swift in Sources */ = {isa = PBXBuildFile; fileRef = 93EAD8979A20E239E16257E4 /* PaymentIntent+API.swift */; }; + 4ADC5356764DC5E3F1C1D51B /* CardBrand.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8DBC2BC8C0A60983D0948A11 /* CardBrand.swift */; }; + 4B2DE4109D876CCBDC53C2A3 /* StripeCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 62ECC1AA5583E4104C073B63 /* StripeCore.framework */; }; + 4CE57B5BF79D1515F27A18A3 /* Address.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF86CF1EFE4AC1A46F055202 /* Address.swift */; }; + 505A82AADE15E2B2B99C5D32 /* Token.swift in Sources */ = {isa = PBXBuildFile; fileRef = 594F0478E94E6FA2F551EA0A /* Token.swift */; }; + 526AE8381F232C9FEABFFDCD /* OHHTTPStubsSwift in Frameworks */ = {isa = PBXBuildFile; productRef = 8ED017D12DE81D3ABD013768 /* OHHTTPStubsSwift */; }; + 53BA33D02E07DFF1393DF0E4 /* PaymentMethod+API.swift in Sources */ = {isa = PBXBuildFile; fileRef = 264583DBFFA42C864220D7FF /* PaymentMethod+API.swift */; }; + 59AEBEB856DB8A0118F1D0DE /* STPAnalyticsClient+Payments.swift in Sources */ = {isa = PBXBuildFile; fileRef = 20A70C7D203A80129DBB9304 /* STPAnalyticsClient+Payments.swift */; }; + 59C1DB9EF052987BD20B65A3 /* BillingDetails+ApplePay.swift in Sources */ = {isa = PBXBuildFile; fileRef = F4FD028203455E50A637227C /* BillingDetails+ApplePay.swift */; }; + 62AFEE5A1E32BD84588CA233 /* STPTelemetryClientTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E4B4DF82BA51CC68265B0795 /* STPTelemetryClientTest.swift */; }; + 7C7C92AFED77FC4C5D26DC36 /* BillingDetails.swift in Sources */ = {isa = PBXBuildFile; fileRef = C563CA8E1D005A453C762703 /* BillingDetails.swift */; }; + 848843F1145350ABF540D69F /* ShippingDetails.swift in Sources */ = {isa = PBXBuildFile; fileRef = C08710B3CD45DE9DCE2055A3 /* ShippingDetails.swift */; }; + 90286A48FA6C350BDEC227D0 /* TelemetryInjectionTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 446F888DB3EC0FC1FE9B9377 /* TelemetryInjectionTest.swift */; }; + 9F50D23599CD0B1270F8C295 /* STPApplePayContext+LegacySupport.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7023208C7E3B3044F553DFDD /* STPApplePayContext+LegacySupport.swift */; }; + A69598FE8095AFA9898297A9 /* Token+API.swift in Sources */ = {isa = PBXBuildFile; fileRef = D81AB96E6201D534220ABB9D /* Token+API.swift */; }; + A79510332C88568637C9E867 /* STPAnalyticsClient+ApplePayTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E6E9055B54A5634B636570E3 /* STPAnalyticsClient+ApplePayTest.swift */; }; + AB48B0CBFFEFC46E01C76B6C /* STPAPIClient+PaymentsCore.swift in Sources */ = {isa = PBXBuildFile; fileRef = 894B7D67F2364A12DD133CF4 /* STPAPIClient+PaymentsCore.swift */; }; + BCC7454DB40673493DF940F5 /* STPAnalyticsClient+PaymentsAPI.swift in Sources */ = {isa = PBXBuildFile; fileRef = FBD5C52C679274D830AA5B93 /* STPAnalyticsClient+PaymentsAPI.swift */; }; + C0D4B1753F584397DD1AD946 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 87A189B89110D3A8EF052425 /* XCTest.framework */; }; + C6552A9ADEA160B9BBAF3A10 /* STPApplePayContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = E4AEF10AF7900AF7ED0EDE21 /* STPApplePayContext.swift */; }; + C6A333FBB72EE91849DD6202 /* STPAPIClient+ApplePay.swift in Sources */ = {isa = PBXBuildFile; fileRef = 855AB5D51581CE6ABCCD2493 /* STPAPIClient+ApplePay.swift */; }; + CB78059C8EB6A072C30A98DA /* STPTelemetryClientFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0B7AA620790EC4257132D738 /* STPTelemetryClientFunctionalTest.swift */; }; + CECAB0CB1CE4D7BD1EF897F0 /* Blocks.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5439AD0D1236F6A21EE93CB3 /* Blocks.swift */; }; + D8D613D3A85B81F8C4386E08 /* OHHTTPStubs in Frameworks */ = {isa = PBXBuildFile; productRef = CB64080D8A41D33BC3DEEAF8 /* OHHTTPStubs */; }; + E1E7B153B169D0A6363ADD4B /* PaymentMethod.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6D1E8A9F849655BAC63DB2FD /* PaymentMethod.swift */; }; + EEF84AC6C1EBF27BF6AAC0BF /* StripeCoreTestUtils.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 0FA663325484F2D515613494 /* StripeCoreTestUtils.framework */; }; + F42BC784EDBD141C90E74A5F /* PKContact+Stripe.swift in Sources */ = {isa = PBXBuildFile; fileRef = 133A7483EDB9F1B1CFFDB9ED /* PKContact+Stripe.swift */; }; + F59D5A3F2C5849CD465062A5 /* StripeCore+Import.swift in Sources */ = {isa = PBXBuildFile; fileRef = 88664A990EE5D99076E88987 /* StripeCore+Import.swift */; }; + FADFD3B7E3832928E254FAF1 /* PaymentMethodParams.swift in Sources */ = {isa = PBXBuildFile; fileRef = 468654E242DFE2F85FF422EB /* PaymentMethodParams.swift */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 6DAB385DAFF8DCB87110CC7E /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 95898E5CA0A1871DDFFB66B0 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 747C9FE1743C0B4444303569; + remoteInfo = StripeApplePay; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + F937383B644A63BD5FC4A081 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + FF3CB7A978895136380088BB /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 0B7AA620790EC4257132D738 /* STPTelemetryClientFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPTelemetryClientFunctionalTest.swift; sourceTree = ""; }; + 0C1D3421B1B2BB91FAA66620 /* StripeApplePay.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = StripeApplePay.h; sourceTree = ""; }; + 0FA663325484F2D515613494 /* StripeCoreTestUtils.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCoreTestUtils.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 133A7483EDB9F1B1CFFDB9ED /* PKContact+Stripe.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKContact+Stripe.swift"; sourceTree = ""; }; + 20A70C7D203A80129DBB9304 /* STPAnalyticsClient+Payments.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAnalyticsClient+Payments.swift"; sourceTree = ""; }; + 229446797CC5FB07DB344D81 /* StripeApplePayTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = StripeApplePayTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 264583DBFFA42C864220D7FF /* PaymentMethod+API.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PaymentMethod+API.swift"; sourceTree = ""; }; + 29BFCC7B0FCEA743A857B51F /* StripeApplePay.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeApplePay.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 30DDE851AD7450BE70381337 /* SetupIntent+API.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "SetupIntent+API.swift"; sourceTree = ""; }; + 3D7CF2B75A797D6B2B5A04B4 /* SetupIntentParams.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SetupIntentParams.swift; sourceTree = ""; }; + 40D5ED47331313B6AD8B6F46 /* Project-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Debug.xcconfig"; sourceTree = ""; }; + 425B19195E4B86F77229252D /* StripeiOS Tests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Debug.xcconfig"; sourceTree = ""; }; + 446F888DB3EC0FC1FE9B9377 /* TelemetryInjectionTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TelemetryInjectionTest.swift; sourceTree = ""; }; + 468654E242DFE2F85FF422EB /* PaymentMethodParams.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentMethodParams.swift; sourceTree = ""; }; + 478361D29DF10F2B43F7A1D2 /* PaymentIntentParams.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentIntentParams.swift; sourceTree = ""; }; + 49B0926F8F65BC0102B99BF4 /* StripeiOS-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS-Debug.xcconfig"; sourceTree = ""; }; + 4EE8157F3536C238AC337337 /* StripeiOS-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS-Release.xcconfig"; sourceTree = ""; }; + 5439AD0D1236F6A21EE93CB3 /* Blocks.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Blocks.swift; sourceTree = ""; }; + 594F0478E94E6FA2F551EA0A /* Token.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Token.swift; sourceTree = ""; }; + 62ECC1AA5583E4104C073B63 /* StripeCore.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCore.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 67711D1BC01BA83323EA1F6B /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 6D1E8A9F849655BAC63DB2FD /* PaymentMethod.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentMethod.swift; sourceTree = ""; }; + 7023208C7E3B3044F553DFDD /* STPApplePayContext+LegacySupport.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPApplePayContext+LegacySupport.swift"; sourceTree = ""; }; + 79B089D69DBA1A0BCF4503B6 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 855AB5D51581CE6ABCCD2493 /* STPAPIClient+ApplePay.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAPIClient+ApplePay.swift"; sourceTree = ""; }; + 87A189B89110D3A8EF052425 /* XCTest.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = XCTest.framework; path = Platforms/iPhoneOS.platform/Developer/Library/Frameworks/XCTest.framework; sourceTree = DEVELOPER_DIR; }; + 88664A990EE5D99076E88987 /* StripeCore+Import.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "StripeCore+Import.swift"; sourceTree = ""; }; + 894B7D67F2364A12DD133CF4 /* STPAPIClient+PaymentsCore.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAPIClient+PaymentsCore.swift"; sourceTree = ""; }; + 8DBC2BC8C0A60983D0948A11 /* CardBrand.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardBrand.swift; sourceTree = ""; }; + 905182496A07DE4F056A3EAA /* STPPaymentMethodFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodFunctionalTest.swift; sourceTree = ""; }; + 906F7217DBF694BF42F23458 /* PKPayment+Stripe.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKPayment+Stripe.swift"; sourceTree = ""; }; + 93EAD8979A20E239E16257E4 /* PaymentIntent+API.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PaymentIntent+API.swift"; sourceTree = ""; }; + BDAEEEB96953ECBB9C68CF1C /* StripeiOS Tests-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Release.xcconfig"; sourceTree = ""; }; + C08710B3CD45DE9DCE2055A3 /* ShippingDetails.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ShippingDetails.swift; sourceTree = ""; }; + C563CA8E1D005A453C762703 /* BillingDetails.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BillingDetails.swift; sourceTree = ""; }; + CB01F67DF47C468825F8FF6A /* Project-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Release.xcconfig"; sourceTree = ""; }; + CC2A354F50C5D563436A0068 /* SetupIntent.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SetupIntent.swift; sourceTree = ""; }; + D2398CA91B601B5FC7C8FB48 /* PaymentIntent.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentIntent.swift; sourceTree = ""; }; + D81AB96E6201D534220ABB9D /* Token+API.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Token+API.swift"; sourceTree = ""; }; + E4AEF10AF7900AF7ED0EDE21 /* STPApplePayContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayContext.swift; sourceTree = ""; }; + E4B4DF82BA51CC68265B0795 /* STPTelemetryClientTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPTelemetryClientTest.swift; sourceTree = ""; }; + E6E9055B54A5634B636570E3 /* STPAnalyticsClient+ApplePayTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAnalyticsClient+ApplePayTest.swift"; sourceTree = ""; }; + F4FD028203455E50A637227C /* BillingDetails+ApplePay.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "BillingDetails+ApplePay.swift"; sourceTree = ""; }; + FBD5C52C679274D830AA5B93 /* STPAnalyticsClient+PaymentsAPI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAnalyticsClient+PaymentsAPI.swift"; sourceTree = ""; }; + FF86CF1EFE4AC1A46F055202 /* Address.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Address.swift; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 2A690CB3AEAD3A05D0DF3D47 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 4B2DE4109D876CCBDC53C2A3 /* StripeCore.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 92797DEFFA4201115E38DEAD /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + C0D4B1753F584397DD1AD946 /* XCTest.framework in Frameworks */, + 01AA289840E0AC9229A8CF63 /* StripeApplePay.framework in Frameworks */, + EEF84AC6C1EBF27BF6AAC0BF /* StripeCoreTestUtils.framework in Frameworks */, + D8D613D3A85B81F8C4386E08 /* OHHTTPStubs in Frameworks */, + 526AE8381F232C9FEABFFDCD /* OHHTTPStubsSwift in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 0D2F315747B25043D0135CBC /* Analytics */ = { + isa = PBXGroup; + children = ( + 20A70C7D203A80129DBB9304 /* STPAnalyticsClient+Payments.swift */, + FBD5C52C679274D830AA5B93 /* STPAnalyticsClient+PaymentsAPI.swift */, + ); + path = Analytics; + sourceTree = ""; + }; + 18E783E8B2360F08F2FCDEB9 /* Frameworks */ = { + isa = PBXGroup; + children = ( + 87A189B89110D3A8EF052425 /* XCTest.framework */, + ); + name = Frameworks; + sourceTree = ""; + }; + 1A99D698FD7EE445DA2366ED /* StripeApplePay */ = { + isa = PBXGroup; + children = ( + B7FC50BAFE8B7A10FA7EA283 /* Source */, + 67711D1BC01BA83323EA1F6B /* Info.plist */, + 0C1D3421B1B2BB91FAA66620 /* StripeApplePay.h */, + ); + path = StripeApplePay; + sourceTree = ""; + }; + 2F76FD687FF6507E5B2C1A3F /* ApplePayContext */ = { + isa = PBXGroup; + children = ( + 855AB5D51581CE6ABCCD2493 /* STPAPIClient+ApplePay.swift */, + E4AEF10AF7900AF7ED0EDE21 /* STPApplePayContext.swift */, + 7023208C7E3B3044F553DFDD /* STPApplePayContext+LegacySupport.swift */, + ); + path = ApplePayContext; + sourceTree = ""; + }; + 3581723D3A3FF58EE3E59033 /* BuildConfigurations */ = { + isa = PBXGroup; + children = ( + 40D5ED47331313B6AD8B6F46 /* Project-Debug.xcconfig */, + CB01F67DF47C468825F8FF6A /* Project-Release.xcconfig */, + 425B19195E4B86F77229252D /* StripeiOS Tests-Debug.xcconfig */, + BDAEEEB96953ECBB9C68CF1C /* StripeiOS Tests-Release.xcconfig */, + 49B0926F8F65BC0102B99BF4 /* StripeiOS-Debug.xcconfig */, + 4EE8157F3536C238AC337337 /* StripeiOS-Release.xcconfig */, + ); + name = BuildConfigurations; + path = ../BuildConfigurations; + sourceTree = ""; + }; + 3E8665C11A450F2FB7D2F2E7 /* Models */ = { + isa = PBXGroup; + children = ( + FF86CF1EFE4AC1A46F055202 /* Address.swift */, + C563CA8E1D005A453C762703 /* BillingDetails.swift */, + 8DBC2BC8C0A60983D0948A11 /* CardBrand.swift */, + D2398CA91B601B5FC7C8FB48 /* PaymentIntent.swift */, + 478361D29DF10F2B43F7A1D2 /* PaymentIntentParams.swift */, + 6D1E8A9F849655BAC63DB2FD /* PaymentMethod.swift */, + 468654E242DFE2F85FF422EB /* PaymentMethodParams.swift */, + CC2A354F50C5D563436A0068 /* SetupIntent.swift */, + 3D7CF2B75A797D6B2B5A04B4 /* SetupIntentParams.swift */, + C08710B3CD45DE9DCE2055A3 /* ShippingDetails.swift */, + 594F0478E94E6FA2F551EA0A /* Token.swift */, + ); + path = Models; + sourceTree = ""; + }; + 56479504C3EB58BC9E0C860A = { + isa = PBXGroup; + children = ( + DC6E0D845E48183E2C732ECF /* Project */, + 18E783E8B2360F08F2FCDEB9 /* Frameworks */, + 7C21384B46F40CB3F23DD515 /* Products */, + ); + sourceTree = ""; + }; + 6FAB736FE07EDFA07F922A75 /* API */ = { + isa = PBXGroup; + children = ( + 3E8665C11A450F2FB7D2F2E7 /* Models */, + 93EAD8979A20E239E16257E4 /* PaymentIntent+API.swift */, + 264583DBFFA42C864220D7FF /* PaymentMethod+API.swift */, + 30DDE851AD7450BE70381337 /* SetupIntent+API.swift */, + D81AB96E6201D534220ABB9D /* Token+API.swift */, + ); + path = API; + sourceTree = ""; + }; + 7261423B1B83AF12EE2FEA40 /* PaymentsCore */ = { + isa = PBXGroup; + children = ( + 0B7AA620790EC4257132D738 /* STPTelemetryClientFunctionalTest.swift */, + E4B4DF82BA51CC68265B0795 /* STPTelemetryClientTest.swift */, + ); + path = PaymentsCore; + sourceTree = ""; + }; + 7C21384B46F40CB3F23DD515 /* Products */ = { + isa = PBXGroup; + children = ( + 29BFCC7B0FCEA743A857B51F /* StripeApplePay.framework */, + 229446797CC5FB07DB344D81 /* StripeApplePayTests.xctest */, + 62ECC1AA5583E4104C073B63 /* StripeCore.framework */, + 0FA663325484F2D515613494 /* StripeCoreTestUtils.framework */, + ); + name = Products; + sourceTree = ""; + }; + 823E8B8D6F322BF2BCC4C030 /* PaymentsCore */ = { + isa = PBXGroup; + children = ( + 0D2F315747B25043D0135CBC /* Analytics */, + 6FAB736FE07EDFA07F922A75 /* API */, + 9FE0B80D161B8DB9C70902BF /* Categories */, + ); + path = PaymentsCore; + sourceTree = ""; + }; + 9FE0B80D161B8DB9C70902BF /* Categories */ = { + isa = PBXGroup; + children = ( + 894B7D67F2364A12DD133CF4 /* STPAPIClient+PaymentsCore.swift */, + ); + path = Categories; + sourceTree = ""; + }; + A4676BE706A785EA50A1FEBE /* StripeApplePayTests */ = { + isa = PBXGroup; + children = ( + 7261423B1B83AF12EE2FEA40 /* PaymentsCore */, + 79B089D69DBA1A0BCF4503B6 /* Info.plist */, + E6E9055B54A5634B636570E3 /* STPAnalyticsClient+ApplePayTest.swift */, + 905182496A07DE4F056A3EAA /* STPPaymentMethodFunctionalTest.swift */, + 446F888DB3EC0FC1FE9B9377 /* TelemetryInjectionTest.swift */, + ); + path = StripeApplePayTests; + sourceTree = ""; + }; + B7FC50BAFE8B7A10FA7EA283 /* Source */ = { + isa = PBXGroup; + children = ( + 2F76FD687FF6507E5B2C1A3F /* ApplePayContext */, + E1B6C6692BA02E5DD1AED20B /* Extensions */, + 823E8B8D6F322BF2BCC4C030 /* PaymentsCore */, + 5439AD0D1236F6A21EE93CB3 /* Blocks.swift */, + 88664A990EE5D99076E88987 /* StripeCore+Import.swift */, + ); + path = Source; + sourceTree = ""; + }; + DC6E0D845E48183E2C732ECF /* Project */ = { + isa = PBXGroup; + children = ( + 3581723D3A3FF58EE3E59033 /* BuildConfigurations */, + 1A99D698FD7EE445DA2366ED /* StripeApplePay */, + A4676BE706A785EA50A1FEBE /* StripeApplePayTests */, + ); + name = Project; + sourceTree = ""; + }; + E1B6C6692BA02E5DD1AED20B /* Extensions */ = { + isa = PBXGroup; + children = ( + F4FD028203455E50A637227C /* BillingDetails+ApplePay.swift */, + 133A7483EDB9F1B1CFFDB9ED /* PKContact+Stripe.swift */, + 906F7217DBF694BF42F23458 /* PKPayment+Stripe.swift */, + ); + path = Extensions; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXHeadersBuildPhase section */ + 82FBB6E9E55342EADEEE1865 /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 1990346BA0B39ADD47210E18 /* StripeApplePay.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXHeadersBuildPhase section */ + +/* Begin PBXNativeTarget section */ + 747C9FE1743C0B4444303569 /* StripeApplePay */ = { + isa = PBXNativeTarget; + buildConfigurationList = 49AD325F0339BB78EE36ABC4 /* Build configuration list for PBXNativeTarget "StripeApplePay" */; + buildPhases = ( + 82FBB6E9E55342EADEEE1865 /* Headers */, + 0FA9CF39340C39235D831C53 /* Sources */, + 52997B8984C3A02398E230F4 /* Resources */, + FF3CB7A978895136380088BB /* Embed Frameworks */, + 2A690CB3AEAD3A05D0DF3D47 /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = StripeApplePay; + productName = StripeApplePay; + productReference = 29BFCC7B0FCEA743A857B51F /* StripeApplePay.framework */; + productType = "com.apple.product-type.framework"; + }; + ACAFA21CF224F80EFAEFDC2F /* StripeApplePayTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 48EC33D43FBA14658B0E9567 /* Build configuration list for PBXNativeTarget "StripeApplePayTests" */; + buildPhases = ( + C5D7D85B45B4D8BECDF5B7CD /* Sources */, + 200F36028817645AC1730398 /* Resources */, + F937383B644A63BD5FC4A081 /* Embed Frameworks */, + 92797DEFFA4201115E38DEAD /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + 9D2CBE0177D04009899ECFED /* PBXTargetDependency */, + ); + name = StripeApplePayTests; + packageProductDependencies = ( + CB64080D8A41D33BC3DEEAF8 /* OHHTTPStubs */, + 8ED017D12DE81D3ABD013768 /* OHHTTPStubsSwift */, + ); + productName = StripeApplePayTests; + productReference = 229446797CC5FB07DB344D81 /* StripeApplePayTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 95898E5CA0A1871DDFFB66B0 /* Project object */ = { + isa = PBXProject; + attributes = { + TargetAttributes = { + }; + }; + buildConfigurationList = 2182982C71AEF088A6D2AA6A /* Build configuration list for PBXProject "StripeApplePay" */; + compatibilityVersion = "Xcode 13.0"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + Base, + en, + ); + mainGroup = 56479504C3EB58BC9E0C860A; + packageReferences = ( + B45F7C9F63270FAF880F5EEF /* XCRemoteSwiftPackageReference "OHHTTPStubs" */, + ); + productRefGroup = 7C21384B46F40CB3F23DD515 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 747C9FE1743C0B4444303569 /* StripeApplePay */, + ACAFA21CF224F80EFAEFDC2F /* StripeApplePayTests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 200F36028817645AC1730398 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 52997B8984C3A02398E230F4 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 0FA9CF39340C39235D831C53 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + C6A333FBB72EE91849DD6202 /* STPAPIClient+ApplePay.swift in Sources */, + 9F50D23599CD0B1270F8C295 /* STPApplePayContext+LegacySupport.swift in Sources */, + C6552A9ADEA160B9BBAF3A10 /* STPApplePayContext.swift in Sources */, + CECAB0CB1CE4D7BD1EF897F0 /* Blocks.swift in Sources */, + 59C1DB9EF052987BD20B65A3 /* BillingDetails+ApplePay.swift in Sources */, + F42BC784EDBD141C90E74A5F /* PKContact+Stripe.swift in Sources */, + 2D6CD6872A00B6FE0243C3F5 /* PKPayment+Stripe.swift in Sources */, + 4CE57B5BF79D1515F27A18A3 /* Address.swift in Sources */, + 7C7C92AFED77FC4C5D26DC36 /* BillingDetails.swift in Sources */, + 4ADC5356764DC5E3F1C1D51B /* CardBrand.swift in Sources */, + 22E1F50D066A294A316052ED /* PaymentIntent.swift in Sources */, + 463680AADB8CED6E962CD45A /* PaymentIntentParams.swift in Sources */, + E1E7B153B169D0A6363ADD4B /* PaymentMethod.swift in Sources */, + FADFD3B7E3832928E254FAF1 /* PaymentMethodParams.swift in Sources */, + 43682CEAB00A93868FA3188A /* SetupIntent.swift in Sources */, + 234FA38DC46927B23871A75D /* SetupIntentParams.swift in Sources */, + 848843F1145350ABF540D69F /* ShippingDetails.swift in Sources */, + 505A82AADE15E2B2B99C5D32 /* Token.swift in Sources */, + 4AA6A66246DD30798E5CC5F7 /* PaymentIntent+API.swift in Sources */, + 53BA33D02E07DFF1393DF0E4 /* PaymentMethod+API.swift in Sources */, + 09EB0F7E346CB22144515E67 /* SetupIntent+API.swift in Sources */, + A69598FE8095AFA9898297A9 /* Token+API.swift in Sources */, + 59AEBEB856DB8A0118F1D0DE /* STPAnalyticsClient+Payments.swift in Sources */, + BCC7454DB40673493DF940F5 /* STPAnalyticsClient+PaymentsAPI.swift in Sources */, + AB48B0CBFFEFC46E01C76B6C /* STPAPIClient+PaymentsCore.swift in Sources */, + F59D5A3F2C5849CD465062A5 /* StripeCore+Import.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + C5D7D85B45B4D8BECDF5B7CD /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + CB78059C8EB6A072C30A98DA /* STPTelemetryClientFunctionalTest.swift in Sources */, + 62AFEE5A1E32BD84588CA233 /* STPTelemetryClientTest.swift in Sources */, + A79510332C88568637C9E867 /* STPAnalyticsClient+ApplePayTest.swift in Sources */, + 2EBB230815383A8402D71146 /* STPPaymentMethodFunctionalTest.swift in Sources */, + 90286A48FA6C350BDEC227D0 /* TelemetryInjectionTest.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 9D2CBE0177D04009899ECFED /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeApplePay; + target = 747C9FE1743C0B4444303569 /* StripeApplePay */; + targetProxy = 6DAB385DAFF8DCB87110CC7E /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin XCBuildConfiguration section */ + 13B57D9AA89321957F3ACF3A /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = BDAEEEB96953ECBB9C68CF1C /* StripeiOS Tests-Release.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeApplePayTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeApplePayTests; + PRODUCT_NAME = StripeApplePayTests; + SDKROOT = iphoneos; + }; + name = Release; + }; + 15219003E9DE09D774FB72BB /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 4EE8157F3536C238AC337337 /* StripeiOS-Release.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeApplePay/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-apple-pay"; + PRODUCT_NAME = StripeApplePay; + SDKROOT = iphoneos; + }; + name = Release; + }; + 6EEB3D72619B3E4C89798C09 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 49B0926F8F65BC0102B99BF4 /* StripeiOS-Debug.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeApplePay/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-apple-pay"; + PRODUCT_NAME = StripeApplePay; + SDKROOT = iphoneos; + }; + name = Debug; + }; + 73AE406DB5197038572F40A7 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = CB01F67DF47C468825F8FF6A /* Project-Release.xcconfig */; + buildSettings = { + }; + name = Release; + }; + 7CA6A9C0F1D4C842935D7CEC /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 40D5ED47331313B6AD8B6F46 /* Project-Debug.xcconfig */; + buildSettings = { + }; + name = Debug; + }; + EB1D00A86E16D71A6153C2A0 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 425B19195E4B86F77229252D /* StripeiOS Tests-Debug.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeApplePayTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeApplePayTests; + PRODUCT_NAME = StripeApplePayTests; + SDKROOT = iphoneos; + }; + name = Debug; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 2182982C71AEF088A6D2AA6A /* Build configuration list for PBXProject "StripeApplePay" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 7CA6A9C0F1D4C842935D7CEC /* Debug */, + 73AE406DB5197038572F40A7 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 48EC33D43FBA14658B0E9567 /* Build configuration list for PBXNativeTarget "StripeApplePayTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + EB1D00A86E16D71A6153C2A0 /* Debug */, + 13B57D9AA89321957F3ACF3A /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 49AD325F0339BB78EE36ABC4 /* Build configuration list for PBXNativeTarget "StripeApplePay" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 6EEB3D72619B3E4C89798C09 /* Debug */, + 15219003E9DE09D774FB72BB /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + +/* Begin XCRemoteSwiftPackageReference section */ + B45F7C9F63270FAF880F5EEF /* XCRemoteSwiftPackageReference "OHHTTPStubs" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/eurias-stripe/OHHTTPStubs"; + requirement = { + branch = master; + kind = branch; + }; + }; +/* End XCRemoteSwiftPackageReference section */ + +/* Begin XCSwiftPackageProductDependency section */ + 8ED017D12DE81D3ABD013768 /* OHHTTPStubsSwift */ = { + isa = XCSwiftPackageProductDependency; + productName = OHHTTPStubsSwift; + }; + CB64080D8A41D33BC3DEEAF8 /* OHHTTPStubs */ = { + isa = XCSwiftPackageProductDependency; + productName = OHHTTPStubs; + }; +/* End XCSwiftPackageProductDependency section */ + }; + rootObject = 95898E5CA0A1871DDFFB66B0 /* Project object */; +} diff --git a/StripeApplePay/StripeApplePay.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/StripeApplePay/StripeApplePay.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/StripeApplePay/StripeApplePay.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/StripeApplePay/StripeApplePay.xcodeproj/xcshareddata/xcschemes/StripeApplePay.xcscheme b/StripeApplePay/StripeApplePay.xcodeproj/xcshareddata/xcschemes/StripeApplePay.xcscheme new file mode 100644 index 00000000..64ccb761 --- /dev/null +++ b/StripeApplePay/StripeApplePay.xcodeproj/xcshareddata/xcschemes/StripeApplePay.xcscheme @@ -0,0 +1,95 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/StripeApplePay/StripeApplePay/Info.plist b/StripeApplePay/StripeApplePay/Info.plist new file mode 100644 index 00000000..cd4a496b --- /dev/null +++ b/StripeApplePay/StripeApplePay/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + $(CURRENT_PROJECT_VERSION) + CFBundleVersion + $(CURRENT_PROJECT_VERSION) + + diff --git a/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPAPIClient+ApplePay.swift b/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPAPIClient+ApplePay.swift new file mode 100644 index 00000000..d00717db --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPAPIClient+ApplePay.swift @@ -0,0 +1,44 @@ +// +// STPAPIClient+ApplePay.swift +// StripeApplePay +// +// Created by Jack Flintermann on 12/19/14. +// Copyright © 2014 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripeCore + +/// STPAPIClient extensions to create Stripe Tokens, Sources, or PaymentMethods from Apple Pay PKPayment objects. +extension STPAPIClient { + /// Converts Stripe errors into the appropriate Apple Pay error, for use in `PKPaymentAuthorizationResult`. + /// If the error can be fixed by the customer within the Apple Pay sheet, we return an NSError that can be displayed in the Apple Pay sheet. + /// Otherwise, the original error is returned, resulting in the Apple Pay sheet being dismissed. You should display the error message to the customer afterwards. + /// Currently, we convert billing address related errors into a PKPaymentError that helpfully points to the billing address field in the Apple Pay sheet. + /// Note that Apple Pay should prevent most card errors (e.g. invalid CVC, expired cards) when you add a card to the wallet. + /// - Parameter stripeError: An error from the Stripe SDK. + @objc(pkPaymentErrorForStripeError:) + public class func pkPaymentError(forStripeError stripeError: Error?) -> Error? { + guard let stripeError = stripeError else { + return nil + } + + if (stripeError as NSError).domain == STPError.stripeDomain + && ((stripeError as NSError).userInfo[STPError.cardErrorCodeKey] as? String + == STPCardErrorCode.incorrectZip.rawValue) + { + var userInfo = (stripeError as NSError).userInfo + var errorCode: PKPaymentError.Code = .unknownError + errorCode = .billingContactInvalidError + userInfo[PKPaymentErrorKey.postalAddressUserInfoKey.rawValue] = + CNPostalAddressPostalCodeKey + return NSError( + domain: STPError.stripeDomain, + code: errorCode.rawValue, + userInfo: userInfo + ) + } + return stripeError + } +} diff --git a/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext+LegacySupport.swift b/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext+LegacySupport.swift new file mode 100644 index 00000000..c71eb499 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext+LegacySupport.swift @@ -0,0 +1,55 @@ +// +// STPApplePayContext+LegacySupport.swift +// StripeApplePay +// +// Created by David Estes on 1/25/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit + +/// Internal Apple Pay class. Do not use. +/// :nodoc: +@objc @_spi(STP) public class _stpinternal_ApplePayContextDidCreatePaymentMethodStorage: NSObject { + @_spi(STP) public weak var delegate: _stpinternal_STPApplePayContextDelegateBase? + @_spi(STP) public var context: STPApplePayContext + @_spi(STP) public var paymentMethod: StripeAPI.PaymentMethod + @_spi(STP) public var paymentInformation: PKPayment + @_spi(STP) public var completion: STPIntentClientSecretCompletionBlock + + @_spi(STP) public init( + delegate: _stpinternal_STPApplePayContextDelegateBase, + context: STPApplePayContext, + paymentMethod: StripeAPI.PaymentMethod, + paymentInformation: PKPayment, + completion: @escaping STPIntentClientSecretCompletionBlock + ) { + self.delegate = delegate + self.context = context + self.paymentMethod = paymentMethod + self.paymentInformation = paymentInformation + self.completion = completion + } +} + +/// Internal Apple Pay class. Do not use. +/// :nodoc: +@objc @_spi(STP) public class _stpinternal_ApplePayContextDidCompleteStorage: NSObject { + @_spi(STP) public weak var delegate: _stpinternal_STPApplePayContextDelegateBase? + @_spi(STP) public var context: STPApplePayContext + @_spi(STP) public var status: STPApplePayContext.PaymentStatus + @_spi(STP) public var error: Error? + + @_spi(STP) public init( + delegate: _stpinternal_STPApplePayContextDelegateBase, + context: STPApplePayContext, + status: STPApplePayContext.PaymentStatus, + error: Error? + ) { + self.delegate = delegate + self.context = context + self.status = status + self.error = error + } +} diff --git a/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext.swift b/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext.swift new file mode 100644 index 00000000..7fdc09de --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/ApplePayContext/STPApplePayContext.swift @@ -0,0 +1,746 @@ +// +// STPApplePayContext.swift +// StripeApplePay +// +// Created by Yuki Tokuhiro on 2/20/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import Foundation +import ObjectiveC +import PassKit +@_spi(STP) import StripeCore + +/// :nodoc: +@objc public protocol _stpinternal_STPApplePayContextDelegateBase: NSObjectProtocol { + /// Called when the user selects a new shipping method. The delegate should determine + /// shipping costs based on the shipping method and either the shipping address supplied in the original + /// PKPaymentRequest or the address fragment provided by the last call to paymentAuthorizationController: + /// didSelectShippingContact:completion:. + /// You must invoke the completion block with an updated array of PKPaymentSummaryItem objects. + @objc(applePayContext:didSelectShippingMethod:handler:) + optional func applePayContext( + _ context: STPApplePayContext, + didSelect shippingMethod: PKShippingMethod, + handler: @escaping (_ update: PKPaymentRequestShippingMethodUpdate) -> Void + ) + + /// Called when the user has selected a new shipping address. You should inspect the + /// address and must invoke the completion block with an updated array of PKPaymentSummaryItem objects. + /// @note To maintain privacy, the shipping information is anonymized. For example, in the United States it only includes the city, state, and zip code. This provides enough information to calculate shipping costs, without revealing sensitive information until the user actually approves the purchase. + /// Receive full shipping information in the paymentInformation passed to `applePayContext:didCreatePaymentMethod:paymentInformation:completion:` + @objc optional func applePayContext( + _ context: STPApplePayContext, + didSelectShippingContact contact: PKContact, + handler: @escaping (_ update: PKPaymentRequestShippingContactUpdate) -> Void + ) + + /// Optionally configure additional information on your PKPaymentAuthorizationResult. + /// This closure will be called after the PaymentIntent or SetupIntent is confirmed, but before + /// the Apple Pay sheet has been closed. + /// In your implementation, you can configure the PKPaymentAuthorizationResult to add custom fields, such as `orderDetails`. + /// See https://developer.apple.com/documentation/passkit/pkpaymentauthorizationresult for all configuration options. + /// This method is optional. If you implement this, you must call the handler block with the PKPaymentAuthorizationResult on the main queue. + /// WARNING: If you do not call the completion handler, your app will hang until the Apple Pay sheet times out. + @objc optional func applePayContext( + _ context: STPApplePayContext, + willCompleteWithResult authorizationResult: PKPaymentAuthorizationResult, + handler: @escaping (_ authorizationResult: PKPaymentAuthorizationResult) -> Void + ) +} + +/// Implement the required methods of this delegate to supply a PaymentIntent to ApplePayContext and be notified of the completion of the Apple Pay payment. +/// You may also implement the optional delegate methods to handle shipping methods and shipping address changes e.g. to verify you can ship to the address, or update the payment amount. +public protocol ApplePayContextDelegate: _stpinternal_STPApplePayContextDelegateBase { + /// Called after the customer has authorized Apple Pay. Implement this method to call the completion block with the client secret of a PaymentIntent or SetupIntent. + /// - Parameters: + /// - paymentMethod: The PaymentMethod that represents the customer's Apple Pay payment method. + /// If you create the PaymentIntent with confirmation_method=manual, pass `paymentMethod.id` as the payment_method and confirm=true. Otherwise, you can ignore this parameter. + /// - paymentInformation: The underlying PKPayment created by Apple Pay. + /// If you create the PaymentIntent with confirmation_method=manual, you can collect shipping information using its `shippingContact` and `shippingMethod` properties. + /// - completion: Call this with the PaymentIntent or SetupIntent client secret, or the error that occurred creating the PaymentIntent or SetupIntent. + func applePayContext( + _ context: STPApplePayContext, + didCreatePaymentMethod paymentMethod: StripeAPI.PaymentMethod, + paymentInformation: PKPayment, + completion: @escaping STPIntentClientSecretCompletionBlock + ) + + /// Called after the Apple Pay sheet is dismissed with the result of the payment. + /// Your implementation could stop a spinner and display a receipt view or error to the customer, for example. + /// - Parameters: + /// - status: The status of the payment + /// - error: The error that occurred, if any. + func applePayContext( + _ context: STPApplePayContext, + didCompleteWith status: STPApplePayContext.PaymentStatus, + error: Error? + ) +} + +/// A helper class that implements Apple Pay. +/// Usage looks like this: +/// 1. Initialize this class with a PKPaymentRequest describing the payment request (amount, line items, required shipping info, etc) +/// 2. Call presentApplePayOnViewController:completion: to present the Apple Pay sheet and begin the payment process +/// 3 (optional): If you need to respond to the user changing their shipping information/shipping method, implement the optional delegate methods +/// 4. When the user taps 'Buy', this class uses the PaymentIntent that you supply in the applePayContext:didCreatePaymentMethod:completion: delegate method to complete the payment +/// 5. After payment completes/errors and the sheet is dismissed, this class informs you in the applePayContext:didCompleteWithStatus: delegate method +/// - seealso: https://stripe.com/docs/apple-pay#native for a full guide +/// - seealso: ApplePayExampleViewController for an example +@objc(STPApplePayContext) +public class STPApplePayContext: NSObject, PKPaymentAuthorizationControllerDelegate { + /// Initializes this class. + /// @note This may return nil if the request is invalid e.g. the user is restricted by parental controls, or can't make payments on any of the request's supported networks + /// @note If using Swift, using ApplePayContextDelegate is recommended over STPApplePayContextDelegate. + /// - Parameters: + /// - paymentRequest: The payment request to use with Apple Pay. + /// - delegate: The delegate. + @objc(initWithPaymentRequest:delegate:) + public required init?( + paymentRequest: PKPaymentRequest, + delegate: _stpinternal_STPApplePayContextDelegateBase? + ) { + STPAnalyticsClient.sharedClient.addClass(toProductUsageIfNecessary: STPApplePayContext.self) + if !StripeAPI.canSubmitPaymentRequest(paymentRequest) { + return nil + } + + authorizationController = PKPaymentAuthorizationController(paymentRequest: paymentRequest) + if authorizationController == nil { + return nil + } + + self.delegate = delegate + + super.init() + authorizationController?.delegate = self + } + + private var presentationWindow: UIWindow? + + /// Presents the Apple Pay sheet from the key window, starting the payment process. + /// @note This method should only be called once; create a new instance of STPApplePayContext every time you present Apple Pay. + /// - Parameters: + /// - completion: Called after the Apple Pay sheet is presented + @available( + iOSApplicationExtension, + unavailable, + message: "Use `presentApplePay(from:completion:)` in App Extensions." + ) + @available( + macCatalystApplicationExtension, + unavailable, + message: "Use `presentApplePay(from:completion:)` in App Extensions." + ) + @objc(presentApplePayWithCompletion:) + public func presentApplePay(completion: STPVoidBlock? = nil) { + let window = UIApplication.shared.windows.first { $0.isKeyWindow } + self.presentApplePay(from: window, completion: completion) + } + + /// Presents the Apple Pay sheet from the specified window, starting the payment process. + /// @note This method should only be called once; create a new instance of STPApplePayContext every time you present Apple Pay. + /// - Parameters: + /// - window: The UIWindow to host the Apple Pay sheet + /// - completion: Called after the Apple Pay sheet is presented + @objc(presentApplePayFromWindow:completion:) + public func presentApplePay(from window: UIWindow?, completion: STPVoidBlock? = nil) { + presentationWindow = window + guard !didPresentApplePay, let applePayController = self.authorizationController else { + assert( + false, + "This method should only be called once; create a new instance of STPApplePayContext every time you present Apple Pay." + ) + return + } + didPresentApplePay = true + + // This instance (and the associated Objective-C bridge object, if any) must live so + // that the apple pay sheet is dismissed; until then, the app is effectively frozen. + objc_setAssociatedObject( + applePayController, + UnsafeRawPointer(&kApplePayContextAssociatedObjectKey), + self, + .OBJC_ASSOCIATION_RETAIN_NONATOMIC + ) + + applePayController.present { (_) in + DispatchQueue.main.async { + completion?() + } + } + } + + /// Presents the Apple Pay sheet from the specified view controller, starting the payment process. + /// @note This method should only be called once; create a new instance of STPApplePayContext every time you present Apple Pay. + /// @deprecated A presenting UIViewController is no longer needed. Use presentApplePay(completion:) instead. + /// - Parameters: + /// - viewController: The UIViewController instance to present the Apple Pay sheet on + /// - completion: Called after the Apple Pay sheet is presented + @objc(presentApplePayOnViewController:completion:) + @available( + *, + deprecated, + message: "Use `presentApplePay(completion:)` instead.", + renamed: "presentApplePay(completion:)" + ) + public func presentApplePay( + on viewController: UIViewController, + completion: STPVoidBlock? = nil + ) { + let window = viewController.viewIfLoaded?.window + presentApplePay(from: window, completion: completion) + } + + /// The API Client to use to make requests. + /// Defaults to `STPAPIClient.shared` + @objc public var apiClient: STPAPIClient = STPAPIClient.shared + /// ApplePayContext passes this to the /confirm endpoint for PaymentIntents if it did not collect shipping details itself. + /// :nodoc: + @_spi(STP) public var shippingDetails: StripeAPI.ShippingDetails? + private weak var delegate: _stpinternal_STPApplePayContextDelegateBase? + @objc var authorizationController: PKPaymentAuthorizationController? + @_spi(STP) public var returnUrl: String? + // Internal state + private var paymentState: PaymentState = .notStarted + private var error: Error? + /// YES if the flow cancelled or timed out. This toggles which delegate method (didFinish or didAuthorize) calls our didComplete delegate method + private var didCancelOrTimeoutWhilePending = false + private var didPresentApplePay = false + + /// :nodoc: + @objc public override func responds(to aSelector: Selector!) -> Bool { + // ApplePayContextDelegate exposes methods that map 1:1 to PKPaymentAuthorizationControllerDelegate methods + // We want this method to return YES for these methods IFF they are implemented by our delegate + + // Why not simply implement the methods to call their equivalents on self.delegate? + // The implementation of e.g. didSelectShippingMethod must call the completion block. + // If the user does not implement e.g. didSelectShippingMethod, we don't know the correct PKPaymentSummaryItems to pass to the completion block + // (it may have changed since we were initialized due to another delegate method) + if let equivalentDelegateSelector = _delegateToAppleDelegateMapping()[aSelector] { + return delegate?.responds(to: equivalentDelegateSelector) ?? false + } else { + return super.responds(to: aSelector) + } + } + + // MARK: - Private Helper + func _delegateToAppleDelegateMapping() -> [Selector: Selector] { + // We need this type to disambiguate from the other PKACDelegate.didSelect:handler: method + // HACK: This signature changed in Xcode 14, we need to check the compiler version to choose the right signature. + #if compiler(>=5.7) + typealias pkDidSelectShippingMethodSignature = + (any PKPaymentAuthorizationControllerDelegate) -> ( + ( + PKPaymentAuthorizationController, + PKShippingMethod, + @escaping (PKPaymentRequestShippingMethodUpdate) -> Void + ) -> Void + )? + #else + typealias pkDidSelectShippingMethodSignature = ( + (PKPaymentAuthorizationControllerDelegate) -> ( + PKPaymentAuthorizationController, PKShippingMethod, + @escaping (PKPaymentRequestShippingMethodUpdate) -> Void + ) -> Void + )? + #endif + + let pk_didSelectShippingMethod = #selector( + (PKPaymentAuthorizationControllerDelegate.paymentAuthorizationController( + _: + didSelectShippingMethod: + handler: + )) as pkDidSelectShippingMethodSignature) + let stp_didSelectShippingMethod = #selector( + _stpinternal_STPApplePayContextDelegateBase.applePayContext(_:didSelect:handler:)) + let pk_didSelectShippingContact = #selector( + PKPaymentAuthorizationControllerDelegate.paymentAuthorizationController( + _: + didSelectShippingContact: + handler: + )) + let stp_didSelectShippingContact = #selector( + _stpinternal_STPApplePayContextDelegateBase.applePayContext( + _: + didSelectShippingContact: + handler: + )) + + return [ + pk_didSelectShippingMethod: stp_didSelectShippingMethod, + pk_didSelectShippingContact: stp_didSelectShippingContact, + ] + } + + func _end() { + if let authorizationController = authorizationController { + objc_setAssociatedObject( + authorizationController, + UnsafeRawPointer(&kApplePayContextAssociatedObjectKey), + nil, + .OBJC_ASSOCIATION_RETAIN_NONATOMIC + ) + } + authorizationController = nil + delegate = nil + } + + func _shippingDetails(from payment: PKPayment) -> StripeAPI.ShippingDetails? { + guard let address = payment.shippingContact?.postalAddress, + let name = payment.shippingContact?.name + else { + // The shipping address street and name are required parameters for a valid .ShippingDetails + // Return `shippingDetails` instead + return shippingDetails + } + + let addressParams = StripeAPI.ShippingDetails.Address( + city: address.city, + country: address.isoCountryCode, + line1: address.street, + postalCode: address.postalCode, + state: address.state + ) + + let formatter = PersonNameComponentsFormatter() + formatter.style = .long + let shippingParams = StripeAPI.ShippingDetails( + address: addressParams, + name: formatter.string(from: name), + phone: payment.shippingContact?.phoneNumber?.stringValue + ) + + return shippingParams + } + + // MARK: - PKPaymentAuthorizationControllerDelegate + /// :nodoc: + @objc(paymentAuthorizationController:didAuthorizePayment:handler:) + public func paymentAuthorizationController( + _ controller: PKPaymentAuthorizationController, + didAuthorizePayment payment: PKPayment, + handler completion: @escaping (PKPaymentAuthorizationResult) -> Void + ) { + // Some observations (on iOS 12 simulator): + // - The docs say localizedDescription can be shown in the Apple Pay sheet, but I haven't seen this. + // - If you call the completion block w/ a status of .failure and an error, the user is prompted to try again. + + _completePayment(with: payment) { status, error in + let errors = [STPAPIClient.pkPaymentError(forStripeError: error)].compactMap({ $0 }) + let result = PKPaymentAuthorizationResult(status: status, errors: errors) + if self.delegate?.responds( + to: #selector( + _stpinternal_STPApplePayContextDelegateBase.applePayContext( + _: + willCompleteWithResult: + handler: + )) + ) + ?? false + { + self.delegate?.applePayContext?( + self, + willCompleteWithResult: result, + handler: { newResult in + completion(newResult) + } + ) + } else { + completion(result) + } + } + } + + /// :nodoc: + @objc + public func paymentAuthorizationController( + _ controller: PKPaymentAuthorizationController, + didSelectShippingMethod shippingMethod: PKShippingMethod, + handler completion: @escaping (PKPaymentRequestShippingMethodUpdate) -> Void + ) { + if delegate?.responds( + to: #selector( + _stpinternal_STPApplePayContextDelegateBase.applePayContext(_:didSelect:handler:)) + ) + ?? false + { + delegate?.applePayContext?(self, didSelect: shippingMethod, handler: completion) + } + } + + /// :nodoc: + @objc + public func paymentAuthorizationController( + _ controller: PKPaymentAuthorizationController, + didSelectShippingContact contact: PKContact, + handler completion: @escaping (PKPaymentRequestShippingContactUpdate) -> Void + ) { + if delegate?.responds( + to: #selector( + _stpinternal_STPApplePayContextDelegateBase.applePayContext( + _: + didSelectShippingContact: + handler: + )) + ) ?? false { + delegate?.applePayContext?(self, didSelectShippingContact: contact, handler: completion) + } + } + + /// :nodoc: + @objc public func paymentAuthorizationControllerDidFinish( + _ controller: PKPaymentAuthorizationController + ) { + // Note: If you don't dismiss the VC, the UI disappears, the VC blocks interaction, and this method gets called again. + // Note: This method is called if the user cancels (taps outside the sheet) or Apple Pay times out (empirically 30 seconds) + switch paymentState { + case .notStarted: + controller.dismiss { + stpDispatchToMainThreadIfNecessary { + self.callDidCompleteDelegate(status: .userCancellation, error: nil) + self._end() + } + } + case .pending: + // We can't cancel a pending payment. If we dismiss the VC now, the customer might interact with the app and miss seeing the result of the payment - risking a double charge, chargeback, etc. + // Instead, we'll dismiss and notify our delegate when the payment finishes. + didCancelOrTimeoutWhilePending = true + case .error: + controller.dismiss { + stpDispatchToMainThreadIfNecessary { + self.callDidCompleteDelegate(status: .error, error: self.error) + self._end() + } + } + case .success: + controller.dismiss { + stpDispatchToMainThreadIfNecessary { + self.callDidCompleteDelegate(status: .success, error: nil) + self._end() + } + } + } + } + + /// :nodoc: + @objc public func presentationWindow( + for controller: PKPaymentAuthorizationController + ) -> UIWindow? { + return presentationWindow + } + + // MARK: - Helpers + func _completePayment( + with payment: PKPayment, + completion: @escaping (PKPaymentAuthorizationStatus, Error?) -> Void + ) { + // Helper to handle annoying logic around "Do I call completion block or dismiss + call delegate?" + let handleFinalState: ((PaymentState, Error?) -> Void) = { state, error in + switch state { + case .error: + self.paymentState = .error + self.error = error + if self.didCancelOrTimeoutWhilePending { + self.authorizationController?.dismiss { + DispatchQueue.main.async { + self.callDidCompleteDelegate(status: .error, error: self.error) + self._end() + } + } + } else { + completion(PKPaymentAuthorizationStatus.failure, error) + } + return + case .success: + self.paymentState = .success + if self.didCancelOrTimeoutWhilePending { + self.authorizationController?.dismiss { + DispatchQueue.main.async { + self.callDidCompleteDelegate(status: .success, error: nil) + self._end() + } + } + } else { + completion(PKPaymentAuthorizationStatus.success, nil) + } + return + case .pending, .notStarted: + assert(false, "Invalid final state") + return + } + } + + // 1. Create PaymentMethod + StripeAPI.PaymentMethod.create(apiClient: apiClient, payment: payment) { result in + guard let paymentMethod = try? result.get(), self.authorizationController != nil else { + if case .failure(let error) = result { + handleFinalState(.error, error) + } else { + handleFinalState(.error, nil) + } + return + } + + let paymentMethodCompletion: STPIntentClientSecretCompletionBlock = { + clientSecret, + intentCreationError in + guard let clientSecret = clientSecret, intentCreationError == nil, + self.authorizationController != nil + else { + handleFinalState(.error, intentCreationError) + return + } + + if StripeAPI.SetupIntentConfirmParams.isClientSecretValid(clientSecret) { + // 3a. Retrieve the SetupIntent and see if we need to confirm it client-side + StripeAPI.SetupIntent.get(apiClient: self.apiClient, clientSecret: clientSecret) + { + result in + guard let setupIntent = try? result.get(), + self.authorizationController != nil + else { + if case .failure(let error) = result { + handleFinalState(.error, error) + } else { + handleFinalState(.error, nil) + } + return + } + + switch setupIntent.status { + case .requiresConfirmation, .requiresAction, .requiresPaymentMethod: + // 4a. Confirm the SetupIntent + self.paymentState = .pending // After this point, we can't cancel + var confirmParams = StripeAPI.SetupIntentConfirmParams( + clientSecret: clientSecret + ) + confirmParams.paymentMethod = paymentMethod.id + confirmParams.useStripeSdk = true + confirmParams.returnUrl = self.returnUrl + + StripeAPI.SetupIntent.confirm( + apiClient: self.apiClient, + params: confirmParams + ) { + result in + guard let setupIntent = try? result.get(), + self.authorizationController != nil, + setupIntent.status == .succeeded + else { + if case .failure(let error) = result { + handleFinalState(.error, error) + } else { + handleFinalState(.error, nil) + } + return + } + + handleFinalState(.success, nil) + } + case .succeeded: + handleFinalState(.success, nil) + case .canceled, .processing, .unknown, .unparsable, .none: + handleFinalState( + .error, + Self.makeUnknownError( + message: + "The SetupIntent is in an unexpected state: \(setupIntent.status!)" + ) + ) + } + } + } else { + let paymentIntentClientSecret = clientSecret + // 3b. Retrieve the PaymentIntent and see if we need to confirm it client-side + StripeAPI.PaymentIntent.get( + apiClient: self.apiClient, + clientSecret: paymentIntentClientSecret + ) { result in + guard let paymentIntent = try? result.get(), + self.authorizationController != nil + else { + if case .failure(let error) = result { + handleFinalState(.error, error) + } else { + handleFinalState(.error, nil) + } + return + } + + if paymentIntent.confirmationMethod == .automatic + && (paymentIntent.status == .requiresPaymentMethod + || paymentIntent.status == .requiresConfirmation) + { + // 4b. Confirm the PaymentIntent + + var paymentIntentParams = StripeAPI.PaymentIntentParams( + clientSecret: paymentIntentClientSecret + ) + paymentIntentParams.paymentMethod = paymentMethod.id + paymentIntentParams.useStripeSdk = true + // If a merchant attaches shipping to the PI on their server, the /confirm endpoint will error if we update shipping with a “requires secret key” error message. + // To accommodate this, don't attach if our shipping is the same as the PI's shipping + if paymentIntent.shipping != self._shippingDetails(from: payment) { + paymentIntentParams.shipping = self._shippingDetails(from: payment) + } + + self.paymentState = .pending // After this point, we can't cancel + + // We don't use PaymentHandler because we can't handle next actions as-is - we'd need to dismiss the Apple Pay VC. + StripeAPI.PaymentIntent.confirm( + apiClient: self.apiClient, + params: paymentIntentParams + ) { + result in + guard let postConfirmPI = try? result.get(), + postConfirmPI.status == .succeeded + || postConfirmPI.status == .requiresCapture + else { + if case .failure(let error) = result { + handleFinalState(.error, error) + } else { + handleFinalState(.error, nil) + } + return + } + handleFinalState(.success, nil) + } + } else if paymentIntent.status == .succeeded + || paymentIntent.status == .requiresCapture + { + handleFinalState(.success, nil) + } else { + let unknownError = Self.makeUnknownError( + message: + "The PaymentIntent is in an unexpected state. If you pass confirmation_method = manual when creating the PaymentIntent, also pass confirm = true. If server-side confirmation fails, double check you are passing the error back to the client." + ) + handleFinalState(.error, unknownError) + } + } + } + } + // 2. Fetch PaymentIntent/SetupIntent client secret from delegate + let legacyDelegateSelector = NSSelectorFromString( + "applePayContext:didCreatePaymentMethod:paymentInformation:completion:" + ) + if let delegate = self.delegate { + if let delegate = delegate as? ApplePayContextDelegate { + delegate.applePayContext( + self, + didCreatePaymentMethod: paymentMethod, + paymentInformation: payment, + completion: paymentMethodCompletion + ) + } else if delegate.responds(to: legacyDelegateSelector), + let helperClass = NSClassFromString("STPApplePayContextLegacyHelper") + { + let legacyStorage = _stpinternal_ApplePayContextDidCreatePaymentMethodStorage( + delegate: delegate, + context: self, + paymentMethod: paymentMethod, + paymentInformation: payment, + completion: paymentMethodCompletion + ) + helperClass.performDidCreatePaymentMethod(legacyStorage) + } else { + assertionFailure( + "An STPApplePayContext's delegate must conform to ApplePayContextDelegate or STPApplePayContextDelegate." + ) + } + } + } + } + + func callDidCompleteDelegate(status: PaymentStatus, error: Error?) { + if let delegate = self.delegate { + if let delegate = delegate as? ApplePayContextDelegate { + delegate.applePayContext(self, didCompleteWith: status, error: error) + } else if delegate.responds( + to: NSSelectorFromString("applePayContext:didCompleteWithStatus:error:") + ) { + if let helperClass = NSClassFromString("STPApplePayContextLegacyHelper") { + let legacyStorage = _stpinternal_ApplePayContextDidCompleteStorage( + delegate: delegate, + context: self, + status: status, + error: error + ) + helperClass.performDidComplete(legacyStorage) + } + } else { + assertionFailure( + "An STPApplePayContext's delegate must conform to ApplePayContextDelegate or STPApplePayContextDelegate." + ) + } + } + + } + + static func makeUnknownError(message: String) -> NSError { + let userInfo = [ + NSLocalizedDescriptionKey: NSError.stp_unexpectedErrorMessage(), + STPError.errorMessageKey: message, + ] + return NSError( + domain: STPError.STPPaymentHandlerErrorDomain, + code: STPPaymentHandlerErrorCodeIntentStatusErrorCode, + userInfo: userInfo + ) + } + + /// This is STPPaymentHandlerErrorCode.intentStatusErrorCode.rawValue, which we don't want to vend from this framework. + fileprivate static let STPPaymentHandlerErrorCodeIntentStatusErrorCode = 3 + + enum PaymentState { + case notStarted + case pending + case error + case success + } + + /// An enum representing the status of a payment requested from the user. + @frozen public enum PaymentStatus { + /// The payment succeeded. + case success + /// The payment failed due to an unforeseen error, such as the user's Internet connection being offline. + case error + /// The user cancelled the payment (for example, by hitting "cancel" in the Apple Pay dialog). + case userCancellation + } +} + +/// :nodoc: +@_spi(STP) extension STPApplePayContext: STPAnalyticsProtocol { + @_spi(STP) public static var stp_analyticsIdentifier: String { + return "STPApplePayContext" + } +} + +/// :nodoc: +class ModernApplePayContext: STPAnalyticsProtocol { + @_spi(STP) public static var stp_analyticsIdentifier: String { + return "ModernApplePayContext" + } +} + +private var kSTPApplePayContextAssociatedObjectKey = 0 +enum STPPaymentState: Int { + case notStarted + case pending + case error + case success +} + +private class _stpinternal_STPApplePayContextLegacyHelper: NSObject { + @objc class func performDidCreatePaymentMethod( + _ storage: _stpinternal_ApplePayContextDidCreatePaymentMethodStorage + ) { + // Placeholder to allow this to be called on AnyObject + } + @objc class func performDidComplete(_ storage: _stpinternal_ApplePayContextDidCompleteStorage) { + // Placeholder to allow this to be called on AnyObject + } +} + +private var kApplePayContextAssociatedObjectKey = 0 diff --git a/StripeApplePay/StripeApplePay/Source/Blocks.swift b/StripeApplePay/StripeApplePay/Source/Blocks.swift new file mode 100644 index 00000000..e434acbc --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/Blocks.swift @@ -0,0 +1,18 @@ +// +// Blocks.swift +// StripeApplePay +// +// Created by David Estes on 1/6/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation + +/// An empty block, called with no arguments, returning nothing. +public typealias STPVoidBlock = () -> Void + +/// A block to be run with the client secret of a PaymentIntent or SetupIntent. +/// - Parameters: +/// - clientSecret: The client secret of the PaymentIntent or SetupIntent. See https://stripe.com/docs/api/payment_intents/object#payment_intent_object-client_secret +/// - error: The error that occurred when creating the Intent, or nil if none occurred. +public typealias STPIntentClientSecretCompletionBlock = (String?, Error?) -> Void diff --git a/StripeApplePay/StripeApplePay/Source/Extensions/BillingDetails+ApplePay.swift b/StripeApplePay/StripeApplePay/Source/Extensions/BillingDetails+ApplePay.swift new file mode 100644 index 00000000..8e1ed5c2 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/Extensions/BillingDetails+ApplePay.swift @@ -0,0 +1,101 @@ +// +// BillingDetails+ApplePay.swift +// StripeApplePay +// +// Created by David Estes on 8/9/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripeCore + +extension StripeContact { + /// Initializes a new Contact with data from an PassKit contact. + /// - Parameter contact: The PassKit contact you want to populate the Contact from. + /// - Returns: A new Contact with data copied from the passed in contact. + init( + pkContact contact: PKContact + ) { + let nameComponents = contact.name + if let nameComponents = nameComponents { + givenName = stringIfHasContentsElseNil(nameComponents.givenName) + familyName = stringIfHasContentsElseNil(nameComponents.familyName) + + name = stringIfHasContentsElseNil( + PersonNameComponentsFormatter.localizedString(from: nameComponents, style: .default) + ) + } + email = stringIfHasContentsElseNil(contact.emailAddress) + if let phoneNumber = contact.phoneNumber { + phone = sanitizedPhoneString(from: phoneNumber) + } else { + phone = nil + } + setAddressFromCNPostal(contact.postalAddress) + } + + private func sanitizedPhoneString(from phoneNumber: CNPhoneNumber) -> String? { + return stringIfHasContentsElseNil( + STPNumericStringValidator.sanitizedNumericString(for: phoneNumber.stringValue) + ) + } + + private mutating func setAddressFromCNPostal(_ address: CNPostalAddress?) { + line1 = stringIfHasContentsElseNil(address?.street) + city = stringIfHasContentsElseNil(address?.city) + state = stringIfHasContentsElseNil(address?.state) + postalCode = stringIfHasContentsElseNil(address?.postalCode) + country = stringIfHasContentsElseNil(address?.isoCountryCode.uppercased()) + } +} + +extension StripeAPI.BillingDetails { + init?( + from payment: PKPayment + ) { + var billingDetails: StripeAPI.BillingDetails? + if payment.billingContact != nil { + billingDetails = StripeAPI.BillingDetails() + if let billingContact = payment.billingContact { + let billingAddress = StripeContact(pkContact: billingContact) + billingDetails?.name = billingAddress.name + billingDetails?.email = billingAddress.email + billingDetails?.phone = billingAddress.phone + if billingContact.postalAddress != nil { + billingDetails?.address = StripeAPI.BillingDetails.Address( + contact: billingAddress + ) + } + } + } + + // The phone number and email in the "Contact" panel in the Apple Pay dialog go into the shippingContact, + // not the billingContact. To work around this, we should fill the billingDetails' email and phone + // number from the shippingDetails. + if payment.shippingContact != nil { + var shippingAddress: StripeContact? + if let shippingContact = payment.shippingContact { + shippingAddress = StripeContact(pkContact: shippingContact) + } + if billingDetails?.email == nil && shippingAddress?.email != nil { + if billingDetails == nil { + billingDetails = StripeAPI.BillingDetails() + } + billingDetails?.email = shippingAddress?.email + } + if billingDetails?.phone == nil && shippingAddress?.phone != nil { + if billingDetails == nil { + billingDetails = StripeAPI.BillingDetails() + } + billingDetails?.phone = shippingAddress?.phone + } + } + + if let billingDetails = billingDetails { + self = billingDetails + } else { + return nil + } + } +} diff --git a/StripeApplePay/StripeApplePay/Source/Extensions/PKContact+Stripe.swift b/StripeApplePay/StripeApplePay/Source/Extensions/PKContact+Stripe.swift new file mode 100644 index 00000000..03db9af0 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/Extensions/PKContact+Stripe.swift @@ -0,0 +1,26 @@ +// +// PKContact+Stripe.swift +// StripeApplePay +// +// Created by David Estes on 11/16/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit + +extension PKContact { + @_spi(STP) public var addressParams: [AnyHashable: Any] { + var params: [AnyHashable: Any] = [:] + let stpAddress = StripeContact(pkContact: self) + + params["name"] = stpAddress.name + params["address_line1"] = stpAddress.line1 + params["address_city"] = stpAddress.city + params["address_state"] = stpAddress.state + params["address_zip"] = stpAddress.postalCode + params["address_country"] = stpAddress.country + + return params + } +} diff --git a/StripeApplePay/StripeApplePay/Source/Extensions/PKPayment+Stripe.swift b/StripeApplePay/StripeApplePay/Source/Extensions/PKPayment+Stripe.swift new file mode 100644 index 00000000..2d85c930 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/Extensions/PKPayment+Stripe.swift @@ -0,0 +1,32 @@ +// +// PKPayment+Stripe.swift +// StripeApplePay +// +// Created by Ben Guo on 7/2/15. +// Copyright © 2015 Stripe, Inc. All rights reserved. +// + +import PassKit + +extension PKPayment { + /// Returns true if the instance is a payment from the simulator. + @_spi(STP) public func stp_isSimulated() -> Bool { + return token.transactionIdentifier == "Simulated Identifier" + } + + /// Returns a fake transaction identifier with the expected ~-separated format. + @_spi(STP) public class func stp_testTransactionIdentifier() -> String { + var uuid = UUID().uuidString + uuid = uuid.replacingOccurrences(of: "~", with: "") + + // Simulated cards don't have enough info yet. For now, use a fake Visa number + let number = "4242424242424242" + + // Without the original PKPaymentRequest, we'll need to use fake data here. + let amount = NSDecimalNumber(string: "0") + let cents = NSNumber(value: amount.multiplying(byPowerOf10: 2).intValue).stringValue + let currency = "USD" + let identifier = ["ApplePayStubs", number, cents, currency, uuid].joined(separator: "~") + return identifier + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Address.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Address.swift new file mode 100644 index 00000000..dc80d6dd --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Address.swift @@ -0,0 +1,43 @@ +// +// Address.swift +// StripeApplePay +// +// Created by David Estes on 8/9/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +/// An internal struct for handling contacts. This is not encodable/decodable for use with the Stripe API. +struct StripeContact { + /// The user's full name (e.g. "Jane Doe") + public var name: String? + + /// The first line of the user's street address (e.g. "123 Fake St") + public var line1: String? + + /// The apartment, floor number, etc of the user's street address (e.g. "Apartment 1A") + public var line2: String? + + /// The city in which the user resides (e.g. "San Francisco") + public var city: String? + + /// The state in which the user resides (e.g. "CA") + public var state: String? + + /// The postal code in which the user resides (e.g. "90210") + public var postalCode: String? + + /// The ISO country code of the address (e.g. "US") + public var country: String? + + /// The phone number of the address (e.g. "8885551212") + public var phone: String? + + /// The email of the address (e.g. "jane@doe.com") + public var email: String? + + internal var givenName: String? + internal var familyName: String? +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/BillingDetails.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/BillingDetails.swift new file mode 100644 index 00000000..1871ec79 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/BillingDetails.swift @@ -0,0 +1,67 @@ +// +// BillingDetails.swift +// StripeApplePay +// +// Created by David Estes on 7/15/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + /// Billing information associated with a `STPPaymentMethod` that may be used or required by particular types of payment methods. + /// - seealso: https://stripe.com/docs/api/payment_methods/object#payment_method_object-billing_details + public struct BillingDetails: UnknownFieldsCodable { + /// Billing address. + public var address: Address? + + /// The billing address, a property sent in a PaymentMethod response + public struct Address: UnknownFieldsCodable { + /// The first line of the user's street address (e.g. "123 Fake St") + public var line1: String? + + /// The apartment, floor number, etc of the user's street address (e.g. "Apartment 1A") + public var line2: String? + + /// The city in which the user resides (e.g. "San Francisco") + public var city: String? + + /// The state in which the user resides (e.g. "CA") + public var state: String? + + /// The postal code in which the user resides (e.g. "90210") + public var postalCode: String? + + /// The ISO country code of the address (e.g. "US") + public var country: String? + + public var _additionalParametersStorage: NonEncodableParameters? + public var _allResponseFieldsStorage: NonEncodableParameters? + } + + /// Email address. + public var email: String? + /// Full name. + public var name: String? + /// Billing phone number (including extension). + public var phone: String? + + public var _additionalParametersStorage: NonEncodableParameters? + public var _allResponseFieldsStorage: NonEncodableParameters? + } + +} + +extension StripeAPI.BillingDetails.Address { + init( + contact: StripeContact + ) { + self.city = contact.city + self.country = contact.country + self.line1 = contact.line1 + self.line2 = contact.line2 + self.postalCode = contact.postalCode + self.state = contact.state + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/CardBrand.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/CardBrand.swift new file mode 100644 index 00000000..0ca26890 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/CardBrand.swift @@ -0,0 +1,33 @@ +// +// CardBrand.swift +// StripeApplePay +// +// Created by David Estes on 4/14/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation + +extension StripeAPI { + /// The various card brands to which a payment card can belong. + enum CardBrand: String, SafeEnumCodable { + /// Visa card + case visa = "Visa" + /// American Express card + case amex = "American Express" + /// Mastercard card + case mastercard = "MasterCard" + /// Discover card + case discover = "Discover" + /// JCB card + case JCB = "JCB" + /// Diners Club card + case dinersClub = "Diners Club" + /// UnionPay card + case unionPay = "UnionPay" + /// An unknown card brand type + case unknown = "Unknown" + /// An unparsable card brand + case unparsable + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntent.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntent.swift new file mode 100644 index 00000000..24cbfcb2 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntent.swift @@ -0,0 +1,149 @@ +// +// PaymentIntent.swift +// StripeApplePay +// +// Created by David Estes on 6/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + @_spi(STP) public struct PaymentIntent: UnknownFieldsDecodable { + // TODO: (MOBILESDK-468) Add modern bindings for more PaymentIntent fields + /// The Stripe ID of the PaymentIntent. + @_spi(STP) public let id: String + + /// The client secret used to fetch this PaymentIntent + @_spi(STP) public let clientSecret: String + + /// Amount intended to be collected by this PaymentIntent. + @_spi(STP) public let amount: Int + + /// If status is `.canceled`, when the PaymentIntent was canceled. + @_spi(STP) public let canceledAt: Date? + + /// Capture method of this PaymentIntent + @_spi(STP) public let captureMethod: CaptureMethod + + /// Confirmation method of this PaymentIntent + @_spi(STP) public let confirmationMethod: ConfirmationMethod + + /// When the PaymentIntent was created. + @_spi(STP) public let created: Date + + /// The currency associated with the PaymentIntent. + @_spi(STP) public let currency: String + + /// The `description` field of the PaymentIntent. + /// An arbitrary string attached to the object. Often useful for displaying to users. + @_spi(STP) public let stripeDescription: String? + + /// Whether or not this PaymentIntent was created in livemode. + @_spi(STP) public let livemode: Bool + + /// Email address that the receipt for the resulting payment will be sent to. + @_spi(STP) public let receiptEmail: String? + + /// The Stripe ID of the Source used in this PaymentIntent. + @_spi(STP) public let sourceId: String? + + /// The Stripe ID of the PaymentMethod used in this PaymentIntent. + @_spi(STP) public let paymentMethodId: String? + + /// Status of the PaymentIntent + @_spi(STP) public let status: Status + + /// Shipping information for this PaymentIntent. + @_spi(STP) public let shipping: ShippingDetails? + + /// Status types for a PaymentIntent + @frozen @_spi(STP) public enum Status: String, SafeEnumCodable { + /// Unknown status + case unknown + /// This PaymentIntent requires a PaymentMethod or Source + case requiresPaymentMethod = "requires_payment_method" + /// This PaymentIntent requires a Source + /// Deprecated: Use STPPaymentIntentStatusRequiresPaymentMethod instead. + @available( + *, + deprecated, + message: "Use STPPaymentIntentStatus.requiresPaymentMethod instead", + renamed: "STPPaymentIntentStatus.requiresPaymentMethod" + ) + case requiresSource = "requires_source" + /// This PaymentIntent needs to be confirmed + case requiresConfirmation = "requires_confirmation" + /// The selected PaymentMethod or Source requires additional authentication steps. + /// Additional actions found via `next_action` + case requiresAction = "requires_action" + /// The selected Source requires additional authentication steps. + /// Additional actions found via `next_source_action` + /// Deprecated: Use STPPaymentIntentStatusRequiresAction instead. + @available( + *, + deprecated, + message: "Use STPPaymentIntentStatus.requiresAction instead", + renamed: "STPPaymentIntentStatus.requiresAction" + ) + case requiresSourceAction = "requires_source_action" + /// Stripe is processing this PaymentIntent + case processing + /// The payment has succeeded + case succeeded + /// Indicates the payment must be captured, for STPPaymentIntentCaptureMethodManual + case requiresCapture = "requires_capture" + /// This PaymentIntent was canceled and cannot be changed. + case canceled + + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + @frozen @_spi(STP) public enum ConfirmationMethod: String, SafeEnumCodable { + /// Unknown confirmation method + case unknown + /// Confirmed via publishable key + case manual + /// Confirmed via secret key + case automatic + + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + @frozen @_spi(STP) public enum CaptureMethod: String, SafeEnumCodable { + /// Unknown capture method + case unknown + /// The PaymentIntent will be automatically captured + case automatic + /// The PaymentIntent must be manually captured once it has the status + /// `.requiresCapture` + case manual + + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + @_spi(STP) public var _allResponseFieldsStorage: NonEncodableParameters? + } +} + +extension StripeAPI.PaymentIntent { + /// Helper function for extracting PaymentIntent id from the Client Secret. + /// This avoids having to pass around both the id and the secret. + /// - Parameter clientSecret: The `client_secret` from the PaymentIntent + internal static func id(fromClientSecret clientSecret: String) -> String? { + // see parseClientSecret from stripe-js-v3 + let components = clientSecret.components(separatedBy: "_secret_") + if components.count >= 2 && components[0].hasPrefix("pi_") { + return components[0] + } else { + return nil + } + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntentParams.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntentParams.swift new file mode 100644 index 00000000..5dfe11a8 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentIntentParams.swift @@ -0,0 +1,101 @@ +// +// PaymentIntentParams.swift +// StripeApplePay +// +// Created by David Estes on 6/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + @_spi(STP) public struct PaymentIntentParams: UnknownFieldsEncodable { + /// The client secret of the PaymentIntent. Required + @_spi(STP) public let clientSecret: String + + @_spi(STP) public init( + clientSecret: String + ) { + self.clientSecret = clientSecret + } + + @_spi(STP) public var id: String? { + return PaymentIntent.id(fromClientSecret: clientSecret) + } + + /// Provide a supported `PaymentMethodParams` object, and Stripe will create a + /// PaymentMethod during PaymentIntent confirmation. + /// @note alternative to `paymentMethodId` + @_spi(STP) public var paymentMethodData: PaymentMethodParams? + + /// Provide an already created PaymentMethod's id, and it will be used to confirm the PaymentIntent. + /// @note alternative to `paymentMethodParams` + @_spi(STP) public var paymentMethod: String? + + /// Provide an already created Source's id, and it will be used to confirm the PaymentIntent. + @_spi(STP) public var sourceId: String? + + /// Email address that the receipt for the resulting payment will be sent to. + @_spi(STP) public var receiptEmail: String? + + /// `@YES` to save this PaymentIntent’s PaymentMethod or Source to the associated Customer, + /// if the PaymentMethod/Source is not already attached. + /// This should be a boolean NSNumber, so that it can be `nil` + @_spi(STP) public var savePaymentMethod: Bool? + + /// The URL to redirect your customer back to after they authenticate or cancel + /// their payment on the payment method’s app or site. + /// This should probably be a URL that opens your iOS app. + @_spi(STP) public var returnURL: String? + + /// When provided, this property indicates how you intend to use the payment method that your customer provides after the current payment completes. + /// If applicable, additional authentication may be performed to comply with regional legislation or network rules required to enable the usage of the same payment method for additional payments. + @_spi(STP) public var setupFutureUsage: SetupFutureUsage? + + /// A boolean number to indicate whether you intend to use the Stripe SDK's functionality to handle any PaymentIntent next actions. + /// If set to false, STPPaymentIntent.nextAction will only ever contain a redirect url that can be opened in a webview or mobile browser. + /// When set to true, the nextAction may contain information that the Stripe SDK can use to perform native authentication within your + /// app. + @_spi(STP) public var useStripeSdk: Bool? + + /// Shipping information. + @_spi(STP) public var shipping: ShippingDetails? + + /// Indicates how you intend to use the payment method that your customer provides after the current payment completes. + /// If applicable, additional authentication may be performed to comply with regional legislation or network rules required to enable the usage of the same payment method for additional payments. + /// - seealso: https://stripe.com/docs/api/payment_intents/object#payment_intent_object-setup_future_usage + @frozen @_spi(STP) public enum SetupFutureUsage: String, SafeEnumCodable { + /// Unknown value. Update your SDK, or use `allResponseFields` for custom handling. + case unknown + /// No value was provided. + case none + /// Indicates you intend to only reuse the payment method when the customer is in your checkout flow. + case onSession + /// Indicates you intend to reuse the payment method when the customer may or may not be in your checkout flow. + case offSession + + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + @_spi(STP) public var _additionalParametersStorage: NonEncodableParameters? + } +} + +extension StripeAPI.PaymentIntentParams { + static internal let isClientSecretValidRegex: NSRegularExpression = try! NSRegularExpression( + pattern: "^pi_[^_]+_secret_[^_]+$", + options: [] + ) + + @_spi(STP) public static func isClientSecretValid(_ clientSecret: String) -> Bool { + return + (isClientSecretValidRegex.numberOfMatches( + in: clientSecret, + options: .anchored, + range: NSRange(location: 0, length: clientSecret.count) + )) == 1 + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethod.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethod.swift new file mode 100644 index 00000000..235e47e2 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethod.swift @@ -0,0 +1,176 @@ +// +// PaymentMethod.swift +// StripeApplePay +// +// Created by David Estes on 6/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + /// PaymentMethod objects represent your customer's payment instruments. They can be used with PaymentIntents to collect payments. + /// - seealso: https://stripe.com/docs/api/payment_methods + public struct PaymentMethod: UnknownFieldsDecodable { + /// The Stripe ID of the PaymentMethod. + public let id: String + + /// Time at which the object was created. Measured in seconds since the Unix epoch. + public var created: Date? + /// `YES` if the object exists in live mode or the value `NO` if the object exists in test mode. + public var livemode = false + + /// The type of the PaymentMethod. The corresponding, similarly named property contains additional information specific to the PaymentMethod type. + /// e.g. if the type is `Card`, the `card` property is also populated. + public var type: PaymentMethodType? + + /// The type of the PaymentMethod. + @frozen public enum PaymentMethodType: String, SafeEnumCodable { + /// A card payment method. + case card + /// An unknown type. + case unknown + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + /// Billing information associated with the PaymentMethod that may be used or required by particular types of payment methods. + public var billingDetails: BillingDetails? + /// The ID of the Customer to which this PaymentMethod is saved. Nil when the PaymentMethod has not been saved to a Customer. + public var customerId: String? + /// If this is a card PaymentMethod (ie `self.type == .card`), this contains additional details. + public var card: Card? + + /// :nodoc: + public struct Card: UnknownFieldsDecodable { + public var _allResponseFieldsStorage: NonEncodableParameters? + /// The issuer of the card. + public private(set) var brand: Brand = .unknown + + /// The various card brands to which a payment card can belong. + @frozen public enum Brand: String, SafeEnumCodable { + /// Visa + case visa + /// American Express + case amex + /// Mastercard + case mastercard + /// Discover + case discover + /// JCB + case jcb + /// Diners Club + case diners + /// UnionPay + case unionpay + /// An unknown card brand + case unknown + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + /// Two-letter ISO code representing the country of the card. + public private(set) var country: String? + /// Two-digit number representing the card’s expiration month. + public private(set) var expMonth: Int + /// Four-digit number representing the card’s expiration year. + public private(set) var expYear: Int + /// Card funding type. Can be credit, debit, prepaid, or unknown. + public private(set) var funding: String? + /// The last four digits of the card. + public private(set) var last4: String? + /// Uniquely identifies this particular card number. You can use this attribute to check whether two customers who’ve signed up with you are using the same card number, for example. + public private(set) var fingerprint: String? + + /// Contains information about card networks that can be used to process the payment. + public private(set) var networks: Networks? + + /// Contains details on how this Card maybe be used for 3D Secure authentication. + public private(set) var threeDSecureUsage: ThreeDSecureUsage? + + /// If this Card is part of a Card Wallet, this contains the details of the Card Wallet. + public private(set) var wallet: Wallet? + + public struct Networks: UnknownFieldsDecodable { + public var _allResponseFieldsStorage: NonEncodableParameters? + + /// All available networks for the card. + public private(set) var available: [String]? + /// The preferred network for the card if one exists. + public private(set) var preferred: String? + } + + /// Contains details on how a `Card` may be used for 3D Secure authentication. + public struct ThreeDSecureUsage: UnknownFieldsDecodable { + public var _allResponseFieldsStorage: NonEncodableParameters? + + /// `true` if 3D Secure is supported on this card. + public private(set) var supported = false + } + + public struct Wallet: UnknownFieldsDecodable { + public var _allResponseFieldsStorage: NonEncodableParameters? + /// The type of the Card Wallet. A matching property is populated if the type is `.masterpass` or `.visaCheckout` containing additional information specific to the Card Wallet type. + public private(set) var type: WalletType = .unknown + /// Contains additional Masterpass information, if the type of the Card Wallet is `STPPaymentMethodCardWalletTypeMasterpass` + public private(set) var masterpass: Masterpass? + /// Contains additional Visa Checkout information, if the type of the Card Wallet is `STPPaymentMethodCardWalletTypeVisaCheckout` + public private(set) var visaCheckout: VisaCheckout? + + /// The type of Card Wallet. + @frozen public enum WalletType: String, SafeEnumCodable { + /// Amex Express Checkout + case amexExpressCheckout = "amex_express_checkout" + /// Apple Pay + case applePay = "apple_pay" + /// Google Pay + case googlePay = "google_pay" + /// Masterpass + case masterpass = "masterpass" + /// Samsung Pay + case samsungPay = "samsung_pay" + /// Visa Checkout + case visaCheckout = "visa_checkout" + /// An unknown Card Wallet type. + case unknown = "unknown" + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + public struct Masterpass: UnknownFieldsDecodable { + public var _allResponseFieldsStorage: NonEncodableParameters? + + /// Owner’s verified email. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var email: String? + /// Owner’s verified email. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var name: String? + /// Owner’s verified billing address. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var billingAddress: BillingDetails.Address? + /// Owner’s verified shipping address. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var shippingAddress: BillingDetails.Address? + } + + /// A Visa Checkout Card Wallet + /// - seealso: https://stripe.com/docs/visa-checkout + public struct VisaCheckout: UnknownFieldsDecodable { + /// Owner’s verified email. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var email: String? + /// Owner’s verified email. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var name: String? + /// Owner’s verified billing address. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var billingAddress: BillingDetails.Address? + /// Owner’s verified shipping address. Values are verified or provided by the payment method directly (and if supported) at the time of authorization or settlement. + public private(set) var shippingAddress: BillingDetails.Address? + + public var _allResponseFieldsStorage: NonEncodableParameters? + } + } + } + + public var _allResponseFieldsStorage: NonEncodableParameters? + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethodParams.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethodParams.swift new file mode 100644 index 00000000..c790a9a7 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/PaymentMethodParams.swift @@ -0,0 +1,73 @@ +// +// PaymentMethodParams.swift +// StripeApplePay +// +// Created by David Estes on 6/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + /// An object representing parameters used to create a PaymentMethod object. + /// - seealso: https://stripe.com/docs/api/payment_methods/create + @_spi(STP) public struct PaymentMethodParams: UnknownFieldsEncodable { + /// The type of payment method. + /// The associated property will contain additional information (e.g. `type == .card` means `card` should also be populated). + @_spi(STP) public var type: PaymentMethod.PaymentMethodType + + /// If this is a card PaymentMethod, this contains the user’s card details. + @_spi(STP) public var card: Card? + + /// Billing information associated with the PaymentMethod that may be used or required by particular types of payment methods. + @_spi(STP) public var billingDetails: BillingDetails? + + /// Used internally to identify the version of the SDK sending the request + @_spi(STP) public var paymentUserAgent: String? = { + return PaymentsSDKVariant.paymentUserAgent + }() + + /// :nodoc: + @_spi(STP) public struct Card: UnknownFieldsEncodable { + /// The card number, as a string without any separators. Ex. "4242424242424242" + @_spi(STP) public var number: String? + /// Number representing the card's expiration month. Ex. 1 + @_spi(STP) public var expMonth: Int? + /// Two- or four-digit number representing the card's expiration year. + @_spi(STP) public var expYear: Int? + /// For backwards compatibility, you can alternatively set this as a Stripe token (e.g., for Apple Pay) + @_spi(STP) public var token: String? + /// Card security code. It is highly recommended to always include this value. + @_spi(STP) public var cvc: String? + + /// The last 4 digits of the card's number, if it's been set, otherwise nil. + @_spi(STP) public var last4: String? { + if number != nil && (number?.count ?? 0) >= 4 { + return (number as NSString?)?.substring(from: (number?.count ?? 0) - 4) + } else { + return nil + } + } + @_spi(STP) public var _additionalParametersStorage: NonEncodableParameters? + } + + @_spi(STP) public var _additionalParametersStorage: NonEncodableParameters? + } +} + +extension StripeAPI.PaymentMethodParams.Card: CustomStringConvertible, CustomDebugStringConvertible, + CustomLeafReflectable +{ + @_spi(STP) public var debugDescription: String { + return description + } + + @_spi(STP) public var description: String { + return "Card \(last4 ?? "")" + } + + @_spi(STP) public var customMirror: Mirror { + return Mirror(reflecting: self.description) + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntent.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntent.swift new file mode 100644 index 00000000..a8369479 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntent.swift @@ -0,0 +1,58 @@ +// +// SetupIntent.swift +// StripeApplePay +// +// Created by David Estes on 6/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + @_spi(STP) public struct SetupIntent: UnknownFieldsDecodable { + @_spi(STP) public let id: String + // TODO: (MOBILESDK-467) Add modern bindings for more SetupIntent fields + @_spi(STP) public let status: SetupIntentStatus? + + /// Status types for an STPSetupIntent + @frozen @_spi(STP) public enum SetupIntentStatus: String, SafeEnumCodable { + /// Unknown status + case unknown + /// This SetupIntent requires a PaymentMethod + case requiresPaymentMethod = "requires_payment_method" + /// This SetupIntent needs to be confirmed + case requiresConfirmation = "requires_confirmation" + /// The selected PaymentMethod requires additional authentication steps. + /// Additional actions found via the `nextAction` property of `STPSetupIntent` + case requiresAction = "requires_action" + /// Stripe is processing this SetupIntent + case processing + /// The SetupIntent has succeeded + case succeeded + /// This SetupIntent was canceled and cannot be changed. + case canceled + + case unparsable + // TODO: This is @frozen because of a bug in the Xcode 12.2 Swift compiler. + // Remove @frozen after Xcode 12.2 support has been dropped. + } + + @_spi(STP) public var _allResponseFieldsStorage: NonEncodableParameters? + } +} + +extension StripeAPI.SetupIntent { + /// Helper function for extracting SetupIntent id from the Client Secret. + /// This avoids having to pass around both the id and the secret. + /// - Parameter clientSecret: The `client_secret` from the SetupIntent + internal static func id(fromClientSecret clientSecret: String) -> String? { + // see parseClientSecret from stripe-js-v3 + let components = clientSecret.components(separatedBy: "_secret_") + if components.count >= 2 && components[0].hasPrefix("seti_") { + return components[0] + } else { + return nil + } + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntentParams.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntentParams.swift new file mode 100644 index 00000000..72d750b1 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/SetupIntentParams.swift @@ -0,0 +1,66 @@ +// +// SetupIntentParams.swift +// StripeApplePay +// +// Created by David Estes on 6/29/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + @_spi(STP) public struct SetupIntentConfirmParams: UnknownFieldsEncodable { + /// Generated by by Editor -> Refactor -> Generate Memberwise Initializer + @_spi(STP) public init( + clientSecret: String, + paymentMethodData: StripeAPI.PaymentMethodParams? = nil, + paymentMethod: String? = nil, + returnUrl: String? = nil, + useStripeSdk: Bool? = nil, + _additionalParametersStorage: NonEncodableParameters? = nil + ) { + self.clientSecret = clientSecret + self.paymentMethodData = paymentMethodData + self.paymentMethod = paymentMethod + self.returnUrl = returnUrl + self.useStripeSdk = useStripeSdk + self._additionalParametersStorage = _additionalParametersStorage + } + + /// The client secret of the SetupIntent. Required. + @_spi(STP) public let clientSecret: String + /// Provide a supported `PaymentMethodParams` object, and Stripe will create a + /// PaymentMethod during PaymentIntent confirmation. + /// @note alternative to `paymentMethodId` + @_spi(STP) public var paymentMethodData: PaymentMethodParams? + /// Provide an already created PaymentMethod's id, and it will be used to confirm the SetupIntent. + /// @note alternative to `paymentMethodParams` + @_spi(STP) public var paymentMethod: String? + /// The URL to redirect your customer back to after they authenticate or cancel + /// their payment on the payment method’s app or site. + /// This should probably be a URL that opens your iOS app. + @_spi(STP) public var returnUrl: String? + /// A boolean number to indicate whether you intend to use the Stripe SDK's functionality to handle any SetupIntent next actions. + /// If set to false, SetupIntent.nextAction will only ever contain a redirect url that can be opened in a webview or mobile browser. + /// When set to true, the nextAction may contain information that the Stripe SDK can use to perform native authentication within your + /// app. + @_spi(STP) public var useStripeSdk: Bool? + + @_spi(STP) public var _additionalParametersStorage: NonEncodableParameters? + + // MARK: - Utilities + static private let regex = try! NSRegularExpression( + pattern: "^seti_[^_]+_secret_[^_]+$", + options: [] + ) + @_spi(STP) public static func isClientSecretValid(_ clientSecret: String) -> Bool { + return + (regex.numberOfMatches( + in: clientSecret, + options: .anchored, + range: NSRange(location: 0, length: clientSecret.count) + )) == 1 + } + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/ShippingDetails.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/ShippingDetails.swift new file mode 100644 index 00000000..cec8a3bb --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/ShippingDetails.swift @@ -0,0 +1,93 @@ +// +// ShippingDetails.swift +// StripeApplePay +// +// Created by Yuki Tokuhiro on 8/4/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI { + @_spi(STP) public struct ShippingDetails: UnknownFieldsCodable, Equatable { + @_spi(STP) public init( + address: StripeAPI.ShippingDetails.Address, + name: String, + carrier: String? = nil, + phone: String? = nil, + trackingNumber: String? = nil, + _allResponseFieldsStorage: NonEncodableParameters? = nil, + _additionalParametersStorage: NonEncodableParameters? = nil + ) { + self.address = address + self.name = name + self.carrier = carrier + self.phone = phone + self.trackingNumber = trackingNumber + self._allResponseFieldsStorage = _allResponseFieldsStorage + self._additionalParametersStorage = _additionalParametersStorage + } + + /// Shipping address. + @_spi(STP) public var address: Address + + /// Recipient name. + @_spi(STP) public var name: String + + /// The delivery service that shipped a physical product, such as Fedex, UPS, USPS, etc. + @_spi(STP) public var carrier: String? + + /// Recipient phone (including extension). + @_spi(STP) public var phone: String? + + /// The tracking number for a physical product, obtained from the delivery service. If multiple tracking numbers were generated for this purchase, please separate them with commas. + @_spi(STP) public var trackingNumber: String? + + @_spi(STP) public var _allResponseFieldsStorage: NonEncodableParameters? + @_spi(STP) public var _additionalParametersStorage: NonEncodableParameters? + + @_spi(STP) public struct Address: UnknownFieldsCodable, Equatable { + @_spi(STP) public init( + city: String? = nil, + country: String? = nil, + line1: String, + line2: String? = nil, + postalCode: String? = nil, + state: String? = nil, + _allResponseFieldsStorage: NonEncodableParameters? = nil, + _additionalParametersStorage: NonEncodableParameters? = nil + ) { + self.city = city + self.country = country + self.line1 = line1 + self.line2 = line2 + self.postalCode = postalCode + self.state = state + self._allResponseFieldsStorage = _allResponseFieldsStorage + self._additionalParametersStorage = _additionalParametersStorage + } + + /// City/District/Suburb/Town/Village. + @_spi(STP) public var city: String? + + /// Two-letter country code (ISO 3166-1 alpha-2). + @_spi(STP) public var country: String? + + /// Address line 1 (Street address/PO Box/Company name). + @_spi(STP) public var line1: String + + /// Address line 2 (Apartment/Suite/Unit/Building). + @_spi(STP) public var line2: String? + + /// ZIP or postal code. + @_spi(STP) public var postalCode: String? + + /// State/County/Province/Region. + @_spi(STP) public var state: String? + + @_spi(STP) public var _allResponseFieldsStorage: NonEncodableParameters? + @_spi(STP) public var _additionalParametersStorage: NonEncodableParameters? + } + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Token.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Token.swift new file mode 100644 index 00000000..db48177d --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Models/Token.swift @@ -0,0 +1,129 @@ +// +// Token.swift +// StripeApplePay +// +// Created by David Estes on 7/14/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripeCore + +extension StripeAPI { + struct Token: UnknownFieldsDecodable { + var _allResponseFieldsStorage: NonEncodableParameters? + + /// The value of the token. You can store this value on your server and use it to make charges and customers. + /// - seealso: https://stripe.com/docs/payments/charges-api + let id: String + /// Whether or not this token was created in livemode. Will be YES if you used your Live Publishable Key, and NO if you used your Test Publishable Key. + var livemode: Bool + /// The type of this token. + var type: TokenType + + /// Possible Token types + enum TokenType: String, SafeEnumCodable { + /// Account token type + case account + /// Bank account token type + case bankAccount = "bank_account" + /// Card token type + case card + /// PII token type + case PII = "pii" + /// CVC update token type + case cvcUpdate = "cvc_update" + case unparsable + } + + /// The credit card details that were used to create the token. Will only be set if the token was created via a credit card or Apple Pay, otherwise it will be + /// nil. + var card: Card? + // /// The bank account details that were used to create the token. Will only be set if the token was created with a bank account, otherwise it will be nil. + // Not yet implemented. + // var bankAccount: BankAccount? + /// When the token was created. + var created: Date? + + struct Card: UnknownFieldsDecodable { + var _allResponseFieldsStorage: NonEncodableParameters? + + /// The last 4 digits of the card. + var last4: String + /// For cards made with Apple Pay, this refers to the last 4 digits of the + /// "Device Account Number" for the tokenized card. For regular cards, it will + /// be nil. + var dynamicLast4: String? + /// Whether or not the card originated from Apple Pay. + var isApplePayCard: Bool { + return (allResponseFields["tokenization_method"] as? String) == "apple_pay" + } + /// The card's expiration month. 1-indexed (i.e. 1 == January) + var expMonth: Int + /// The card's expiration year. + var expYear: Int + /// The cardholder's name. + var name: String? + + /// City/District/Suburb/Town/Village. + var addressCity: String? + + /// Billing address country, if provided when creating card. + var addressCountry: String? + + /// Address line 1 (Street address/PO Box/Company name). + var addressLine1: String? + + /// If address_line1 was provided, results of the check. + var addressLine1Check: AddressCheck? + + /// Results of an address check. + enum AddressCheck: String, SafeEnumCodable { + case pass + case fail + case unavailable + case unchecked + case unparsable + } + + /// Address line 2 (Apartment/Suite/Unit/Building). + var addressLine2: String? + + /// State/County/Province/Region. + var addressState: String? + + /// ZIP or postal code. + var addressZip: String? + + /// If address_zip was provided, results of the check. + var addressZipCheck: AddressCheck? + + /// The issuer of the card. + var brand: CardBrand = .unknown + + /// The funding source for the card (credit, debit, prepaid, or other) + var funding: FundingType = .unknown + + /// The various funding sources for a payment card. + enum FundingType: String, SafeEnumCodable { + /// Debit card funding + case debit + /// Credit card funding + case credit + /// Prepaid card funding + case prepaid + /// An other or unknown type of funding source. + case unknown + case unparsable + } + + /// Two-letter ISO code representing the issuing country of the card. + var country: String? + /// This is only applicable when tokenizing debit cards to issue payouts to managed + /// accounts. You should not set it otherwise. The card can then be used as a + /// transfer destination for funds in this currency. + var currency: String? + } + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentIntent+API.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentIntent+API.swift new file mode 100644 index 00000000..76c49cf6 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentIntent+API.swift @@ -0,0 +1,85 @@ +// +// PaymentIntent+API.swift +// StripeApplePay +// +// Created by David Estes on 8/10/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI.PaymentIntent { + /// A callback to be run with a PaymentIntent response from the Stripe API. + /// - Parameters: + /// - paymentIntent: The Stripe PaymentIntent from the response. Will be nil if an error occurs. - seealso: PaymentIntent + /// - error: The error returned from the response, or nil if none occurs. - seealso: StripeError.h for possible values. + @_spi(STP) public typealias PaymentIntentCompletionBlock = ( + Result + ) -> Void + + /// Retrieves the PaymentIntent object using the given secret. - seealso: https://stripe.com/docs/api#retrieve_payment_intent + /// - Parameters: + /// - secret: The client secret of the payment intent to be retrieved. Cannot be nil. + /// - completion: The callback to run with the returned PaymentIntent object, or an error. + @_spi(STP) public static func get( + apiClient: STPAPIClient = .shared, + clientSecret: String, + completion: @escaping PaymentIntentCompletionBlock + ) { + assert( + StripeAPI.PaymentIntentParams.isClientSecretValid(clientSecret), + "`secret` format does not match expected client secret formatting." + ) + guard let identifier = StripeAPI.PaymentIntent.id(fromClientSecret: clientSecret) else { + completion(.failure(StripeError.invalidRequest)) + return + } + let endpoint = "\(Resource)/\(identifier)" + let parameters: [String: String] = ["client_secret": clientSecret] + + apiClient.get(resource: endpoint, parameters: parameters, completion: completion) + } + + /// Confirms the PaymentIntent object with the provided params object. + /// At a minimum, the params object must include the `clientSecret`. + /// - seealso: https://stripe.com/docs/api#confirm_payment_intent + /// @note Use the `confirmPayment:withAuthenticationContext:completion:` method on `PaymentHandler` instead + /// of calling this method directly. It handles any authentication necessary for you. - seealso: https://stripe.com/docs/payments/3d-secure + /// - Parameters: + /// - paymentIntentParams: The `PaymentIntentParams` to pass to `/confirm` + /// - completion: The callback to run with the returned PaymentIntent object, or an error. + @_spi(STP) public static func confirm( + apiClient: STPAPIClient = .shared, + params: StripeAPI.PaymentIntentParams, + completion: @escaping PaymentIntentCompletionBlock + ) { + assert( + StripeAPI.PaymentIntentParams.isClientSecretValid(params.clientSecret), + "`paymentIntentParams.clientSecret` format does not match expected client secret formatting." + ) + + guard let identifier = StripeAPI.PaymentIntent.id(fromClientSecret: params.clientSecret) + else { + completion(.failure(StripeError.invalidRequest)) + return + } + let endpoint = "\(Resource)/\(identifier)/confirm" + + let type = params.paymentMethodData?.type.rawValue + STPAnalyticsClient.sharedClient.logPaymentIntentConfirmationAttempt( + paymentMethodType: type + ) + + // Add telemetry + var paramsWithTelemetry = params + if let pmAdditionalParams = paramsWithTelemetry.paymentMethodData?.additionalParameters { + paramsWithTelemetry.paymentMethodData?.additionalParameters = STPTelemetryClient.shared + .paramsByAddingTelemetryFields(toParams: pmAdditionalParams) + } + + apiClient.post(resource: endpoint, object: paramsWithTelemetry, completion: completion) + } + + static let Resource = "payment_intents" +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentMethod+API.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentMethod+API.swift new file mode 100644 index 00000000..f48e682c --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/PaymentMethod+API.swift @@ -0,0 +1,61 @@ +// +// PaymentMethod+API.swift +// StripeApplePay +// +// Created by David Estes on 8/10/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripeCore + +extension StripeAPI.PaymentMethod { + /// A callback to be run with a PaymentMethod response from the Stripe API. + /// - Parameters: + /// - paymentMethod: The Stripe PaymentMethod from the response. Will be nil if an error occurs. - seealso: PaymentMethod + /// - error: The error returned from the response, or nil if none occurs. - seealso: StripeError.h for possible values. + @_spi(STP) public typealias PaymentMethodCompletionBlock = ( + Result + ) -> Void + + static func create( + apiClient: STPAPIClient = .shared, + params: StripeAPI.PaymentMethodParams, + completion: @escaping PaymentMethodCompletionBlock + ) { + STPAnalyticsClient.sharedClient.logPaymentMethodCreationAttempt( + paymentMethodType: params.type.rawValue + ) + apiClient.post(resource: Resource, object: params, completion: completion) + } + + /// Converts a PKPayment object into a Stripe Payment Method using the Stripe API. + /// - Parameters: + /// - payment: The user's encrypted payment information as returned from a PKPaymentAuthorizationController. Cannot be nil. + /// - completion: The callback to run with the returned Stripe source (and any errors that may have occurred). + @_spi(STP) public static func create( + apiClient: STPAPIClient = .shared, + payment: PKPayment, + completion: @escaping PaymentMethodCompletionBlock + ) { + StripeAPI.Token.create(apiClient: apiClient, payment: payment) { (result) in + guard let token = try? result.get() else { + if case .failure(let error) = result { + completion(.failure(error)) + } else { + completion(.failure(NSError.stp_genericConnectionError())) + } + return + } + var cardParams = StripeAPI.PaymentMethodParams.Card() + cardParams.token = token.id + let billingDetails = StripeAPI.BillingDetails(from: payment) + var paymentMethodParams = StripeAPI.PaymentMethodParams(type: .card, card: cardParams) + paymentMethodParams.billingDetails = billingDetails + Self.create(apiClient: apiClient, params: paymentMethodParams, completion: completion) + } + } + + static let Resource = "payment_methods" +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/SetupIntent+API.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/SetupIntent+API.swift new file mode 100644 index 00000000..74fbfd82 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/SetupIntent+API.swift @@ -0,0 +1,84 @@ +// +// SetupIntent+API.swift +// StripeApplePay +// +// Created by David Estes on 8/10/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension StripeAPI.SetupIntent { + /// A callback to be run with a SetupIntent response from the Stripe API. + /// - Parameters: + /// - setupIntent: The Stripe SetupIntent from the response. Will be nil if an error occurs. - seealso: SetupIntent + /// - error: The error returned from the response, or nil if none occurs. - seealso: StripeError.h for possible values. + @_spi(STP) public typealias SetupIntentCompletionBlock = (Result) + -> Void + + /// Retrieves the SetupIntent object using the given secret. - seealso: https://stripe.com/docs/api/setup_intents/retrieve + /// - Parameters: + /// - secret: The client secret of the SetupIntent to be retrieved. Cannot be nil. + /// - completion: The callback to run with the returned SetupIntent object, or an error. + @_spi(STP) public static func get( + apiClient: STPAPIClient = .shared, + clientSecret: String, + completion: @escaping SetupIntentCompletionBlock + ) { + assert( + StripeAPI.SetupIntentConfirmParams.isClientSecretValid(clientSecret), + "`secret` format does not match expected client secret formatting." + ) + guard let identifier = StripeAPI.SetupIntent.id(fromClientSecret: clientSecret) else { + completion(.failure(StripeError.invalidRequest)) + return + } + let endpoint = "\(Resource)/\(identifier)" + let parameters: [String: String] = ["client_secret": clientSecret] + + apiClient.get(resource: endpoint, parameters: parameters, completion: completion) + } + + /// Confirms the SetupIntent object with the provided params object. + /// At a minimum, the params object must include the `clientSecret`. + /// - seealso: https://stripe.com/docs/api/setup_intents/confirm + /// @note Use the `confirmSetupIntent:withAuthenticationContext:completion:` method on `PaymentHandler` instead + /// of calling this method directly. It handles any authentication necessary for you. - seealso: https://stripe.com/docs/payments/3d-secure + /// - Parameters: + /// - setupIntentParams: The `SetupIntentConfirmParams` to pass to `/confirm` + /// - completion: The callback to run with the returned PaymentIntent object, or an error. + @_spi(STP) public static func confirm( + apiClient: STPAPIClient = .shared, + params: StripeAPI.SetupIntentConfirmParams, + completion: @escaping SetupIntentCompletionBlock + ) { + assert( + StripeAPI.SetupIntentConfirmParams.isClientSecretValid(params.clientSecret), + "`setupIntentConfirmParams.clientSecret` format does not match expected client secret formatting." + ) + + guard let identifier = StripeAPI.SetupIntent.id(fromClientSecret: params.clientSecret) + else { + completion(.failure(StripeError.invalidRequest)) + return + } + let endpoint = "\(Resource)/\(identifier)/confirm" + + let type = params.paymentMethodData?.type.rawValue + STPAnalyticsClient.sharedClient.logSetupIntentConfirmationAttempt( + paymentMethodType: type + ) + + // Add telemetry + var paramsWithTelemetry = params + if let pmAdditionalParams = paramsWithTelemetry.paymentMethodData?.additionalParameters { + paramsWithTelemetry.paymentMethodData?.additionalParameters = STPTelemetryClient.shared + .paramsByAddingTelemetryFields(toParams: pmAdditionalParams) + } + + apiClient.post(resource: endpoint, object: paramsWithTelemetry, completion: completion) + } + + static let Resource = "setup_intents" +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Token+API.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Token+API.swift new file mode 100644 index 00000000..b02ec961 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/API/Token+API.swift @@ -0,0 +1,91 @@ +// +// Token+API.swift +// StripeApplePay +// +// Created by David Estes on 8/10/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import PassKit +@_spi(STP) import StripeCore + +extension StripeAPI.Token { + /// A callback to be run with a token response from the Stripe API. + /// - Parameters: + /// - token: The Stripe token from the response. Will be nil if an error occurs. - seealso: STPToken + /// - error: The error returned from the response, or nil if none occurs. - seealso: StripeError.h for possible values. + typealias TokenCompletionBlock = (Result) -> Void + + /// Converts a PKPayment object into a Stripe token using the Stripe API. + /// - Parameters: + /// - payment: The user's encrypted payment information as returned from a PKPaymentAuthorizationController. Cannot be nil. + /// - completion: The callback to run with the returned Stripe token (and any errors that may have occurred). + static func create( + apiClient: STPAPIClient = .shared, + payment: PKPayment, + completion: @escaping TokenCompletionBlock + ) { + let params = payment.stp_tokenParameters(apiClient: apiClient) + create( + apiClient: apiClient, + parameters: params, + completion: completion + ) + } + + static func create( + apiClient: STPAPIClient = .shared, + parameters: [String: Any], + completion: @escaping TokenCompletionBlock + ) { + let tokenType = STPAnalyticsClient.tokenType(fromParameters: parameters) + var mutableParams = parameters + STPTelemetryClient.shared.addTelemetryFields(toParams: &mutableParams) + mutableParams = STPAPIClient.paramsAddingPaymentUserAgent(mutableParams) + STPAnalyticsClient.sharedClient.logTokenCreationAttempt(tokenType: tokenType) + apiClient.post(resource: Resource, parameters: mutableParams, completion: completion) + STPTelemetryClient.shared.sendTelemetryData() + } + + static let Resource = "tokens" +} + +extension PKPayment { + func stp_tokenParameters(apiClient: STPAPIClient) -> [String: Any] { + let paymentString = String(data: self.token.paymentData, encoding: .utf8) + var payload: [String: Any] = [:] + payload["pk_token"] = paymentString + if let billingContact = self.billingContact { + payload["card"] = billingContact.addressParams + } + + assert( + !((paymentString?.count ?? 0) == 0 + && apiClient.publishableKey?.hasPrefix("pk_live") ?? false), + "The pk_token is empty. Using Apple Pay with an iOS Simulator while not in Stripe Test Mode will always fail." + ) + + let paymentInstrumentName = self.token.paymentMethod.displayName + if let paymentInstrumentName = paymentInstrumentName { + payload["pk_token_instrument_name"] = paymentInstrumentName + } + + let paymentNetwork = self.token.paymentMethod.network + if let paymentNetwork = paymentNetwork { + // Note: As of SDK 20.0.0, this will return `PKPaymentNetwork(_rawValue: MasterCard)`. + // We're intentionally leaving it this way: See RUN_MOBILESDK-125. + payload["pk_token_payment_network"] = paymentNetwork + } + + var transactionIdentifier = self.token.transactionIdentifier + if transactionIdentifier != "" { + if self.stp_isSimulated() { + transactionIdentifier = PKPayment.stp_testTransactionIdentifier() + } + payload["pk_token_transaction_id"] = transactionIdentifier + } + + return payload + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+Payments.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+Payments.swift new file mode 100644 index 00000000..91cfbd1e --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+Payments.swift @@ -0,0 +1,28 @@ +// +// STPAnalyticsClient+Payments.swift +// StripeApplePay +// +// Created by David Estes on 1/24/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +/// An analytic specific to payments that serializes payment-specific +/// information into its params. +@_spi(STP) public protocol PaymentAnalytic: Analytic { + var productUsage: Set { get } + var additionalParams: [String: Any] { get } +} + +@_spi(STP) extension PaymentAnalytic { + public var params: [String: Any] { + var params = additionalParams + + params["apple_pay_enabled"] = NSNumber(value: StripeAPI.deviceSupportsApplePay()) + params["ocr_type"] = PaymentsSDKVariant.ocrTypeString + params["pay_var"] = PaymentsSDKVariant.variant + return params + } +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+PaymentsAPI.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+PaymentsAPI.swift new file mode 100644 index 00000000..2793d27a --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/Analytics/STPAnalyticsClient+PaymentsAPI.swift @@ -0,0 +1,72 @@ +// +// STPAnalyticsClient+PaymentsAPI.swift +// StripeApplePay +// +// Created by David Estes on 1/21/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension STPAnalyticsClient { + // MARK: - Log events + + func logPaymentMethodCreationAttempt(paymentMethodType: String?) { + log( + analytic: PaymentAPIAnalytic( + event: .paymentMethodCreation, + productUsage: productUsage, + additionalParams: [ + "source_type": paymentMethodType ?? "unknown" + ] + ) + ) + } + + func logTokenCreationAttempt(tokenType: String?) { + log( + analytic: PaymentAPIAnalytic( + event: .tokenCreation, + productUsage: productUsage, + additionalParams: [ + "token_type": tokenType ?? "unknown" + ] + ) + ) + } + + func logPaymentIntentConfirmationAttempt( + paymentMethodType: String? + ) { + log( + analytic: PaymentAPIAnalytic( + event: .paymentMethodIntentCreation, + productUsage: productUsage, + additionalParams: [ + "source_type": paymentMethodType ?? "unknown" + ] + ) + ) + } + + func logSetupIntentConfirmationAttempt( + paymentMethodType: String? + ) { + log( + analytic: PaymentAPIAnalytic( + event: .setupIntentConfirmationAttempt, + productUsage: productUsage, + additionalParams: [ + "source_type": paymentMethodType ?? "unknown" + ] + ) + ) + } +} + +struct PaymentAPIAnalytic: PaymentAnalytic { + let event: STPAnalyticEvent + let productUsage: Set + let additionalParams: [String: Any] +} diff --git a/StripeApplePay/StripeApplePay/Source/PaymentsCore/Categories/STPAPIClient+PaymentsCore.swift b/StripeApplePay/StripeApplePay/Source/PaymentsCore/Categories/STPAPIClient+PaymentsCore.swift new file mode 100644 index 00000000..2ed46cf8 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/PaymentsCore/Categories/STPAPIClient+PaymentsCore.swift @@ -0,0 +1,20 @@ +// +// STPAPIClient+PaymentsCore.swift +// StripeApplePay +// +// Created by David Estes on 1/25/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore + +extension STPAPIClient { + @_spi(STP) public class func paramsAddingPaymentUserAgent( + _ params: [String: Any] + ) -> [String: Any] { + var newParams = params + newParams["payment_user_agent"] = PaymentsSDKVariant.paymentUserAgent + return newParams + } +} diff --git a/StripeApplePay/StripeApplePay/Source/StripeCore+Import.swift b/StripeApplePay/StripeApplePay/Source/StripeCore+Import.swift new file mode 100644 index 00000000..123a8e98 --- /dev/null +++ b/StripeApplePay/StripeApplePay/Source/StripeCore+Import.swift @@ -0,0 +1,10 @@ +// +// StripeCore+Import.swift +// StripeApplePay +// +// Created by David Estes on 11/15/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_exported import StripeCore diff --git a/StripeApplePay/StripeApplePay/StripeApplePay.h b/StripeApplePay/StripeApplePay/StripeApplePay.h new file mode 100644 index 00000000..e6b9e0f8 --- /dev/null +++ b/StripeApplePay/StripeApplePay/StripeApplePay.h @@ -0,0 +1,18 @@ +// +// StripeApplePay.h +// StripeApplePay +// +// Created by David Estes on 11/8/21. +// + +#import + +//! Project version number for StripeApplePay. +FOUNDATION_EXPORT double StripeApplePayVersionNumber; + +//! Project version string for StripeApplePay. +FOUNDATION_EXPORT const unsigned char StripeApplePayVersionString[]; + +// In this header, you should import all the public headers of your framework using statements like #import + + diff --git a/StripeApplePay/StripeApplePayTests/Info.plist b/StripeApplePay/StripeApplePayTests/Info.plist new file mode 100644 index 00000000..64d65ca4 --- /dev/null +++ b/StripeApplePay/StripeApplePayTests/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + + diff --git a/StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientFunctionalTest.swift b/StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientFunctionalTest.swift new file mode 100644 index 00000000..9f70bfb5 --- /dev/null +++ b/StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientFunctionalTest.swift @@ -0,0 +1,67 @@ +// +// STPTelemetryClientFunctionalTest.swift +// StripeApplePayTests +// +// Created by Yuki Tokuhiro on 5/21/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import XCTest + +// swift-format-ignore +@testable @_spi(STP) import StripeApplePay + +// swift-format-ignore +@testable @_spi(STP) import StripeCore + +class STPTelemetryClientFunctionalTest: XCTestCase { + func testSendFraudDetectionData() { + // Sending telemetry without any FraudDetectionData... + FraudDetectionData.shared.sid = nil + FraudDetectionData.shared.sidCreationDate = nil + FraudDetectionData.shared.muid = nil + FraudDetectionData.shared.guid = nil + let sendTelemetry1 = expectation(description: "") + STPTelemetryClient.shared.sendTelemetryData(forceSend: true) { _ in + sendTelemetry1.fulfill() + } + waitForExpectations(timeout: 10, handler: nil) + // ...populates FraudDetectionData + let sid = FraudDetectionData.shared.sid + let muid = FraudDetectionData.shared.muid + let guid = FraudDetectionData.shared.guid + XCTAssertNotNil(sid) + XCTAssertNotNil(muid) + XCTAssertNotNil(guid) + + let sendTelemetry2 = expectation(description: "") + // Sending telemetry again... + STPTelemetryClient.shared.sendTelemetryData(forceSend: true) { _ in + sendTelemetry2.fulfill() + } + // ...gives the same FraudDetectionData + XCTAssertEqual(FraudDetectionData.shared.sid, sid) + XCTAssertEqual(FraudDetectionData.shared.muid, muid) + XCTAssertEqual(FraudDetectionData.shared.guid, guid) + guard let sidCreationDate = FraudDetectionData.shared.sidCreationDate else { + XCTFail() + return + } + // sanity check creation date looks right + XCTAssertTrue(sidCreationDate > Date(timeIntervalSinceNow: -10)) + waitForExpectations(timeout: 10, handler: nil) + + // Expiring the FraudDetectionData + FraudDetectionData.shared.sidCreationDate = Date(timeIntervalSinceNow: -999999) + let sendTelemetry3 = expectation(description: "") + // ...and sending telemetry again + STPTelemetryClient.shared.sendTelemetryData(forceSend: true) { _ in + sendTelemetry3.fulfill() + } + waitForExpectations(timeout: 10, handler: nil) + // ...gives the same muid and guid but different sid + XCTAssertEqual(FraudDetectionData.shared.muid, muid) + XCTAssertEqual(FraudDetectionData.shared.guid, guid) + XCTAssertNotEqual(FraudDetectionData.shared.sid, sid) + } +} diff --git a/StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientTest.swift b/StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientTest.swift new file mode 100644 index 00000000..8a730008 --- /dev/null +++ b/StripeApplePay/StripeApplePayTests/PaymentsCore/STPTelemetryClientTest.swift @@ -0,0 +1,75 @@ +// +// STPTelemetryClientTest.swift +// StripeApplePayTests +// +// Created by Yuki Tokuhiro on 9/24/20. +// Copyright © 2020 Stripe, Inc. All rights reserved. +// + +import XCTest + +// swift-format-ignore +@testable @_spi(STP) import StripeApplePay + +// swift-format-ignore +@testable @_spi(STP) import StripeCore + +class STPTelemetryClientTest: XCTestCase { + + func testAddTelemetryData() { + let sut = STPTelemetryClient.shared + var params: [String: Any] = [ + "foo": "bar" + ] + let exp = expectation(description: "delay") + DispatchQueue.main.asyncAfter( + deadline: DispatchTime.now() + Double(Int64(0.1 * Double(NSEC_PER_SEC))) + / Double(NSEC_PER_SEC), + execute: { + sut.addTelemetryFields(toParams: ¶ms) + XCTAssertNotNil(params) + exp.fulfill() + } + ) + waitForExpectations(timeout: 2, handler: nil) + } + + func testAdvancedFraudSignalsSwitch() { + XCTAssertTrue(StripeAPI.advancedFraudSignalsEnabled) + StripeAPI.advancedFraudSignalsEnabled = false + XCTAssertFalse(StripeAPI.advancedFraudSignalsEnabled) + } + + func testAddTelemetryFieldsWhenFraudDetectionDataEmpty() { + // Should not add any fields if fraudDetectionData is empty + FraudDetectionData.shared.reset() + var params: [String: Any] = [:] + STPTelemetryClient.shared.addTelemetryFields(toParams: ¶ms) + XCTAssertTrue(params.isEmpty) + } + + func testAddTelemetryFieldsWhenSIDExpired() { + // Should add muid, but not add sid if it's expired + var params: [String: Any] = [:] + FraudDetectionData.shared.sid = "expired" + FraudDetectionData.shared.sidCreationDate = Date(timeInterval: -30 * 60, since: Date()) + FraudDetectionData.shared.muid = "muid value" + FraudDetectionData.shared.guid = "guid value" + STPTelemetryClient.shared.addTelemetryFields(toParams: ¶ms) + XCTAssertEqual(params["muid"] as? String, "muid value") + XCTAssertEqual(params["guid"] as? String, "guid value") + XCTAssertNil(params["sid"] as? String) + } + + func testAddTelemetryFields() { + var params: [String: Any] = [:] + FraudDetectionData.shared.sid = "sid value" + FraudDetectionData.shared.muid = "muid value" + FraudDetectionData.shared.guid = "guid value" + FraudDetectionData.shared.sidCreationDate = Date() + STPTelemetryClient.shared.addTelemetryFields(toParams: ¶ms) + XCTAssertEqual(params["muid"] as? String, "muid value") + XCTAssertEqual(params["sid"] as? String, "sid value") + XCTAssertEqual(params["guid"] as? String, "guid value") + } +} diff --git a/StripeApplePay/StripeApplePayTests/STPAnalyticsClient+ApplePayTest.swift b/StripeApplePay/StripeApplePayTests/STPAnalyticsClient+ApplePayTest.swift new file mode 100644 index 00000000..a39e180c --- /dev/null +++ b/StripeApplePay/StripeApplePayTests/STPAnalyticsClient+ApplePayTest.swift @@ -0,0 +1,30 @@ +// +// STPAnalyticsClient+ApplePayTest.swift +// StripeApplePayTests +// +// Created by David Estes on 2/3/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import Foundation +import XCTest + +// swift-format-ignore +@_spi(STP) @testable import StripeApplePay + +// swift-format-ignore +@_spi(STP) @testable import StripeCore + +class STPAnalyticsClientApplePayTest: XCTestCase { + func testApplePaySDKVariantPayload() throws { + // setup + let analytic = PaymentAPIAnalytic( + event: .paymentMethodCreation, + productUsage: [], + additionalParams: [:] + ) + let client = STPAnalyticsClient() + let payload = client.payload(from: analytic) + XCTAssertEqual("applepay", payload["pay_var"] as? String) + } +} diff --git a/StripeApplePay/StripeApplePayTests/STPPaymentMethodFunctionalTest.swift b/StripeApplePay/StripeApplePayTests/STPPaymentMethodFunctionalTest.swift new file mode 100644 index 00000000..d2037222 --- /dev/null +++ b/StripeApplePay/StripeApplePayTests/STPPaymentMethodFunctionalTest.swift @@ -0,0 +1,96 @@ +// +// STPPaymentMethodFunctionalTest.swift +// StripeApplePayTests +// +// Created by David Estes on 8/9/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +@_spi(STP) import StripeCore // for StripeError +import XCTest + +// swift-format-ignore +@_spi(STP) @testable import StripeApplePay + +let STPTestingDefaultPublishableKey = "pk_test_ErsyMEOTudSjQR8hh0VrQr5X008sBXGOu6" +public let STPTestingNetworkRequestTimeout: TimeInterval = 8 + +class STPPaymentMethodModernTest: XCTestCase { + func testCreateCardPaymentMethod() { + let expectation = self.expectation(description: "Created") + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + var params = StripeAPI.PaymentMethodParams(type: .card) + var card = StripeAPI.PaymentMethodParams.Card() + card.number = "4242424242424242" + card.expYear = 28 + card.expMonth = 12 + card.cvc = "100" + var billingAddress = StripeAPI.BillingDetails.Address() + billingAddress.city = "San Francisco" + billingAddress.country = "US" + billingAddress.line1 = "150 Townsend St" + billingAddress.line2 = "4th Floor" + billingAddress.postalCode = "94103" + billingAddress.state = "CA" + + var billingDetails = StripeAPI.BillingDetails() + billingDetails.address = billingAddress + billingDetails.email = "email@email.com" + billingDetails.name = "Isaac Asimov" + billingDetails.phone = "555-555-5555" + + params.card = card + params.billingDetails = billingDetails + + StripeAPI.PaymentMethod.create(apiClient: apiClient, params: params) { result in + let paymentMethod = try! result.get() + XCTAssertEqual(paymentMethod.card?.last4, "4242") + expectation.fulfill() + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout, handler: nil) + } + + func testCreateCardPaymentMethodWithAdditionalAPIStuff() { + let expectation = self.expectation(description: "Created") + let apiClient = STPAPIClient(publishableKey: STPTestingDefaultPublishableKey) + var params = StripeAPI.PaymentMethodParams(type: .card) + var card = StripeAPI.PaymentMethodParams.Card() + card.number = "4242424242424242" + card.expYear = 28 + card.expMonth = 12 + card.cvc = "100" + var billingAddress = StripeAPI.BillingDetails.Address() + billingAddress.city = "San Francisco" + billingAddress.country = "US" + billingAddress.line1 = "150 Townsend St" + billingAddress.line2 = "4th Floor" + billingAddress.postalCode = "94103" + billingAddress.state = "CA" + billingAddress.additionalParameters = ["invalid_thing": "yes"] + + var billingDetails = StripeAPI.BillingDetails() + billingDetails.address = billingAddress + billingDetails.email = "email@email.com" + billingDetails.name = "Isaac Asimov" + billingDetails.phone = "555-555-5555" + + params.card = card + params.billingDetails = billingDetails + + StripeAPI.PaymentMethod.create(apiClient: apiClient, params: params) { result in + do { + _ = try result.get() + XCTFail("This request should fail") + } catch { + let stripeError = error as? StripeError + if case .apiError(let apiError) = stripeError { + XCTAssertEqual(apiError.code, "parameter_unknown") + XCTAssertEqual(apiError.param, "billing_details[address][invalid_thing]") + expectation.fulfill() + } + } + } + waitForExpectations(timeout: STPTestingNetworkRequestTimeout, handler: nil) + } +} diff --git a/StripeApplePay/StripeApplePayTests/TelemetryInjectionTest.swift b/StripeApplePay/StripeApplePayTests/TelemetryInjectionTest.swift new file mode 100644 index 00000000..b337a440 --- /dev/null +++ b/StripeApplePay/StripeApplePayTests/TelemetryInjectionTest.swift @@ -0,0 +1,95 @@ +// +// TelemetryInjectionTest.swift +// StripeApplePayTests +// +// Created by David Estes on 2/9/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import AVFoundation +import Foundation +import OHHTTPStubs +import OHHTTPStubsSwift +import StripeCoreTestUtils +import XCTest + +// swift-format-ignore +@_spi(STP) @testable import StripeApplePay + +// swift-format-ignore +@_spi(STP) @testable import StripeCore + +class TelemetryInjectionTest: APIStubbedTestCase { + func testIntentConfirmAddsTelemetry() { + let apiClient = stubbedAPIClient() + + let piTelemetryExpectation = self.expectation(description: "saw pi telemetry") + let siTelemetryExpectation = self.expectation(description: "saw si telemetry") + + // As an implementation detail, OHHTTPStubs will run this block in `canInitWithRequest` in addition to + // `initWithRequest`. So it could be called more times than we expect. + // We don't have control over this behavior (CFNetwork drives it), so let's not worry + // about overfulfillment. + piTelemetryExpectation.assertForOverFulfill = false + siTelemetryExpectation.assertForOverFulfill = false + + stub { urlRequest in + if urlRequest.url!.absoluteString.contains("_intent") { + let ua = urlRequest.queryItems!.first(where: { + $0.name == "payment_method_data[payment_user_agent]" + })!.value! + XCTAssertTrue(ua.hasPrefix("stripe-ios/")) + let muid = urlRequest.queryItems!.first(where: { + $0.name == "payment_method_data[muid]" + })!.value! + let guid = urlRequest.queryItems!.first(where: { + $0.name == "payment_method_data[guid]" + })!.value! + XCTAssertNotNil(muid) + XCTAssertNotNil(guid) + if urlRequest.url!.absoluteString.contains("payment_intent") { + piTelemetryExpectation.fulfill() + } + if urlRequest.url!.absoluteString.contains("setup_intent") { + siTelemetryExpectation.fulfill() + } + return true + } + return false + } response: { _ in + // We don't care about the response + return HTTPStubsResponse() + } + + var params = StripeAPI.PaymentMethodParams(type: .card) + var card = StripeAPI.PaymentMethodParams.Card() + card.number = "4242424242424242" + card.expYear = 28 + card.expMonth = 12 + card.cvc = "100" + params.card = card + + // Set up telemetry data + StripeAPI.advancedFraudSignalsEnabled = true + FraudDetectionData.shared.sid = "sid" + FraudDetectionData.shared.muid = "muid" + FraudDetectionData.shared.guid = "guid" + FraudDetectionData.shared.sidCreationDate = Date() + + let piExpectation = self.expectation(description: "PI Confirmed") + var pip = StripeAPI.PaymentIntentParams(clientSecret: "pi_123_secret_abc") + pip.paymentMethodData = params + StripeAPI.PaymentIntent.confirm(apiClient: apiClient, params: pip) { _ in + piExpectation.fulfill() + } + + let siExpectation = self.expectation(description: "SI Confirmed") + var sip = StripeAPI.SetupIntentConfirmParams(clientSecret: "seti_123_secret_abc") + sip.paymentMethodData = params + StripeAPI.SetupIntent.confirm(apiClient: apiClient, params: sip) { _ in + siExpectation.fulfill() + } + + waitForExpectations(timeout: STPTestingNetworkRequestTimeout, handler: nil) + } +} diff --git a/StripeCameraCore/Project.swift b/StripeCameraCore/Project.swift new file mode 100644 index 00000000..d755cfda --- /dev/null +++ b/StripeCameraCore/Project.swift @@ -0,0 +1,11 @@ +import ProjectDescription +import ProjectDescriptionHelpers + +let project = Project.stripeFramework( + name: "StripeCameraCore", + dependencies: [ + .project(target: "StripeCore", path: "//StripeCore"), + ], + testUtilsOptions: .testOptions(), + unitTestOptions: .testOptions() +) diff --git a/StripeCameraCore/StripeCameraCore.xcodeproj/project.pbxproj b/StripeCameraCore/StripeCameraCore.xcodeproj/project.pbxproj new file mode 100644 index 00000000..ccfeffb1 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore.xcodeproj/project.pbxproj @@ -0,0 +1,640 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 52; + objects = { + +/* Begin PBXBuildFile section */ + 0485C4A62D9444E75877E170 /* CameraExifMetadata.swift in Sources */ = {isa = PBXBuildFile; fileRef = D76A4D11F53602B77F9F5B2E /* CameraExifMetadata.swift */; }; + 064567ECD2B71E5EB7D3A201 /* MockCameraPermissionsManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 832A750C892D9D3FC54D9CF3 /* MockCameraPermissionsManager.swift */; }; + 06FEEA450541E2D36D5A10FC /* CameraPermissionsManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3D0F925920743B648B04C949 /* CameraPermissionsManager.swift */; }; + 0F02BB3F9F7936D7BB593226 /* Torch.swift in Sources */ = {isa = PBXBuildFile; fileRef = D750C5508AB271C099351E42 /* Torch.swift */; }; + 10BCE429AA79F7F0D1A9F2D3 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7158E127DC7294CDABDEF5B8 /* XCTest.framework */; }; + 1EF9F2AB3E43769AB2BA6D04 /* StripeCameraCoreTestUtils.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 2720B7BD13954418A8AC52E5 /* StripeCameraCoreTestUtils.framework */; }; + 28F23B4322876ED2BAC8C933 /* StripeCameraCoreTestUtils.h in Headers */ = {isa = PBXBuildFile; fileRef = DDD1039F321BAF147F162F7D /* StripeCameraCoreTestUtils.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 2A1AAD8E5A589E4E74B02A35 /* MockSimulatorCameraSession.swift in Sources */ = {isa = PBXBuildFile; fileRef = BCCE205F4CCF2D486967BA25 /* MockSimulatorCameraSession.swift */; }; + 2CEF8A27C8864A6198C69A5D /* CVPixelBuffer+StripeCameraCore.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0A0B9FDC8622F6CDF9F8A4B4 /* CVPixelBuffer+StripeCameraCore.swift */; }; + 48B09ECBED820F3097779208 /* CGRect+StripeCameraCore.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3B4106DF455718394E209C00 /* CGRect+StripeCameraCore.swift */; }; + 5F57E0BF10039F7055013066 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7158E127DC7294CDABDEF5B8 /* XCTest.framework */; }; + 6D44C32189C0CB409C9E4712 /* StripeCameraCore.h in Headers */ = {isa = PBXBuildFile; fileRef = E7FB32269B8E36C3CBDB370F /* StripeCameraCore.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 88A1AF2B173B4055D8F44A8C /* MockAppSettingsHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9F04D387EEF3A5D07ECD1F7E /* MockAppSettingsHelper.swift */; }; + 8F3073987F30B9B8DDF0D956 /* MockTestCameraSession.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8091DB42296B7EC745567AF7 /* MockTestCameraSession.swift */; }; + 9AFBF50F47562CB34903064E /* StripeCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 8E7BCD0896ADAFFF5CCA738E /* StripeCore.framework */; }; + 9CE6EE572439AD5B22B77294 /* CameraPreviewView.swift in Sources */ = {isa = PBXBuildFile; fileRef = A79D46F44C7BC55B9B595AEF /* CameraPreviewView.swift */; }; + 9FC42D068D1719AB28C3068F /* AppSettingsHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 48DFFB1E2F90176D156E315B /* AppSettingsHelper.swift */; }; + A95255494DF27A7CAE9A70CE /* CameraSession.swift in Sources */ = {isa = PBXBuildFile; fileRef = FBD8FEBAE0069BE88B75B0C5 /* CameraSession.swift */; }; + AB24784F49EB99E7C36509BD /* CGRect_StripeCameraCoreTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2B5B6BCFF035A9E214EA9EA0 /* CGRect_StripeCameraCoreTest.swift */; }; + D2186D6E491660D0C2A9D577 /* StripeCameraCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 6DBC37790A9F68168A89095B /* StripeCameraCore.framework */; }; + D9687FAD52D1896F28309580 /* UIImage+Buffer.swift in Sources */ = {isa = PBXBuildFile; fileRef = BA969E606F5A3829BC9D4445 /* UIImage+Buffer.swift */; }; + E01A054861E5D82D18C7D224 /* StripeCameraCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 6DBC37790A9F68168A89095B /* StripeCameraCore.framework */; }; + F181EEBCE3D2CAD043DAF0F0 /* UIDeviceOrientation+StripeCameraCore.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7D9BDABE163D20DF20F2237D /* UIDeviceOrientation+StripeCameraCore.swift */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 0321B7B71CC9BDDCF236B375 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = B93DF1E4460962AFB28CF144 /* Project object */; + proxyType = 1; + remoteGlobalIDString = B50782C89D54809DFFCF80D0; + remoteInfo = StripeCameraCore; + }; + 55BD94DFEA7738F26C23A545 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = B93DF1E4460962AFB28CF144 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 6A948E338F0BF55DBFD83170; + remoteInfo = StripeCameraCoreTestUtils; + }; + BA0100EB31091CFA64BF12A6 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = B93DF1E4460962AFB28CF144 /* Project object */; + proxyType = 1; + remoteGlobalIDString = B50782C89D54809DFFCF80D0; + remoteInfo = StripeCameraCore; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 2F2CDDE07BFFE88706C6A908 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + BCA0A45A41520FB95F4E8D10 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + F97E2A90DA9F982FC293A4A9 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 0A0B9FDC8622F6CDF9F8A4B4 /* CVPixelBuffer+StripeCameraCore.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "CVPixelBuffer+StripeCameraCore.swift"; sourceTree = ""; }; + 0F4174046507759A9668F4EE /* Project-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Release.xcconfig"; sourceTree = ""; }; + 17E9A614AF4543DB62E7F5B4 /* StripeiOS Tests-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Release.xcconfig"; sourceTree = ""; }; + 226FED2248EE3F96F217BEE9 /* StripeCameraCoreTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = StripeCameraCoreTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 2720B7BD13954418A8AC52E5 /* StripeCameraCoreTestUtils.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCameraCoreTestUtils.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 2B5B6BCFF035A9E214EA9EA0 /* CGRect_StripeCameraCoreTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CGRect_StripeCameraCoreTest.swift; sourceTree = ""; }; + 3B4106DF455718394E209C00 /* CGRect+StripeCameraCore.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "CGRect+StripeCameraCore.swift"; sourceTree = ""; }; + 3D0F925920743B648B04C949 /* CameraPermissionsManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CameraPermissionsManager.swift; sourceTree = ""; }; + 44E9A4900C4B8D08A617488C /* StripeiOS Tests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Debug.xcconfig"; sourceTree = ""; }; + 48DFFB1E2F90176D156E315B /* AppSettingsHelper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppSettingsHelper.swift; sourceTree = ""; }; + 4E0D85BEF8C269A46D5ECEAC /* StripeiOS-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS-Release.xcconfig"; sourceTree = ""; }; + 5DFD03E78034CFB59E244199 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 609D9ADB9E9898694A2A67FF /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 6DBC37790A9F68168A89095B /* StripeCameraCore.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCameraCore.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 7158E127DC7294CDABDEF5B8 /* XCTest.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = XCTest.framework; path = Platforms/iPhoneOS.platform/Developer/Library/Frameworks/XCTest.framework; sourceTree = DEVELOPER_DIR; }; + 7D9BDABE163D20DF20F2237D /* UIDeviceOrientation+StripeCameraCore.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIDeviceOrientation+StripeCameraCore.swift"; sourceTree = ""; }; + 8091DB42296B7EC745567AF7 /* MockTestCameraSession.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MockTestCameraSession.swift; sourceTree = ""; }; + 832A750C892D9D3FC54D9CF3 /* MockCameraPermissionsManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MockCameraPermissionsManager.swift; sourceTree = ""; }; + 83596C8ED3829B17A93490A6 /* StripeiOS-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS-Debug.xcconfig"; sourceTree = ""; }; + 8E7BCD0896ADAFFF5CCA738E /* StripeCore.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCore.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 9F04D387EEF3A5D07ECD1F7E /* MockAppSettingsHelper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MockAppSettingsHelper.swift; sourceTree = ""; }; + A79D46F44C7BC55B9B595AEF /* CameraPreviewView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CameraPreviewView.swift; sourceTree = ""; }; + ABCCADF0E7337301A472E75C /* Project-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Debug.xcconfig"; sourceTree = ""; }; + B8D39F3EE77B9DF5FFBA4CAE /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + BA969E606F5A3829BC9D4445 /* UIImage+Buffer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIImage+Buffer.swift"; sourceTree = ""; }; + BCCE205F4CCF2D486967BA25 /* MockSimulatorCameraSession.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MockSimulatorCameraSession.swift; sourceTree = ""; }; + D750C5508AB271C099351E42 /* Torch.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Torch.swift; sourceTree = ""; }; + D76A4D11F53602B77F9F5B2E /* CameraExifMetadata.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CameraExifMetadata.swift; sourceTree = ""; }; + DDD1039F321BAF147F162F7D /* StripeCameraCoreTestUtils.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = StripeCameraCoreTestUtils.h; sourceTree = ""; }; + E7FB32269B8E36C3CBDB370F /* StripeCameraCore.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = StripeCameraCore.h; sourceTree = ""; }; + FBD8FEBAE0069BE88B75B0C5 /* CameraSession.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CameraSession.swift; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 267776DB1189FACBD4FA2AEA /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 5F57E0BF10039F7055013066 /* XCTest.framework in Frameworks */, + E01A054861E5D82D18C7D224 /* StripeCameraCore.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 9CC1AF0A3093D51C5085D503 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 10BCE429AA79F7F0D1A9F2D3 /* XCTest.framework in Frameworks */, + D2186D6E491660D0C2A9D577 /* StripeCameraCore.framework in Frameworks */, + 1EF9F2AB3E43769AB2BA6D04 /* StripeCameraCoreTestUtils.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + E3A6C5C1A5D47CEFAFD1853F /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 9AFBF50F47562CB34903064E /* StripeCore.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 00F9BB709FA21D59419FA5AE /* BuildConfigurations */ = { + isa = PBXGroup; + children = ( + ABCCADF0E7337301A472E75C /* Project-Debug.xcconfig */, + 0F4174046507759A9668F4EE /* Project-Release.xcconfig */, + 44E9A4900C4B8D08A617488C /* StripeiOS Tests-Debug.xcconfig */, + 17E9A614AF4543DB62E7F5B4 /* StripeiOS Tests-Release.xcconfig */, + 83596C8ED3829B17A93490A6 /* StripeiOS-Debug.xcconfig */, + 4E0D85BEF8C269A46D5ECEAC /* StripeiOS-Release.xcconfig */, + ); + name = BuildConfigurations; + path = ../BuildConfigurations; + sourceTree = ""; + }; + 03256F1FDB473FE93A57A38A /* Source */ = { + isa = PBXGroup; + children = ( + 170C84B7FB58FAA15283903E /* Categories */, + 7E719691677D20AEAEF0C120 /* Coordinators */, + 72FEB5C092565B365BBC3906 /* Views */, + D76A4D11F53602B77F9F5B2E /* CameraExifMetadata.swift */, + ); + path = Source; + sourceTree = ""; + }; + 0FB9D3D4625E1AEA6EE1CC22 /* Frameworks */ = { + isa = PBXGroup; + children = ( + 7158E127DC7294CDABDEF5B8 /* XCTest.framework */, + ); + name = Frameworks; + sourceTree = ""; + }; + 170C84B7FB58FAA15283903E /* Categories */ = { + isa = PBXGroup; + children = ( + 3B4106DF455718394E209C00 /* CGRect+StripeCameraCore.swift */, + 0A0B9FDC8622F6CDF9F8A4B4 /* CVPixelBuffer+StripeCameraCore.swift */, + 7D9BDABE163D20DF20F2237D /* UIDeviceOrientation+StripeCameraCore.swift */, + BA969E606F5A3829BC9D4445 /* UIImage+Buffer.swift */, + ); + path = Categories; + sourceTree = ""; + }; + 171BE48EE95E66F94D6CCC33 /* StripeCameraCoreTests */ = { + isa = PBXGroup; + children = ( + CC0771AB4883EEDAB5AA7D1C /* Unit */, + 5DFD03E78034CFB59E244199 /* Info.plist */, + ); + path = StripeCameraCoreTests; + sourceTree = ""; + }; + 18A8A17329DE3C8891103ECD /* Categories */ = { + isa = PBXGroup; + children = ( + 2B5B6BCFF035A9E214EA9EA0 /* CGRect_StripeCameraCoreTest.swift */, + ); + path = Categories; + sourceTree = ""; + }; + 27E56A8242FD577334930F5B /* StripeCameraCoreTestUtils */ = { + isa = PBXGroup; + children = ( + BCB41D804D8A44F6DEEF4B8E /* Mocks */, + B8D39F3EE77B9DF5FFBA4CAE /* Info.plist */, + DDD1039F321BAF147F162F7D /* StripeCameraCoreTestUtils.h */, + ); + path = StripeCameraCoreTestUtils; + sourceTree = ""; + }; + 5CE9B5F8E73CBC18E22AF375 /* Project */ = { + isa = PBXGroup; + children = ( + 00F9BB709FA21D59419FA5AE /* BuildConfigurations */, + E2CD33A048BB3FD49B4D47FD /* StripeCameraCore */, + 171BE48EE95E66F94D6CCC33 /* StripeCameraCoreTests */, + 27E56A8242FD577334930F5B /* StripeCameraCoreTestUtils */, + ); + name = Project; + sourceTree = ""; + }; + 72FEB5C092565B365BBC3906 /* Views */ = { + isa = PBXGroup; + children = ( + A79D46F44C7BC55B9B595AEF /* CameraPreviewView.swift */, + ); + path = Views; + sourceTree = ""; + }; + 7E719691677D20AEAEF0C120 /* Coordinators */ = { + isa = PBXGroup; + children = ( + 48DFFB1E2F90176D156E315B /* AppSettingsHelper.swift */, + 3D0F925920743B648B04C949 /* CameraPermissionsManager.swift */, + FBD8FEBAE0069BE88B75B0C5 /* CameraSession.swift */, + BCCE205F4CCF2D486967BA25 /* MockSimulatorCameraSession.swift */, + D750C5508AB271C099351E42 /* Torch.swift */, + ); + path = Coordinators; + sourceTree = ""; + }; + BCB41D804D8A44F6DEEF4B8E /* Mocks */ = { + isa = PBXGroup; + children = ( + 9F04D387EEF3A5D07ECD1F7E /* MockAppSettingsHelper.swift */, + 832A750C892D9D3FC54D9CF3 /* MockCameraPermissionsManager.swift */, + 8091DB42296B7EC745567AF7 /* MockTestCameraSession.swift */, + ); + path = Mocks; + sourceTree = ""; + }; + CC0771AB4883EEDAB5AA7D1C /* Unit */ = { + isa = PBXGroup; + children = ( + 18A8A17329DE3C8891103ECD /* Categories */, + ); + path = Unit; + sourceTree = ""; + }; + E2B3561EEB5629A212FD2AB4 /* Products */ = { + isa = PBXGroup; + children = ( + 6DBC37790A9F68168A89095B /* StripeCameraCore.framework */, + 226FED2248EE3F96F217BEE9 /* StripeCameraCoreTests.xctest */, + 2720B7BD13954418A8AC52E5 /* StripeCameraCoreTestUtils.framework */, + 8E7BCD0896ADAFFF5CCA738E /* StripeCore.framework */, + ); + name = Products; + sourceTree = ""; + }; + E2CD33A048BB3FD49B4D47FD /* StripeCameraCore */ = { + isa = PBXGroup; + children = ( + 03256F1FDB473FE93A57A38A /* Source */, + 609D9ADB9E9898694A2A67FF /* Info.plist */, + E7FB32269B8E36C3CBDB370F /* StripeCameraCore.h */, + ); + path = StripeCameraCore; + sourceTree = ""; + }; + F72F2AFE9BE5E2CCBE7C6436 = { + isa = PBXGroup; + children = ( + 5CE9B5F8E73CBC18E22AF375 /* Project */, + 0FB9D3D4625E1AEA6EE1CC22 /* Frameworks */, + E2B3561EEB5629A212FD2AB4 /* Products */, + ); + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXHeadersBuildPhase section */ + 21D2298D6D8328B46AB437BB /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 6D44C32189C0CB409C9E4712 /* StripeCameraCore.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 8FED65A79644B7F1CB19619B /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 28F23B4322876ED2BAC8C933 /* StripeCameraCoreTestUtils.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXHeadersBuildPhase section */ + +/* Begin PBXNativeTarget section */ + 6A948E338F0BF55DBFD83170 /* StripeCameraCoreTestUtils */ = { + isa = PBXNativeTarget; + buildConfigurationList = 12D1B5D975CF0E6EC4956438 /* Build configuration list for PBXNativeTarget "StripeCameraCoreTestUtils" */; + buildPhases = ( + 8FED65A79644B7F1CB19619B /* Headers */, + F2FB0D7ACB63AFA0A81A5FF6 /* Sources */, + 83C726705BB615A9940B98CC /* Resources */, + BCA0A45A41520FB95F4E8D10 /* Embed Frameworks */, + 267776DB1189FACBD4FA2AEA /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + D1513DE143A5DDFAB1AA4069 /* PBXTargetDependency */, + ); + name = StripeCameraCoreTestUtils; + productName = StripeCameraCoreTestUtils; + productReference = 2720B7BD13954418A8AC52E5 /* StripeCameraCoreTestUtils.framework */; + productType = "com.apple.product-type.framework"; + }; + B50782C89D54809DFFCF80D0 /* StripeCameraCore */ = { + isa = PBXNativeTarget; + buildConfigurationList = 7056F053793C453C8750F440 /* Build configuration list for PBXNativeTarget "StripeCameraCore" */; + buildPhases = ( + 21D2298D6D8328B46AB437BB /* Headers */, + 62DE5865F3F7B8CB25F62395 /* Sources */, + 445990B4DBEECCD2FBA8B324 /* Resources */, + 2F2CDDE07BFFE88706C6A908 /* Embed Frameworks */, + E3A6C5C1A5D47CEFAFD1853F /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = StripeCameraCore; + productName = StripeCameraCore; + productReference = 6DBC37790A9F68168A89095B /* StripeCameraCore.framework */; + productType = "com.apple.product-type.framework"; + }; + F3DED9AB60FDAB786A737384 /* StripeCameraCoreTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 6BB26614346BD154B399D073 /* Build configuration list for PBXNativeTarget "StripeCameraCoreTests" */; + buildPhases = ( + 85B8B0A9CAD01C91AD19797A /* Sources */, + 98362E27FEF7FBD502400444 /* Resources */, + F97E2A90DA9F982FC293A4A9 /* Embed Frameworks */, + 9CC1AF0A3093D51C5085D503 /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + F102799B90F2BBF57EBDBBDC /* PBXTargetDependency */, + 24ECB0864CAE52D6ABF4503E /* PBXTargetDependency */, + ); + name = StripeCameraCoreTests; + productName = StripeCameraCoreTests; + productReference = 226FED2248EE3F96F217BEE9 /* StripeCameraCoreTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + B93DF1E4460962AFB28CF144 /* Project object */ = { + isa = PBXProject; + attributes = { + TargetAttributes = { + }; + }; + buildConfigurationList = C1216BCDB94950EBFA541D74 /* Build configuration list for PBXProject "StripeCameraCore" */; + compatibilityVersion = "Xcode 13.0"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + Base, + en, + ); + mainGroup = F72F2AFE9BE5E2CCBE7C6436; + productRefGroup = E2B3561EEB5629A212FD2AB4 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + B50782C89D54809DFFCF80D0 /* StripeCameraCore */, + 6A948E338F0BF55DBFD83170 /* StripeCameraCoreTestUtils */, + F3DED9AB60FDAB786A737384 /* StripeCameraCoreTests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 445990B4DBEECCD2FBA8B324 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 83C726705BB615A9940B98CC /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 98362E27FEF7FBD502400444 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 62DE5865F3F7B8CB25F62395 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 0485C4A62D9444E75877E170 /* CameraExifMetadata.swift in Sources */, + 48B09ECBED820F3097779208 /* CGRect+StripeCameraCore.swift in Sources */, + 2CEF8A27C8864A6198C69A5D /* CVPixelBuffer+StripeCameraCore.swift in Sources */, + F181EEBCE3D2CAD043DAF0F0 /* UIDeviceOrientation+StripeCameraCore.swift in Sources */, + D9687FAD52D1896F28309580 /* UIImage+Buffer.swift in Sources */, + 9FC42D068D1719AB28C3068F /* AppSettingsHelper.swift in Sources */, + 06FEEA450541E2D36D5A10FC /* CameraPermissionsManager.swift in Sources */, + A95255494DF27A7CAE9A70CE /* CameraSession.swift in Sources */, + 2A1AAD8E5A589E4E74B02A35 /* MockSimulatorCameraSession.swift in Sources */, + 0F02BB3F9F7936D7BB593226 /* Torch.swift in Sources */, + 9CE6EE572439AD5B22B77294 /* CameraPreviewView.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 85B8B0A9CAD01C91AD19797A /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + AB24784F49EB99E7C36509BD /* CGRect_StripeCameraCoreTest.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + F2FB0D7ACB63AFA0A81A5FF6 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 88A1AF2B173B4055D8F44A8C /* MockAppSettingsHelper.swift in Sources */, + 064567ECD2B71E5EB7D3A201 /* MockCameraPermissionsManager.swift in Sources */, + 8F3073987F30B9B8DDF0D956 /* MockTestCameraSession.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 24ECB0864CAE52D6ABF4503E /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeCameraCoreTestUtils; + target = 6A948E338F0BF55DBFD83170 /* StripeCameraCoreTestUtils */; + targetProxy = 55BD94DFEA7738F26C23A545 /* PBXContainerItemProxy */; + }; + D1513DE143A5DDFAB1AA4069 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeCameraCore; + target = B50782C89D54809DFFCF80D0 /* StripeCameraCore */; + targetProxy = 0321B7B71CC9BDDCF236B375 /* PBXContainerItemProxy */; + }; + F102799B90F2BBF57EBDBBDC /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeCameraCore; + target = B50782C89D54809DFFCF80D0 /* StripeCameraCore */; + targetProxy = BA0100EB31091CFA64BF12A6 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin XCBuildConfiguration section */ + 1555A494851A82726CF41684 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 17E9A614AF4543DB62E7F5B4 /* StripeiOS Tests-Release.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeCameraCoreTestUtils/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeCameraCoreTestUtils; + PRODUCT_NAME = StripeCameraCoreTestUtils; + SDKROOT = iphoneos; + }; + name = Release; + }; + 8101CFE889F8AC21AF12FC99 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 83596C8ED3829B17A93490A6 /* StripeiOS-Debug.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeCameraCore/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-camera-core"; + PRODUCT_NAME = StripeCameraCore; + SDKROOT = iphoneos; + }; + name = Debug; + }; + 85DD0786DE8D23550B2113B7 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = ABCCADF0E7337301A472E75C /* Project-Debug.xcconfig */; + buildSettings = { + }; + name = Debug; + }; + A6D92FFDFE11DCBEC32190D4 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 44E9A4900C4B8D08A617488C /* StripeiOS Tests-Debug.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeCameraCoreTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeCameraCoreTests; + PRODUCT_NAME = StripeCameraCoreTests; + SDKROOT = iphoneos; + }; + name = Debug; + }; + B26CBFD3B42072A398AE6605 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 17E9A614AF4543DB62E7F5B4 /* StripeiOS Tests-Release.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeCameraCoreTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeCameraCoreTests; + PRODUCT_NAME = StripeCameraCoreTests; + SDKROOT = iphoneos; + }; + name = Release; + }; + E1A07E117DC9E8051D871578 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 4E0D85BEF8C269A46D5ECEAC /* StripeiOS-Release.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeCameraCore/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-camera-core"; + PRODUCT_NAME = StripeCameraCore; + SDKROOT = iphoneos; + }; + name = Release; + }; + F11D6A5494AA72875ECC53A4 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 0F4174046507759A9668F4EE /* Project-Release.xcconfig */; + buildSettings = { + }; + name = Release; + }; + FD80773E1582BF61724D6EDE /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 44E9A4900C4B8D08A617488C /* StripeiOS Tests-Debug.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeCameraCoreTestUtils/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeCameraCoreTestUtils; + PRODUCT_NAME = StripeCameraCoreTestUtils; + SDKROOT = iphoneos; + }; + name = Debug; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 12D1B5D975CF0E6EC4956438 /* Build configuration list for PBXNativeTarget "StripeCameraCoreTestUtils" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + FD80773E1582BF61724D6EDE /* Debug */, + 1555A494851A82726CF41684 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 6BB26614346BD154B399D073 /* Build configuration list for PBXNativeTarget "StripeCameraCoreTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + A6D92FFDFE11DCBEC32190D4 /* Debug */, + B26CBFD3B42072A398AE6605 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 7056F053793C453C8750F440 /* Build configuration list for PBXNativeTarget "StripeCameraCore" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 8101CFE889F8AC21AF12FC99 /* Debug */, + E1A07E117DC9E8051D871578 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + C1216BCDB94950EBFA541D74 /* Build configuration list for PBXProject "StripeCameraCore" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 85DD0786DE8D23550B2113B7 /* Debug */, + F11D6A5494AA72875ECC53A4 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + }; + rootObject = B93DF1E4460962AFB28CF144 /* Project object */; +} diff --git a/StripeCameraCore/StripeCameraCore.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/StripeCameraCore/StripeCameraCore.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/StripeCameraCore/StripeCameraCore.xcodeproj/xcshareddata/xcschemes/StripeCameraCore.xcscheme b/StripeCameraCore/StripeCameraCore.xcodeproj/xcshareddata/xcschemes/StripeCameraCore.xcscheme new file mode 100644 index 00000000..d616e1ac --- /dev/null +++ b/StripeCameraCore/StripeCameraCore.xcodeproj/xcshareddata/xcschemes/StripeCameraCore.xcscheme @@ -0,0 +1,95 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/StripeCameraCore/StripeCameraCore/Info.plist b/StripeCameraCore/StripeCameraCore/Info.plist new file mode 100644 index 00000000..cd4a496b --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + $(CURRENT_PROJECT_VERSION) + CFBundleVersion + $(CURRENT_PROJECT_VERSION) + + diff --git a/StripeCameraCore/StripeCameraCore/Source/CameraExifMetadata.swift b/StripeCameraCore/StripeCameraCore/Source/CameraExifMetadata.swift new file mode 100644 index 00000000..4860f94b --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/CameraExifMetadata.swift @@ -0,0 +1,46 @@ +// +// CameraExifMetadata.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 4/14/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import CoreMedia +import Foundation +import ImageIO + +/// A helper to extract properties from an EXIF metadata dictionary +@_spi(STP) public struct CameraExifMetadata: Equatable { + public let brightnessValue: Double? + public let focalLength: Double? + public let lensModel: String? +} + +extension CameraExifMetadata { + public init?( + exifDictionary: [CFString: Any]? + ) { + guard let exifDictionary = exifDictionary else { + return nil + } + + self.init( + brightnessValue: exifDictionary[kCGImagePropertyExifBrightnessValue] as? Double, + focalLength: exifDictionary[kCGImagePropertyExifFocalLength] as? Double, + lensModel: exifDictionary[kCGImagePropertyExifLensModel] as? String + ) + } + + public init?( + sampleBuffer: CMSampleBuffer + ) { + self.init( + exifDictionary: CMGetAttachment( + sampleBuffer, + key: kCGImagePropertyExifDictionary, + attachmentModeOut: nil + ) as? [CFString: Any] + ) + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Categories/CGRect+StripeCameraCore.swift b/StripeCameraCore/StripeCameraCore/Source/Categories/CGRect+StripeCameraCore.swift new file mode 100644 index 00000000..29c5661b --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Categories/CGRect+StripeCameraCore.swift @@ -0,0 +1,80 @@ +// +// CGRect+StripeCameraCore.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 12/14/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import CoreGraphics +import Foundation + +@_spi(STP) extension CGRect { + + /// Represents the bounds of a normalized coordinate system with range from (0,0) to (1,1) + public static let normalizedBounds = CGRect(x: 0, y: 0, width: 1, height: 1) + + /// - Returns: A `CGRect` that has its y-coordinates inverted between the + /// upper-left corner and lower-left corner. + /// + /// - Note: + /// This should only be used for rects that are using a normalized + /// coordinate system, meaning that the coordinate of the corner opposite + /// origin is (1,1) + public var invertedNormalizedCoordinates: CGRect { + return CGRect( + x: minX, + y: 1 - minY - height, + width: width, + height: height + ) + } + + /// Converts a rectangle that's using a normalized coordinate system from a + /// center-crop coordinate system to an un-cropped coordinate system + /// + /// Example, if the original size has a portrait aspect ratio, center-cropping + /// the rect will result in the square area: + /// ``` + /// +---------+ + /// | | + /// |---------| + /// | | + /// | | + /// | | + /// |---------| + /// | | + /// +---------+ + /// ``` + /// + /// This method converts the rect's coordinate relative to the center-cropped + /// area into coordinates relative to the original un-cropped area: + /// ``` + /// +---------+ + /// | | + /// +---------+ | | + /// | +--+ | | +--+ | + /// | | | | --> | | | | + /// | +--+ | | +--+ | + /// +---------+ | | + /// | | + /// +---------+ + /// ``` + /// + /// - Parameters: + /// - size: The original size of the un-cropped area. + public func convertFromNormalizedCenterCropSquare( + toOriginalSize originalSize: CGSize + ) -> CGRect { + let croppedWidth = min(originalSize.width, originalSize.height) + let scaleX = croppedWidth / originalSize.width + let scaleY = croppedWidth / originalSize.height + + return CGRect( + x: (minX - 0.5) * scaleX + 0.5, + y: (minY - 0.5) * scaleY + 0.5, + width: width * scaleX, + height: height * scaleY + ) + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Categories/CVPixelBuffer+StripeCameraCore.swift b/StripeCameraCore/StripeCameraCore/Source/Categories/CVPixelBuffer+StripeCameraCore.swift new file mode 100644 index 00000000..6d147652 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Categories/CVPixelBuffer+StripeCameraCore.swift @@ -0,0 +1,18 @@ +// +// CVPixelBuffer+StripeCameraCore.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 3/16/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import CoreVideo +import VideoToolbox + +@_spi(STP) extension CVPixelBuffer { + public func cgImage() -> CGImage? { + var cgImage: CGImage? + VTCreateCGImageFromCVPixelBuffer(self, options: nil, imageOut: &cgImage) + return cgImage + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Categories/UIDeviceOrientation+StripeCameraCore.swift b/StripeCameraCore/StripeCameraCore/Source/Categories/UIDeviceOrientation+StripeCameraCore.swift new file mode 100644 index 00000000..e9bdbdf5 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Categories/UIDeviceOrientation+StripeCameraCore.swift @@ -0,0 +1,26 @@ +// +// UIDeviceOrientation+StripeCameraCore.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 1/20/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +import AVKit +import Foundation +import UIKit + +@_spi(STP) extension UIDeviceOrientation { + public var videoOrientation: AVCaptureVideoOrientation { + switch UIDevice.current.orientation { + case .portraitUpsideDown: + return .portraitUpsideDown + case .landscapeLeft: + return .landscapeRight + case .landscapeRight: + return .landscapeLeft + default: + return .portrait + } + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Categories/UIImage+Buffer.swift b/StripeCameraCore/StripeCameraCore/Source/Categories/UIImage+Buffer.swift new file mode 100644 index 00000000..9d13e713 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Categories/UIImage+Buffer.swift @@ -0,0 +1,118 @@ +// Ignoring file format becase of compiler directive which would force the whole file to be +// indented, including import statements. +// swift-format-ignore-file + +// Taken from https://gist.github.com/createwithswift/30a058c2745c8b09e64e7b073485e516 +// +// MIT License +// +// Copyright (c) 2021 Create with Swift +// +// 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. + +#if targetEnvironment(simulator) + +import CoreMedia +import UIKit + +extension UIImage { + @_spi(STP) public func convertToPixelBuffer() -> CVPixelBuffer? { + + let attributes = + [ + kCVPixelBufferCGImageCompatibilityKey: kCFBooleanTrue, + kCVPixelBufferCGBitmapContextCompatibilityKey: kCFBooleanTrue, + ] as CFDictionary + + var pixelBuffer: CVPixelBuffer? + + let status = CVPixelBufferCreate( + kCFAllocatorDefault, + Int(self.size.width), + Int(self.size.height), + kCVPixelFormatType_32ARGB, + attributes, + &pixelBuffer + ) + + guard status == kCVReturnSuccess else { + return nil + } + + CVPixelBufferLockBaseAddress(pixelBuffer!, CVPixelBufferLockFlags(rawValue: 0)) + + let pixelData = CVPixelBufferGetBaseAddress(pixelBuffer!) + let rgbColorSpace = CGColorSpaceCreateDeviceRGB() + + let context = CGContext( + data: pixelData, + width: Int(self.size.width), + height: Int(self.size.height), + bitsPerComponent: 8, + bytesPerRow: CVPixelBufferGetBytesPerRow(pixelBuffer!), + space: rgbColorSpace, + bitmapInfo: CGImageAlphaInfo.noneSkipFirst.rawValue + ) + + context?.translateBy(x: 0, y: self.size.height) + context?.scaleBy(x: 1.0, y: -1.0) + + UIGraphicsPushContext(context!) + self.draw(in: CGRect(x: 0, y: 0, width: self.size.width, height: self.size.height)) + UIGraphicsPopContext() + + CVPixelBufferUnlockBaseAddress(pixelBuffer!, CVPixelBufferLockFlags(rawValue: 0)) + + return pixelBuffer + } + + @_spi(STP) public func convertToSampleBuffer() -> CMSampleBuffer? { + guard let pixelBuffer = convertToPixelBuffer() else { + return nil + } + + var sampleBuffer: CMSampleBuffer? + var optionalVideoInfo: CMVideoFormatDescription? + var timingInfo: CMSampleTimingInfo = .invalid + + CMVideoFormatDescriptionCreateForImageBuffer( + allocator: nil, + imageBuffer: pixelBuffer, + formatDescriptionOut: &optionalVideoInfo + ) + guard let videoInfo = optionalVideoInfo else { + return nil + } + + CMSampleBufferCreateForImageBuffer( + allocator: kCFAllocatorDefault, + imageBuffer: pixelBuffer, + dataReady: true, + makeDataReadyCallback: nil, + refcon: nil, + formatDescription: videoInfo, + sampleTiming: &timingInfo, + sampleBufferOut: &sampleBuffer + ) + + return sampleBuffer + } +} + +#endif diff --git a/StripeCameraCore/StripeCameraCore/Source/Coordinators/AppSettingsHelper.swift b/StripeCameraCore/StripeCameraCore/Source/Coordinators/AppSettingsHelper.swift new file mode 100644 index 00000000..8168aef0 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Coordinators/AppSettingsHelper.swift @@ -0,0 +1,44 @@ +// +// AppSettingsHelper.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 12/3/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import Foundation +import UIKit + +public protocol AppSettingsHelperProtocol { + var canOpenAppSettings: Bool { get } + func openAppSettings() +} + +/// Helper class that opens the app's settings screen in the Settings app. +@available(iOSApplicationExtension, unavailable) +@_spi(STP) public class AppSettingsHelper: AppSettingsHelperProtocol { + + public static let shared = AppSettingsHelper() + + private(set) lazy var appSettingsUrl: URL? = URL(string: UIApplication.openSettingsURLString) + + private init() { + // Use shared instance instead of init + } + + /// `true` if the system is able to open the app's settings screen. + public var canOpenAppSettings: Bool { + guard let settingsUrl = appSettingsUrl else { + return false + } + return UIApplication.shared.canOpenURL(settingsUrl) + } + + /// Opens the app's settings screen, if possible. + public func openAppSettings() { + guard let settingsUrl = appSettingsUrl else { + return + } + UIApplication.shared.open(settingsUrl) + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraPermissionsManager.swift b/StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraPermissionsManager.swift new file mode 100644 index 00000000..36db983b --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraPermissionsManager.swift @@ -0,0 +1,85 @@ +// +// CameraPermissionsManager.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 12/3/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import AVKit +import Foundation + +@_spi(STP) public protocol CameraPermissionsManagerProtocol { + /// Completion block called when done requesting permissions. + /// + /// - Parameter granted: If camera permissions are granted for the app. + /// Value is `nil` if the authorization status cannot be determined. + typealias CompletionBlock = (_ granted: Bool?) -> Void + + var hasCameraAccess: Bool { get } + func requestCameraAccess( + completeOnQueue queue: DispatchQueue, + completion: @escaping CompletionBlock + ) +} + +/// Dependency-injectable class to assist with accessing and requesting camera authorization +@_spi(STP) public final class CameraPermissionsManager: CameraPermissionsManagerProtocol { + + public static let shared = CameraPermissionsManager() + + /// If this app currently has authorization to access the device's camera + public var hasCameraAccess: Bool { + return AVCaptureDevice.authorizationStatus(for: .video) == .authorized + } + + private init() { + // Use shared instance instead of init + } + + /// Requests camera permissions and calls completion block with result after retrieving them. + /// + /// - Parameters: + /// - completion: + /// - queue: DispatchQueue to complete the + /// + /// - Note: + /// If the user has already granted or denied camera permissions to the app, + /// this callback will respond immediately after `requestCameraAccess` is + /// called on the video feed and `showedPrompt` will be false. + /// + /// If the user has not yet granted or denied camera permissions to the app, + /// they will be prompted to do so. This callback will respond after the user + /// selects a response and `showedPrompt` will be true. + /// + public func requestCameraAccess( + completeOnQueue queue: DispatchQueue = .main, + completion: @escaping CompletionBlock + ) { + let wrappedCompletion: CompletionBlock = { granted in + queue.async { + completion(granted) + } + } + + switch AVCaptureDevice.authorizationStatus(for: .video) { + case .authorized: + wrappedCompletion(true) + + case .notDetermined: + AVCaptureDevice.requestAccess( + for: .video, + completionHandler: { granted in + wrappedCompletion(granted) + } + ) + + case .restricted, + .denied: + wrappedCompletion(false) + + default: + wrappedCompletion(nil) + } + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraSession.swift b/StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraSession.swift new file mode 100644 index 00000000..c8307c36 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Coordinators/CameraSession.swift @@ -0,0 +1,514 @@ +// +// CameraSession.swift +// StripeCameraCore +// +// Created by Jaime Park on 12/16/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import AVKit +@_spi(STP) import StripeCore + +@_spi(STP) @frozen public enum CameraSessionError: Error { + /// Can't find capture device to add + case captureDeviceNotFound + /// Session configuration has failed + case configurationFailed +} + +@_spi(STP) public protocol CameraSessionProtocol: AnyObject { + + var previewView: CameraPreviewView? { get set } + + func configureSession( + configuration: CameraSession.Configuration, + delegate: AVCaptureVideoDataOutputSampleBufferDelegate, + completeOn queue: DispatchQueue, + completion: @escaping (CameraSession.SetupResult) -> Void + ) + + func setVideoOrientation( + orientation: AVCaptureVideoOrientation + ) + + func toggleCamera( + to position: CameraSession.CameraPosition, + completeOn queue: DispatchQueue, + completion: @escaping (CameraSession.SetupResult) -> Void + ) + + func toggleTorch() + + func getCameraProperties() -> CameraSession.DeviceProperties? + + func startSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) + + func stopSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) +} + +@_spi(STP) public final class CameraSession: CameraSessionProtocol { + @frozen public enum SetupResult { + /// Session has successfully updated + case success + /// Session did not update due to an error + case failed(error: Error) + } + + public enum CameraPosition { + case front + case back + } + + public struct Configuration { + /// The initial position of camera: front or back + public let initialCameraPosition: CameraPosition + /// The initial video orientation of the camera session + public let initialOrientation: AVCaptureVideoOrientation + /// The capture device’s focus mode. + /// - Seealso: https://developer.apple.com/documentation/avfoundation/avcapturedevice/focusmode + public let focusMode: AVCaptureDevice.FocusMode? + /// The point of interest for focusing. + /// - Seealso: + /// https://developer.apple.com/documentation/avfoundation/avcapturedevice/focuspointofinterest + public let focusPointOfInterest: CGPoint? + /// A preset value of the quality of the capture session + public let sessionPreset: AVCaptureSession.Preset + /// Video settings for the video output + /// - Seealso: https://developer.apple.com/documentation/avfoundation/avcapturephotosettings/video_settings + public let outputSettings: [String: Any] + + /// - Parameters: + /// - initialCameraPosition: The initial position of camera: front or back + /// - initialOrientation: The initial video orientation of the camera session + /// - focusMode: The focus mode of the camera session + /// - focusPointOfInterest: The point of interest for focusing + /// - sessionPreset: A preset value of the quality of the capture session + /// - outputSettings: Video settings for the video output + public init( + initialCameraPosition: CameraPosition, + initialOrientation: AVCaptureVideoOrientation, + focusMode: AVCaptureDevice.FocusMode? = nil, + focusPointOfInterest: CGPoint? = nil, + sessionPreset: AVCaptureSession.Preset = .high, + outputSettings: [String: Any] = [:] + ) { + self.initialCameraPosition = initialCameraPosition + self.initialOrientation = initialOrientation + self.focusMode = focusMode + self.focusPointOfInterest = focusPointOfInterest + self.sessionPreset = sessionPreset + self.outputSettings = outputSettings + } + } + + public struct DeviceProperties: Equatable { + public let exposureDuration: CMTime + public let cameraDeviceType: AVCaptureDevice.DeviceType + public let isVirtualDevice: Bool? + public let lensPosition: Float + public let exposureISO: Float + public let isAdjustingFocus: Bool + } + + // MARK: - Properties + + public weak var previewView: CameraPreviewView? { + didSet { + guard oldValue !== previewView else { + return + } + + // Remove captureSession from previous view and add it to new one + oldValue?.setCaptureSession(nil, on: sessionQueue) + previewView?.setCaptureSession(session, on: sessionQueue) + } + } + + private let session: AVCaptureSession = AVCaptureSession() + private var captureConnection: AVCaptureConnection? + private let sessionQueue = DispatchQueue(label: "com.stripe.camera-session") + private var torchDevice: Torch? + private var setupResult: SetupResult? + + private var videoDeviceInput: AVCaptureDeviceInput? { + didSet { + if let videoDeviceInput = videoDeviceInput { + // Set torch with new capture input + self.torchDevice = Torch(device: videoDeviceInput.device) + } + } + } + + // MARK: - Public + + public init() { + // This is needed to expose init publicly + } + + /// Configures the camera session with the initial inputs and outputs. + /// + /// If the camera session has been configured already, then the configuration + /// is ignored and the previous setup result is passed to the completion block. + /// + /// - Parameters: + /// - configuration: Configuration settings for the session + /// - delegate: + /// - queue: DispatchQueue the completion block should be called on + /// - completion: A block executed when the session is done being configured + public func configureSession( + configuration: Configuration, + delegate: AVCaptureVideoDataOutputSampleBufferDelegate, + completeOn queue: DispatchQueue, + completion: @escaping (SetupResult) -> Void + ) { + sessionQueue.async { [weak self] in + guard let self = self else { return } + + // Check if already configured + if let setupResult = self.setupResult { + completion(setupResult) + return + } + + self.session.beginConfiguration() + self.session.sessionPreset = configuration.sessionPreset + self.session.commitConfiguration() + + self.configureSessionInput(with: configuration.initialCameraPosition).chained { + [weak self] _ -> Future in + guard let self = self else { + // If self has been deallocated before configuring output, return failure + let promise = Promise() + promise.reject(with: CameraSessionError.configurationFailed) + return promise + } + + return self.configureSessionOutput( + with: configuration.outputSettings, + orientation: configuration.initialOrientation, + focusMode: configuration.focusMode, + focusPointOfInterest: configuration.focusPointOfInterest, + delegate: delegate + ) + }.observe(on: queue) { [weak self] result in + self?.setupResult = result.setupResult + completion(result.setupResult) + } + } + } + + public func setFocus( + focusMode: AVCaptureDevice.FocusMode, + focusPointOfInterest: CGPoint? = nil, + completion: @escaping (Error?) -> Void + ) { + sessionQueue.async { [weak self] in + do { + try self?.setFocusOnCurrentQueue( + focusMode: focusMode, + focusPointOfInterest: focusPointOfInterest + ) + completion(nil) + } catch { + completion(error) + } + } + } + + /// Attempts to change the video orientation of both the session output + /// and the preview view layer. + /// + /// - Parameters: + /// - orientation: The desired video orientation + public func setVideoOrientation( + orientation: AVCaptureVideoOrientation + ) { + DispatchQueue.main.async { [weak self] in + guard let self = self else { return } + + self.captureConnection?.videoOrientation = orientation + self.previewView?.videoPreviewLayer.connection?.videoOrientation = orientation + } + } + + /// Returns the properties from the camera device. + /// + /// - Note: This method can only be called on the camera session thread, + /// meaning it's only meant to be called from the output delegate's + /// `captureOutput` method. + public func getCameraProperties() -> CameraSession.DeviceProperties? { + dispatchPrecondition(condition: .onQueue(sessionQueue)) + + guard let device = videoDeviceInput?.device else { + return nil + } + + var isVirtualDevice: Bool? + if #available(iOS 13, *) { + isVirtualDevice = device.isVirtualDevice + } + + return .init( + exposureDuration: device.exposureDuration, + cameraDeviceType: device.deviceType, + isVirtualDevice: isVirtualDevice, + lensPosition: device.lensPosition, + exposureISO: device.iso, + isAdjustingFocus: device.isAdjustingFocus + ) + } + + /// Attempts to switch camera input to a new camera position. + /// - Parameters: + /// - position: The camera position to toggle to + /// - queue: DispatchQueue the completion block should be called on + /// - completion: A block executed when the camera has finished toggling + public func toggleCamera( + to position: CameraPosition, + completeOn queue: DispatchQueue, + completion: @escaping (SetupResult) -> Void + ) { + configureSessionInput(with: position).observe(on: queue) { result in + completion(result.setupResult) + } + } + + /// Attempts to toggle the torch on or off. + public func toggleTorch() { + self.torchDevice?.toggle() + } + + /// Starts the camera session, calling a completion block when the session has + /// been started. + /// + /// - Parameters: + /// - queue: The queue to call the completion block on + /// - completion: A block executed when the session has been started + public func startSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) { + sessionQueue.async { [weak self] in + defer { + queue.async { + completion() + } + } + + guard let self = self, + case .success = self.setupResult + else { + return + } + + self.session.startRunning() + } + } + + /// Stop the camera session + public func stopSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) { + sessionQueue.async { [weak self] in + defer { + queue.async { + completion() + } + } + + guard let self = self, + case .success = self.setupResult + else { + return + } + + self.session.stopRunning() + } + } +} + +// MARK: - Private + +extension CameraSession { + fileprivate func configureSessionInput( + with position: CameraPosition + ) -> Future { + let promise = Promise() + + sessionQueue.async { [weak self] in + guard let self = self else { return } + + self.session.beginConfiguration() + defer { + self.session.commitConfiguration() + } + + do { + // Remove inputs + self.session.inputs.forEach { + self.session.removeInput($0) + } + + let newVideoDeviceInput = try self.captureDeviceInput(position: position) + + // Add video input + guard self.session.canAddInput(newVideoDeviceInput) + else { + promise.reject(with: CameraSessionError.configurationFailed) + return + } + self.session.addInput(newVideoDeviceInput) + + // Keep reference to video device input + self.videoDeviceInput = newVideoDeviceInput + + promise.resolve(with: ()) + } catch { + promise.reject(with: error) + } + } + + return promise + } + + fileprivate func configureSessionOutput( + with videoSettings: [String: Any], + orientation: AVCaptureVideoOrientation, + focusMode: AVCaptureDevice.FocusMode?, + focusPointOfInterest: CGPoint?, + delegate: AVCaptureVideoDataOutputSampleBufferDelegate + ) -> Future { + let promise = Promise() + + sessionQueue.async { [weak self] in + guard let self = self else { return } + + self.session.beginConfiguration() + defer { + self.session.commitConfiguration() + } + + let videoOutput = AVCaptureVideoDataOutput() + videoOutput.videoSettings = videoSettings + videoOutput.alwaysDiscardsLateVideoFrames = true + videoOutput.setSampleBufferDelegate(delegate, queue: self.sessionQueue) + + guard self.session.canAddOutput(videoOutput) else { + promise.reject(with: CameraSessionError.configurationFailed) + return + } + + // Add output to session + self.session.addOutput(videoOutput) + + // Update output connection reference + self.captureConnection = videoOutput.connection(with: .video) + + // Update new output and previewLayer orientation + self.setVideoOrientation(orientation: orientation) + + // Set focus if needed + guard let focusMode = focusMode else { + promise.resolve(with: ()) + return + } + + promise.fulfill { [weak self] in + try self?.setFocusOnCurrentQueue( + focusMode: focusMode, + focusPointOfInterest: focusPointOfInterest + ) + } + } + + return promise + } + + fileprivate func captureDeviceInput(position: CameraPosition) throws -> AVCaptureDeviceInput { + let captureDevices = AVCaptureDevice.DiscoverySession( + deviceTypes: position.captureDeviceTypes, + mediaType: .video, + position: position.captureDevicePosition + ) + + guard let captureDevice = captureDevices.devices.first else { + throw CameraSessionError.captureDeviceNotFound + } + + return try AVCaptureDeviceInput(device: captureDevice) + } + + fileprivate func setFocusOnCurrentQueue( + focusMode: AVCaptureDevice.FocusMode, + focusPointOfInterest: CGPoint? + ) throws { + dispatchPrecondition(condition: .onQueue(sessionQueue)) + + guard let device = videoDeviceInput?.device else { + return + } + + try device.lockForConfiguration() + if device.isFocusModeSupported(focusMode) { + device.focusMode = focusMode + } + + if let focusPointOfInterest = focusPointOfInterest, + device.isFocusPointOfInterestSupported + { + device.focusPointOfInterest = focusPointOfInterest + } + + device.unlockForConfiguration() + } +} + +// MARK: - CameraPosition + +extension CameraSession.CameraPosition { + /// Returns a list of camera devices, ordered by preferred device, for this + /// camera position. + var captureDeviceTypes: [AVCaptureDevice.DeviceType] { + switch self { + case .front: + return [.builtInTrueDepthCamera, .builtInWideAngleCamera] + + case .back: + if #available(iOS 13.0, *) { + return [.builtInDualCamera, .builtInDualWideCamera, .builtInWideAngleCamera] + } else { + return [.builtInDualCamera, .builtInWideAngleCamera] + } + } + } + + var captureDevicePosition: AVCaptureDevice.Position { + switch self { + case .front: + return .front + case .back: + return .back + } + } +} + +// MARK: - Result + +/// Helper to convert a Result into SetupResult +extension Result where Success == Void, Failure == Error { + fileprivate var setupResult: CameraSession.SetupResult { + switch self { + case .success: + return .success + case .failure(let error): + return .failed(error: error) + } + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Coordinators/MockSimulatorCameraSession.swift b/StripeCameraCore/StripeCameraCore/Source/Coordinators/MockSimulatorCameraSession.swift new file mode 100644 index 00000000..ef0bdf74 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Coordinators/MockSimulatorCameraSession.swift @@ -0,0 +1,220 @@ +// +// MockSimulatorCameraSession.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 1/21/22. +// Copyright © 2022 Stripe, Inc. All rights reserved. +// + +// Ignoring file formatt becase of compiler directive which would force the whole file to be +// indented, including import statements. +// swift-format-ignore-file + +#if targetEnvironment(simulator) + +import AVKit +import Foundation +@_spi(STP) import StripeCore + +/// Mocks a CameraSession on the simulator. +@_spi(STP) public final class MockSimulatorCameraSession: CameraSessionProtocol { + + enum Error: Swift.Error { + case sessionNotConfigured + case noImages + } + + static let mockSampleBufferTimeInterval: TimeInterval = 0.1 + + private let images: [UIImage] + private var nextImageToReturn: Int = 0 + private var currentImage: UIImage? + private let sessionQueue = DispatchQueue(label: "com.stripe.mock-simulator-camera-session") + private let videoOutput = AVCaptureVideoDataOutput() + private lazy var captureConnection = AVCaptureConnection( + inputPorts: [], + output: videoOutput + ) + private var mockSampleBufferTimer: Timer? + weak private var delegate: AVCaptureVideoDataOutputSampleBufferDelegate? + + // MARK: - Public + + private var isConfigured: Bool = false + private var cameraPosition: CameraSession.CameraPosition = .front + + public weak var previewView: CameraPreviewView? { + didSet { + setPreviewViewToCurrentImage() + } + } + + public init( + images: [UIImage] + ) { + self.images = images + } + + public func configureSession( + configuration: CameraSession.Configuration, + delegate: AVCaptureVideoDataOutputSampleBufferDelegate, + completeOn queue: DispatchQueue, + completion: @escaping (CameraSession.SetupResult) -> Void + ) { + sessionQueue.async { [weak self] in + let wrappedCompletion = { setupResult in + queue.async { + completion(setupResult) + } + } + + guard let self = self else { return } + self.delegate = delegate + + guard !self.images.isEmpty else { + self.isConfigured = false + wrappedCompletion(.failed(error: Error.noImages)) + return + } + + self.cameraPosition = configuration.initialCameraPosition + self.isConfigured = true + wrappedCompletion(.success) + } + } + + public func setVideoOrientation(orientation: AVCaptureVideoOrientation) { + // no-op + } + + public func toggleCamera( + to position: CameraSession.CameraPosition, + completeOn queue: DispatchQueue, + completion: @escaping (CameraSession.SetupResult) -> Void + ) { + sessionQueue.async { [weak self] in + guard let self = self else { return } + + let wrappedCompletion = { setupResult in + queue.async { + completion(setupResult) + } + } + + guard self.isConfigured else { + wrappedCompletion(.failed(error: Error.sessionNotConfigured)) + return + } + + self.cameraPosition = position + wrappedCompletion(.success) + } + } + + public func getCameraProperties() -> CameraSession.DeviceProperties? { + return .init( + exposureDuration: CMTime(value: 0, timescale: 1), + cameraDeviceType: .builtInDualCamera, + isVirtualDevice: nil, + lensPosition: 0, + exposureISO: 0, + isAdjustingFocus: false + ) + } + + public func toggleTorch() { + // no-op + } + + public func startSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) { + sessionQueue.async { [weak self] in + guard let self = self else { return } + + defer { + queue.async { + completion() + } + } + + if self.isConfigured && self.currentImage == nil { + self.currentImage = self.images.stp_boundSafeObject(at: self.nextImageToReturn) + self.setPreviewViewToCurrentImage() + } + + DispatchQueue.main.async { [weak self] in + guard let self = self else { return } + self.mockSampleBufferTimer = Timer.scheduledTimer( + timeInterval: MockSimulatorCameraSession.mockSampleBufferTimeInterval, + target: self, + selector: #selector(self.mockSampleBufferDelegateCallback), + userInfo: nil, + repeats: true + ) + } + } + } + + public func stopSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) { + sessionQueue.async { [weak self] in + guard let self = self else { return } + + defer { + queue.async { + completion() + } + } + + self.mockSampleBufferTimer?.invalidate() + + if self.currentImage != nil { + self.nextImageToReturn += 1 + } + self.currentImage = nil + } + } +} + +extension MockSimulatorCameraSession { + @objc fileprivate func mockSampleBufferDelegateCallback() { + sessionQueue.async { [weak self] in + guard let self = self, + var image = self.currentImage + else { + return + } + + // Flip image horizontally if mocking front-facing camera + if self.cameraPosition == .front, + let cgImage = image.cgImage + { + image = UIImage(cgImage: cgImage, scale: image.scale, orientation: .upMirrored) + } + + guard let sampleBuffer = image.convertToSampleBuffer() else { + return + } + + self.delegate?.captureOutput?( + self.videoOutput, + didOutput: sampleBuffer, + from: self.captureConnection + ) + } + } + + fileprivate func setPreviewViewToCurrentImage() { + DispatchQueue.main.async { [weak self, weak currentImage] in + guard let self = self else { return } + self.previewView?.layer.contents = currentImage?.cgImage + self.previewView?.layer.contentsGravity = .resizeAspectFill + } + } +} + +#endif diff --git a/StripeCameraCore/StripeCameraCore/Source/Coordinators/Torch.swift b/StripeCameraCore/StripeCameraCore/Source/Coordinators/Torch.swift new file mode 100644 index 00000000..caa5ef05 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Coordinators/Torch.swift @@ -0,0 +1,53 @@ +// +// Torch.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 12/1/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import AVFoundation +import Foundation + +struct Torch { + enum State { + case off + case on + } + let device: AVCaptureDevice? + var state: State + var lastStateChange: Date + var level: Float + + init( + device: AVCaptureDevice + ) { + self.state = .off + self.lastStateChange = Date() + if device.hasTorch { + self.device = device + if device.isTorchActive { + self.state = .on + } + } else { + self.device = nil + } + self.level = 1.0 + } + + mutating func toggle() { + self.state = self.state == .on ? .off : .on + do { + try self.device?.lockForConfiguration() + if self.state == .on { + try self.device?.setTorchModeOn(level: self.level) + } else { + self.device?.torchMode = .off + } + } catch { + // no-op + } + // Always unlock when we're done even if `setTorchModeOn` threw + self.device?.unlockForConfiguration() + } +} diff --git a/StripeCameraCore/StripeCameraCore/Source/Views/CameraPreviewView.swift b/StripeCameraCore/StripeCameraCore/Source/Views/CameraPreviewView.swift new file mode 100644 index 00000000..61a8ed97 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/Source/Views/CameraPreviewView.swift @@ -0,0 +1,81 @@ +// +// CameraPreviewView.swift +// StripeCameraCore +// +// Created by Mel Ludowise on 12/1/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import AVFoundation +import UIKit + +@_spi(STP) public class CameraPreviewView: UIView { + var videoPreviewLayer: AVCaptureVideoPreviewLayer { + return layer as! AVCaptureVideoPreviewLayer + } + + /// Camera session that configures the video feed for this view. + /// + /// - Note: + /// A session can only be used on one `PreviewView` at a time. Setting the same + /// session on a different `PreviewView` will remove it from the previous one. + public weak var session: CameraSessionProtocol? { + didSet { + guard oldValue !== session else { + return + } + oldValue?.previewView = nil + session?.previewView = self + } + } + + // MARK: Initialization + + public init() { + super.init(frame: .zero) + + videoPreviewLayer.videoGravity = .resizeAspectFill + } + + required init?( + coder aDecoder: NSCoder + ) { + super.init(coder: aDecoder) + } + + // MARK: UIView + + public override class var layerClass: AnyClass { + return AVCaptureVideoPreviewLayer.self + } + + // MARK: Internal + + /// Sets the video capture session in thread-safe way that doesn't block main + /// thread when configuring the session. + /// + /// - Parameters: + /// - captureSession: The capture session to set on this view + /// - cameraSessionQueue: The CameraSession's queue to configure the session + func setCaptureSession( + _ captureSession: AVCaptureSession?, + on cameraSessionQueue: DispatchQueue + ) { + // Get reference to videoPreviewLayer on main queue then dispatch to + // worker queue to set session so it doesn't block main. + + let mainWorkItem = DispatchWorkItem { [weak self] in + guard let self = self else { return } + cameraSessionQueue.async { + [weak captureSession, weak videoPreviewLayer = self.videoPreviewLayer] in + videoPreviewLayer?.session = captureSession + } + } + + if Thread.isMainThread { + mainWorkItem.perform() + } else { + DispatchQueue.main.async(execute: mainWorkItem) + } + } +} diff --git a/StripeCameraCore/StripeCameraCore/StripeCameraCore.h b/StripeCameraCore/StripeCameraCore/StripeCameraCore.h new file mode 100644 index 00000000..6cb45611 --- /dev/null +++ b/StripeCameraCore/StripeCameraCore/StripeCameraCore.h @@ -0,0 +1,18 @@ +// +// StripeCameraCore.h +// StripeCameraCore +// +// Created by Mel Ludowise on 11/15/21. +// + +#import + +//! Project version number for StripeCameraCore. +FOUNDATION_EXPORT double StripeCameraCoreVersionNumber; + +//! Project version string for StripeCameraCore. +FOUNDATION_EXPORT const unsigned char StripeCameraCoreVersionString[]; + +// In this header, you should import all the public headers of your framework using statements like #import + + diff --git a/StripeCameraCore/StripeCameraCoreTestUtils/Info.plist b/StripeCameraCore/StripeCameraCoreTestUtils/Info.plist new file mode 100644 index 00000000..9bcb2444 --- /dev/null +++ b/StripeCameraCore/StripeCameraCoreTestUtils/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + 1.0 + CFBundleVersion + $(CURRENT_PROJECT_VERSION) + + diff --git a/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockAppSettingsHelper.swift b/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockAppSettingsHelper.swift new file mode 100644 index 00000000..608e1d1f --- /dev/null +++ b/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockAppSettingsHelper.swift @@ -0,0 +1,21 @@ +// +// MockAppSettingsHelper.swift +// StripeCameraCoreTestUtils +// +// Created by Mel Ludowise on 12/3/21. +// + +import Foundation +@_spi(STP) import StripeCameraCore + +@_spi(STP) public class MockAppSettingsHelper: AppSettingsHelperProtocol { + + public var canOpenAppSettings = false + public private(set) var didOpenAppSettings = false + + public init() {} + + public func openAppSettings() { + didOpenAppSettings = true + } +} diff --git a/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockCameraPermissionsManager.swift b/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockCameraPermissionsManager.swift new file mode 100644 index 00000000..811451be --- /dev/null +++ b/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockCameraPermissionsManager.swift @@ -0,0 +1,41 @@ +// +// MockCameraPermissionsManager.swift +// StripeCameraCoreTestUtils +// +// Created by Mel Ludowise on 12/3/21. +// + +import Foundation +@_spi(STP) import StripeCameraCore +import XCTest + +@_spi(STP) public class MockCameraPermissionsManager: CameraPermissionsManagerProtocol { + + public var hasCameraAccess = false + + public private(set) var didRequestCameraAccess = false + public let didCompleteExpectation = XCTestExpectation( + description: "MockCameraPermissionsManager completion did finish" + ) + + private var completion: CompletionBlock = { _ in } + + public init() {} + + public func requestCameraAccess( + completeOnQueue queue: DispatchQueue, + completion: @escaping CompletionBlock + ) { + self.completion = { granted in + queue.async { [weak self] in + completion(granted) + self?.didCompleteExpectation.fulfill() + } + } + didRequestCameraAccess = true + } + + public func respondToRequest(granted: Bool?) { + completion(granted) + } +} diff --git a/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockTestCameraSession.swift b/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockTestCameraSession.swift new file mode 100644 index 00000000..5ba9bea0 --- /dev/null +++ b/StripeCameraCore/StripeCameraCoreTestUtils/Mocks/MockTestCameraSession.swift @@ -0,0 +1,180 @@ +// +// MockTestCameraSession.swift +// StripeCameraCoreTestUtils +// +// Created by Mel Ludowise on 1/21/22. +// + +import AVKit +import Foundation +@_spi(STP)@testable import StripeCameraCore +import XCTest + +@_spi(STP) public final class MockTestCameraSession: CameraSessionProtocol { + + /// Mock image to display in previewView. Should be used for snapshot tests + public var mockImage: UIImage? { + didSet { + setPreviewViewToMockImage() + } + } + + public var previewView: CameraPreviewView? { + didSet { + setPreviewViewToMockImage() + } + } + + public var mockDeviceProperties = CameraSession.DeviceProperties( + exposureDuration: CMTime(), + cameraDeviceType: .builtInDualCamera, + isVirtualDevice: nil, + lensPosition: 0, + exposureISO: 0, + isAdjustingFocus: false + ) + + public init() { + // no-op + } + + // MARK: configureSession + + private var configureCompletion: ((CameraSession.SetupResult) -> Void)? + public private(set) var configureSessionCompletionExp = XCTestExpectation( + description: "configureSession completion block called" + ) + + public private(set) var sessionPreset: AVCaptureSession.Preset? + public private(set) var outputSettings: [String: Any]? + + public func configureSession( + configuration: CameraSession.Configuration, + delegate: AVCaptureVideoDataOutputSampleBufferDelegate, + completeOn queue: DispatchQueue, + completion: @escaping (CameraSession.SetupResult) -> Void + ) { + cameraPosition = configuration.initialCameraPosition + videoOrientation = configuration.initialOrientation + sessionPreset = configuration.sessionPreset + outputSettings = configuration.outputSettings + configureCompletion = { setupResult in + queue.async { [weak self] in + self?.configureSessionCompletionExp.fulfill() + completion(setupResult) + } + } + } + + public func respondToConfigureSession(setupResult: CameraSession.SetupResult) { + configureCompletion?(setupResult) + } + + // MARK: setVideoOrientation + + public private(set) var videoOrientation: AVCaptureVideoOrientation? + + public func setVideoOrientation(orientation: AVCaptureVideoOrientation) { + self.videoOrientation = orientation + } + + // MARK: toggleCamera + + private var toggleCameraCompletion: ((CameraSession.SetupResult) -> Void)? + public private(set) var cameraPosition: CameraSession.CameraPosition? + + public func toggleCamera( + to position: CameraSession.CameraPosition, + completeOn queue: DispatchQueue, + completion: @escaping (CameraSession.SetupResult) -> Void + ) { + toggleCameraCompletion = { setupResult in + queue.async { + completion(setupResult) + } + } + cameraPosition = position + } + + public func respondToToggleCamera(setupResult: CameraSession.SetupResult) { + toggleCameraCompletion?(setupResult) + } + + // MARK: toggleTorch + + public private(set) var didToggleTorch = false + + public func toggleTorch() { + didToggleTorch = true + } + + // MARK: getCameraProperties + + public func getCameraProperties() -> CameraSession.DeviceProperties? { + return mockDeviceProperties + } + + // MARK: startSession + + private var startSessionCompletion: (() -> Void)? + public private(set) var startSessionCompletionExp = XCTestExpectation( + description: "startSession completion block called" + ) + + public func startSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) { + startSessionCompletion = { + queue.async { [weak self] in + self?.startSessionCompletionExp.fulfill() + completion() + } + } + } + + public func respondToStartSession() { + startSessionCompletion?() + } + + // MARK: stopSession + + private var stopSessionCompletion: (() -> Void)? + public private(set) var stopSessionCompletionExp = XCTestExpectation( + description: "stopSession completion block called" + ) + + public func stopSession( + completeOn queue: DispatchQueue, + completion: @escaping () -> Void + ) { + stopSessionCompletion = { + queue.async { [weak self] in + self?.stopSessionCompletionExp.fulfill() + completion() + } + } + } + + public func respondToStopSession() { + stopSessionCompletion?() + } + + // MARK: previewView + + func setPreviewViewToMockImage() { + let block = { [weak self] in + guard let self = self else { return } + self.previewView?.layer.contents = self.mockImage?.cgImage + self.previewView?.layer.contentsGravity = .resizeAspectFill + } + + // Only dispatch to main async if necessary so this can run + // synchronously for snapshot tests. + if Thread.isMainThread { + block() + } else { + DispatchQueue.main.async(execute: block) + } + } +} diff --git a/StripeCameraCore/StripeCameraCoreTestUtils/StripeCameraCoreTestUtils.h b/StripeCameraCore/StripeCameraCoreTestUtils/StripeCameraCoreTestUtils.h new file mode 100644 index 00000000..1f3c16b0 --- /dev/null +++ b/StripeCameraCore/StripeCameraCoreTestUtils/StripeCameraCoreTestUtils.h @@ -0,0 +1,18 @@ +// +// StripeCameraCoreTestUtils.h +// StripeCameraCoreTestUtils +// +// Created by Mel Ludowise on 11/15/21. +// + +#import + +//! Project version number for StripeCameraCoreTestUtils. +FOUNDATION_EXPORT double StripeCameraCoreTestUtilsVersionNumber; + +//! Project version string for StripeCameraCoreTestUtils. +FOUNDATION_EXPORT const unsigned char StripeCameraCoreTestUtilsVersionString[]; + +// In this header, you should import all the public headers of your framework using statements like #import + + diff --git a/StripeCameraCore/StripeCameraCoreTests/Info.plist b/StripeCameraCore/StripeCameraCoreTests/Info.plist new file mode 100644 index 00000000..64d65ca4 --- /dev/null +++ b/StripeCameraCore/StripeCameraCoreTests/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + 1.0 + CFBundleVersion + 1 + + diff --git a/StripeCameraCore/StripeCameraCoreTests/Unit/Categories/CGRect_StripeCameraCoreTest.swift b/StripeCameraCore/StripeCameraCoreTests/Unit/Categories/CGRect_StripeCameraCoreTest.swift new file mode 100644 index 00000000..317135ec --- /dev/null +++ b/StripeCameraCore/StripeCameraCoreTests/Unit/Categories/CGRect_StripeCameraCoreTest.swift @@ -0,0 +1,68 @@ +// +// CGRect_StripeCameraCoreTest.swift +// StripeCameraCoreTests +// +// Created by Mel Ludowise on 12/14/21. +// Copyright © 2021 Stripe, Inc. All rights reserved. +// + +import CoreGraphics +@_spi(STP) import StripeCameraCore +import XCTest + +class CGRect_StripeCameraCoreTest: XCTestCase { + func testInvertedNormalizedCoordinates() { + let rect = CGRect(x: 0.1, y: 0.1, width: 0.5, height: 0.5) + let invertedRect = rect.invertedNormalizedCoordinates + + XCTAssertEqual(invertedRect.minX, 0.1) + XCTAssertEqual(invertedRect.minY, 0.4) + XCTAssertEqual(invertedRect.width, 0.5) + XCTAssertEqual(invertedRect.height, 0.5) + } + + func testConvertFromNormalizedCenterCropSquareLandscape() { + // Centered-square rect corresponds to (350,0) 900x900 + let originalSize = CGSize(width: 1600, height: 900) + + // Corresponds to actual coordinates of + // (350+900*0.25,900*0.25) 900*0.25 x 900*0.25 + // = (575,225) 225x225 + let normalizedSquareRect = CGRect(x: 0.25, y: 0.25, width: 0.25, height: 0.25) + + // Should have coordinates of + // (575/1600, 225/900) 225/1600 x 225/900 + // = (0.359375,0.25) 0.140625 x 0.25 + let convertedRect = normalizedSquareRect.convertFromNormalizedCenterCropSquare( + toOriginalSize: originalSize + ) + + XCTAssertEqual(convertedRect.minX, 0.359375) + XCTAssertEqual(convertedRect.minY, 0.25) + XCTAssertEqual(convertedRect.width, 0.140625) + XCTAssertEqual(convertedRect.height, 0.25) + } + + func testConvertFromNormalizedCenterCropSquarePortrait() { + // Centered-square rect corresponds to (0,350) 900x900 + let originalSize = CGSize(width: 900, height: 1600) + + // Corresponds to actual coordinates of + // (900*0.25,350+900*0.25) 900*0.25 x 900*0.25 + // = (225,575) 225x225 + let normalizedSquareRect = CGRect(x: 0.25, y: 0.25, width: 0.25, height: 0.25) + + // Should have coordinates of + // (225/900, 575/1600) 225/900 x 225/1600 + // = (0.25,0.359375) 0.25 x 0.140625 + let convertedRect = normalizedSquareRect.convertFromNormalizedCenterCropSquare( + toOriginalSize: originalSize + ) + + XCTAssertEqual(convertedRect.minX, 0.25) + XCTAssertEqual(convertedRect.minY, 0.359375) + XCTAssertEqual(convertedRect.width, 0.25) + XCTAssertEqual(convertedRect.height, 0.140625) + } + +} diff --git a/StripeCardScan/BuildConfigurations/StripeCardScan-Debug.xcconfig b/StripeCardScan/BuildConfigurations/StripeCardScan-Debug.xcconfig new file mode 100644 index 00000000..0f5765ca --- /dev/null +++ b/StripeCardScan/BuildConfigurations/StripeCardScan-Debug.xcconfig @@ -0,0 +1,7 @@ +// +// StripeCardScan-Debug.xcconfig +// + +#include "../../BuildConfigurations/StripeiOS-Debug.xcconfig" + +APPLICATION_EXTENSION_API_ONLY = NO diff --git a/StripeCardScan/BuildConfigurations/StripeCardScan-Release.xcconfig b/StripeCardScan/BuildConfigurations/StripeCardScan-Release.xcconfig new file mode 100644 index 00000000..e99bbccc --- /dev/null +++ b/StripeCardScan/BuildConfigurations/StripeCardScan-Release.xcconfig @@ -0,0 +1,7 @@ +// +// StripeCardScan-Release.xcconfig +// + +#include "../../BuildConfigurations/StripeiOS-Release.xcconfig" + +APPLICATION_EXTENSION_API_ONLY = NO diff --git a/StripeCardScan/Project.swift b/StripeCardScan/Project.swift new file mode 100644 index 00000000..cf588d14 --- /dev/null +++ b/StripeCardScan/Project.swift @@ -0,0 +1,32 @@ +import ProjectDescription +import ProjectDescriptionHelpers + +let project = Project.stripeFramework( + name: "StripeCardScan", + targetSettings: .settings( + configurations: [ + .debug( + name: "Debug", + xcconfig: "BuildConfigurations/StripeCardScan-Debug.xcconfig" + ), + .release( + name: "Release", + xcconfig: "BuildConfigurations/StripeCardScan-Release.xcconfig" + ), + ], + defaultSettings: .none + ), + resources: "StripeCardScan/Resources/**", + dependencies: [ + .project(target: "StripeCore", path: "//StripeCore"), + ], + unitTestOptions: .testOptions( + resources: [ + "StripeCardScanTests/Mock Data/**", + "StripeCardScanTests/Resources/**", + ], + dependencies: [ + .project(target: "StripeCoreTestUtils", path: "//StripeCore"), + ] + ) +) diff --git a/StripeCardScan/README.md b/StripeCardScan/README.md new file mode 100644 index 00000000..f82d8a31 --- /dev/null +++ b/StripeCardScan/README.md @@ -0,0 +1,87 @@ +# Stripe CardScan iOS SDK + +This library provides support for the standalone Stripe CardScan product. + +## Overview + +This library provides a user interface through which users can scan payment cards and extract information from them. It uses the Stripe Publishable Key to authenticate with Stripe services. + +Note that this is a standalone SDK and, while compatible with, does not directly integrate with the [PaymentIntent](https://stripe.com/docs/api/payment_intents) API nor with [next_action](https://stripe.com/docs/api/errors#errors-payment_intent-next_action). + +This library can be used entirely outside of a Stripe integration and with other payment processing providers. + +## Requirements + +- iOS 13.0 or higher +- XCode 14.1 or higher + +## Example + +See the [CardImageVerification Example](https://github.com/stripe/stripe-ios/tree/master/Example/CardImageVerification%20Example) directory for an example application that you can try for yourself! + +## Installation + +- Cocoapod + - `pod install StripeCardScan` +- SPM + - In Xcode, select File > Add Packages… and enter https://github.com/stripe/stripe-ios-spm as the repository URL. + - Select the latest version number from our releases page. + - Add the `StripeCardScan` product to the target of your app. + +## Integration +### Credit Card OCR +Add `CardScanSheet` in your view controller where you want to invoke the credit card scanning flow. + +1. Set up camera permissions + * The SDK uses the camera, so you'll need to add an description of camera usage to your Info.plist file: +![info.plist camera permissions](https://gblobscdn.gitbook.com/assets%2F-MAfqrnL3d-uke0sAFsI%2Fsync%2F573e3f05043e4d903189b5fb107d4b3565bdb11b.png?alt=media) +![camera permissions prompt](https://gblobscdn.gitbook.com/assets%2F-MAfqrnL3d-uke0sAFsI%2Fsync%2F0d7119d3cbe2f519e5e5c04b56fe43539e4435e1.png?alt=media) + + * Alternatively, you can add this permission directly to your Info.plist file: + ``` + NSCameraUsageDescriptionkey> + We need access to your camera to scan your cardstring> + ``` +2. Add `CardScanSheet` in your app where you want to invoke the scan flow + * Initialize `CardScanSheet` + * When it’s time to invoke the scan flow, display the sheet with `CardScanSheet.present()` + * When the verification flow is finished, the sheet will be dismissed and the completion block will be called with a [Result](https://stripe.dev/stripe-ios/) + +### Example Implementation +```swift + +import UIKit +import StripeCardScan + +class ViewController: UIViewController { + + @IBAction func cardScanSheetButtonPressed() { + let cardScanSheet = CardScanSheet() + + cardScanSheet.present(from: self) { [weak self] result in + switch result { + case .completed(let scannedCard): + /* + * The user scanned a card. The result of the scan are detailed + * in the `scannedCard` field of the result. + */ + print("scan success") + case .canceled: + /* + * The scan was canceled by the user. + */ + print("scan canceled") + case .failed(let error): + /* + * The scan failed. The displayable error is + * included in the `localizedDescription`. + */ + print("scan failed: \(error.localizedDescription)") + } + } + } +} +``` + +## Credit Card Verification +🚧 diff --git a/StripeCardScan/StripeCardScan.xcodeproj/project.pbxproj b/StripeCardScan/StripeCardScan.xcodeproj/project.pbxproj new file mode 100644 index 00000000..0264da3a --- /dev/null +++ b/StripeCardScan/StripeCardScan.xcodeproj/project.pbxproj @@ -0,0 +1,1212 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 52; + objects = { + +/* Begin PBXBuildFile section */ + 040020B8558B0489DF99F8B5 /* DetectedAllBoxes.swift in Sources */ = {isa = PBXBuildFile; fileRef = A7EE5FDCA3B075FE5BBAF42A /* DetectedAllBoxes.swift */; }; + 050B462804602A9F4DB58A9D /* Expiry.swift in Sources */ = {isa = PBXBuildFile; fileRef = 96F3DDAD14A9F2DEB98E236E /* Expiry.swift */; }; + 05188062E522359CE24912CF /* ScannedCardImageData+Verification.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6C3EE6315FB1507E40F10D85 /* ScannedCardImageData+Verification.swift */; }; + 06711423AB563634EADFCCC0 /* VerifyCardViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = F912323832888F88EA218BA9 /* VerifyCardViewController.swift */; }; + 0838FEA7071EDCE1B633F093 /* PredictionAPI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4B452EE8B96AF1B0691E7D43 /* PredictionAPI.swift */; }; + 0A4DCE2C98659B92DDB29B9A /* ScanAnalyticsManager+Tasks.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6563E5CE5CD0439D5DBF35D7 /* ScanAnalyticsManager+Tasks.swift */; }; + 0ADD10809FEABDCB59A8FA72 /* CardType.swift in Sources */ = {isa = PBXBuildFile; fileRef = 236AE86CE7DCB0DFA3B7C978 /* CardType.swift */; }; + 0D062E99363099A5728F3DCD /* ScanAnalyticsManagerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 27F6ECCDC721E4D46DAC4F48 /* ScanAnalyticsManagerTests.swift */; }; + 0D62200AA65D71F040037CC8 /* CardImageVerification_CardAdd_200.json in Resources */ = {isa = PBXBuildFile; fileRef = CF90067085243D998A1D6030 /* CardImageVerification_CardAdd_200.json */; }; + 0D8CA0E3EFF9694CAE4FB8F7 /* ScanConfiguration.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF9618F7B15B33EC5B339A94 /* ScanConfiguration.swift */; }; + 0EAA2314FEA8D05D24C498BD /* Torch.swift in Sources */ = {isa = PBXBuildFile; fileRef = C827434739027B2DD43449EA /* Torch.swift */; }; + 10E840B00703A92CC487B324 /* CGRectExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7633166B4D9096FE995F08F0 /* CGRectExtension.swift */; }; + 164BF41B5C522B7338376BB2 /* BlurView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 938958FCB9B918AD6CDF0F4E /* BlurView.swift */; }; + 17E34C9EF882AEAE47BECAB4 /* ErrorCorrection.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6BEC906101069A1E3C50FA51 /* ErrorCorrection.swift */; }; + 189DEBAF38F2FB88CCA39EA2 /* UxModel+Utils.swift in Sources */ = {isa = PBXBuildFile; fileRef = CF3FCD1B2283A2B02EBAEF4F /* UxModel+Utils.swift */; }; + 18B63245A933E292345C9410 /* UxAndOcrMainLoop.swift in Sources */ = {isa = PBXBuildFile; fileRef = 87448BAE438EF08C55AA4B29 /* UxAndOcrMainLoop.swift */; }; + 19EF4634631CFC121DE21D1B /* CardImageVerificationDetailsResponseTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7DDAF8E33A3D7CD020AC904A /* CardImageVerificationDetailsResponseTest.swift */; }; + 19F7EC09C9B0ED11A4601E08 /* ScanStatsPayload.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2D61E754E520B4E21920D1DA /* ScanStatsPayload.swift */; }; + 1AD72764EA8BCB66A9D7B637 /* CreditCardOcrPrediction.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3DEE8C18A3CECFF0751AB0D5 /* CreditCardOcrPrediction.swift */; }; + 1B10222A5121C9B2C3479FAB /* StripeCardScan.h in Headers */ = {isa = PBXBuildFile; fileRef = AB56DDBBF0E5BC10682C7D96 /* StripeCardScan.h */; settings = {ATTRIBUTES = (Public, ); }; }; + 1B71815D5B3EC8AC21E2A202 /* UIImage+pixelBuffer.swift in Sources */ = {isa = PBXBuildFile; fileRef = B2D9886B0ED119E8D1FB6448 /* UIImage+pixelBuffer.swift */; }; + 1DE075A700592250D87DF558 /* ScannedCard.swift in Sources */ = {isa = PBXBuildFile; fileRef = 76C70F9A1879ED8B41639945 /* ScannedCard.swift */; }; + 20DCEB955D9E0660EAB3ABEB /* InterfaceOrientation.swift in Sources */ = {isa = PBXBuildFile; fileRef = 77D0D8C2909455B76E246468 /* InterfaceOrientation.swift */; }; + 20DF2E9D5A543360DF3A4DDA /* DetectedBox.swift in Sources */ = {isa = PBXBuildFile; fileRef = B53D407C0AC8D44ECC0C3C52 /* DetectedBox.swift */; }; + 213A91107E76434972A1F848 /* SSDOcr.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2F3E329C37141862BD4A4DE7 /* SSDOcr.swift */; }; + 25FAF64D4FE93BF38E807ECA /* VerifyFramesAPIBindingsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4301B281F5E0A5BACBBD68CC /* VerifyFramesAPIBindingsTests.swift */; }; + 2A9ECCF086685AD607D82492 /* AppleOcr.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3E24CF248F312DDE025D662A /* AppleOcr.swift */; }; + 2D042FD2E8A0C8236596CE54 /* CardScanFraudData.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7D41E45A285FEF48E9528997 /* CardScanFraudData.swift */; }; + 2F3FA5E8CCBF7F77106A8268 /* EndToEndTestDataSource.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA3ECBF071E7507C9D41F232 /* EndToEndTestDataSource.swift */; }; + 30E3E90F9C8E3D3E8FCA869E /* PreviewView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DC12B6E0657203AAB765468C /* PreviewView.swift */; }; + 35F2BB10395405DB6C1E31B5 /* SSDOcrOutputExtensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = EA89DA7FF10BC3A7B2D102FB /* SSDOcrOutputExtensions.swift */; }; + 36525A0F774E7FA055D8B525 /* CardBase.swift in Sources */ = {isa = PBXBuildFile; fileRef = BEDE71D7051DDC98698C4057 /* CardBase.swift */; }; + 3736756FBB875060C86E2777 /* CardScanSheet.swift in Sources */ = {isa = PBXBuildFile; fileRef = 14BC8B1C28C3F6C8330164E0 /* CardScanSheet.swift */; }; + 3F5B9466E9FC13CA29218750 /* ImageHelpers.swift in Sources */ = {isa = PBXBuildFile; fileRef = 091163ED5E37922332F502B1 /* ImageHelpers.swift */; }; + 3FCC183583090FD0246D9336 /* ScanBaseViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = A790FC6FAEC3F88FB2C26E27 /* ScanBaseViewController.swift */; }; + 41366AB64512BB7E4248A904 /* OcrDDUtils.swift in Sources */ = {isa = PBXBuildFile; fileRef = BC0B91384C63CBFB790C8EE6 /* OcrDDUtils.swift */; }; + 41F2E4475B9B19350C31F255 /* PaymentCard.swift in Sources */ = {isa = PBXBuildFile; fileRef = 67C8AB382B81DE27C8614F6D /* PaymentCard.swift */; }; + 420FF119A7147335D802AB14 /* CardImageVerificationSheet.swift in Sources */ = {isa = PBXBuildFile; fileRef = 12F09D21BAC44C461B1AD7CB /* CardImageVerificationSheet.swift */; }; + 43AA83BFF8DEFC09D406669F /* CardVerifyStateMachine.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9B3242DC2FCDF20D51D23AE1 /* CardVerifyStateMachine.swift */; }; + 449DD2A5D1F3FF94524D3CD6 /* STPAPIClient+CardImageVerificationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3BC4F647EB512813078B9A6F /* STPAPIClient+CardImageVerificationTest.swift */; }; + 45C8D17FED02529B7FC4E8C3 /* STPLocalizedString.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6418F87D5168A2A08D65A482 /* STPLocalizedString.swift */; }; + 47DCE237223D40B1EB183704 /* AppState.swift in Sources */ = {isa = PBXBuildFile; fileRef = DCA2733CEC85F8E427CA99CB /* AppState.swift */; }; + 48580A7E92CC8EBFD2FD4C91 /* DetectedAllOcrBoxes.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1441A7A39C36C2A634A882F5 /* DetectedAllOcrBoxes.swift */; }; + 48E4577B073DD9485BC62902 /* ImageCompressionTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0AA128D69F3C32A4EF67CF0 /* ImageCompressionTests.swift */; }; + 49E1959C680AD68D1D345D6B /* DeviceUtils.swift in Sources */ = {isa = PBXBuildFile; fileRef = CDB03A9253E8FAD07B739438 /* DeviceUtils.swift */; }; + 4ACBD9754F304CE4F70CAE43 /* MainLoopStateMachine.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7D550382D6320388A5BA75A1 /* MainLoopStateMachine.swift */; }; + 4B01B9F4FB647908AA5276E2 /* StripeCoreTestUtils.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 17FA90EE37CF1E6A49532491 /* StripeCoreTestUtils.framework */; }; + 4D1016654AFB3D9EE26092B7 /* PredictionUtilOcr.swift in Sources */ = {isa = PBXBuildFile; fileRef = 795E739553651C8AC40E6AC1 /* PredictionUtilOcr.swift */; }; + 52519CA3928967768049164E /* CardImageVerification_CardSet_200.json in Resources */ = {isa = PBXBuildFile; fileRef = BDD0C14472FA163BD395DBD3 /* CardImageVerification_CardSet_200.json */; }; + 5382B471868AA7D95BBD2B3A /* SSDOcrDetect.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8AEB912151CB564AE21D073C /* SSDOcrDetect.swift */; }; + 54469A1A5D77BAD54061BEA1 /* ScanStatsPayload+Tasks.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4A1CD398A6AFB6747D095C59 /* ScanStatsPayload+Tasks.swift */; }; + 5488DA1817A0C9F780EF69B2 /* CardVerifyFraudData.swift in Sources */ = {isa = PBXBuildFile; fileRef = 63FA7635BAFDC7DD54B1A731 /* CardVerifyFraudData.swift */; }; + 59889C85D6D25B3222776B28 /* synthetic_test_image.jpg in Resources */ = {isa = PBXBuildFile; fileRef = 4D9820BC2FFB68C808FB9507 /* synthetic_test_image.jpg */; }; + 5B1749A19231B20E2A081EB9 /* OcrObject.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0DC83A169A82484B9F60E452 /* OcrObject.swift */; }; + 60AF52B9EFFF12EBABE4D221 /* CreditCardOcrImplementation.swift in Sources */ = {isa = PBXBuildFile; fileRef = 05B28BBCB062B6AE5A3B7CBC /* CreditCardOcrImplementation.swift */; }; + 65C1EB5447CD5DF162CDEC2B /* Bouncer.swift in Sources */ = {isa = PBXBuildFile; fileRef = CA16EAC68E04D4FDE67DB807 /* Bouncer.swift */; }; + 6E9908C612ECD2E0AEA3DFF8 /* AtomicPropertyWrapper.swift in Sources */ = {isa = PBXBuildFile; fileRef = D9B548B9626366BB0F110873 /* AtomicPropertyWrapper.swift */; }; + 70B5FF75EF5DC8FFB23B95F7 /* UxModelTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = C038EFADF52B2477FC934EE0 /* UxModelTests.swift */; }; + 74330F51A91DC3A617F7AE42 /* SimpleScanViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = A6E1DA01CB00CBDE36735EDD /* SimpleScanViewController.swift */; }; + 77443628E02F3AEFF112314D /* CardScanMockData.swift in Sources */ = {isa = PBXBuildFile; fileRef = 746AAB9327BDE6791F9393B0 /* CardScanMockData.swift */; }; + 77BB4BE6E5E03626936453C6 /* STPAPIClient+CardImageVerification.swift in Sources */ = {isa = PBXBuildFile; fileRef = 84E54DFA3C47D94639297588 /* STPAPIClient+CardImageVerification.swift */; }; + 79A96C88970E4E61BAC47412 /* ScanStatsPayloadAPIBindingsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5CF9ADC4CC0B4ADFE0A5427E /* ScanStatsPayloadAPIBindingsTests.swift */; }; + 7B0CF214778E17FEC721602E /* PredictionResult.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4BBE2CADB8D8E98D6EEE10C0 /* PredictionResult.swift */; }; + 7C71166DACDB829F8CBC383D /* NMS.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACD9FF5C9DD64BE92DB0927D /* NMS.swift */; }; + 7CED1B42C94A49D6BF98C9E4 /* VideoFeed.swift in Sources */ = {isa = PBXBuildFile; fileRef = D42510E09D32547455CDDBEF /* VideoFeed.swift */; }; + 7D9C9C2A26EF11F0C5D67D7C /* CGrect+utils.swift in Sources */ = {isa = PBXBuildFile; fileRef = BD87F9971C25C4B99138318F /* CGrect+utils.swift */; }; + 8463F2DB039C481D26A24BAA /* PostDetectionAlgorithm.swift in Sources */ = {isa = PBXBuildFile; fileRef = AC9F1B20AEB433DE3CE0042C /* PostDetectionAlgorithm.swift */; }; + 84975E58102A5D8C62C6C4E5 /* String+Localized.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69A2DB0837C11D8829C027CF /* String+Localized.swift */; }; + 858CDA751309CA04202B6203 /* MachineLearningResult.swift in Sources */ = {isa = PBXBuildFile; fileRef = AF17FD18E52CB3508AD895E3 /* MachineLearningResult.swift */; }; + 86635536450EE7D583B8BD59 /* StripeCore+Import.swift in Sources */ = {isa = PBXBuildFile; fileRef = 64A1E35C6DF336F34B6C7AEA /* StripeCore+Import.swift */; }; + 8CF112F4889C96617504A931 /* Localizable.strings in Resources */ = {isa = PBXBuildFile; fileRef = 6C27E58C1953533C68D43361 /* Localizable.strings */; }; + 8E4B117272F5B17A1E6AD609 /* UxModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = DDF173627F545C88CB1551B0 /* UxModel.swift */; }; + 8FC373E15C1169677F563901 /* OcrPriorsGen.swift in Sources */ = {isa = PBXBuildFile; fileRef = ED780E8B4C14EE39F853B3A1 /* OcrPriorsGen.swift */; }; + 9188179F13E5EA15E0419580 /* ScanStats.swift in Sources */ = {isa = PBXBuildFile; fileRef = 66A49708CF5D37C7CB267732 /* ScanStats.swift */; }; + 93D4785D8B91F12B6DDDE203 /* UxAnalyzer.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4F85031ADAE5928D977465C2 /* UxAnalyzer.swift */; }; + 9FA5A312032FC54B35BB604E /* SoftNMS.swift in Sources */ = {isa = PBXBuildFile; fileRef = 00A93BFDEE4072A2A9C30397 /* SoftNMS.swift */; }; + A060350B93E3369463AE2898 /* StripeCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 2E889051D7CD76D4FB9084A1 /* StripeCore.framework */; }; + A7014CF97250D97BC683FAC8 /* CreditCardUtils.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8735A24B634F3B9060CADE45 /* CreditCardUtils.swift */; }; + A7F92FB7213E66A7D15A1BE3 /* SSDOcr.mlmodelc in Resources */ = {isa = PBXBuildFile; fileRef = E73DC7DD9C3E3395DF3F5A29 /* SSDOcr.mlmodelc */; }; + A8D0A57687A7CF9F389D7BDB /* ScanAnalyticsManager+Managers.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1A337FE5C3BE4FAB001E5520 /* ScanAnalyticsManager+Managers.swift */; }; + AB21982DC43977754F237756 /* VerificationFramesData.swift in Sources */ = {isa = PBXBuildFile; fileRef = 24C2345660A95FBF90BF4850 /* VerificationFramesData.swift */; }; + ACC3C1A295E43983F67F858E /* FrameData.swift in Sources */ = {isa = PBXBuildFile; fileRef = A3C926149A434F4F3BC961D9 /* FrameData.swift */; }; + AE144927F21D5DF9A8C0EF79 /* ScannedCardDetails.swift in Sources */ = {isa = PBXBuildFile; fileRef = 15541EEEE784EECDD3E7399D /* ScannedCardDetails.swift */; }; + AFA334F5007A4C141A96FC2E /* VerifyFrames.swift in Sources */ = {isa = PBXBuildFile; fileRef = E4FAA99A64BCD00A336A5E01 /* VerifyFrames.swift */; }; + AFB2482D559BCC08E96C3615 /* CardImageVerificationIntent.swift in Sources */ = {isa = PBXBuildFile; fileRef = CD5004FC0706B2660B4C6EA0 /* CardImageVerificationIntent.swift */; }; + B00954CF5F2A15B72CB94FA5 /* VerifyCardAddViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69B38020E8E33CA893CBA6F3 /* VerifyCardAddViewController.swift */; }; + B2DF7862E5F80ACD8DEE9317 /* UxModel.mlmodelc in Resources */ = {isa = PBXBuildFile; fileRef = F4721524B6285C507681CC4D /* UxModel.mlmodelc */; }; + B47F583CE0F239881877E5D3 /* ZoomedInCGImage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 10A26712A38A91726C314329 /* ZoomedInCGImage.swift */; }; + B66697826FEA9FCE81DDD6F4 /* ActiveStateComputation.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6EBC1F86A03CF9AAC6FE5825 /* ActiveStateComputation.swift */; }; + BDEA39DE5D3775AD2A4BFB6C /* CardImageVerificationControllerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F24DEA029F94A359B7A571BB /* CardImageVerificationControllerTests.swift */; }; + BEE2BFA103DE985D057A0F9D /* ScanStatsPayload+Common.swift in Sources */ = {isa = PBXBuildFile; fileRef = 028DA95BD9A0DA8AD69162CB /* ScanStatsPayload+Common.swift */; }; + BFFA3E8CE79BFFE47530FAF6 /* DetectedSSDBox.swift in Sources */ = {isa = PBXBuildFile; fileRef = A2EF6799FCE0817D9E3A536B /* DetectedSSDBox.swift */; }; + C2C41F5E568BF767C8232E74 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 14E5A51A77928A2700E23321 /* XCTest.framework */; }; + C3B9A4A30443A38C2D17BE8E /* CardNetwork.swift in Sources */ = {isa = PBXBuildFile; fileRef = 311A5A7EF789248E35BB1FEC /* CardNetwork.swift */; }; + C57BCF07EDF419D36ED4E690 /* ScanEventsProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0D1FBA1B4676CC5E9F785D51 /* ScanEventsProtocol.swift */; }; + C5DBC36A41FB70C5DCCC97C7 /* SimpleScanViewController+Verify.swift in Sources */ = {isa = PBXBuildFile; fileRef = 34F806DF921888A9657E2928 /* SimpleScanViewController+Verify.swift */; }; + C633115B02A0CE24AC55A747 /* OcrDD.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1FD9AF72B522A1E535A4EC81 /* OcrDD.swift */; }; + C7259DA76E6AB9BF53735D19 /* Array+utils.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8CA457DF27DC45455B8E1345 /* Array+utils.swift */; }; + C7A941539DBCFA790DCDB0B0 /* NonNameWords.swift in Sources */ = {isa = PBXBuildFile; fileRef = F96A85E323CFDA8BA35BC293 /* NonNameWords.swift */; }; + C8E2E98B7ED108804E0A0E24 /* SSDCreditCardOcr.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0B00395C6A2B14A6D56735B9 /* SSDCreditCardOcr.swift */; }; + CBA4FE649A3B21DAC3A61E5B /* CancellationReason.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53C171A1336B7FB0C321D194 /* CancellationReason.swift */; }; + CC07F702B9EC043ACB0AC1E5 /* ScannedCardImageData.swift in Sources */ = {isa = PBXBuildFile; fileRef = BAE45D8AC7BBA5DC433A9383 /* ScannedCardImageData.swift */; }; + CCB7722FDB8E6E68E90F3D12 /* CreditCardOcrPrediction+expiry.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5E72DD18B9FCB80ABEDE2689 /* CreditCardOcrPrediction+expiry.swift */; }; + CED42084C5FC46C17E357AD5 /* ScanAnalyticsManager+Helpers.swift in Sources */ = {isa = PBXBuildFile; fileRef = B97CA4D8286B196A49B8D535 /* ScanAnalyticsManager+Helpers.swift */; }; + D2030CA1B3B7DAF3229189AD /* Image+utils.swift in Sources */ = {isa = PBXBuildFile; fileRef = AEE9C295E982F92DE2418AE1 /* Image+utils.swift */; }; + D27E02429E5DC8B28DFE8945 /* CardImageVerificationDetailsResponse.swift in Sources */ = {isa = PBXBuildFile; fileRef = 256455DAEDCDDB1AEAF6A278 /* CardImageVerificationDetailsResponse.swift */; }; + D42EEFBB72CD0AB00C6B4728 /* String+Sha256.swift in Sources */ = {isa = PBXBuildFile; fileRef = 732D3AB4CE25CC5C26A352B1 /* String+Sha256.swift */; }; + D6F0E1C93997BB36AF0608C0 /* StringResourceTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5A72927DA8A946FDD03843DE /* StringResourceTests.swift */; }; + DB74787AF4EEF0E506DD0E1C /* Data+Sha256.swift in Sources */ = {isa = PBXBuildFile; fileRef = 63662689AD097C859B8759B1 /* Data+Sha256.swift */; }; + DBE3EE5DAEEEA44D6C7ECD4E /* SSDOcr+Utils.swift in Sources */ = {isa = PBXBuildFile; fileRef = 505DF34D76A56FEC2B8453F9 /* SSDOcr+Utils.swift */; }; + DDBCE05A3794129A9A04D511 /* AppleCreditCardOcr.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8EDF39435211B456B7579739 /* AppleCreditCardOcr.swift */; }; + DF352D12199B55B659A11795 /* DetectedSSDOcrBox.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5D58BEEC9D88537E32260CBA /* DetectedSSDOcrBox.swift */; }; + E43FE03D43CB0D7BAA73745A /* StripeCardScan.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = DD6884D0B3347BBF2E61B1D6 /* StripeCardScan.framework */; }; + E4EC278027AF802C39C74C48 /* CardImageVerificationController.swift in Sources */ = {isa = PBXBuildFile; fileRef = B313BA373EF6B567BF40A240 /* CardImageVerificationController.swift */; }; + E70A7E38A7D858031F900649 /* CardScanSheetError.swift in Sources */ = {isa = PBXBuildFile; fileRef = DCF4C0BF8E55D53B004FABCA /* CardScanSheetError.swift */; }; + E90CC9715EC99F6B7FAC98FA /* CardImageVerificationSheetConfiguration.swift in Sources */ = {isa = PBXBuildFile; fileRef = 545E03D5167516D28C8A9F08 /* CardImageVerificationSheetConfiguration.swift */; }; + EA0F179DF887D94E19138A5E /* AsyncModelLoading.swift in Sources */ = {isa = PBXBuildFile; fileRef = EF53C194DDC96160505D54D5 /* AsyncModelLoading.swift */; }; + EA2DBA78722CD65ED599190D /* CardScanMisc.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4CA3AD0FDF25BE452AE4ED34 /* CardScanMisc.swift */; }; + EB061FA550AFFE2AB2E348DF /* ScanAnalyticsManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69E2630C0FC7C9DDD551FD67 /* ScanAnalyticsManager.swift */; }; + EE1CB7D3FA2B44DC17E6FF4C /* OcrMainLoop.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7E38717F47ED06C115E058DE /* OcrMainLoop.swift */; }; + EF96103F82491640651C49F6 /* AppInfoUtils.swift in Sources */ = {isa = PBXBuildFile; fileRef = 362A27BB81840A22104C9A49 /* AppInfoUtils.swift */; }; + F0ED2BFE143DD07337D389E5 /* CornerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = C3E012270975B6DC09A8199F /* CornerView.swift */; }; + F16FAC158C6B0A7687547B4B /* FadeInAnimation.swift in Sources */ = {isa = PBXBuildFile; fileRef = B3CAFFFE55918BB4939868B6 /* FadeInAnimation.swift */; }; + F4E4941F5AA2EC2C2CC4F7FB /* StripeCardScanBundleLocator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 06F97A4CE5104309B720D603 /* StripeCardScanBundleLocator.swift */; }; + F9F19A6F2245863FEA485335 /* CreditCardOcrResult.swift in Sources */ = {isa = PBXBuildFile; fileRef = EDE92803C69CE8253D73A25F /* CreditCardOcrResult.swift */; }; + FE55286899CFB0E34356D4D6 /* StrictModeFramesTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = CD6D4DB87B26D439686392AE /* StrictModeFramesTest.swift */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + F7C4E731844D6B46A4FBCACD /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 9A6BF50E5B02355004AC6020 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 03CF79975A56288F02F20E52; + remoteInfo = StripeCardScan; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 20B092D23CA44561EF997447 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; + 451E97BA25D9CE29EDAB7618 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 00A93BFDEE4072A2A9C30397 /* SoftNMS.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SoftNMS.swift; sourceTree = ""; }; + 0197A1615C3FA86907CB6186 /* StripeCardScan-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeCardScan-Debug.xcconfig"; sourceTree = ""; }; + 028DA95BD9A0DA8AD69162CB /* ScanStatsPayload+Common.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScanStatsPayload+Common.swift"; sourceTree = ""; }; + 03E2C12C0D172DD7DAE33399 /* fi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fi; path = fi.lproj/Localizable.strings; sourceTree = ""; }; + 048ED08A6959593F896B3AC8 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 05B28BBCB062B6AE5A3B7CBC /* CreditCardOcrImplementation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CreditCardOcrImplementation.swift; sourceTree = ""; }; + 06F97A4CE5104309B720D603 /* StripeCardScanBundleLocator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StripeCardScanBundleLocator.swift; sourceTree = ""; }; + 091163ED5E37922332F502B1 /* ImageHelpers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageHelpers.swift; sourceTree = ""; }; + 0B00395C6A2B14A6D56735B9 /* SSDCreditCardOcr.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SSDCreditCardOcr.swift; sourceTree = ""; }; + 0BB35800FB840EFC76240DC7 /* vi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = vi; path = vi.lproj/Localizable.strings; sourceTree = ""; }; + 0C7BAE904C7E1D7A9FDF35F1 /* hu */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = hu; path = hu.lproj/Localizable.strings; sourceTree = ""; }; + 0D1FBA1B4676CC5E9F785D51 /* ScanEventsProtocol.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanEventsProtocol.swift; sourceTree = ""; }; + 0DC83A169A82484B9F60E452 /* OcrObject.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OcrObject.swift; sourceTree = ""; }; + 10311DC6AC19B7A73A8F930F /* id */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = id; path = id.lproj/Localizable.strings; sourceTree = ""; }; + 10A26712A38A91726C314329 /* ZoomedInCGImage.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ZoomedInCGImage.swift; sourceTree = ""; }; + 11D3465E527ABADBD2485A93 /* ms-MY */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ms-MY"; path = "ms-MY.lproj/Localizable.strings"; sourceTree = ""; }; + 12F09D21BAC44C461B1AD7CB /* CardImageVerificationSheet.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardImageVerificationSheet.swift; sourceTree = ""; }; + 1441A7A39C36C2A634A882F5 /* DetectedAllOcrBoxes.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DetectedAllOcrBoxes.swift; sourceTree = ""; }; + 14795A45279E8F328AB54920 /* Project-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Debug.xcconfig"; sourceTree = ""; }; + 14BC8B1C28C3F6C8330164E0 /* CardScanSheet.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardScanSheet.swift; sourceTree = ""; }; + 14E5A51A77928A2700E23321 /* XCTest.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = XCTest.framework; path = Platforms/iPhoneOS.platform/Developer/Library/Frameworks/XCTest.framework; sourceTree = DEVELOPER_DIR; }; + 15541EEEE784EECDD3E7399D /* ScannedCardDetails.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScannedCardDetails.swift; sourceTree = ""; }; + 17BCBB8B34EBA904BB245CBD /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; + 17FA90EE37CF1E6A49532491 /* StripeCoreTestUtils.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCoreTestUtils.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 1A337FE5C3BE4FAB001E5520 /* ScanAnalyticsManager+Managers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScanAnalyticsManager+Managers.swift"; sourceTree = ""; }; + 1C94EE6DB0E3114A38DD9CFE /* pl-PL */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pl-PL"; path = "pl-PL.lproj/Localizable.strings"; sourceTree = ""; }; + 1FD9AF72B522A1E535A4EC81 /* OcrDD.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OcrDD.swift; sourceTree = ""; }; + 236AE86CE7DCB0DFA3B7C978 /* CardType.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardType.swift; sourceTree = ""; }; + 24C2345660A95FBF90BF4850 /* VerificationFramesData.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VerificationFramesData.swift; sourceTree = ""; }; + 256455DAEDCDDB1AEAF6A278 /* CardImageVerificationDetailsResponse.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardImageVerificationDetailsResponse.swift; sourceTree = ""; }; + 2667E0E0C7DC6CDAF03F8B40 /* pt-PT */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pt-PT"; path = "pt-PT.lproj/Localizable.strings"; sourceTree = ""; }; + 27F4BF66273070C2054BBAC0 /* cs-CZ */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "cs-CZ"; path = "cs-CZ.lproj/Localizable.strings"; sourceTree = ""; }; + 27F6ECCDC721E4D46DAC4F48 /* ScanAnalyticsManagerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanAnalyticsManagerTests.swift; sourceTree = ""; }; + 2D61E754E520B4E21920D1DA /* ScanStatsPayload.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanStatsPayload.swift; sourceTree = ""; }; + 2E889051D7CD76D4FB9084A1 /* StripeCore.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCore.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 2F3E329C37141862BD4A4DE7 /* SSDOcr.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SSDOcr.swift; sourceTree = ""; }; + 2F9447632965B9BCE1E595E4 /* StripeCardScanTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = StripeCardScanTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 311A5A7EF789248E35BB1FEC /* CardNetwork.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardNetwork.swift; sourceTree = ""; }; + 34F806DF921888A9657E2928 /* SimpleScanViewController+Verify.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "SimpleScanViewController+Verify.swift"; sourceTree = ""; }; + 362A27BB81840A22104C9A49 /* AppInfoUtils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppInfoUtils.swift; sourceTree = ""; }; + 3938F9E3F0F267045D3FDDEB /* tr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = tr; path = tr.lproj/Localizable.strings; sourceTree = ""; }; + 3A9C2B8A3B96E194CED6841C /* fr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fr; path = fr.lproj/Localizable.strings; sourceTree = ""; }; + 3BC4F647EB512813078B9A6F /* STPAPIClient+CardImageVerificationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAPIClient+CardImageVerificationTest.swift"; sourceTree = ""; }; + 3DEE8C18A3CECFF0751AB0D5 /* CreditCardOcrPrediction.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CreditCardOcrPrediction.swift; sourceTree = ""; }; + 3E24CF248F312DDE025D662A /* AppleOcr.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppleOcr.swift; sourceTree = ""; }; + 42454BA950282D9ADB9C6557 /* ru */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ru; path = ru.lproj/Localizable.strings; sourceTree = ""; }; + 4301B281F5E0A5BACBBD68CC /* VerifyFramesAPIBindingsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VerifyFramesAPIBindingsTests.swift; sourceTree = ""; }; + 45C9A1A26405DEAD8E11F13D /* pt-BR */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "pt-BR"; path = "pt-BR.lproj/Localizable.strings"; sourceTree = ""; }; + 4A1CD398A6AFB6747D095C59 /* ScanStatsPayload+Tasks.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScanStatsPayload+Tasks.swift"; sourceTree = ""; }; + 4B452EE8B96AF1B0691E7D43 /* PredictionAPI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PredictionAPI.swift; sourceTree = ""; }; + 4B89D81A4099CC71A3FBC133 /* da */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = da; path = da.lproj/Localizable.strings; sourceTree = ""; }; + 4BBE2CADB8D8E98D6EEE10C0 /* PredictionResult.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PredictionResult.swift; sourceTree = ""; }; + 4CA3AD0FDF25BE452AE4ED34 /* CardScanMisc.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardScanMisc.swift; sourceTree = ""; }; + 4D9820BC2FFB68C808FB9507 /* synthetic_test_image.jpg */ = {isa = PBXFileReference; lastKnownFileType = image.jpeg; path = synthetic_test_image.jpg; sourceTree = ""; }; + 4DA27FFBD426C871C5BD254E /* zh-HK */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-HK"; path = "zh-HK.lproj/Localizable.strings"; sourceTree = ""; }; + 4F85031ADAE5928D977465C2 /* UxAnalyzer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UxAnalyzer.swift; sourceTree = ""; }; + 505DF34D76A56FEC2B8453F9 /* SSDOcr+Utils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "SSDOcr+Utils.swift"; sourceTree = ""; }; + 53C171A1336B7FB0C321D194 /* CancellationReason.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CancellationReason.swift; sourceTree = ""; }; + 545E03D5167516D28C8A9F08 /* CardImageVerificationSheetConfiguration.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardImageVerificationSheetConfiguration.swift; sourceTree = ""; }; + 58A5F2CEEAAB78AA425A5737 /* lt-LT */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "lt-LT"; path = "lt-LT.lproj/Localizable.strings"; sourceTree = ""; }; + 5969433D88B5BD3B7F56B2BD /* sk-SK */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "sk-SK"; path = "sk-SK.lproj/Localizable.strings"; sourceTree = ""; }; + 5A6D5C8E51C23370E889F75F /* es-419 */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "es-419"; path = "es-419.lproj/Localizable.strings"; sourceTree = ""; }; + 5A72927DA8A946FDD03843DE /* StringResourceTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StringResourceTests.swift; sourceTree = ""; }; + 5CF9ADC4CC0B4ADFE0A5427E /* ScanStatsPayloadAPIBindingsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanStatsPayloadAPIBindingsTests.swift; sourceTree = ""; }; + 5D58BEEC9D88537E32260CBA /* DetectedSSDOcrBox.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DetectedSSDOcrBox.swift; sourceTree = ""; }; + 5E72DD18B9FCB80ABEDE2689 /* CreditCardOcrPrediction+expiry.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "CreditCardOcrPrediction+expiry.swift"; sourceTree = ""; }; + 6186A0AAB96E236DEF0B5B79 /* en-GB */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "en-GB"; path = "en-GB.lproj/Localizable.strings"; sourceTree = ""; }; + 63662689AD097C859B8759B1 /* Data+Sha256.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Data+Sha256.swift"; sourceTree = ""; }; + 63FA7635BAFDC7DD54B1A731 /* CardVerifyFraudData.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardVerifyFraudData.swift; sourceTree = ""; }; + 6418F87D5168A2A08D65A482 /* STPLocalizedString.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPLocalizedString.swift; sourceTree = ""; }; + 64A1E35C6DF336F34B6C7AEA /* StripeCore+Import.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "StripeCore+Import.swift"; sourceTree = ""; }; + 6563E5CE5CD0439D5DBF35D7 /* ScanAnalyticsManager+Tasks.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScanAnalyticsManager+Tasks.swift"; sourceTree = ""; }; + 66A49708CF5D37C7CB267732 /* ScanStats.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanStats.swift; sourceTree = ""; }; + 67C8AB382B81DE27C8614F6D /* PaymentCard.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentCard.swift; sourceTree = ""; }; + 69A2DB0837C11D8829C027CF /* String+Localized.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "String+Localized.swift"; sourceTree = ""; }; + 69B38020E8E33CA893CBA6F3 /* VerifyCardAddViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VerifyCardAddViewController.swift; sourceTree = ""; }; + 69E2630C0FC7C9DDD551FD67 /* ScanAnalyticsManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanAnalyticsManager.swift; sourceTree = ""; }; + 6BEC906101069A1E3C50FA51 /* ErrorCorrection.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ErrorCorrection.swift; sourceTree = ""; }; + 6C3EE6315FB1507E40F10D85 /* ScannedCardImageData+Verification.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScannedCardImageData+Verification.swift"; sourceTree = ""; }; + 6EBC1F86A03CF9AAC6FE5825 /* ActiveStateComputation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ActiveStateComputation.swift; sourceTree = ""; }; + 6F41FBE3D90B5F6518E42051 /* fil */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fil; path = fil.lproj/Localizable.strings; sourceTree = ""; }; + 732D3AB4CE25CC5C26A352B1 /* String+Sha256.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "String+Sha256.swift"; sourceTree = ""; }; + 746AAB9327BDE6791F9393B0 /* CardScanMockData.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardScanMockData.swift; sourceTree = ""; }; + 7633166B4D9096FE995F08F0 /* CGRectExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CGRectExtension.swift; sourceTree = ""; }; + 76C70F9A1879ED8B41639945 /* ScannedCard.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScannedCard.swift; sourceTree = ""; }; + 77D0D8C2909455B76E246468 /* InterfaceOrientation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InterfaceOrientation.swift; sourceTree = ""; }; + 795E739553651C8AC40E6AC1 /* PredictionUtilOcr.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PredictionUtilOcr.swift; sourceTree = ""; }; + 79A802EA46840CD45CBECEEA /* es */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = es; path = es.lproj/Localizable.strings; sourceTree = ""; }; + 7D41E45A285FEF48E9528997 /* CardScanFraudData.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardScanFraudData.swift; sourceTree = ""; }; + 7D550382D6320388A5BA75A1 /* MainLoopStateMachine.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MainLoopStateMachine.swift; sourceTree = ""; }; + 7DBB233BD36CBAE2700A4511 /* Project-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Project-Release.xcconfig"; sourceTree = ""; }; + 7DDAF8E33A3D7CD020AC904A /* CardImageVerificationDetailsResponseTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardImageVerificationDetailsResponseTest.swift; sourceTree = ""; }; + 7E38717F47ED06C115E058DE /* OcrMainLoop.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OcrMainLoop.swift; sourceTree = ""; }; + 8094FE7CCD3CDA8B914EC534 /* ja */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ja; path = ja.lproj/Localizable.strings; sourceTree = ""; }; + 84E54DFA3C47D94639297588 /* STPAPIClient+CardImageVerification.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAPIClient+CardImageVerification.swift"; sourceTree = ""; }; + 8735A24B634F3B9060CADE45 /* CreditCardUtils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CreditCardUtils.swift; sourceTree = ""; }; + 87448BAE438EF08C55AA4B29 /* UxAndOcrMainLoop.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UxAndOcrMainLoop.swift; sourceTree = ""; }; + 88F89EC392AFBE060336B63F /* sl-SI */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "sl-SI"; path = "sl-SI.lproj/Localizable.strings"; sourceTree = ""; }; + 89BBC06EC107C8BDEF79BB3D /* sv */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = sv; path = sv.lproj/Localizable.strings; sourceTree = ""; }; + 8AEB912151CB564AE21D073C /* SSDOcrDetect.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SSDOcrDetect.swift; sourceTree = ""; }; + 8C227CA761586744976C952D /* nn-NO */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "nn-NO"; path = "nn-NO.lproj/Localizable.strings"; sourceTree = ""; }; + 8CA457DF27DC45455B8E1345 /* Array+utils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Array+utils.swift"; sourceTree = ""; }; + 8EDF39435211B456B7579739 /* AppleCreditCardOcr.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppleCreditCardOcr.swift; sourceTree = ""; }; + 8FB602C9E909E31BAE6703D6 /* ca-ES */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ca-ES"; path = "ca-ES.lproj/Localizable.strings"; sourceTree = ""; }; + 938958FCB9B918AD6CDF0F4E /* BlurView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlurView.swift; sourceTree = ""; }; + 9593D6788C4DBF554735E2B6 /* mt */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = mt; path = mt.lproj/Localizable.strings; sourceTree = ""; }; + 96F3DDAD14A9F2DEB98E236E /* Expiry.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Expiry.swift; sourceTree = ""; }; + 9A256BCB7822DD94572DDA07 /* nb */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = nb; path = nb.lproj/Localizable.strings; sourceTree = ""; }; + 9B3242DC2FCDF20D51D23AE1 /* CardVerifyStateMachine.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardVerifyStateMachine.swift; sourceTree = ""; }; + 9C3890F6A5A4FF68F95746DC /* hr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = hr; path = hr.lproj/Localizable.strings; sourceTree = ""; }; + 9E4CF9A676A86169070DD54D /* ro-RO */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ro-RO"; path = "ro-RO.lproj/Localizable.strings"; sourceTree = ""; }; + A2EF6799FCE0817D9E3A536B /* DetectedSSDBox.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DetectedSSDBox.swift; sourceTree = ""; }; + A3C926149A434F4F3BC961D9 /* FrameData.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FrameData.swift; sourceTree = ""; }; + A612DDE1110EA970BF69B33D /* zh-Hant */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-Hant"; path = "zh-Hant.lproj/Localizable.strings"; sourceTree = ""; }; + A67EDE2BCCC081DEDD684AAC /* StripeiOS Tests-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Release.xcconfig"; sourceTree = ""; }; + A6E1DA01CB00CBDE36735EDD /* SimpleScanViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SimpleScanViewController.swift; sourceTree = ""; }; + A790FC6FAEC3F88FB2C26E27 /* ScanBaseViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanBaseViewController.swift; sourceTree = ""; }; + A7EE5FDCA3B075FE5BBAF42A /* DetectedAllBoxes.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DetectedAllBoxes.swift; sourceTree = ""; }; + AB56DDBBF0E5BC10682C7D96 /* StripeCardScan.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = StripeCardScan.h; sourceTree = ""; }; + AB64890F80D91D9D7B92197D /* nl */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = nl; path = nl.lproj/Localizable.strings; sourceTree = ""; }; + ABB38869FD175D7BD6BB3890 /* it */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = it; path = it.lproj/Localizable.strings; sourceTree = ""; }; + AC9F1B20AEB433DE3CE0042C /* PostDetectionAlgorithm.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PostDetectionAlgorithm.swift; sourceTree = ""; }; + ACD9FF5C9DD64BE92DB0927D /* NMS.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NMS.swift; sourceTree = ""; }; + ADCB34B81919043FA35E8BC3 /* zh-Hans */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "zh-Hans"; path = "zh-Hans.lproj/Localizable.strings"; sourceTree = ""; }; + AEE9C295E982F92DE2418AE1 /* Image+utils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Image+utils.swift"; sourceTree = ""; }; + AF17FD18E52CB3508AD895E3 /* MachineLearningResult.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MachineLearningResult.swift; sourceTree = ""; }; + B0A7946C3A1E943FB72F3FB7 /* bg-BG */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "bg-BG"; path = "bg-BG.lproj/Localizable.strings"; sourceTree = ""; }; + B2D9886B0ED119E8D1FB6448 /* UIImage+pixelBuffer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIImage+pixelBuffer.swift"; sourceTree = ""; }; + B313BA373EF6B567BF40A240 /* CardImageVerificationController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardImageVerificationController.swift; sourceTree = ""; }; + B3CAFFFE55918BB4939868B6 /* FadeInAnimation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FadeInAnimation.swift; sourceTree = ""; }; + B53D407C0AC8D44ECC0C3C52 /* DetectedBox.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DetectedBox.swift; sourceTree = ""; }; + B97CA4D8286B196A49B8D535 /* ScanAnalyticsManager+Helpers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ScanAnalyticsManager+Helpers.swift"; sourceTree = ""; }; + BAE45D8AC7BBA5DC433A9383 /* ScannedCardImageData.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScannedCardImageData.swift; sourceTree = ""; }; + BC0B91384C63CBFB790C8EE6 /* OcrDDUtils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OcrDDUtils.swift; sourceTree = ""; }; + BD87F9971C25C4B99138318F /* CGrect+utils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "CGrect+utils.swift"; sourceTree = ""; }; + BDD0C14472FA163BD395DBD3 /* CardImageVerification_CardSet_200.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = CardImageVerification_CardSet_200.json; sourceTree = ""; }; + BEDE71D7051DDC98698C4057 /* CardBase.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardBase.swift; sourceTree = ""; }; + BEEE30227F7D34546F0FBB58 /* en */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = en; path = en.lproj/Localizable.strings; sourceTree = ""; }; + C038EFADF52B2477FC934EE0 /* UxModelTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UxModelTests.swift; sourceTree = ""; }; + C264A676DA41344551BA6661 /* StripeCardScan-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeCardScan-Release.xcconfig"; sourceTree = ""; }; + C3E012270975B6DC09A8199F /* CornerView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CornerView.swift; sourceTree = ""; }; + C827434739027B2DD43449EA /* Torch.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Torch.swift; sourceTree = ""; }; + CA16EAC68E04D4FDE67DB807 /* Bouncer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Bouncer.swift; sourceTree = ""; }; + CB48DDCA458A62DCE342F517 /* et-EE */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "et-EE"; path = "et-EE.lproj/Localizable.strings"; sourceTree = ""; }; + CD5004FC0706B2660B4C6EA0 /* CardImageVerificationIntent.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardImageVerificationIntent.swift; sourceTree = ""; }; + CD6D4DB87B26D439686392AE /* StrictModeFramesTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StrictModeFramesTest.swift; sourceTree = ""; }; + CDB03A9253E8FAD07B739438 /* DeviceUtils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DeviceUtils.swift; sourceTree = ""; }; + CF3FCD1B2283A2B02EBAEF4F /* UxModel+Utils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UxModel+Utils.swift"; sourceTree = ""; }; + CF90067085243D998A1D6030 /* CardImageVerification_CardAdd_200.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = CardImageVerification_CardAdd_200.json; sourceTree = ""; }; + D0AA128D69F3C32A4EF67CF0 /* ImageCompressionTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageCompressionTests.swift; sourceTree = ""; }; + D42510E09D32547455CDDBEF /* VideoFeed.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VideoFeed.swift; sourceTree = ""; }; + D532C440ADCD017AD326299B /* el-GR */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "el-GR"; path = "el-GR.lproj/Localizable.strings"; sourceTree = ""; }; + D9B548B9626366BB0F110873 /* AtomicPropertyWrapper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AtomicPropertyWrapper.swift; sourceTree = ""; }; + DA3ECBF071E7507C9D41F232 /* EndToEndTestDataSource.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EndToEndTestDataSource.swift; sourceTree = ""; }; + DC12B6E0657203AAB765468C /* PreviewView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PreviewView.swift; sourceTree = ""; }; + DCA2733CEC85F8E427CA99CB /* AppState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppState.swift; sourceTree = ""; }; + DCF4C0BF8E55D53B004FABCA /* CardScanSheetError.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardScanSheetError.swift; sourceTree = ""; }; + DD6884D0B3347BBF2E61B1D6 /* StripeCardScan.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCardScan.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + DDF173627F545C88CB1551B0 /* UxModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UxModel.swift; sourceTree = ""; }; + E4FAA99A64BCD00A336A5E01 /* VerifyFrames.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VerifyFrames.swift; sourceTree = ""; }; + E73DC7DD9C3E3395DF3F5A29 /* SSDOcr.mlmodelc */ = {isa = PBXFileReference; path = SSDOcr.mlmodelc; sourceTree = ""; }; + EA89DA7FF10BC3A7B2D102FB /* SSDOcrOutputExtensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SSDOcrOutputExtensions.swift; sourceTree = ""; }; + ED780E8B4C14EE39F853B3A1 /* OcrPriorsGen.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OcrPriorsGen.swift; sourceTree = ""; }; + EDE92803C69CE8253D73A25F /* CreditCardOcrResult.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CreditCardOcrResult.swift; sourceTree = ""; }; + EF53C194DDC96160505D54D5 /* AsyncModelLoading.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AsyncModelLoading.swift; sourceTree = ""; }; + F1FCC855CCB0816414A4BEC1 /* StripeiOS Tests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Debug.xcconfig"; sourceTree = ""; }; + F24DEA029F94A359B7A571BB /* CardImageVerificationControllerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardImageVerificationControllerTests.swift; sourceTree = ""; }; + F3AF32D7DF7EE1420D16291D /* fr-CA */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "fr-CA"; path = "fr-CA.lproj/Localizable.strings"; sourceTree = ""; }; + F4721524B6285C507681CC4D /* UxModel.mlmodelc */ = {isa = PBXFileReference; path = UxModel.mlmodelc; sourceTree = ""; }; + F87D884128F85E7F16BCBA9B /* ko */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ko; path = ko.lproj/Localizable.strings; sourceTree = ""; }; + F912323832888F88EA218BA9 /* VerifyCardViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VerifyCardViewController.swift; sourceTree = ""; }; + F96A85E323CFDA8BA35BC293 /* NonNameWords.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NonNameWords.swift; sourceTree = ""; }; + FD40E90452A83FFBBF65C8D8 /* lv-LV */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "lv-LV"; path = "lv-LV.lproj/Localizable.strings"; sourceTree = ""; }; + FF9618F7B15B33EC5B339A94 /* ScanConfiguration.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScanConfiguration.swift; sourceTree = ""; }; + FFE89E65E767D03B92383C76 /* de */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = de; path = de.lproj/Localizable.strings; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 1639B69C00B8C21CEBF08A55 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + C2C41F5E568BF767C8232E74 /* XCTest.framework in Frameworks */, + E43FE03D43CB0D7BAA73745A /* StripeCardScan.framework in Frameworks */, + 4B01B9F4FB647908AA5276E2 /* StripeCoreTestUtils.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 29BDDDCECDEC71451BAAB324 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + A060350B93E3369463AE2898 /* StripeCore.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 01BEE24ED60FE104004E8F2C /* CompiledModels */ = { + isa = PBXGroup; + children = ( + E73DC7DD9C3E3395DF3F5A29 /* SSDOcr.mlmodelc */, + F4721524B6285C507681CC4D /* UxModel.mlmodelc */, + ); + path = CompiledModels; + sourceTree = ""; + }; + 0DFCF5271EC90615420288B3 /* Api */ = { + isa = PBXGroup; + children = ( + C8B14FA3DC29B84F382E6D1F /* Models */, + 256455DAEDCDDB1AEAF6A278 /* CardImageVerificationDetailsResponse.swift */, + 84E54DFA3C47D94639297588 /* STPAPIClient+CardImageVerification.swift */, + ); + path = Api; + sourceTree = ""; + }; + 0E245ED03746B2F8EE73B5F9 /* StripeCardScanTests */ = { + isa = PBXGroup; + children = ( + C8CC6DB5A60043E8A373BC6F /* Helpers */, + 40C830CD6973E7AD722C2213 /* Mock Data */, + 3172AE79E1A187D904BD1022 /* Resources */, + A35971ECF566C2769A608A06 /* Unit */, + 048ED08A6959593F896B3AC8 /* Info.plist */, + ); + path = StripeCardScanTests; + sourceTree = ""; + }; + 10507D99D0E81A181A25B4BC /* MLModels */ = { + isa = PBXGroup; + children = ( + EF53C194DDC96160505D54D5 /* AsyncModelLoading.swift */, + 2F3E329C37141862BD4A4DE7 /* SSDOcr.swift */, + 505DF34D76A56FEC2B8453F9 /* SSDOcr+Utils.swift */, + DDF173627F545C88CB1551B0 /* UxModel.swift */, + CF3FCD1B2283A2B02EBAEF4F /* UxModel+Utils.swift */, + ); + path = MLModels; + sourceTree = ""; + }; + 105B9420C58B9406F7939324 = { + isa = PBXGroup; + children = ( + 36C8C7335EC9F37DE4446270 /* Project */, + D275033F1B615793C2F8B8E0 /* Frameworks */, + 72B7637B9160BBA39556BA0A /* Products */, + ); + sourceTree = ""; + }; + 20CA55CAA02DC2833AD1C426 /* ML Models */ = { + isa = PBXGroup; + children = ( + C038EFADF52B2477FC934EE0 /* UxModelTests.swift */, + ); + path = "ML Models"; + sourceTree = ""; + }; + 25575C2DBB0C8AAA658EC26E /* JSON */ = { + isa = PBXGroup; + children = ( + CF90067085243D998A1D6030 /* CardImageVerification_CardAdd_200.json */, + BDD0C14472FA163BD395DBD3 /* CardImageVerification_CardSet_200.json */, + ); + path = JSON; + sourceTree = ""; + }; + 3172AE79E1A187D904BD1022 /* Resources */ = { + isa = PBXGroup; + children = ( + 4D9820BC2FFB68C808FB9507 /* synthetic_test_image.jpg */, + ); + path = Resources; + sourceTree = ""; + }; + 3228566B727B9D25D13A3BAB /* Resources */ = { + isa = PBXGroup; + children = ( + 01BEE24ED60FE104004E8F2C /* CompiledModels */, + 6A6547B8AC5A170A0092DCF5 /* Localizations */, + ); + path = Resources; + sourceTree = ""; + }; + 324D2CF5354B3283BC0C49ED /* CardVerify */ = { + isa = PBXGroup; + children = ( + 0DFCF5271EC90615420288B3 /* Api */, + BC0070EA31577F5573D7B787 /* Card Image Verification */, + 69D6875F84C12C2E18B96B01 /* Helpers */, + CA16EAC68E04D4FDE67DB807 /* Bouncer.swift */, + BEDE71D7051DDC98698C4057 /* CardBase.swift */, + 7D41E45A285FEF48E9528997 /* CardScanFraudData.swift */, + 4CA3AD0FDF25BE452AE4ED34 /* CardScanMisc.swift */, + 63FA7635BAFDC7DD54B1A731 /* CardVerifyFraudData.swift */, + 9B3242DC2FCDF20D51D23AE1 /* CardVerifyStateMachine.swift */, + B3CAFFFE55918BB4939868B6 /* FadeInAnimation.swift */, + A3C926149A434F4F3BC961D9 /* FrameData.swift */, + 67C8AB382B81DE27C8614F6D /* PaymentCard.swift */, + 34F806DF921888A9657E2928 /* SimpleScanViewController+Verify.swift */, + 06F97A4CE5104309B720D603 /* StripeCardScanBundleLocator.swift */, + 4F85031ADAE5928D977465C2 /* UxAnalyzer.swift */, + 87448BAE438EF08C55AA4B29 /* UxAndOcrMainLoop.swift */, + 69B38020E8E33CA893CBA6F3 /* VerifyCardAddViewController.swift */, + F912323832888F88EA218BA9 /* VerifyCardViewController.swift */, + 10A26712A38A91726C314329 /* ZoomedInCGImage.swift */, + ); + path = CardVerify; + sourceTree = ""; + }; + 36C8C7335EC9F37DE4446270 /* Project */ = { + isa = PBXGroup; + children = ( + 79C01535652A2F89EB22D9A9 /* BuildConfigurations */, + 6C3738C5D8A82E9A3C5AFC85 /* BuildConfigurations */, + DF93066F033C46A7B100FC82 /* StripeCardScan */, + 0E245ED03746B2F8EE73B5F9 /* StripeCardScanTests */, + ); + name = Project; + sourceTree = ""; + }; + 3C728AAC6B6FF615A064C757 /* CardUtils */ = { + isa = PBXGroup; + children = ( + 311A5A7EF789248E35BB1FEC /* CardNetwork.swift */, + 236AE86CE7DCB0DFA3B7C978 /* CardType.swift */, + 8735A24B634F3B9060CADE45 /* CreditCardUtils.swift */, + 96F3DDAD14A9F2DEB98E236E /* Expiry.swift */, + ); + path = CardUtils; + sourceTree = ""; + }; + 40C830CD6973E7AD722C2213 /* Mock Data */ = { + isa = PBXGroup; + children = ( + 25575C2DBB0C8AAA658EC26E /* JSON */, + ); + path = "Mock Data"; + sourceTree = ""; + }; + 620D5B89096A8B2B2202E6EB /* CreditCardOcr */ = { + isa = PBXGroup; + children = ( + 8EDF39435211B456B7579739 /* AppleCreditCardOcr.swift */, + 05B28BBCB062B6AE5A3B7CBC /* CreditCardOcrImplementation.swift */, + 3DEE8C18A3CECFF0751AB0D5 /* CreditCardOcrPrediction.swift */, + EDE92803C69CE8253D73A25F /* CreditCardOcrResult.swift */, + 6BEC906101069A1E3C50FA51 /* ErrorCorrection.swift */, + AF17FD18E52CB3508AD895E3 /* MachineLearningResult.swift */, + 7D550382D6320388A5BA75A1 /* MainLoopStateMachine.swift */, + F96A85E323CFDA8BA35BC293 /* NonNameWords.swift */, + 7E38717F47ED06C115E058DE /* OcrMainLoop.swift */, + 0DC83A169A82484B9F60E452 /* OcrObject.swift */, + 0B00395C6A2B14A6D56735B9 /* SSDCreditCardOcr.swift */, + ); + path = CreditCardOcr; + sourceTree = ""; + }; + 66F259BA8091A6C5D28C4963 /* MLRuntime */ = { + isa = PBXGroup; + children = ( + 6EBC1F86A03CF9AAC6FE5825 /* ActiveStateComputation.swift */, + DCA2733CEC85F8E427CA99CB /* AppState.swift */, + A7EE5FDCA3B075FE5BBAF42A /* DetectedAllBoxes.swift */, + 1441A7A39C36C2A634A882F5 /* DetectedAllOcrBoxes.swift */, + B53D407C0AC8D44ECC0C3C52 /* DetectedBox.swift */, + A2EF6799FCE0817D9E3A536B /* DetectedSSDBox.swift */, + 5D58BEEC9D88537E32260CBA /* DetectedSSDOcrBox.swift */, + ACD9FF5C9DD64BE92DB0927D /* NMS.swift */, + 1FD9AF72B522A1E535A4EC81 /* OcrDD.swift */, + BC0B91384C63CBFB790C8EE6 /* OcrDDUtils.swift */, + ED780E8B4C14EE39F853B3A1 /* OcrPriorsGen.swift */, + AC9F1B20AEB433DE3CE0042C /* PostDetectionAlgorithm.swift */, + 4B452EE8B96AF1B0691E7D43 /* PredictionAPI.swift */, + 4BBE2CADB8D8E98D6EEE10C0 /* PredictionResult.swift */, + 795E739553651C8AC40E6AC1 /* PredictionUtilOcr.swift */, + 00A93BFDEE4072A2A9C30397 /* SoftNMS.swift */, + 8AEB912151CB564AE21D073C /* SSDOcrDetect.swift */, + EA89DA7FF10BC3A7B2D102FB /* SSDOcrOutputExtensions.swift */, + ); + path = MLRuntime; + sourceTree = ""; + }; + 69D6875F84C12C2E18B96B01 /* Helpers */ = { + isa = PBXGroup; + children = ( + DA3ECBF071E7507C9D41F232 /* EndToEndTestDataSource.swift */, + 6418F87D5168A2A08D65A482 /* STPLocalizedString.swift */, + 69A2DB0837C11D8829C027CF /* String+Localized.swift */, + ); + path = Helpers; + sourceTree = ""; + }; + 6A6547B8AC5A170A0092DCF5 /* Localizations */ = { + isa = PBXGroup; + children = ( + 6C27E58C1953533C68D43361 /* Localizable.strings */, + ); + path = Localizations; + sourceTree = ""; + }; + 6C3738C5D8A82E9A3C5AFC85 /* BuildConfigurations */ = { + isa = PBXGroup; + children = ( + 14795A45279E8F328AB54920 /* Project-Debug.xcconfig */, + 7DBB233BD36CBAE2700A4511 /* Project-Release.xcconfig */, + F1FCC855CCB0816414A4BEC1 /* StripeiOS Tests-Debug.xcconfig */, + A67EDE2BCCC081DEDD684AAC /* StripeiOS Tests-Release.xcconfig */, + ); + name = BuildConfigurations; + path = ../BuildConfigurations; + sourceTree = ""; + }; + 72B7637B9160BBA39556BA0A /* Products */ = { + isa = PBXGroup; + children = ( + DD6884D0B3347BBF2E61B1D6 /* StripeCardScan.framework */, + 2F9447632965B9BCE1E595E4 /* StripeCardScanTests.xctest */, + 2E889051D7CD76D4FB9084A1 /* StripeCore.framework */, + 17FA90EE37CF1E6A49532491 /* StripeCoreTestUtils.framework */, + ); + name = Products; + sourceTree = ""; + }; + 79C01535652A2F89EB22D9A9 /* BuildConfigurations */ = { + isa = PBXGroup; + children = ( + 0197A1615C3FA86907CB6186 /* StripeCardScan-Debug.xcconfig */, + C264A676DA41344551BA6661 /* StripeCardScan-Release.xcconfig */, + ); + path = BuildConfigurations; + sourceTree = ""; + }; + 8AA53DD0D15C5BC1040EE334 /* Scan Analytics */ = { + isa = PBXGroup; + children = ( + 2D61E754E520B4E21920D1DA /* ScanStatsPayload.swift */, + 028DA95BD9A0DA8AD69162CB /* ScanStatsPayload+Common.swift */, + 4A1CD398A6AFB6747D095C59 /* ScanStatsPayload+Tasks.swift */, + ); + path = "Scan Analytics"; + sourceTree = ""; + }; + 9AE51241602B3092F45811E8 /* Extensions */ = { + isa = PBXGroup; + children = ( + 8CA457DF27DC45455B8E1345 /* Array+utils.swift */, + BD87F9971C25C4B99138318F /* CGrect+utils.swift */, + 7633166B4D9096FE995F08F0 /* CGRectExtension.swift */, + 5E72DD18B9FCB80ABEDE2689 /* CreditCardOcrPrediction+expiry.swift */, + AEE9C295E982F92DE2418AE1 /* Image+utils.swift */, + B2D9886B0ED119E8D1FB6448 /* UIImage+pixelBuffer.swift */, + ); + path = Extensions; + sourceTree = ""; + }; + 9FDF572663A7F311B8506E13 /* CardScan */ = { + isa = PBXGroup; + children = ( + A1C38CDC3470D8E7A9F39671 /* AppleOcr */, + 3C728AAC6B6FF615A064C757 /* CardUtils */, + 620D5B89096A8B2B2202E6EB /* CreditCardOcr */, + 9AE51241602B3092F45811E8 /* Extensions */, + 10507D99D0E81A181A25B4BC /* MLModels */, + 66F259BA8091A6C5D28C4963 /* MLRuntime */, + E2C9F4FF2603FAF1BD4567B4 /* UI */, + E86A9678BB840BD3EA2AEB94 /* Utils */, + ); + path = CardScan; + sourceTree = ""; + }; + A1C38CDC3470D8E7A9F39671 /* AppleOcr */ = { + isa = PBXGroup; + children = ( + 3E24CF248F312DDE025D662A /* AppleOcr.swift */, + ); + path = AppleOcr; + sourceTree = ""; + }; + A35971ECF566C2769A608A06 /* Unit */ = { + isa = PBXGroup; + children = ( + D6C5CA3B4336E2024BFB5037 /* API Bindings */, + 20CA55CAA02DC2833AD1C426 /* ML Models */, + F24DEA029F94A359B7A571BB /* CardImageVerificationControllerTests.swift */, + 7DDAF8E33A3D7CD020AC904A /* CardImageVerificationDetailsResponseTest.swift */, + D0AA128D69F3C32A4EF67CF0 /* ImageCompressionTests.swift */, + 27F6ECCDC721E4D46DAC4F48 /* ScanAnalyticsManagerTests.swift */, + CD6D4DB87B26D439686392AE /* StrictModeFramesTest.swift */, + 5A72927DA8A946FDD03843DE /* StringResourceTests.swift */, + ); + path = Unit; + sourceTree = ""; + }; + BC0070EA31577F5573D7B787 /* Card Image Verification */ = { + isa = PBXGroup; + children = ( + 53C171A1336B7FB0C321D194 /* CancellationReason.swift */, + B313BA373EF6B567BF40A240 /* CardImageVerificationController.swift */, + CD5004FC0706B2660B4C6EA0 /* CardImageVerificationIntent.swift */, + 12F09D21BAC44C461B1AD7CB /* CardImageVerificationSheet.swift */, + 545E03D5167516D28C8A9F08 /* CardImageVerificationSheetConfiguration.swift */, + DCF4C0BF8E55D53B004FABCA /* CardScanSheetError.swift */, + 69E2630C0FC7C9DDD551FD67 /* ScanAnalyticsManager.swift */, + B97CA4D8286B196A49B8D535 /* ScanAnalyticsManager+Helpers.swift */, + 1A337FE5C3BE4FAB001E5520 /* ScanAnalyticsManager+Managers.swift */, + 6563E5CE5CD0439D5DBF35D7 /* ScanAnalyticsManager+Tasks.swift */, + 76C70F9A1879ED8B41639945 /* ScannedCard.swift */, + BAE45D8AC7BBA5DC433A9383 /* ScannedCardImageData.swift */, + 6C3EE6315FB1507E40F10D85 /* ScannedCardImageData+Verification.swift */, + 64A1E35C6DF336F34B6C7AEA /* StripeCore+Import.swift */, + ); + path = "Card Image Verification"; + sourceTree = ""; + }; + C8B14FA3DC29B84F382E6D1F /* Models */ = { + isa = PBXGroup; + children = ( + 8AA53DD0D15C5BC1040EE334 /* Scan Analytics */, + 24C2345660A95FBF90BF4850 /* VerificationFramesData.swift */, + E4FAA99A64BCD00A336A5E01 /* VerifyFrames.swift */, + ); + path = Models; + sourceTree = ""; + }; + C8CC6DB5A60043E8A373BC6F /* Helpers */ = { + isa = PBXGroup; + children = ( + 746AAB9327BDE6791F9393B0 /* CardScanMockData.swift */, + 63662689AD097C859B8759B1 /* Data+Sha256.swift */, + 091163ED5E37922332F502B1 /* ImageHelpers.swift */, + 15541EEEE784EECDD3E7399D /* ScannedCardDetails.swift */, + 732D3AB4CE25CC5C26A352B1 /* String+Sha256.swift */, + ); + path = Helpers; + sourceTree = ""; + }; + D275033F1B615793C2F8B8E0 /* Frameworks */ = { + isa = PBXGroup; + children = ( + 14E5A51A77928A2700E23321 /* XCTest.framework */, + ); + name = Frameworks; + sourceTree = ""; + }; + D6C5CA3B4336E2024BFB5037 /* API Bindings */ = { + isa = PBXGroup; + children = ( + 5CF9ADC4CC0B4ADFE0A5427E /* ScanStatsPayloadAPIBindingsTests.swift */, + 3BC4F647EB512813078B9A6F /* STPAPIClient+CardImageVerificationTest.swift */, + 4301B281F5E0A5BACBBD68CC /* VerifyFramesAPIBindingsTests.swift */, + ); + path = "API Bindings"; + sourceTree = ""; + }; + DF93066F033C46A7B100FC82 /* StripeCardScan */ = { + isa = PBXGroup; + children = ( + 3228566B727B9D25D13A3BAB /* Resources */, + F28CFA0B72751113E838303F /* Source */, + 17BCBB8B34EBA904BB245CBD /* Info.plist */, + AB56DDBBF0E5BC10682C7D96 /* StripeCardScan.h */, + ); + path = StripeCardScan; + sourceTree = ""; + }; + E2C9F4FF2603FAF1BD4567B4 /* UI */ = { + isa = PBXGroup; + children = ( + 938958FCB9B918AD6CDF0F4E /* BlurView.swift */, + 14BC8B1C28C3F6C8330164E0 /* CardScanSheet.swift */, + C3E012270975B6DC09A8199F /* CornerView.swift */, + 77D0D8C2909455B76E246468 /* InterfaceOrientation.swift */, + DC12B6E0657203AAB765468C /* PreviewView.swift */, + A790FC6FAEC3F88FB2C26E27 /* ScanBaseViewController.swift */, + FF9618F7B15B33EC5B339A94 /* ScanConfiguration.swift */, + 0D1FBA1B4676CC5E9F785D51 /* ScanEventsProtocol.swift */, + 66A49708CF5D37C7CB267732 /* ScanStats.swift */, + A6E1DA01CB00CBDE36735EDD /* SimpleScanViewController.swift */, + C827434739027B2DD43449EA /* Torch.swift */, + D42510E09D32547455CDDBEF /* VideoFeed.swift */, + ); + path = UI; + sourceTree = ""; + }; + E86A9678BB840BD3EA2AEB94 /* Utils */ = { + isa = PBXGroup; + children = ( + 362A27BB81840A22104C9A49 /* AppInfoUtils.swift */, + D9B548B9626366BB0F110873 /* AtomicPropertyWrapper.swift */, + CDB03A9253E8FAD07B739438 /* DeviceUtils.swift */, + ); + path = Utils; + sourceTree = ""; + }; + F28CFA0B72751113E838303F /* Source */ = { + isa = PBXGroup; + children = ( + 9FDF572663A7F311B8506E13 /* CardScan */, + 324D2CF5354B3283BC0C49ED /* CardVerify */, + ); + path = Source; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXHeadersBuildPhase section */ + B4E70071212998150E28FF9A /* Headers */ = { + isa = PBXHeadersBuildPhase; + buildActionMask = 2147483647; + files = ( + 1B10222A5121C9B2C3479FAB /* StripeCardScan.h in Headers */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXHeadersBuildPhase section */ + +/* Begin PBXNativeTarget section */ + 03CF79975A56288F02F20E52 /* StripeCardScan */ = { + isa = PBXNativeTarget; + buildConfigurationList = 3942B170A6D222D5B5128339 /* Build configuration list for PBXNativeTarget "StripeCardScan" */; + buildPhases = ( + B4E70071212998150E28FF9A /* Headers */, + 44F5D31B0E7A7BFCB9425841 /* Sources */, + 7FD545F62DB7C0787E9CD12D /* Resources */, + 451E97BA25D9CE29EDAB7618 /* Embed Frameworks */, + 29BDDDCECDEC71451BAAB324 /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = StripeCardScan; + productName = StripeCardScan; + productReference = DD6884D0B3347BBF2E61B1D6 /* StripeCardScan.framework */; + productType = "com.apple.product-type.framework"; + }; + DCC6FFCFBE0B51B33F4CBB26 /* StripeCardScanTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 89AE00D6B3B382DD3D3A3C70 /* Build configuration list for PBXNativeTarget "StripeCardScanTests" */; + buildPhases = ( + 201682B08469C4AC551E9BF8 /* Sources */, + 54119F2010F36CB483CA88CC /* Resources */, + 20B092D23CA44561EF997447 /* Embed Frameworks */, + 1639B69C00B8C21CEBF08A55 /* Frameworks */, + ); + buildRules = ( + ); + dependencies = ( + 82DAA855443FCAA25F64C5C0 /* PBXTargetDependency */, + ); + name = StripeCardScanTests; + productName = StripeCardScanTests; + productReference = 2F9447632965B9BCE1E595E4 /* StripeCardScanTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 9A6BF50E5B02355004AC6020 /* Project object */ = { + isa = PBXProject; + attributes = { + TargetAttributes = { + }; + }; + buildConfigurationList = CBB5D42795303F47D9D5F2F5 /* Build configuration list for PBXProject "StripeCardScan" */; + compatibilityVersion = "Xcode 13.0"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + Base, + "bg-BG", + "ca-ES", + "cs-CZ", + da, + de, + "el-GR", + en, + "en-GB", + es, + "es-419", + "et-EE", + fi, + fil, + fr, + "fr-CA", + hr, + hu, + id, + it, + ja, + ko, + "lt-LT", + "lv-LV", + "ms-MY", + mt, + nb, + nl, + "nn-NO", + "pl-PL", + "pt-BR", + "pt-PT", + "ro-RO", + ru, + "sk-SK", + "sl-SI", + sv, + tr, + vi, + "zh-HK", + "zh-Hans", + "zh-Hant", + ); + mainGroup = 105B9420C58B9406F7939324; + productRefGroup = 72B7637B9160BBA39556BA0A /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 03CF79975A56288F02F20E52 /* StripeCardScan */, + DCC6FFCFBE0B51B33F4CBB26 /* StripeCardScanTests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 54119F2010F36CB483CA88CC /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 0D62200AA65D71F040037CC8 /* CardImageVerification_CardAdd_200.json in Resources */, + 52519CA3928967768049164E /* CardImageVerification_CardSet_200.json in Resources */, + 59889C85D6D25B3222776B28 /* synthetic_test_image.jpg in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 7FD545F62DB7C0787E9CD12D /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + A7F92FB7213E66A7D15A1BE3 /* SSDOcr.mlmodelc in Resources */, + B2DF7862E5F80ACD8DEE9317 /* UxModel.mlmodelc in Resources */, + 8CF112F4889C96617504A931 /* Localizable.strings in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 201682B08469C4AC551E9BF8 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 77443628E02F3AEFF112314D /* CardScanMockData.swift in Sources */, + DB74787AF4EEF0E506DD0E1C /* Data+Sha256.swift in Sources */, + 3F5B9466E9FC13CA29218750 /* ImageHelpers.swift in Sources */, + AE144927F21D5DF9A8C0EF79 /* ScannedCardDetails.swift in Sources */, + D42EEFBB72CD0AB00C6B4728 /* String+Sha256.swift in Sources */, + 449DD2A5D1F3FF94524D3CD6 /* STPAPIClient+CardImageVerificationTest.swift in Sources */, + 79A96C88970E4E61BAC47412 /* ScanStatsPayloadAPIBindingsTests.swift in Sources */, + 25FAF64D4FE93BF38E807ECA /* VerifyFramesAPIBindingsTests.swift in Sources */, + BDEA39DE5D3775AD2A4BFB6C /* CardImageVerificationControllerTests.swift in Sources */, + 19EF4634631CFC121DE21D1B /* CardImageVerificationDetailsResponseTest.swift in Sources */, + 48E4577B073DD9485BC62902 /* ImageCompressionTests.swift in Sources */, + 70B5FF75EF5DC8FFB23B95F7 /* UxModelTests.swift in Sources */, + 0D062E99363099A5728F3DCD /* ScanAnalyticsManagerTests.swift in Sources */, + FE55286899CFB0E34356D4D6 /* StrictModeFramesTest.swift in Sources */, + D6F0E1C93997BB36AF0608C0 /* StringResourceTests.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 44F5D31B0E7A7BFCB9425841 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 2A9ECCF086685AD607D82492 /* AppleOcr.swift in Sources */, + C3B9A4A30443A38C2D17BE8E /* CardNetwork.swift in Sources */, + 0ADD10809FEABDCB59A8FA72 /* CardType.swift in Sources */, + A7014CF97250D97BC683FAC8 /* CreditCardUtils.swift in Sources */, + 050B462804602A9F4DB58A9D /* Expiry.swift in Sources */, + DDBCE05A3794129A9A04D511 /* AppleCreditCardOcr.swift in Sources */, + 60AF52B9EFFF12EBABE4D221 /* CreditCardOcrImplementation.swift in Sources */, + 1AD72764EA8BCB66A9D7B637 /* CreditCardOcrPrediction.swift in Sources */, + F9F19A6F2245863FEA485335 /* CreditCardOcrResult.swift in Sources */, + 17E34C9EF882AEAE47BECAB4 /* ErrorCorrection.swift in Sources */, + 858CDA751309CA04202B6203 /* MachineLearningResult.swift in Sources */, + 4ACBD9754F304CE4F70CAE43 /* MainLoopStateMachine.swift in Sources */, + C7A941539DBCFA790DCDB0B0 /* NonNameWords.swift in Sources */, + EE1CB7D3FA2B44DC17E6FF4C /* OcrMainLoop.swift in Sources */, + 5B1749A19231B20E2A081EB9 /* OcrObject.swift in Sources */, + C8E2E98B7ED108804E0A0E24 /* SSDCreditCardOcr.swift in Sources */, + C7259DA76E6AB9BF53735D19 /* Array+utils.swift in Sources */, + 10E840B00703A92CC487B324 /* CGRectExtension.swift in Sources */, + 7D9C9C2A26EF11F0C5D67D7C /* CGrect+utils.swift in Sources */, + CCB7722FDB8E6E68E90F3D12 /* CreditCardOcrPrediction+expiry.swift in Sources */, + D2030CA1B3B7DAF3229189AD /* Image+utils.swift in Sources */, + 1B71815D5B3EC8AC21E2A202 /* UIImage+pixelBuffer.swift in Sources */, + EA0F179DF887D94E19138A5E /* AsyncModelLoading.swift in Sources */, + DBE3EE5DAEEEA44D6C7ECD4E /* SSDOcr+Utils.swift in Sources */, + 213A91107E76434972A1F848 /* SSDOcr.swift in Sources */, + 189DEBAF38F2FB88CCA39EA2 /* UxModel+Utils.swift in Sources */, + 8E4B117272F5B17A1E6AD609 /* UxModel.swift in Sources */, + B66697826FEA9FCE81DDD6F4 /* ActiveStateComputation.swift in Sources */, + 47DCE237223D40B1EB183704 /* AppState.swift in Sources */, + 040020B8558B0489DF99F8B5 /* DetectedAllBoxes.swift in Sources */, + 48580A7E92CC8EBFD2FD4C91 /* DetectedAllOcrBoxes.swift in Sources */, + 20DF2E9D5A543360DF3A4DDA /* DetectedBox.swift in Sources */, + BFFA3E8CE79BFFE47530FAF6 /* DetectedSSDBox.swift in Sources */, + DF352D12199B55B659A11795 /* DetectedSSDOcrBox.swift in Sources */, + 7C71166DACDB829F8CBC383D /* NMS.swift in Sources */, + C633115B02A0CE24AC55A747 /* OcrDD.swift in Sources */, + 41366AB64512BB7E4248A904 /* OcrDDUtils.swift in Sources */, + 8FC373E15C1169677F563901 /* OcrPriorsGen.swift in Sources */, + 8463F2DB039C481D26A24BAA /* PostDetectionAlgorithm.swift in Sources */, + 0838FEA7071EDCE1B633F093 /* PredictionAPI.swift in Sources */, + 7B0CF214778E17FEC721602E /* PredictionResult.swift in Sources */, + 4D1016654AFB3D9EE26092B7 /* PredictionUtilOcr.swift in Sources */, + 5382B471868AA7D95BBD2B3A /* SSDOcrDetect.swift in Sources */, + 35F2BB10395405DB6C1E31B5 /* SSDOcrOutputExtensions.swift in Sources */, + 9FA5A312032FC54B35BB604E /* SoftNMS.swift in Sources */, + 164BF41B5C522B7338376BB2 /* BlurView.swift in Sources */, + 3736756FBB875060C86E2777 /* CardScanSheet.swift in Sources */, + F0ED2BFE143DD07337D389E5 /* CornerView.swift in Sources */, + 20DCEB955D9E0660EAB3ABEB /* InterfaceOrientation.swift in Sources */, + 30E3E90F9C8E3D3E8FCA869E /* PreviewView.swift in Sources */, + 3FCC183583090FD0246D9336 /* ScanBaseViewController.swift in Sources */, + 0D8CA0E3EFF9694CAE4FB8F7 /* ScanConfiguration.swift in Sources */, + C57BCF07EDF419D36ED4E690 /* ScanEventsProtocol.swift in Sources */, + 9188179F13E5EA15E0419580 /* ScanStats.swift in Sources */, + 74330F51A91DC3A617F7AE42 /* SimpleScanViewController.swift in Sources */, + 0EAA2314FEA8D05D24C498BD /* Torch.swift in Sources */, + 7CED1B42C94A49D6BF98C9E4 /* VideoFeed.swift in Sources */, + EF96103F82491640651C49F6 /* AppInfoUtils.swift in Sources */, + 6E9908C612ECD2E0AEA3DFF8 /* AtomicPropertyWrapper.swift in Sources */, + 49E1959C680AD68D1D345D6B /* DeviceUtils.swift in Sources */, + D27E02429E5DC8B28DFE8945 /* CardImageVerificationDetailsResponse.swift in Sources */, + BEE2BFA103DE985D057A0F9D /* ScanStatsPayload+Common.swift in Sources */, + 54469A1A5D77BAD54061BEA1 /* ScanStatsPayload+Tasks.swift in Sources */, + 19F7EC09C9B0ED11A4601E08 /* ScanStatsPayload.swift in Sources */, + AB21982DC43977754F237756 /* VerificationFramesData.swift in Sources */, + AFA334F5007A4C141A96FC2E /* VerifyFrames.swift in Sources */, + 77BB4BE6E5E03626936453C6 /* STPAPIClient+CardImageVerification.swift in Sources */, + 65C1EB5447CD5DF162CDEC2B /* Bouncer.swift in Sources */, + CBA4FE649A3B21DAC3A61E5B /* CancellationReason.swift in Sources */, + E4EC278027AF802C39C74C48 /* CardImageVerificationController.swift in Sources */, + AFB2482D559BCC08E96C3615 /* CardImageVerificationIntent.swift in Sources */, + 420FF119A7147335D802AB14 /* CardImageVerificationSheet.swift in Sources */, + E90CC9715EC99F6B7FAC98FA /* CardImageVerificationSheetConfiguration.swift in Sources */, + E70A7E38A7D858031F900649 /* CardScanSheetError.swift in Sources */, + CED42084C5FC46C17E357AD5 /* ScanAnalyticsManager+Helpers.swift in Sources */, + A8D0A57687A7CF9F389D7BDB /* ScanAnalyticsManager+Managers.swift in Sources */, + 0A4DCE2C98659B92DDB29B9A /* ScanAnalyticsManager+Tasks.swift in Sources */, + EB061FA550AFFE2AB2E348DF /* ScanAnalyticsManager.swift in Sources */, + 1DE075A700592250D87DF558 /* ScannedCard.swift in Sources */, + 05188062E522359CE24912CF /* ScannedCardImageData+Verification.swift in Sources */, + CC07F702B9EC043ACB0AC1E5 /* ScannedCardImageData.swift in Sources */, + 86635536450EE7D583B8BD59 /* StripeCore+Import.swift in Sources */, + 36525A0F774E7FA055D8B525 /* CardBase.swift in Sources */, + 2D042FD2E8A0C8236596CE54 /* CardScanFraudData.swift in Sources */, + EA2DBA78722CD65ED599190D /* CardScanMisc.swift in Sources */, + 5488DA1817A0C9F780EF69B2 /* CardVerifyFraudData.swift in Sources */, + 43AA83BFF8DEFC09D406669F /* CardVerifyStateMachine.swift in Sources */, + F16FAC158C6B0A7687547B4B /* FadeInAnimation.swift in Sources */, + ACC3C1A295E43983F67F858E /* FrameData.swift in Sources */, + 2F3FA5E8CCBF7F77106A8268 /* EndToEndTestDataSource.swift in Sources */, + 45C8D17FED02529B7FC4E8C3 /* STPLocalizedString.swift in Sources */, + 84975E58102A5D8C62C6C4E5 /* String+Localized.swift in Sources */, + 41F2E4475B9B19350C31F255 /* PaymentCard.swift in Sources */, + C5DBC36A41FB70C5DCCC97C7 /* SimpleScanViewController+Verify.swift in Sources */, + F4E4941F5AA2EC2C2CC4F7FB /* StripeCardScanBundleLocator.swift in Sources */, + 93D4785D8B91F12B6DDDE203 /* UxAnalyzer.swift in Sources */, + 18B63245A933E292345C9410 /* UxAndOcrMainLoop.swift in Sources */, + B00954CF5F2A15B72CB94FA5 /* VerifyCardAddViewController.swift in Sources */, + 06711423AB563634EADFCCC0 /* VerifyCardViewController.swift in Sources */, + B47F583CE0F239881877E5D3 /* ZoomedInCGImage.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 82DAA855443FCAA25F64C5C0 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + name = StripeCardScan; + target = 03CF79975A56288F02F20E52 /* StripeCardScan */; + targetProxy = F7C4E731844D6B46A4FBCACD /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin PBXVariantGroup section */ + 6C27E58C1953533C68D43361 /* Localizable.strings */ = { + isa = PBXVariantGroup; + children = ( + B0A7946C3A1E943FB72F3FB7 /* bg-BG */, + 8FB602C9E909E31BAE6703D6 /* ca-ES */, + 27F4BF66273070C2054BBAC0 /* cs-CZ */, + 4B89D81A4099CC71A3FBC133 /* da */, + FFE89E65E767D03B92383C76 /* de */, + D532C440ADCD017AD326299B /* el-GR */, + BEEE30227F7D34546F0FBB58 /* en */, + 6186A0AAB96E236DEF0B5B79 /* en-GB */, + 79A802EA46840CD45CBECEEA /* es */, + 5A6D5C8E51C23370E889F75F /* es-419 */, + CB48DDCA458A62DCE342F517 /* et-EE */, + 03E2C12C0D172DD7DAE33399 /* fi */, + 6F41FBE3D90B5F6518E42051 /* fil */, + 3A9C2B8A3B96E194CED6841C /* fr */, + F3AF32D7DF7EE1420D16291D /* fr-CA */, + 9C3890F6A5A4FF68F95746DC /* hr */, + 0C7BAE904C7E1D7A9FDF35F1 /* hu */, + 10311DC6AC19B7A73A8F930F /* id */, + ABB38869FD175D7BD6BB3890 /* it */, + 8094FE7CCD3CDA8B914EC534 /* ja */, + F87D884128F85E7F16BCBA9B /* ko */, + 58A5F2CEEAAB78AA425A5737 /* lt-LT */, + FD40E90452A83FFBBF65C8D8 /* lv-LV */, + 11D3465E527ABADBD2485A93 /* ms-MY */, + 9593D6788C4DBF554735E2B6 /* mt */, + 9A256BCB7822DD94572DDA07 /* nb */, + AB64890F80D91D9D7B92197D /* nl */, + 8C227CA761586744976C952D /* nn-NO */, + 1C94EE6DB0E3114A38DD9CFE /* pl-PL */, + 45C9A1A26405DEAD8E11F13D /* pt-BR */, + 2667E0E0C7DC6CDAF03F8B40 /* pt-PT */, + 9E4CF9A676A86169070DD54D /* ro-RO */, + 42454BA950282D9ADB9C6557 /* ru */, + 5969433D88B5BD3B7F56B2BD /* sk-SK */, + 88F89EC392AFBE060336B63F /* sl-SI */, + 89BBC06EC107C8BDEF79BB3D /* sv */, + 3938F9E3F0F267045D3FDDEB /* tr */, + 0BB35800FB840EFC76240DC7 /* vi */, + ADCB34B81919043FA35E8BC3 /* zh-Hans */, + A612DDE1110EA970BF69B33D /* zh-Hant */, + 4DA27FFBD426C871C5BD254E /* zh-HK */, + ); + name = Localizable.strings; + sourceTree = ""; + }; +/* End PBXVariantGroup section */ + +/* Begin XCBuildConfiguration section */ + 5EA7D712D38589E2EE95AB3A /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 0197A1615C3FA86907CB6186 /* StripeCardScan-Debug.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeCardScan/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-card-scan"; + PRODUCT_NAME = StripeCardScan; + SDKROOT = iphoneos; + }; + name = Debug; + }; + 7F230CA2FE5A1AE4D1DC64D7 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 7DBB233BD36CBAE2700A4511 /* Project-Release.xcconfig */; + buildSettings = { + }; + name = Release; + }; + DCF5B81249E77660231925A7 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = F1FCC855CCB0816414A4BEC1 /* StripeiOS Tests-Debug.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeCardScanTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeCardScanTests; + PRODUCT_NAME = StripeCardScanTests; + SDKROOT = iphoneos; + }; + name = Debug; + }; + EDDF50C51DE9D6BEF2ACB9E5 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = C264A676DA41344551BA6661 /* StripeCardScan-Release.xcconfig */; + buildSettings = { + INFOPLIST_FILE = StripeCardScan/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = "com.stripe.stripe-card-scan"; + PRODUCT_NAME = StripeCardScan; + SDKROOT = iphoneos; + }; + name = Release; + }; + EE63655B6AEB11CB8934D19B /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = A67EDE2BCCC081DEDD684AAC /* StripeiOS Tests-Release.xcconfig */; + buildSettings = { + FRAMEWORK_SEARCH_PATHS = ( + "$(inherited)", + "$(PLATFORM_DIR)/Developer/Library/Frameworks", + ); + INFOPLIST_FILE = StripeCardScanTests/Info.plist; + PRODUCT_BUNDLE_IDENTIFIER = com.stripe.StripeCardScanTests; + PRODUCT_NAME = StripeCardScanTests; + SDKROOT = iphoneos; + }; + name = Release; + }; + FBC92FF6082EA51664498AE4 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 14795A45279E8F328AB54920 /* Project-Debug.xcconfig */; + buildSettings = { + }; + name = Debug; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 3942B170A6D222D5B5128339 /* Build configuration list for PBXNativeTarget "StripeCardScan" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 5EA7D712D38589E2EE95AB3A /* Debug */, + EDDF50C51DE9D6BEF2ACB9E5 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 89AE00D6B3B382DD3D3A3C70 /* Build configuration list for PBXNativeTarget "StripeCardScanTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + DCF5B81249E77660231925A7 /* Debug */, + EE63655B6AEB11CB8934D19B /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + CBB5D42795303F47D9D5F2F5 /* Build configuration list for PBXProject "StripeCardScan" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + FBC92FF6082EA51664498AE4 /* Debug */, + 7F230CA2FE5A1AE4D1DC64D7 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + }; + rootObject = 9A6BF50E5B02355004AC6020 /* Project object */; +} diff --git a/StripeCardScan/StripeCardScan.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/StripeCardScan/StripeCardScan.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/StripeCardScan/StripeCardScan.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/StripeCardScan/StripeCardScan.xcodeproj/xcshareddata/xcschemes/StripeCardScan.xcscheme b/StripeCardScan/StripeCardScan.xcodeproj/xcshareddata/xcschemes/StripeCardScan.xcscheme new file mode 100644 index 00000000..be21f1df --- /dev/null +++ b/StripeCardScan/StripeCardScan.xcodeproj/xcshareddata/xcschemes/StripeCardScan.xcscheme @@ -0,0 +1,95 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/StripeCardScan/StripeCardScan/Info.plist b/StripeCardScan/StripeCardScan/Info.plist new file mode 100644 index 00000000..cd4a496b --- /dev/null +++ b/StripeCardScan/StripeCardScan/Info.plist @@ -0,0 +1,22 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + $(PRODUCT_BUNDLE_PACKAGE_TYPE) + CFBundleShortVersionString + $(CURRENT_PROJECT_VERSION) + CFBundleVersion + $(CURRENT_PROJECT_VERSION) + + diff --git a/StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/analytics/coremldata.bin b/StripeCardScan/StripeCardScan/Resources/CompiledModels/SSDOcr.mlmodelc/analytics/coremldata.bin new file mode 100644 index 0000000000000000000000000000000000000000..77dbb917577a88650119ccc74637b654826fab03 GIT binary patch literal 438 zcmY+AUr)j?6vd4i6JLxmzW5n1DeJ}`ANDZZprRC5flc1JQP;AqUA8$AKD`7>0{7)6 z_nz}R$$7c@AIyGBnqgM&RGCiHhGj0RsmQ8ZEqKs7KmJ6{WF?fm(O1~+odi-I?FSU1 z6@S%&-Vgu}ngw}lO}atY!5z~}QMuZ@hY%`nD-}n1+{5kF(tgw)O(L8Y?3}e5>&Nk?^`Yr>O-PC0rzyIV;zV7kygu zuWbk{FLDD8ihO~|0MT^ZZyfts(@4Q$@grhdA3D%QUF+Yy-p8WOgT~9LQ9uJVJRN{< mg|qGwqz0;z+L7C!75fod`+gLP56ixQV4AbLQ!;UpYCPD!R?QBh*0LVlV;aYkZ6s)B~0jsgT3n;01Y*#??=3LzP(3Tc@| z#U%=fc_|9Tsmb|3DkU>FHLo}`Kd)E;XdckO0+47)K2W}}G&L_d6|6WZu_QS|p*XWD zRYw7&CN;Svvn(~IQi@HBMPNY#OHdCZ7i&^}1<*a!#JIGegpb9fR01#)(}{nVDgi znYkx(w*1X}IrGjtbNc6cuHIT%k8Obk4yhFKfBjPn7W{9c;FbTcr9wig2tG_0R24i% zU^Rh8fz<^DQ9~HS2#rF6F0f_=uT{Zo3mziW*9p47x&i|N>j?~^esCbnzd_IiHWU~T z7$z`?M#3QY`Wp)#Be01;NMKWe!9L9d&k)|YaKVFUE)0V2Pm7=nj1U+Q-q=XNgNPCa z!TRW+3v4N{f$%}a2p*hoYz2=CE-x@%V6cCJ;6Wq`gJA!ppbJbE7_1iw9)wsJ1nVV1 z7bq1NoIfCVaQ>j+Mu9ScLCAvxVMPnp1^*is1{uQog#-hEil7TC9vXCkn80AYvO=yB zJox&9+g)%mjW93@)CL2ADM1(d>w+#&FEH5OP$9>IF4P-?E-+PKus*Fq-b(P`{99MZ z+X!wH*tSC6F6cu4_CXhzE-=`?LxsF!(1rR=K^NFrV6eVRg}kfa!TEQqkarK(3+z!L z?-_Jq!|fGxfxUw+>|go>U0~m!3oBl~pbP9DbfJDg&;yv7wX3ZUEo-O!TFD?kdF`M0w+|+Ck9>UKPl(} zCkI{VKPBh_rv_c9pB8k1(}OP5&j`A}nF53JpH(5B9n1yJsgTbNy3l`K&;`y9y3l_? z&;>3Gx=_C;=mHl9U8r9Ybb(6+2IrqqAzvnVK;ZHU`HG+m8}2_r7r0Vj@c6c>LcTia z!Uwk|=mOUY4A!r!kgpfqC>(eZCS`Szd-^*e$t zaHqgv{jLi6Zoz}|-%}yqD|kTQz6yC}(1re4K^M4RV6gvz3VC+Wh5DSJ3(OT5tk0{E z=L;U3e?f)3Fql{RZ(ypB7X_XFZ(t6(K#Ra&e`|%@CU~$vQ6VQQ)Ke95x7WZdBQV(iY=!(>(1rT*K^J&IV6gsTh5VA>!TDdVkY5ozIRC2^ z@@qjC`d<&az#9UC{cl#tZv|bbza4aecLWCO?^ej~2_BsP{R;U5!80oVH+Wbfe-!k7 z|1I!1=mP%@dgK2FPl7J+Y0!<;|KHKGpbLB+bfNx5&;`B>`hfoiuYxY{bv(l|Nl6nX=h`%SCx#m2XFB2r8d099f>itVT=)Y zDmNNSs~WG)pK6T0e+!?nHqxl6zW^`ZKifELN~96Ey%+b7k1_sKd{8kem1Cnf!E1q^ zC`O=((w^#3>Uz|AZ(MOYunjFhs~0vv8)My|Yy4oxPvTk9vV>Zc#eUDKPW)w?nUpS) z1Ha-s@sAwyZ98nUNLyTf;yUU1{5s%bdX?#A;w$U%*l(gS;x^*y*)^gMiKeslqyt0+ zR6ZCeI&YRHu1N0ht|cC2Vcg%KKS^Bp3sI#wn3*lvo;$7a2q03Fp}MR2Dq8AT<=g_T zq_^;yd{1qefmUV}6D5QAp_V*5Dw?BtrBh>DJSDy~XR+jtjS;tE(hIVo5ag5NF%b>b zmuz9C@gu447^^zte#C@9DoD-rr4T9q+ZyOMOO5d7BNeyes&#ZKkD`bzRYOBF8#APj3no%%%valK3Rmlb?Y?QZaFG z|0!lMq=Wa;E*Gl|M;2R_QJ*cdr4$&5>+g7g7^Qox`}eIA^@lFOlQQp%15od{SA{J^ z^=y~KN6j$>3xK=wPNowL>ytM`C*^aop-GRDw#w_7x|pj-45E7$LsEab2A56-m@AHI z+|?v{?<;fn9H?-v`JDOR98u9FD<+wmFjiiS?2zZdr?IujMrFI=P>q~e9H?pR&q5_z z4bkW-k3rpA$p*R^{OV;TGMyT&=S|b};+hA(8k+MXd^ZhY)M_wy*9yl;QDp<}OTebW`>xEAiLSsphdzZ-qlX9%`%TE+3e99y%^E zQBAq`w20AjXUU6&dzpVNHO#Ojx$%u0wfVBCN)(VSZ|N|Pg36#F>boftbr08MAMbI} zms(CS@6pb~({-CY?fk8LUre_O&xrb~{W`7YlIw)8fwyJO7)cyG+4?cAu48=B?VN7h zi6o`jS#%{l(iTd+u(c_^;yZ-?cFR@A+IKHY@$J+n2Hq;q zYDI@{sU`lhBkBHuhNkTsYi20H;y&nb_2jZdSA3E^O1(bZF?HG+xMKPR5Fd6X=}bmio7;X1lJeU zE=(eO=5O;2(Py=n9Pe~+oH6N8z2NTtfBYB@tZR)osT&qJ{kX1iSLgS*)Z9>?zCvRh z)i5M=kNYV8KNbqXW_Uh2%rOny!aZ}2L0>{6`OCx)#YB@xZb7r)V&`C`iyTT_#vaP1 zNN1~_v}^$_Q#?^NbIf!uR?SmB1*2@kY`NH2dtJwwJiY2Db&dbUC#uZwV|a+_6J1i+ zU4bL4ED^b3j^T{Vm%78rz2U#T#5WfSXsfG59M_!25!04F_AK z`KKi=G%kHTz2xR8FFxvcp7E|0#Rnd}7dTt}LTY{EA45jhe~sR13-Cuo<%2gyg%~5Y z(ShsZ+WF&SJ;tP#tBgc_gE1sICeS437M}aM5dZvSpMPWjbEU^eRyB5*^C3{Vafopf z9>CwGpEs&)b5h?KzT#8Q^bRz_(~XOMCK{hEKWOZSB#_J13;0A&J5ScZrVhyQt+b&& z&41igv&g55E>5&(x__ZNnM}LK{@S@ZU7w3e2^ zF)YCwcn>PE&qi734L1?Seq!+lif;CkV;{snV?~7w30NkMT04@qS=({e))1}4Y*xoiMzD|;L4a9mhFy{l5T7^Z&J1VvDN&OR^P>_ilu))Sb z2U|M%Lyt=vt9RpFp%q9PIj^8ET!@AsH24zO294Z1H}V!3CSL~F5>#Hd>I@XhcU0C- zV2biQ)ii&ZTEso&MTJi8M&G%|Y1V@skyBVQ5J20dEJ<0T`CW*hGF|@y?%4?ZM?|)HUQI!CarX&Nmd`o-Gb`Cetf9!Grq`4 z`Pw8E;qlx>OskeUE}{LCVD=<-pZ}z^vEjBXbd)YccUA;yZaF3bTV1>2hPu+UZAFN9 zv!g0HgWkax0ay6xbcB~w?4n_0KuR=!%G1X`P?m~NSft|)wwZ}`3_< z-cL3#U16n|7)>SSsFoU$!^x28@Duz!G_eE;#Fehpeb!;Rg@&ER9eOKJGx042nINv6 z;wkjG#dM3A8e+GB1Bv^V+lft>hTuxicll{#xAOqEr1&0Y^VLjgsw?ou_t(aPAn~9kGPk)ysncv7}|skaFM!K%IofpIu|=XrH(5zrX3T) zPe8ADwfrKsK6}Km2g#;C^9kabj){!dTQuqz{$R~zJf>A4Zl8x3Z~H16Ml7mpd>pza zFt0~5JezNRP+1=~M%+AajCJ;PYBh|;!|rv)g0$9So7hG7&XHWAl9U68kkxE8Pmk18sAy z&8&dzPS|;dspb9ZuAAG%d6$=Ar&XBh465^9Q&66IhB2;=$TasjXur3JZ>f38CKNwo z5;W=T@#2k6nt!E#?2(gIaj~;hep^>2H(;UI1jRG;dSxBBJ5~?vt8%AYRPEC=)fcN9 zs$Qx0$s$oy8(Q)=Wf)%Th*Z}WpK<82tGfqqRB>TYY5 zShhA&{X}k%t!5Rx(u;YIIeL0#xHoa3?tTsoe$Q8N`Q_ud2abIStT)YeL$bviCBJQY z&WFnTHclr-k)4QdbOD)f-AO#N^a75N$LUn>wvzt71)6Kx-I{f}tEKH!%kYkej%!ha zqjTtKTtYoJ-^IiwyuQH?4u%C@1s&_>`MLWW_f2t zA6ZPnH&R7)wbiE6h}Pt0+gP)P9cq8cSnP4m6Ly&SX+OY@HFqQLT9e4(jG0i{){+O! zV%Gp~sQZq4k#Dm5rZdSq%c){+*!65ad(ZKW8^+w`_ENQ-QpbMugNJooSC#K00UMSn z%>xWMRpr$&5{v~sNF3DEKFgWQg}T;zUOB>@Q{Bz&*;(_{=Nz`ETPbF@Uph+F$ngsR zVJB1rI)*+&bn=17R{1W7rPBF6MJi9Gn_^eHxAC{B=lPeA_WWW;dnL=Z;*AQIDi%!@ zSJR~F9Ow|mDRn3Hz?4}!v2t2zor7onGn5;Y-L+xrktr$2QzS;oQD2zfcF3)S{!zBX z+M##UE|uB+xFpFpkx_8ps0H+XW(5;Zex&vk#7hrJ#)}exr?PJ1MA1^pACg(zhKm9bc3oZdJ@cO?vq^;(xK;j)lR?bpPYmpxayi z@ec><1Z@BA$L|(53N$&>&|kNEWMJH+Vncmd4ddwkmeL0k4;iO=zxj9H2{TswLr=2W za>t9h$%_l9of^qm{?uCCFCjp=@|^k?1E} zGsY%rlR%nAMn^(TVk~+63+9R=r6Th(X$hSKY$sABfb9N(M)pj&Bm7r<#@q<5QkW^# zOV$v3@?}7ENtcLx06bh z*r_G_FsLtISvJ(%8HqA&5`VFG0-syMY?<;uWEj2LJ}c@sGaC3P?ZJL#Opw{~g1-;k zQ4Ccq-~Ws58@&s&dVg7Z8t>w!Py6{L*IZOM6vTp-gYpdZQf%`5BLf@C2=k1jJ;TXo<7_5nE7Gv$zIC4U@4O&5^qgQfKd1!I8Sn) z>2K|3>kAztUQ_=Dh`+yV@eBY@3O5 z&@cWIdy(3UMWPM(;nrjPGPYXz`D#5NH7f_Hoim@U&rFA&<}Zzn;BQ--I;H?Ot#!?r z&|~&K8V%hAXwYlf3&bZWOpB^U%GPm%lGn-Bv%Aon340UI=lzlCpgwtj#N*&Ayp9Ru zc&&NfHt zwk@*E1TEr+R4t;G{pFrfg~K32oSp5+pytOWjWqzR6g{Ksaj11p>}I%YLORn%?l-GB zKwh0+iL}^XYW_<;;5V74*ycjpm{w6^WmP4o#6{dpq7x@EA;9axh=L|amKg=wTW=<$ zCVZ7H1U42uW?skb2MVNztRY+(i|2ICc7i9shNLC>q3khm8tB6ef||1Rh?md;Xt?#C z!fIm6zE6POY)f>o6L}rkgmDn>$YQEg)*p!HtHjqU|1RkfZzTU^n>j$xM{;>vMeDH_EF7Q9Ohebn1}2|pEdZ6*&(aCH zNTgGg^<1KtJT9>TFQ_s)iaC*Qv?=WU3KrS)md=D!av#7XFQxJ1eee*yB>o8SPFyCh zBYDD%0q011AxX4bmS%n=na)qKJtr_J04i)pllIC+il0l&>}o}yg!*z^Qiq8pE5qrc z54IWR`_kuk-%9KoLrAaHx-)p*hIe~dMMRx(!mxz3+o zi^H3!@8F4dlKh%!1l~gN78h^7jQ5)I3|EY=Y;@b2_}{jfiLZB6PgRY&fDixt0H3nv zEgpWX1@7te4X-KgiO;Sz5l<+}#LWSaw;a^Y*fOUW*Xcgsb;J+w zYj^*_-)`Gcy6s*7uXP~I7{6oF|7U+)asRy^o=8t7q@s0{+;$!~1kB(%a$ljn)MDvt zXea4lJA*h~6-qX@BTceFzXti$QL~zCdzu*4<}Uv580P6HmaK2M|;s<>Vmwvq(<4m zz1RDVhwRm~U6@esN^&$7a4vP6Wo*nZXGcC+(MGg^e;(6DV?tg-tCcgAQ*9wJA3Y=C z(Z#2A^MFhPW%^5WRa`DPQ@qxFMs>%t3Rw+)g=;|9%m>Ba#c7a;7~KNJHqmR6bc$K_ z)^UIHn(QVj>cBMTZ#bWw!el~C#ZT-Jw%XJgNgpZ1XL1rg9c8g|_Tl1oM0@+FtQ*+;gzZ#> zWSeA_Qq9$qXyP(R4m%t7f$t$}i5!6raSIg3361$_LJT*V98?HHYxqt~J0-ZctDWV4 zfw*)Vw;CB@xd^9;>Y=4@A9ERZ-Ca$=xyBf-X?A6KwGR1_f`Impey;MRI~!I><~B}% zj;Yt!J8>i6_iQz1nw{WgC5u1^`W<{>NCerH5Ul&!PPMDCJBqXD9?)n|U6w}*-q?O{WU z+7tI(vw)xQCbAL}0-cka?G1?b)&b6Qk~|-h66$Imzl}QsUWK=625H~n>Fi?fZ0@lA zvg0H)(Q!AWy#`a>1T5S#v6D%(d=vxXB>GL^b7X;ZnRHKVZ~jE?2`5q7P1)LU-J8pA z_J%>fRiAW=bp)`)ne1vJ8?V~SW`h>C9}?pF0-Oh$6%5uscSrdw$`{6jL${2L(Cyx; z#`xAd7~b|QxefByoE)8wu0}GY4S|T{{gRtzZNW(S(j-Ho)_ljz6E%_9rU_BU6WngK z`=z5jGLMyDoxwBESD;N^Gqp$E+ckyi#mDiL5z0CNHL+$i%5gGl44(^LC_HB#;!1Es zF{PeuniwUCc2GR#`&)3nSd%B=m~4J4RMqj1&dL8(kCnA7xtzBeIqr;5OQEm4QE63A zbB*P90Uz==nY%gShauLSBo$a>FO==f?Qh!UU@#5pL|DkcEmHSlk{s8p7mykb zr3gjZn;|MvDW%RTyy7obmGqahR&iI&L+@5~I5w3ZoN@=B+tjGItdMJ4po=6s-JM|s zoeo{+d$3v3iQFFU7_fmm&*;G4<>!q=QX*O(H8t-CU&V$seuqvWGS~;u(tNMHmiZLg z%Tt>?4?N@ggVhSXg_QD^9I|p82Fu|Qjb0HCw=>qZiTC*kbeST}-V&XS&6Gv+Ges8u zv$c|GgEcf^n>mY^DPifQMMp)g6Yf&8+0KlaeIij{(TeEYy69EMT(Oc}2V(YGxwqy0 z=79{f=tk>|h@S&Eyet9>0!jD7xo* z0e2!4viT5a9|QL#3!w3`9+Zb1k=)ah^!8SnKr8O_BP3~j1xFt(Vw%b1e(3TJgcO8UTq@$2n2;aiFO`1y>G)T-1$ zW5;!;1Cw|6!b657`+J1QOXEj6@P;ER84kCh1L^Yz26{BAY8>2cv;WY85BOXqRgx7} z9X~Q(Q+`G0q~Cg?Woq4hr~G4PJ@WTFa~VIsvr=l_y4C&@%2?ceqH@W>P>aFNKF6E< z-efqlutDmu-I1whXYDO9jvnsEwu~`emv6%(9&|PK(%X!~VwMMTO2-;&eKhIAvvzA+ z-qZxXV6{`@FDy3He1aOz_)z>a7i~Oy$8D_I(c+)U)xg6HJ4^q`xP*tS?~}H=@3NAs z5uFdlM{f7mPwjLtZO-S?yyvpg&d(E4=>x6**W>?4R|DNLB49MP3O}B}opGvBiTcr@ zS0yvx@49RQ<53#FDt-WcwQ3J48^OIQDHeCpUh_S6J%npz-z`nk6sk7|Bzmhn#Ph@5 zNH5a-)PF6h!mo7{K(HrTU(a{kQ`bnMbKJvBTh z%$-puA3Ua!IoHy#58a?+kY|*PlTbX%HmYSIZN{$UVzcQyn{qFyDO5{*q5x*uB+_ zP|bYR^&^eXil1Q3T+KcD(ig2v%JI(UhTrVk(so)c_Y=%wKgyP9YAZx!YEc5Tmp#sA zP+Ng_T2f~u0?e`Eubxcqy|1e3qk6K=>{ySw6>>J531@d&dO27Eu+>UvMmu!Il&*e- zy1QnCccOoD$&u8anmo9bKGR`RUxHTqhIz_->E=RyhvE+jdPnN6vN3Easy6mf&d_f! zK{Q8w%ZvY(A4y}ZA^tJm+x(Z3hmPaWDsQ2EneMv#H&bBvNqEq1>Y;ACqcQr~u~jy$ zsDZJab}Z)g&SZxi8m_p4*D_?gyZSceN!1(O&D`^hkTYMk>2MQVY08LR{4e4; z+)le$^IR0~PIvuAd%-oGi%Ycn){a!i4zO+FC&v(f$YGJrpsU=vlItom&kAEA9e8jz z{ny(kEzP^yKE^QJu};M&sFQW7j=J^IGUre4>WCCa(h*Z~geI}gCay2Gr}Ps$#ba^2 zQh!LT;ybPA9oVP7>{!6IL~&I+%`EYD*;q>tZ;5%2y9&J)JD2j0X1pWDvD*EYm!og- z^~KBlUs9Z!JAS5o`%QPvw*N$y$YSNvoTdVAC z*yLG}wAa5(lVIARlsa0VQHh!M!@6knJM`3Z*r8>j1CQ~1E!ZmU$RpPP&nfkDHK;Dp zYv9^UI`h(f%hR|dgV&a9Gn`b!`L3uXt|RJB*iBJQdp*T{vY%qSy0f$jQ-yD0|5u@N z*Ha(WZ1cV0z1;1RK-g{gtR_t}-MiU1kL~Mg$l2g=o;9kb=;lM+Q|~Cpm5kT^@Kr9h zmFz(u(l@lzwW~FD-Pqh7B?0Y4Ra3GavJtIT(hnM^smm;u)jd+}^dH@6eV%fLZ@Ig> z?oIJRyq)35p@-h9`r1s3;yT7xnx(Fzsh4zm-wH#hqn7rIeO$@CLqK}t!;3u%_cM0` z**5JrAm2SnZ^gewS4C%P#yURgjyQSXy!vR8lxV`9Lf6TE>e8?WDQl4Tib<|{Sar{I zUAX@cGrV|)R(+@ekX+hKx4QhU^#+!$5=SJg}DxyWycdz_{FRT9I5EcG*GQp-{TM4 z-R4}XpCX&zM;5D3CM=bqv=}k zL#~Q^DAPoJ*tJ$ZnQhBGAy96YC#_IV-$Vw>KIEU`Kf((8Kt#@y&YozDHiZA23o>$Y zmVF$vOFowUS}1Z{p;Fj%i%zq}+y{Bcd@0(F#Y5Kop3q>$M3!>gaUM}vAq^97-#{Jm z^OB8BrktTk(R6D=YyozfuSEToCBk3PG`;XIlz09q3&MmBs zra1}GPT-4TF?2#iu|u2<*#xvj90o^|7UyKS0@@)tBfpD4%xiXMwn(0Vjx6Zq`qcak zwZs*}PP2_d2k^h-kFXz#P(>4U9azFQhK%yo1t?LMZAHvQU-HhvFZ^DlzjUv{C2{dm zDM=h-QnBHQTj*K}35xSy(c?^I#Ye|;$XT$Cy%}DU^rqAXcKcxYyQ)kSY-Y zDx(LP`2Z>X7tLaG!SWL)_KR!)zZjepS7v)h4Pu)jCxC05oIPSQ181#MxHh6kkjvgA zs|$KR?x>@L`c3`GtuL*?x8pX5PbV%$U&);e0a0;va;LM86^)3S7Ay3F%H&vPnIwR{ zlC6*V#b=T(^dm5wjpqfed(OO(@R+^l}yZ~r2f(oE{P8+=+Ib}JQs850q zTtGG>o$V9p?OYr%Z^LKUNg4T}t80<_!2p`XuPQu68zj%c$p>m%H-lTFqp2zKqpIFa zSfLV4#wMzg=|jYL5GU8m+OvOU0~PCC&kGjYo)o1zI*Q&ZMgVQmONtZRaZ*dAidH(` zP`}_zdOguVv(AL@A6192A?i?gy&}`u7}<_CXZ6xqNHzW;kML@2EtRFG{06B%@rUm-HBYdeK+%ur=)0-GDo!#jbIM(f6#m8 z9q1hPnlp@lg+(b-`4x_JPdRPnXqn@uqjSMQNDoD-&hsCg zPl2I|dCXjRzbuKp4qxL^;obJR=u-KQm`vAUxFdTF!m+b(8Q+;Y=MqtUox@#T=qB5- zaF*TgEa2zBVtY^QJ$#tTq&G)oB30o|Y&cSdx`$4c-9QNND`p~o6PfZ)lAY{G@I2F6 zVKs}04Y3RPc;2vY09S`JGtC_bJ<^%MC+u$gFmPT>WQQfRYI$#UYC44k` z2lFWJsKwL;q_sLt6ROUD*+PY~xA~m)4!tv8Cw0M7z`^hVHU;QLt%G)fvlFF|$JQBK zjA$&UU9Xt%yzzVzmuyF|=HPNw5rm@}_i$K@?t~{pImCXp1-aEW+W9Zqn5ZVtCiFBx z4rA&;rxTa5w=7H1<0yvpW7;~uw>+q>D_2I2O1uS$l(W#$j!gF#wH6$x_#;g!jFye& zMk;@U3fBM-RRC7NJsm5JeU1)g_ynAvM6NKuWI8cF$pduNmV*-B!PDta)HyV1XchmS zWjXlWV<+&sn-&-{sTcUt+x%dY@`b$dSs-d_4%kPtZZ=?&)IMp!F zVU*XW|6+Ym+#< zS0@Tzb3hwt_czhN-KuUJuAR(B}Mo{AuIq z{%3Qh`a8F)gs*;bsPyQx4gQ7o&f#xM*W)=I%=jesx8Yv$Exg8cvu;Ms3x31s*8bVM zTN+Q9XO+&Yk&X}R{Q#fz>n${k{V5lkPDb+D9QIh2JkudQ*^Kc=^mCDJ+hBGVpA5oNGkOS(m@uuR;)g;-AQ zGLJ@TBhw4cirR~dMElWm(i4_lMF%Y7ELU?)mcz2^Nh3rZ^5#mf=hTT+NU~&it>+mv zA#=l^)wgl4kP4*)I#QUDtu>sl%JoM2M!1J$m3XCk^pQ+{)I0i7q9^d zgZBPO_OK8&#L}J|Xj3^;?ME%|;gRws3Cm=;MW#Y6_y##BIYr*Kih&x)CU!ZOlfTv4 zIc_PvI_YA3vqZDavu}xLwe^H#y80aVH}_=DE_DY-3uw8ErlzZp@ZZrIku3j5)HDAH zGM|8{lLg(uGO|PBJ}CI1~6Z z=mD?}_)Pl5G2-^p9YS43`>#CYmgsO%+ ziUx^vbi{#fh*vSemM$@ZrRY`jF2{MS~`nu~9TmsioezVnZUnrmv zy*N%%-Ep)0J7f*wHdcnr*W7f6xzB;|(6xeRGPA9w<%c<$QHl;oMo72WT3P2r|0B*a zi(F&fOVBx9cFlmxX0Y3BXv zlEDUF3p&=?z%rOHy6g;+-xr$U9KiBBGcXL2*!N;Oot&GkSmDdn5Ac<7z1{8Dd4$Ja zg&)*%JU3asjB0@{ArIz+Ds}=*Pz_;W{iuk0OKpYrL7(gxs}ny!&&fp0O>{k`XTpG) zRClt?GVTK=74HnYUF9Gt-`(5c)u$$8}iHeaQ;?~#Y4)aBN-RXX=_1$bT=YO3aaH6W)dQj~y>so!^6c4c0f0v+mn}R2mLtN>Z8r zWGt^ zXd*cs%Vcw4sjQANi!F~GGrBivKIKMAvpY2K^+J9=pHR2fLs z1vEu#k*9DyqWb=KMO*AL`3Ib!tY76IYpFXj_VT&BwTtiB9_L!6r@tx4RN!1Qai_6TZUN8t4BkGe3T3zASOiXkH`qV)qIm`4|;@eY$ZNW~&ijaW?%{AeUONFo5 z!*U0FnJ-fqX}`Q$bbqp@JBC|;M6=ILADC8h2QW@jXl;eeEP4%(C%zo`2AzrIKqWQV z{J}iA;HIRlOiZ7(Wmu2N{*BupSsfiv#Y0`dSau_Hf!)UZWRAmW(n#{5XqN3~bUU;L zl)z^bKdF6=I$|q|i+W>Al;;eK7Mw9|KD5q%;?5ewkZV97VREC?RaXlFbH|T2Tt~{! zg>Uot2N7;qS-Y8`_y!iZbu!+lTrkzxp>|K>y3a1d`0=Dsa;ePNVrz-vPdA6L!-tTA zVOM+W)^-5>1+R4a#%0^_(KCsX?3&F|S1tikRlWR%PLebJvnMv<59r9$R_POsBWu)8 z?K@J7S6)5FP%UM$u`f~6SR;9c;mn_e)bJ|VM!r?=R99)UK==F?#*Nc&8glnO)E^$c z9vIO7+O+`p?*-PDfr_LojtyUNi1a-LDu*lH;J`dDvxh2iAXYN=r= zOn>I?r@%>WEk5JDO4oa~5CWezmeb;cH{2@N#TwMnUUXePc`Gp%%?zON=q z-#{tRT+&=gZl-+6)^R=}yP{vEH#Jr8C;Ca+QwCIDg}vmBEN-Rb^|aw@%4lO~>RGSe zu!TKW^3Vvlzx&egGkk1pEtOyCA;ZwJ@~5pYbF-5fyA~+Hx_cg>-BY@HM2V+WyV>ii z*_!X{1$~NP1M^ciN7tC%4)N}I*r)Tl^2)F1TtioS-XVk#s@c9}-FSs9{Y6FizW9S3@}u$!52xhoCz=_tE#D%Dvh*Dsa)? zP4(Md$BG!BwDoD*nG22v4ptne|E~K-9Io1;ti|?rC%QiZaq5ohhcY<%uJt%k+cRBR z)iloC8-E@@$vZ=TAud|kU2QXy%9ZLZMIYf-m=2h#7>~}8S5?1J4;NoUNo;K~>3dE7 zDlYN!$S~ridkR{_sMxysJGfB$OA$l-OEj03-?v34D#fazlsoDP$`#rKRh<;GQHo!O zmXwHneM_94hRgu90Y8KD$RSOb?y0+uubNN8<-j|L8+I>qhEYs!G&ZQ0J6+dO22K>(EY#L-LQZo$aG!BN1tQp<gk*>+=JB-CDH!J z@XwMJo*F<+FdMC*9E>Jt*K0;9l-L*yq?VbFOZqS!JkPyZwmi?x;@NNy)lzIelBEgP zHkM65m#H=<$0&Wujo?T14b3}frJB;HBh>9zbcCwtyXm_StzA7_ zN;}V|@@K6d_!IUBuC6=Vv7C!=F6J^d(YmFo^c1^#0q0b%U~3ky(zW;b$fNibe=nfE zKGQd={5njW{*1Z`x(eXkeoLD7x9=t~4!I(`mlRh1>}YXP1MC7i$Px~?(3_eq`ZTS? z_q`<4*HOM3Ekq`|ueiJODc)D!s_qQ5tFDW#3wFow*RT_#bP`=nO?*m>Vxtc)-Ba8| zxk9s39bOXUU+in0lBAcV#48h(vHS?e;Tj~~;Og&-bM1EIdu5(g@KEGw;&dnpaR6sc zxyk#@i=7I(01kNUenoLT*@dy|K+$Pp*TzjqaD$=pLSu zlrqpe4s+|z=?V2@{UbeqG0Ka|?FIc*t9^UnI;vZxmlZnfO=%^O(mq1js9-%aPt!m5 zD_Ik(V;^Skmz(0Coa^G!*yZ-VrsH5g&3E@>#jKQcUuPg!Inldc{Z%_fJ+0VVI;Uh3 zYNh*0uLAMf0VyrveYyz4GbA+y)TP;r9L=1~u#4(ktg^a#3je?U-lpRIGgY1$xPE5> zzBplp@$pBAf9rCYf8En5{?q^R{zIXN@%gD-L&wjJjp57M8-EqnFlP10!C$nS?>SoSb#uC0%#o&7cRJZ1@K1lw9(i0eVA;1?uO*3lFOEoL%JBsDhaV0;4HmZ)xR z3)9j~Vo~zWq)p%1mW zjmcE$P1`pz6RK8tD;~|M4%L!xCP$0J;K`z7`SOAmL>~2l+_V2k;WP0qMrRokzaD%o zzRpNu=|VKIz#QFljU`(&*1AsC*}?(y8WkjTr45RNc-Hzve#kNts@(FNbX+3G{(`F8 z#7r-uh&*F1F+)%&05DCND0DmfgZgB9L?%MNSP?yk+MKjlWPvx5RjqRpKTvjajA@@~ zy@;2e220@ii9ZUa!XC1OlOkeE{6_dE$A|wKy+b$?ju*6tw{kat;WDubR5Zn#2zk zn}JxODlyhlB-s=>IBaulf=P;wmoHD6s)(V#FE>AYXi6?tXz ztfEaOqVRsfN8&fsoIC)xf$GPdK$fv~`Mtg0D7PsOng+_5%aR7b&4O#P;b0B(Yui6? z2w0hLS?-HgCB<1*7DRJ@#;Ip4ewH z-(&zJf@deiGN|+h`CS5uS}=&HIuJ(>H0!yxaUEn~bQRGpqB;dLJ7oc&c49R}Hld8a z4!#4Nti_ZOcM0ev%OOSPQSzF^N3gRbl6Fek$Y+Bj#N>}CJQ7nDsbtS{OPN?xf~Brv zF4&$KS~$+SfVjPF8#qMDNC6s=rCReP{>V#mXF>z$ws7!psR#>GJXRa5I6Pql_@(c)w8%(6z8v-}zRCq4xHhm)7C3OR&;&Aq}ZCzm|=b>}O zZ3&W0f%2rPmS(Cp6#`wf_9*|Y$qRWO^HwuQj3JiO1&~TMn{J-)kliPLL1&Vy#VokT znkZ=noY`NMUTXei887+_79{kQe2LdFq1J8kR`Pxz#yo`gu#ov3J%O1TM}zap?#ye5 zC59x=D0EQw&FRVa;c;*nL_)W1CoE1HwS5uqD}U`b*vrxkX_<^nerT1%4zs2=j{{Pm z!Ne+Y8dt)#m2Cl9O5Oom%)Ml_n4>X?1^0{Ya2%Z=`DVEc&1G8Imv~db z0_IfWD$y@6B=?H#dFD@|3*-k+iel-BKvPI2x?z6PvMt?(^3j9oKDl>*TGV~{L}Hw{ zd16=D6|%EP%eEoDS=$uU%%2BfRygk=8%19MKQM*RP~rqv421(#fn-J-`;ciZV-p8M zP0j0U!=bHNlVjc#hDxT|>hbF25Uz^E0ZyT+*qT^kCD(1+%AZWBUbqq5Aa)h)h9U?M z*=JuGS-bEE(|ugN_Evh_5>-HhbqX^J!{xOT>#-lpe|tOHa)do7$$%rIL)jRh zi2O>wl8$a+v+idaTZhx1McbJ&_%KDBjarI7%IcHhML z@w*_qxQb5%1>lZ0}*lIyo07=!(>jGMC_3@0VX7;b6C#YLY=6yc&K<1Q%gF-+?RElx{69f zE5XmQUb2OfKOznoZmH6;rHo4W%v7`9bp1aJodshPSr>+JC{V0Oky4AyB;%QhHEmMd z-Q8tzcXwZ`P^dSa@l33i;O?+EELzwFKHOoy{DaBlCimWRp7%NFoCDYow{oN>xw4wE zI<}|mou`y&DpXNW@|kLn^#y&<`AF6Q94F*>_K6_ULYB(jRq*+Lfy3xmp7xLmKZJyd zP51%rH?*&6mOI;a%)KQyA07_@*eh&88^5ZzVmPhlYN_4mbVWZpo^Gf(n!1O-NPOmd z^EkT-9miiR_5syfeq$FUobl$d@4!vWE%gNCof@)Y&f553nIEIzMy|p1O|Ui2qL;4ZcRw z`Fo!BZOW`~*uUYo^Z;l`L0vFPcGI0yN$?5VJXD$qB*;QKkqmM z(adi6D>K3V%E`Nqr(A(K#Y^NN+Q+uQdzU*Ui^oLHl5j0|1Uws>#B^q!q2L?I{73;rt{tgmX*bYm(Pbr3q&{P3EVx+iVn3!283`}b2UHD?FSqq+#@;zbQDwT{PF#F!P~Tkgdf z`CWFHE1R!2pQ!)T{QZBI%^z*g!?BlJTN*lJE%8s*W^@IzD*El2WzPIo*Yc>=Ld(yM zvn>fN_M3mMpKt!!yoqIGKW+Hh+}q~8hV+b%HGi6q-282Byfnvbo7mBkl3dr4IbuZ|*L$YE>PJbyh)m%TlJ?vGSgI!2)~5$oC=hf!;kVL&i6?K&}2x z?>Hnk{9;^8w9bFb%-Rf=4d-f_BO}|HnNVwsc(95kZgUIE*r|wR6#vW&Wz07>uK+CD zbIlf8kH(gjvK;f;T1_mEs~t3V0sb*>3qLVy|A~)W=@D7M{U4 zunM6pL>SL{_p<+L&jzXyE~$n8rtehXFxE7<-POCzb1AIP#2S}&$STC&kxICax!BMI zKVpiH_@YNK46G`ciR$POt~Rj{gZ6aga`OW##WLmwUmGarS9*K!6VVYW zC9&Q=0Ix@6x!bt+!snAt5HJ0Y_yd8q1cIk&qQVOCEAbI3r24xT`1*2gU44Wn54tun zzu23Q)lmvgcRh3-!&qSpccx%2x0r9wk50YrQUJTr8-a$@Lw=KIHoZdJ?(a>uq00OY zFhxA*>n{zV(vf)o9^b!A3vxNSnBPtG#;5r>Pd9xXoryAv<+^h^mG-yEqa9nin@r_Z zMxS;`aA;aq$(ocD(-ZnQ_{2H8I3T`J?L|Hdi_r0^vAd7MOPTsD`Vmc~wve;rB@zpA zp-WM2U&D;&TIkK%A<|HBNy+kHo3vKm+3q*MYt=bu5kH-6jWWK9Z&jqQl-;~Z{aUE&T1^gWT*>y*w>Kk>fgz~_NVH1v*y4VYz{lX7eM3z zt9FLeL_AI{PmeN{)KvJ0A;nZJc#*hK23GW`_(y*_eQuy%uy$sSs+Vg9Fh-y~3VN=0 zf9h0H#~cy9h=a%;+|Rb@ijYQCTFqA?yk-+5no}1oh(1diHIF{0?ZB3xYW}A4f3|g2 z9UkH`kqvmZD;b_1{l6B3zkqF6t%A1nT>Z@8Y2QZMBG(XNyEK$5)|VPuiTT0yvfcV; znoVhih}`uEa@ogd=8GB9aPYj;lf1`NfZee~?>TECqygKb?`b8Q;H#o#q?v)m`sw;3 ztg&QBl+LP6yfW%l$LzgR3S7fIp}2{7kqbl`V8`f-30;y;dkSqBAjk>+hu9xX1$WRb ztZCLQaLrB4xvt!EJPsP0MESSuTNE=hckjObS)T0bEpP9i6}PErOkB-`drd~}Yh3sA zzH6Qn`mVnAOdI_vWV&W&%P!i6)^|ZHeXzHP93T2l-4^asRg_oJ40k8833i;FOANqg zXqM*wOWedz{EjdJnyl7gDcl^gJ^#0;5jbWRLm_Ph5U%N*U{AxA&}^#GIo6TO{LaGE zrsSRS&cJ76KBT81?yhVhP`&)LdcWoXU=O{@=ukF3NLt#I_3?gSYNg#Kn(_AXtiUz+ zpm>=$#A2?$k>!#X`5dg{U2OQww=BEBT+^nKesP=EW?u&HZq*j)AYjM`bO1kBaB2cr zIHsd>`^ClYk$QHFS(g z>|aPY5$k&J0>C>hir)oSX=(CXJ)zin(c0^ak11yX3N3Su|`;j|Hts@?i zeW0p9b*Y?~&uw%sr9S3Yfc;fYxhJ729M*`Ce}44edxa{*^^&zFK5+Ft zwWu$;Tz?1RaY-lbUh00(0Nyg3rv?%OeO|5G?+#6D{R$9CeKd7kiS3h?p!K9lK8ddj z&&6(tPdSkG$zoh4X?!MHSGb86(xs|(#6&@2R#JarUxnp(u4axe z2rc)0C~C|t_ErKNfr`|LlF7fzHc`Jb_>Z=xn5UbjeW@X(k>YU8qGT1>&Y+UrXG@ih zvCi;DSy!|J> zvHnafUw5|TM>NU%R@2l!S+`pP#QNxWUv>PsXCu;|?q{D2cg>w8O*p9Yk3{@{Jj%n5 z3Kxb}l)Ea9l^+VU_c|?&Osh(?C5wY&ZS~WhE1qLNFchopv!|H78hp3_1Uu?FLFH%@ z)f`!0LFR4f91D0{SCIO`4zUGy4r#4u#crl=lVjW}ZR-OkqE2PSUwHVgsLvIUH(MAjg57^bAw_#WGJ`SwN1XzxdwZ$ zZVv6$GH#AIgpBldK{5i5Q-@R6={Nd4WG$aTKR`R2yb7fUzBnHWYW5kv4091Nz7SA| zMwm&h6P#xgjut%Pz3hWFd9t5uI^BXAC!X}5O}v3T2rUyyZ|}eWKDi`bA0t*Ji#3;! zuKpUL#&n|WVh}?%;d7nUi&w&T_=~OLWiR+>PaW`JGRLpf68<*ZNa=SfAUC^0-h(p0 zdiNR(kQM<2czyII(hYpU7i&97dM-rG7k;WsxONDnsg-n4DwG!C$@#OeZBOih-r$G@>Jk&2=bc}P&Gz-!0Av)q zQ`Lz}(3ELEiiqw!Y|%J53pg6iv$x0A^HW_f(TU<-)dNQia@&3y&nFyyrS1qlg)}-Q zi7$X%8Y}(~A5l`KtD06dRKqA|UoZnb&E9SW>TIxcdq>Bgm8MOOA+t=)BPy#Y)IP-Ge`hosm@EK{Vi8*t8)%lIiMJ z@?kgw$%^(DH?SQ&a`;SCYL-X;59^u=IgNB}mGaN^2`8MpJsAaOk#W&n;VR-7vli(L zs??pCcFcRGB{W#styLE_?90<}EwaErppW7K|4A}db6-;!%w~V-K2XCGMyRjszK5S6 zJ8Kp}RlREL7zc{A+WN7(Oa|YD{fdkb-b!79!#u@?6z-lTln+sy?g-vPdZ_%Iz9Ljy z*0TtpuW4>z$C*9$6@(W&Yu}^Ys#wX+6{>0siiIrAmr2b-tA&pEGU$agz}FA%NoFS0 z6n<$3vvTZ(FRUAFAfPHld);S}P3!9~z*;K0rTx@BExWFdK_(z2;s&DhkuapcE@?cwB7ERsd7 zt(ekrktJhd(~6A!vnzT&>=l-IcUlITjFu00iwID8qN2&uU6Bcu#_~q@wV4xBhGtgz zyKlydYVj84ra3$)ub(AnrYiGG!r<`DI4E-T>{Rodjir_ek1t!0(W}fY+&LBkTI5tx4K*W(KgV<>2)$otF8(+I#3dJ)59{>s#`_~^R&p<-Xp`4yAL!Uy5BYP z+DdQuVAHnY`v*>xMLjB!x}7$aFKJp}Iol_xY{NCDu~yf@j6K^`790P_(zoEUd27#s zkq_5eST-sagje_Mm-*^Shscx|XZTmVJp9yJC6n~Uo7?A9DStAraRlprI-}`!F*5u6 zV$1X4Z8H~xS1orp1}smpvWn^kb!5?oPUc^pBjE;1yGKIyJz>{BS1O|BE#F`znbxO# z7u|;J8SEx@vAiumf8xGzNML4JRIOuLZlZvi;#Fjg8zi^lm1+uBS*I~CK!olt{F|g< zPPn&569OGLj!N=&;7*8h@yFms{|K1(*F}1WhsahHCru^(Kd^tWB14aGKV-A6G<|aL zvgVZjSSVgSkv7N{r*q+6>Ag~qY4XHr+z4%!Z%O+62-)tZ|BRtWS=WwZI^6Fxx8g~; z7ixB7ZRCIcbIgT;hIAwE6t=Z+7oUc;#`~D`9OJ6h7@yK{fIKH zR3-Evd6_R=Sb;4y)grnxmBGurOla$@aD9X4;{Ca2_~nqy@|4m_m_7%lqGxiKp_Hgo ze!??_@1|H0oK$hgnydAjd`2UcAieY+fxq*(s;RMy)XD&BHu#dzYN~im6S$7lAngPC z5ZO!4qY-gBKScT|%n}oPU$jHXUi$rpYnG$NKLTrlyM2@3KuLT5RaKE+jMngTbT73l z@#L~2KwWK5qbZ;}oEF*>IL3Iwm*HstkE)kDP_uTi`#Bz-%&$>bt;pmrb-{`RbIXP7adQmCcjdCG(&fQn9a_>21$2KFYq{a zaeTw**|3MHQkSpqWO{8H9Qsz)sXSo+3%M!R!A!|#cPhGAe-N4Cr|EN&&Xi*cq)(7y zsYk?SC5S0Po#u*64JHRljbojk$giQf=+$5yi;o%{ zVZ7+o_t(>o<2NHWTK4#IJNC}BI~Cjf_xHwcROvw1sz! z)>b8|hRJBMBDcb59Bc@dQPcwCZ#d4p(NRzO6u#?iAOQI{Tzk|1iT*BUAH8C!c{Itm zs)WM7Mf=5mcq%^Kw+OxD8j{px%>ZjRVruRSaIUTkb`jpmvZ}HA%;0D80a3Lqq)jkP z2#`@T3CN65O)2wRW(U6cRHnt!Ut(2rR-~4GLMKe5%JRyRj*ckp7Rl0)JDcgSJN_u-TaucodxIZ=T#3}&XgPENZ_Ij1eugH zdIh%A)<1oxFr%FFpLOh}H25Q9E2Ggjv#ggikRIvROVx`G6Z?UNsLudNv(>Zs_4Hdj zrNm5X4j*aXBC<)_ID9rO8E2t9`x>Xy6oj5IG39B}Xjn$qhyD|CGvs{w-Xw0iyx3ng zU@FPb#2HNbtGREf?9n~{2eQcc2Ce}v)qV2juw~MI zb}<&9PNODC6bDP+rQhD~0Yq5tS4kT+{Y_u=jm5QIr3+w}mQW~!e)7ad?wFvIMNgh?t! z{o63vRU_e@a5mwo;X~OyAy2wj(KOhPyyS0QnrvtvS{^F%m0L!dXl`L-N5x@kOyq&? zBK64pXZbYpmG7Cb7JKIRlYd6JkWGdr>TPUY>|Tas?48jdxZi&>R3{YmjrvoW7V;|a zO1c=lKzxVaOWU=}e18LnQW}L4sP%3t{WJSbPq6CX1LJPjVj4Hq_U1}%<4x-y*nPzx z&J)JI)DfqWJRjXpnky%Z_xvp?K&c2_ZHd)Y4hx0P&*3cfUtKis0nVG3ltPhSPE(}T zq$=jNyZ#C<+_uk*44WP~vV4`i!`3g;AduY zveVg=E%waz^pX|&wtC~y$vn+;Vf@Y-o;iF~><*6--n5U7fHl$`oHRVV?HW{zf$(UeJH4p2MtvjNQ-5H~WuDQMMfbsXu5>gRUq#o% zn&MmBB(sctC0*jR>N4p&pDiSE&*{g^c8`cmO`-S)OtSn>)gxZO8iBRklWlMD7W@Y2 zZPev55$nkuQ7kIRch^?WLkoc0bb|Gs_!v1Nbdnw&z9?Cl!%Qx-jVN`hWwG$MB9{ym zuEUKzPw^*o@2J1wg(?@k$6Z4P;M;N}>>609`tC}^I%B__+r3t74dYL)Q_#X*%Rb3D z)bkFU$NW_g6jD>p6g`Z3TUxQ_;FUl_*IHsW)RF-ld!b=PBhf1Stq2J>~J$rKH?fOcW(d3kJ%UGAmW*FqQhIGORk`*aKC%dyrc-{lpAm<>L=cPq`-k%bJk}D$I5WABzn@W5uEBwrmVPR>R1D0yg*u_!e=wNO630 z6>_ex&M)fiz*=~vv${(b?Y$Kg${l4$kiMdL;rvJ1ExY0(HDkr9++s$|!-REM03U|E z5(#@AFUU=73V0RYMXU!GxhC6c<9|WxTW2CoiLahKY$pCzwV4|(J8DlwH1J+ePi_Uj z7PitY2|bt&?B+MqtN9y_0w%zK<2e>ueb|tf@ca<6(au@Thp}sa4oE}t3GoD z-{`hlr!f6JE$J2N{xHlx6|RAGkYspoz8)Pz{2~bVOD9S6LbtezU5o8q7|6SWug1Ls z7kh8zTtz;%+)sbS?yy%-41_$D&;_71x&r@>ts&Z+{*X@r+58K+F1aI~gbkEoIp??}>mIf?o&~L6uq4f6`tjH<<_SE z6EOM`u}$~_y~o$WCCDuK3F10d2Ca>DUo*Wrtv$-==X5m*OJZ|LU15By)u zO<6bem+gz}t35$88X66k5v>F({RduQi(?Xzqnr#o$1Wsdv44@x@LEjS+Pt?~?T${aZbD^*QtpR2!d<52vEM3U}FcKvoJqgO6#t zvY+X9wmrlfWOB=Pw(8!kX#0Z2;I_Q(#7R`ikH<=w$DUn$Ke!y~Pu~}&c;~lj$qg;M zA_^Xv?QZf6=1}5xS1aMLXRxq}K8-dY`_OA}L^zmKo-#vI*O4qWq9=G7(pk8lS%7>KnZ7YR!5Tsp?pSQR1S_}Q}-mKP^gZb2cN46ZerANmn|UsV}(4NXCS~JuVmxJZQzN*UPUV0igs7G zfno$}&MR&nyU4O0;$=l+Zyly_ zybeobbE3|lPfc&ZcN}Zbt=3&w1sKbBqYp82`C-Utx>#dIyW0n29-=Cn0qW?*^eB#& z>{xHc0u4}35~@LqT~=lp^pQ9W1)vVO?-cdnO~v1l$%!wEZtyX5Yjzv78?{5DvD;8< z<^=Q|J_^mVedlLEqv_M+Y!WX*4-O#PSx{HZaoQ^A)4HAL=E8&PY&-=pLcnIr_Nhmh<}>% z778%?RV$-ety}yiK@;{8;w;Z4LH=PaNL(Vxb>HytQ_ z=1_0(g;bsG2whNFv3qFP?ZoT55>%ISz32 zYHuItE=Mz6SL9x#qwJA$DgTLH=a`(cojyp^XmbHm$vbg*_MU>r`W}00pU0i0tMPN& zzEPHQAKVwX^RD6aaq5003yZ%1euv-5BVtf{KCqm8$rvR2y{M00GaHAb^QdY4qMnH*h&-B$cZ^Q?^( zxSr}H-Ve-`PHAYqjT%E5Vo#*IY#DJEZ%7Tu>4bLo)^!YF>x%=~Cz^yDwP#0^@@WCr zC(nzOc{RM#vc5Q~VM{b_b|KT;n;DoST?ZS8xMK~hCVo2?7gJ(pQH-wy%vP;sP8zxj&nrny((w>j@> zQbw1*8dyf0d1+Se8yS9ebfV?Lt9j-L!^w;`*(qUJ)%=R0PxH(T!)MK>PYnzoe&fkF zGPp)0i)dr%cH_EvWMid;Ivxx!J=)OnbmV~W|9;lX*s!Kf*{Cr^W=F6rTJA40L zSojrdncCrjxi(NEvghkfbIn~F%r`R?k;Ydpm=ky3H_sGmTVAiuHm7xHWHHPzhr=;a z*t@FI{3h7m{EE?;v)Wg&JWXkB@$OezZtwbN?)PD{<+bXbxz4tCX^%?&XNFdPG+#Rw zYngfbjJcn&mSy7I`Qf6N2j;4OpEv*b{Mh_M*3v?qZ)R!lF@zgmT2;~dXVd}My0bZ< zZ+)}3@eXs!gqz{U1B2$XBTCIRvcH;ZmDtSv?(Z}&?fb%f>-Z0IpV;%-3XOsKXc z^N%=7x8JobouZsf7j$I!?V1+mkxMGg=MC+BmxT=JEA^-6BfXRT>20o=OiiS=A{M+e zmkTuUu0Rgq-vuSzT+`0&CN~4)$!|y;^2cEatzq-grPL~VFnPhQV@G%%OJ1^p_CnNy zdIMZf{TuDq%pwONh#hn>$Qy`IZ${s=*~PZ^P~bPk-~3$qEjq-x$KMW~u3uQvj`~-B zSvHy9q*1{>@)W4|4`nA{ose|=Gp#=K4C>;X2#=*}7i~uppn6H2i-t1q^K)G|??z{- z{u0)wY=f$ayP$!9FR2>6w?!Q$JC&hjo-^)lDu6Tb|M3IVQ(W2j4Y43Ejrb4I>D#Iv zpz*>$zML8frx2Z($wVf9LQ<$wj5Ek1q0^}`{%*)T>JWKbQ;k%(C+oi8o0T?iE$TjR zGu+hnbvMQzYRlj_;%s4h?k+A2WkRQzbhwZPt)tn(Lte%0f<4w^4`Z8@x{`Zj^^j-z z`RI0j96SNL4*w5}@tzf50=GQX*w3E(E?n6iYvd{|p5s!&lXFU;t3bTan6FJYQ=4Qs z_J$wizAQb|%@K;|enB7p4mD_+dAs3_wPSn<#5tA9vCy#$g=je|DaTo_1J!|nY?Bs; z!HFru4>n?F@LIMWT_rxZ?Mf909H!^NbLbWH^4!nlDg?vqXiIXsuP0sU?T?)ogZzG` zmAnm+h1J)56($JFnbfE`$Kct6;o zkZskqnD!ziI@Q_4G%v_iU>?e*evH*ksmHEoRcLZx2-7E}oM;^MCuO?2cnJ8FUZT1Q zL;O&}fZ$@p5vVFY!nT23{q0<8-#cXz^{s>=Who6n8oi)jK`&BvD;pX*mlk2SP@9Y? zUOqTe^9l!~h2m4^A^jw3Q`!u&ws9!F&3iza2W1mS#GyrQLBbxx3q2dSB>6iJ7v%A< z7>{*M>rNim>!fKsp}!<$D~^#4>IkG|SNRuFXy8U5_EIw-h!pm+hZ*SE6mg!~XZ;Oz#G5W9~C^iu@W_&M!zC zMD5KRZOZg*)lTxh@m>w}Gb|IAVtb?aqlU&GhVI(z)bnKb=qA(`?T0D&lEOV6L^x}! zuYRbqCya5w2eR=*_Zyp9JSfbAZ`(E{VlI?BE#>gtNUy!dfdrwLc_PG0Xm%5-FVvYC zh#tq&ag}E#O~P+;ztc9MopU_=Kenc<8h=n)Dh%_cL2Cd0WG(&)yr^45_JGh3B7N1- zC42Rk$Sde{EXLo2=v1=B)m0qstA-_q26~m`k&?O^CwL2}A+A(iR;h={3KnH0 zu_k*1eb?!8>`@iTX?hsi4xc4~K-A+Tw9vKH^d@6;3w>kQ2~u6`OG**?4jf4DQSF2C z;K{Z+&U*Mcyo$92&J4L(&jL6}RX5)D%p`}?c%gsYlURBfpNUWV@G38`&P zf+>PNRE%Vua0-8(#IQ3`bprkKpsGR>K{|N1Lb;xQH9R+z9*Q-=)aNp8&XM?Tv^7Iw84u zV1J_WeW4qlAnO2C;eW%dY!)z#v^r?LElk|GkA*2xf1C{*vHNqK0|_G}Qk$M^W#DyNHX4(O7(PX*sJc zXvq(--BdNxuR}of0yLXkO>Odg(wor5!YA^q1~yD~f8{4q16**jTC*0cud}#rGY6Pk z^6j=S>PGypD1UGd*nuCzHw)9?txyXr5gW$eQ>Pl|>i3bu3kpq3{pY+h^ruQC{;+X` zfzch*x=i=w7)ArN#hRkzn&aH;-Ue2nYvJU=Z0w?H7BmQ6%RW$l)ZP`sOa*fUSqNe`U%D88sd$hmOf~j;BRn9 z{MS29oCNpGxrz>lQk`AN8jb~MxG3ti(;VR48b}?7vd|TT$MmqRKnYi&^O;U;U-zfl zTv8Xu5YgzbtL`UzB=%6m!*im}s29R;Y&2Dqoj^T6R#O|eddQ{X1@IZ?Z#liID(!^6 z(qmiIN&Z{uRi&|BSB|P9bT4%mk{8_+q9ctkQ~gq>s6d+FH={z|Yp#uDTuvuv_t(os@&h;#E+RadiPmUUAceI}H1I5ilwn5YG+-3|xBI=j zjnE#y#eOe@;ElYB-s-HB?9^zeew5WlamLpZA1XX`za>gp4bs9T z{jPwddF8F=Ye3x!;$%PlHQ!cv18;GT3qghvd?U((wHH?UApDa5Kl)#7zW*sD>HZ*Q zaQijY@%>m=x6xi%c+)Y+m6-feCUT%6n@(c8svbMmHtquarpLG{6Rs3I;FMMh>)f;{ zy*|MDBETjgW84EiNdyLChM6+zjaLe1p7a|trd5p-zfr-8F6lY~v`n;3*? zlE&a6)eHJHFqhxOyiciv#5qu*roF)Z#nz!~hqixJO7mC?r-V z_u)FYJ)7+M8%xOhpZ$RwrE8;y99i6dNL#)cJ5HPiq$40!0L|jobEi}pYPYIQ%6fUR zIN$9?e$raRhUEpjz*l`60$$zSKpGX1st0et8vc@T9h@&r)0;KdJm0xusjsP>VF~rd zH!844;;17?YSgl4kmf7)@ve_C&FqFAastbQi&WLh9!Ji#|1TB0AE zS@OEY`@pGr!@O7Fc{Lzx*eux&HD%Lmc%FDk-B%`1w>Oyj4@FePnotuRt)qF0i^_o~DocE<05^ z;)#=%IQ?Q<+$?yJo@^pG0{Y^f1wL(hB)2wlxgZs73te_a>3jTx`xZO{J~eStc*d~U z3hL<#)1ro(%}CX55f7orthvv8O>3xpngzx1ykHoyaZ@xcQ6&b;IusqrE z%xoRhJOV#DP?7wCFfZFbEaH4rVOG{`l=;%XB)xFL$?(S=!@`B1#+sj)zE*sG?n$p7 zu3|P#t#@Q_oGNo>PW$kC%{9yC70H(6GgRTd=S-IU1IS3r#`zVyZw$)VM2!xc9v;nj zLer7`J*!)^rupVC6ApyiuR9zr?b^^sU;J<@GWEz6NDn=Nf_PYw6hrACIOBu56OPBD+H^3`(5-pKMR zw=j&?n;CAojIxXXUYJj%T#Rg^kZX3Yj;Y&3npJAJ(70ap?^9J+;d)$aC;R9vJ22&oBDgYZq5T)zZa zmR1l-5B5)6c*1Dx2u8}^QZ65nKAHY#s?*Mx7OUA)){9cA2<->uVE9-174nX%Q2Y|j zyR1POv^P`VxKP=^Hq$?c$;K&iqb4VCBJG6~fzJ_XsJG-bZYlQ~{s1O{U%n>f;FkTci9uy) zzwm+3F6?53O8BNN$TX*&;M(gN(2YaUcf(|ddYiJ;orp!IOuUOj9eU2ccefJ{%DNn! zt$m80vON~RyS6yi@R%@FyF^#Q_w<|F*RMFHtk6+`Q?5?QQS#5RTZ&N4v5wjrWOs5| zP-TGAzWMr@I=iNqSBk^&JX0P8c;@(bNRtF-u$O-U>Bp7Fv^p z?uXuo$7x@I|7fhJ&evJ}9#e;c;5cTv&Ibv_Qjq1!1%E z-7%Z~Lv#l%_utR1L;mU7l005p3hXE@6~B7kV!b0hsr(Y~IFq?p8(;C+w1|4)Db@bg zO)33GZBUJh`m0;Z2arlSeg9eF4cyK>Pp5#FNUeQsy)%4GnthSJXhtzruol`S%--S@ z%O~GrEo3b8H45C8<(J+RrsycDn&}at%$)B3?VDqotN-ZVVBSVX_}|8b8mD_T=m9@L z-_zouA$+1Mwh`;DNw2Xf6`g$)yVsUtXbJ~#Uvwr?eCdaFcW7_uwdo@|!CSjLs9x*c zWH#l>5-Za_vn?#%$U5V%vXANJ@_F8{sbN`iL8Ul}GBrMCZD|^WO$`=mAK*-y1W(hq zH!cn|AYKY*{44Pkdmpe4^e|0ISr~QjJpg9%E8xvioaJ=+&@_|uo=gecqTWSQ2a1x8 zens$oaFL%ceL~jOq!EAUlmU!Vkc$EgFrG1$gG9nPay}laJNMLT#el zT)!0@%8OY(QdK3pvT~Nq1Yx;D0kc11H&7czDTksHrP4 zlp8%Gr3NaDeal)@jLO_AXnify98tcuFL_Jy8%#zyny9mDN4jJDce(z$=OrI-B`_o; zCbf$8jUSh8#5ZQ79J@!RNb``@sNS(XSi7iG#srG?_A|~VO{T78)5P=XD~@DRMZvQ{ zr*^UOr0z0e5$+uSEH*<%nWkx2CE^>bU*&C79&2%+qeCmxR#PRW>HGlcW!N6*jl9p$ z>0gnTr62my^a1Bw$qb3~c<%#$$g*B%re2!Lse>7bCHYk6U`()gsEBTyabEq8_Zy`n z>);8-8=Cx#wfdFD4ZhC$0m8mCi7e6IDIMv}56$$om4Dr#)-aOjh`uJPqhB31Z&zNA{N?aFp21++<|dS{H@Luii)s{9r&XrD_a{;3 zOq)u37}I$6*x7P7J~6m8?M=o`Q!st0Pp<4*`ULNq_K2YIf#%k-8^W8Tn$S#J149Mb z)bMX;cBp5tk2Ec6KIkC4E`!NbZvnQsT$WKCt1K-yj%AF{I;n=Zj+#*NP*;!3PtP&@ zOIDzRs0XmcB7_fk_G%mZdWxcPGA)B!=^DzeAG{6sXVbt5(!VLKffmto-9=>=?aTaZ zzMVKzfk0`&ZiRcPF~NHFrD|N0XZ=hJvgA-FL&5Uuq2_*Num`<6@jMg}Ep2w&-ohx-}2MCQ%!8HxLz7CEtbwt0R0M$6`wA#=^{eJu%@e_7TpSReUt zzDn4!KxgUFC(nY>EOt$n>FvV4Q^s(d>x(81?ghS}cX{{7XQmFOI(pFJsOwnzqv$pA zmR`#z3In9}JWkdj`t#!y&*3W4IeL@qs;e&-Paf5@@$JXzFm>P>MVXva6H8siU~VY< zf*u4<@@^6TV{1aMqBGHAk-hjx)ikM_BoV7rca=Ns|KroSIqCxZyLBE=7wc?$0PqJs z+Ui1ap0!Dv;4biK6~g!Aud8#sP_% znqX6?N@l-Qggh2G+@N{udN0jYY+~Mcb5fQevs|-OgTX=2Hl&ny;ycLf)XB_TsuO2L z46Y#DiBw8X_6JsCk7EbP#;$JZOo-Ond(mIzFQf|XHUC;@0;a(0_};O13uUS-UL~Ef zod7oR57qCJa)iOrDPlUnMOEku2pLpO^{3`7;{rC}8OR)>Bi2Z^(zy*A#ofis@M^e{ zeI!gd=p_0|G5ixE!y$McU}se;v8CJ`d{U}bbwRjJq`P9Q7hoCrR+^7~D8A)A>~J&B z88h~jfNVXbzHFkSZPESYIZ;*xm;Oh&unJi=`5hT9ZE~%5wwFKTYr_8kcf=#vW3m0f za?g{ZTFBg1wLN?3;p`fESWzeX7PisZ5E)x^4{KPI?YQQAM2wOyNH_V~)JDw+U?ma8 zUy6f)T|kkiYf6k4!m2YByiMp(@ReC*Z6P1fa;VspsT7jA`Co$3Q$g{@N8{=X|Iq_Fwo=S zXGIygV@f~Y$kl~7O2+2&!J_k|;*Y|;iO!no$R7mYy-S+3R1VKRm3u`2-Vqvb5%%M@T8@`emOE1S75b=UsTNQ84 zdE7xJ>^x=L$p`Vpo@Q`uxHVb_l|gl#!>t>TV~UoNA3F@z6*rrP)PZvvEN93wBIkkd&Ol-7w1$HA=A%|Oy+<_011F!^q znT!$RJpZZ@LDKkLeet761K(ky5&l#1;E&xi84S9pv9UYF^(07+_w=G~XlH<0DHA!P zNbtOuu8A_oMBIeGMXo^^>|x&r?20m_kcoa9IFu|sklN@LCjT#~Kx2}c*mw9l0D^k9 zqDxV4B1}(UpU7r=A4G3^U2vB$l(I{mX%}5$eGi|)A1F{$-3k$?I z+BdB_pmmAMx%H${VV?If`V9Z2|00}a?rNI*&p>l@ms$@ZTaja=vGzZO83KjZVq?j4 z$2Cb^++X-tQ2U2_+i|xwamiC1q_Y{)kw4{ZgKUFi*)?sax}IXI;obB)&oSo@MVn+} z(qi}@5aL(zGW!qZUcm1zjm`=kNnZyaxc|fkd!lIxDZm&7J^ce=kwxrO_dDBqE2`FF z`rKPsTy#aQ@Z3U_c^=gyY8vy9{ss?}rZ~0|>(D=va_nC4c(FSU;Prf0#cFt? z)5$*)YBT+bQqa)mPtEgaw!WKt5#11J4e#S?h-O6Pe9i2DO->U2&ctEe;U-8;Tb1aU zZY&%}zEoy#AHjD-XQ?mL4MW^VU?^(EK8RsN8|M3Dz?;Mc@z(G@z}S3t}{ z&10;PYl5t&#zs>UHczbYtT=~v3*(S^^@&%3Z z7l=+!5vGu=Y#egRp2Hs##bg)P-`|6`@)dz4!5{o8RL4y9kA>e9%$Hw5FR@MZ6Yy4~ z2A_(~^fK6Lp^WP&ZEOA7@ZSDSwZM;Hi9{dx7rztRqj|{@Y7(C;bjMcf`UMWa{lya! zWc*~47hU2``>JU6iXHK8Opa*guJaD6k~^VuX#Z1ea6V>E5sk6g(r_SJBOq15sZnZM z?^s4&BiaP+5!Wd%6z`vnnXqb1DshasoKWKWuMI%TsTZ_JW8y~gZ+8QC5AKrmP^$NT ziq0}TYP{>ixI?i5h01g^KFLf*swuMA;_k3G6kFWg-Adg?lT4CHM%t+k?z{Nni#@p8 z;_&9}=lRepSMry0&hNg(XWs8J*Y_x{LY|vf!rO5Zv{fdCFZX5&-RE#An<-sUMr-Q*-C>pKhM5E97z*^9jCY z8o)m>HBfwtorrdie@iriGu_SczYPd`MsqCA?cE|Y^c8a##Np&A!o^=V#rclKU(l~D zn&2Hy+N4eXtKu)?vsi-d1pP_p5as3~>oj~JW`GwP=88Mq^1M%g5kt7uKNLxo;>=d!=nl%7Wd3tDr-HwWP^F=&`t#@;Z0P6cM^4%lIrMZG{HnEal#!{Zf7Wf58~j zSE;6WpM2|mgirUk!n^nx^1Jarr;CU}{{v$39k}#4bSK5m+lS3;MvIYwfn7W z0zE8IfqDM6xwfw2Jn6Nig&UVH>YG~I8t&sxGPZQNJIu8t#+Kx;6zGU@qG_zKDx)DG7b>0YMcxN42K7dEh6}3VTc* zBgV^Vt&lOltQourD8OHp{6-Rn_fmhdxBnKG>Frt)2`Q!#76{Luua9J)H`4 z#!Ko%uYw)s`$Jk{e4|*Y)_OV;94Ruzfo_;vCU1Z6tssxZn*APW3U?lx$S;X%%y|97 z`~#7_UL3FDzZ-F*Ec>%n;nN90SHaHn4aVbD7ufmU&f1F}qpPi0NqqIZMtUQ=^&@=q zxq5gTt~)ufq=VRrj5SLFS~L2s3}J7|?=n1318n064q0!{F~W~j84`5Lz{*iw}pSZXQ{?h;G& zpIxPWM?`IsL?mhEY*ZJC(${BZ}+PUpEH83YS4L5tjRcW266kb8S-cmc2 zgRX|Jbh}f%wCO`?dmLs3?;U?i) zX~}s+%II59@VXTq=TMbzu$|Ak*&&8JTFEIjySYHljj;ZSPk=Zn;tI#O5) zJ)i_>dn&zfKDDAp!!#qlD!jLHc-n^^$Efq?u2NIaXNEsW;p!I7e$?`gAH&JVRMZRj zLFz;1UW)6Js(Pj=C%o|`l5+6zjBx#3@551wGgOw<5>ASI67HPWG7XP>A69K!PhqtO zhZoj9M>Q9DAMPk6dFmyw9 zNpp>7$Ybj7XpHMJTo3>3s+Kuyt=0Qn1UFZ^&bb$u?_J^f$B~JC=T<5tjPyU%BXme0 z@b}`G)`eszAzx| zglDjsox~j=oA_q}c}NDKH(lfE5fP$9@Yn;pWZ784@}&Wjr=;+v&x>c0jhI}LA|2#+ zcco7OjS_>E{a7rF^+O(HH2`LEV;pbW7q~K+*&KlmjPHrb zCOhrE{UMhYZfrei+%5eM$Cxe{zWDoL-@MJpzsUWD-rn`*hvtpNTGL{(z1Yd$owXTe zp;HoBV64F^&p{K8*5X0uHQMO=OIodUOZ_bJ|4v>gx`zJ1&mvvO*}%rwN9G)Y1IO#H zDeAR<2(@tC09)va*&CYQ(RVJN`?Tsi+uzg8keo2dm1KHm7;Utp!%CK$8UkTqrl}vP zERQF8q;xj12mMHN(l&^hT6ium!I70eMs-%VOnoMP z-@YTjRp&-_r6bL~IVPh0U*>gILHg5hgRDm9NbD=4;|$vlXKWJBHUV0 zFauj?TISU2FmVVU@MNHoVl=cBsDKZnO{9i_b;dt*DL?}l^gPAeBKa~uzf0j?%Gvsn zkzctrNMmg$wsxG&IM1{$U=585@jJbWJUUtkehYM_kujdP9Nq|ZFX%>uvLmU0;P6fYJB z0jG^SffA$*eFy)eu!P(w%|T7bD1OAg?oekb2Wm&&F?80CkmUIdW08A+&xWo>njxFr z8qa3kJZ7cNiS$KR>f`V(;t=8xbOD%-Ca}#7ns`p=XWyopa^Ms8h_1oikR8*L>FT1< zOa(c>4Y^m+NA@o9+~wLbzhJc8X2W{)4yoe$;|rmtG}5*W5z995u28o^=PA5UDR&?I z1O2X9QLLmtv)KL^`DD=tG0H}PVD>(?8aV>QsYiL1=x2aM4nMwwm|@>4`)ks1Ua`_L zAa@_WRrXcZZF9A7BeD&s20u0JEA}S#D6YVNK?Lxbiw6FZJ4B$T3ZV5h^td=P={DIZ zxU8%QI%R3bUk)8k*eLwZI>HDhXl#8lRB4a zoB08Yamo7q#%Y+(%fVwiG|4HKZwyKVfs)ugv8P<;oy~J6@n4m%0fhhHStW4#Ja&P$ z0@pIV7!B~%5XQM|1v?yXqgyG{jRy(o{Aav|&QS2w7cO_^@!l9&}m1k z@BsfP-%QsBd*c(NSUI}|U=j(4Il^Z?%RZa?;p~D874^t>!)5*kzZP&KaoFFg+4ML- z2|09gT|xAze~~&BZwn1}jqsFnIMa-|t*G?+^yBmwkXl6s$5$jR_bVQGj~n5yVOF9l;Y zEx7;e(cX74vzQggQ^hDH;(qM9;vXH&`IoW%h*{FAc*OM;N&{38a<09T1UuI1u2a}W<2&~}^z+LpGXf^72 z8!?fX&y>U6@je2TTa+<@?ZGZ1`sudg-RvJ)EYhEWEPM}dfmUb`t=A{<+I)bm_>P-y0w&$PsI73XC>e_3*WvXd`0VX76j)F|tW zPfKb>9uwmX`Jpao+fXOH3a^Wmn`(hQGv5gqIfio)V*~Y>X+&poOKuoOft{lx-q_!G&nwh-pGUfp7puez7bw}VPxTUZTbCY?e8SQGoK6Kr7 z&A>wH0_mva#5ed3px4_?0W?Uu&=%b)MaK^_rX$xq4aq`!JM)aAq5nD6!f|j6LNSoB zlip|8%KZagz@Mm{w!Xwd@+Cenc)_^P+yJ?bF2`qRc9IdiNh)v%s_v$X!S?h_vA$Rq z+XDH`H3xyo{YYEnC7=^Ckqa?%<@IenHk0-T0Pr!oTkL|4MP5L>;;v^NlOcc0T>@8- zLg_c-X`jW>(I17R5&FH8kxm9Zs*2ww8QfhFoa=3VAx$M`gB|e^#K(;Ph!xN!c#Tyy zc$NI7CH*~+t5}NXHU5Krfvk1z78Lkq3Gg-G{>E3hy0OcNC1|^72au1w5VFzR_!{yZ z`GM$x^$56#_84p!6=66f0YU*P+cXm#rO>K6d3`|(v6tvxofCzo&gHJ zEW;mU9m62^R&-q8q4ugY!7v?9Bf*(rVrnB0_*0ophrk}nYS&`Cu{Rx9T%sY%qlVF_&BZ>@*JXoTB$O|X*El>AbOG%Y&qz;0 zshe&+_ZB@8!$;q8K6hpc$(VfK=JyJR3%2r{)Gsg-`K)O`_Ei7F{Xw=@&kRMjqCvjEc0CqkmM5I~I{T{K?C( z`iqqMqjm2zJA1x5X;ng6k^5p=&E^NISKT~Z9i?m9ae2!Hsi_aHhI{Ybp3-v6m9znG zzN9wz_gO_VG_88RcTZ~gtvcmv7b>lJlNPBTme=c;wM>(GJ@l};19Ubua~MvoI4q_` zKD~Kx!J9Lcijs{7f7Y#$dRjX;EuvH#PF-!MPIuUy$}2~tE!p@b_4}{&spF=PORFk) zLEUrIKGbPPjkKJvrnGP5b86{NHf_^4XKI4tWK!1_uPO0)-IPAj)2b!y+qA2(h>I`BYtRu_yIZ=Vz&J`{@o|v9wIN19v`{umeoV$*-H5 zuW@5z2)pKmsimcD{ws2T<*Nls{A!VkpSr(ht#mx1zpHJAX4VF#diYLj0;#IFU726L z0sCQnVcm#7O#GfW5$l!kB;f)%i~lX4H4RQEOX^UORh3y`1Doq}ksZWo>qzrxZ?4qe zi-0eN_Uqw;_M-BD4)wH*%*54jrVPlhWZjpx(5$Svw|-F z0mngLDOuwDiq=V7P&6I0Cma>01#$xzN0c=!Z?D-LDlYF0iN^27d2HBk4NR2-56gTu zXciGAPX-p5o0>`x*ffqjz@19In4D$Wo7^;|p=C-!M8Y-e9_fs;U-J3jkHX5Nn1Dj) zg6+24D9y$rlP6WSK$8+iRJ0d|}*Uyr0>0oX_s47&!&BXPy?;Q$MCO+en@}D*+)&eeCaMW;Cu8NQv!Wnh z8EI~37F(o zybpN-rwnRKJHs=qwP_?WfN=$91nq2YsTfLR&z80dO~(I8i1s!%3{Bo81t-8?;u zJ6KyXueI}%-}!Qo_Q?UMCo+VL6BEJR*2JJ3Ot2mZ4DeQ%KC*XwKw<;Y8rYh!O*|s~ z7n+pZ-h00yob~|TRi2f$N@p}B2g30mt-psViyB(fO4l>_CcSlys(W%M8PG%}w@$$s zs*FxtFnSr6S?1H^B@*Pe3LqG3qJUJqdF4E~;H|I=LNMS-rpG68S0K?RXsi zpFRRT5rfCyi|g&mW(GQ~*bnn|U2VjbFkR8AAZTH^$wHSb3*o zT>)f_=uDYT%p(Vq-lU*OXUrmnre>0!jzFiurs-*T$ENydT&2?+MPm)74?a+@W$fyQ z#opRYIaC0os|04)fhQaoZ@Dbg1?wf&3&h166TbRtp<_%BiIqg{glzLMw3GF)d1aZV z{BK{sl0DK!BHR0lCxCCnfJ99~x}jmxl|%}+CgfO0p-=IL`j=Q|a)Mz2YBci1a-qE? z+qYgkk+?H>K_U`=_%zt&V146p!`{%yL;(9%`ZO`u=u5b2^^nh#|4inIHi?GBi@Zn3 z@F;cN(HqcoSHI{bn(uU$&+UJNZU{yMW3iOL3+Wg)jNcpCF7zSPtyD68F;6l)1L@a?MCF^V8r5R2B=Jl9jl3E2&dXV-5 zRaf}!@k46;jxx_mh+yLLV@S7rDMWK%Sw8sRL-b0I;*#& z*#R+0S=CXyBxz?=ls1c-9Wz3wGCVW)0hf{Yt^M^bG7vC1LN0ec?TWw^F_7)!}f<_2FH%@oB9mlvcHvur92e zb1=32&X#Erje8ut^w0U~-i|^lug;$EF?%x7N6rOADx1>7qkLjM^DVA9+glyM?UC=8 zhmdY?3s0t|DKL|11(z}dfUO9nJqGHyVR|?FL!R5nQp>ypaATW|I!WdgOl^Oa8>`)k z<$5}@e*9l3P20@5)`Q?#wxi;)h&{mOqG4RTW3?0n6>f-~1W$ARsj4TGBTMXOOc8Iw0~0w~vc}Xy6q6Omh_;0L_p)rH|;*>b}Su(BN7nTtgb^Zz1i?X7ahGCNYN~acH6G@lXr8{y&5N(My`r7OHdYqE)1ubvlEHwto#!$0 zLYM*m{Qtyev6xrssZ_5q=uivr#=C%~*ivt?_*!#Wdz>ApQ@GnX-I{m}sSLw+#rXK4 zxeuKiG?ALY&N88`2sopKqe6g9fwlNWK@SeL*TuWVyaoqH_9YIYm$g5YuaH&wMJ?ZZ z&bT@OQ}|x2R$CXj3v4F_i8!`7?jv%Dz99bLN@1?k>%BD?KleBEl^&yMo0a4_U6c-- z1NZ1-5MI-mUgf$Xwnu^yGvQ5|cSsDoF?t3Z70+V#!Sj5NxZ9dV`@8Dyay_7_@OW-a zQ7lp<9JNL9_4QVJmLgqsomr#tJMuNP*h#iV!rw*C4*l>C!YKY?L83I1Tf{+8FPR_F zSB0j0L+GpUQu~}&+m5q!m?h8&`ll{e_UbaxyYxX|M6_C1BIo>hu%mOVrz@uhqP#@L|fPBo(%FV!9I8H+e-U*uC!b(qU_CEBo zQ}W(a^x}5MC#haD|Irn$ld8k$sFqdA0wkNe=NSrO(XZ&kAg1h~E_c!TI)<7aJl5?n zWe)(`MW1(FVwb`;G|sb|9|)JJJLqHO4Fh+~vLzI+ir-iqfj(1~#jVtoBTtbT&g*eC zLH(}1nwQu_OyGA9ySK<3IvslMs0+$EBH#)$9nC44uL}qf z?#|u>uF!r?`&8XXv!IBBi`m1-;37o(7r(J!CsU8jz(g<;Z7g093!EkNdUgfOsc&d& z%0><#b}M_Z=_TMOTOl)~hI;!U$J+Vu;e~q;mS&&~x-(n?q&Qa6x~yew!gPa}w=RX| z4BZmm&c$=ZJc3>XdIGDpoiIBaql|P^s^1ojLK2vx1@C#8<;Xaq-_gmu83-Uv#j7@h zzOL&KAT!IsNUk-6!gs*m9i8KAq5V|^G?)LA8LM5Uhq(iQ2g>y1gQNJhhz4E@jAs(z zOl%Gu!L;H(6t4w`&@VI{>5CYs=q|gA9>!7Zb8U0_i2D(G1Xg)Y+pZD+Qv_TTT#Q5r z0sdCJGI9{u8{DF%z#*}bII4Mu7O^I<1NQ-bs;lRore7o5w!e7pv8i^iwu9J`D-kz? zcd-tRTB@fq6|Skc3Og@u;-9?=h!tCJ61LuJ##4&YzS zM#nlN8-!G2?4WcSw|OhXQG9Balk4j#hP6aa`=`opy06F(XAjp)XpOkvvCm&bFW0rg zFUp|mtuhg43g5^+4Gnqz(pR&UbQRZEGg<2Dd9QiTuEAE#&-zqZ%6&9{Fn{N%ui z{SFMd#ELReX&0WX+g!YsQ!_)QGQK|j#?i`s8;IwU;o{7c++Mc9u``N35)08c@Obv7 zS}Rv3V~aJe$k=Z*ANNw@DynBDAWfCJVl_CMt_GfhTM9T9#^%{F*!G1liu!fxeokZiEM%$9SOr$?nL$?T3 zvgv^gZK zbLp=a9}=4oOW3~rCSa^>%7F~eQf6@cA%&6t2c4y`^N*l)ygb8UpDc4V4&^Gm8(~g< z*1tm<^>uVHHkc~~Xrv`t5B*O&Sg@$ZL-*Au9G{DCaMSpgz^ussusJbt&?xi^o#0rL za~fQye86pDjv|+|ka&T)p(Md`s!vEKPd)c_;1JXt?nAEuhtWgP$?Dr39I+8fd-GI& zkh{ng3$|ic=*6NnXd=D=9RT-4F3~D{D*Z3|iXI%F0uE{Uf}iVJgfc>V{G$Va#xR@P zM8$Q+de|30n^bA6$+63^31YdRctI#+v)MN4V@_xDI?Qg|iOtijE9PV3ncs?|JjvpI z=3lzLIts$k3A$n4KCXLM4RD;dD6V~TgD0%aX6|cw$8a9XeT^=1wr194)y+IdzsV1w z5m{3T=6J10U7=Fl0_-jo(7(`Mig=lLbYJfPr*SQzQq;!oiZ5o5>PDboRBddYyjK1R zbkg|vj@nm1FF~zs!9)6vSpT?6dJ|q({R4TZwz$T5b6CHeS1ZQeNI3i_v{y8tJFt5| zsme-!R@~Hd5teiR7OSx&cOhVNU*$g@(9#>C5oiWmgslR1i3xzh`58OQ3{g+Ctu9o1 z2kVy7eIuH9y2v%_&P8dCOEx3B(9w-+=y~VNM~}L6++TPn^b~SJSt7LL61nZVhuX8u zXR&$FVR|H1p4~M1iKmXn5-V_|5m?)(2Tj+1a*=#?(4gHLCz?KUtwNh%V z``7)GT?6%%dqv&Awdf*cF0>O211}LRGCn#3JP6lU@!HXv`p}$qc+si0Yh8EUGmtRq z12Xj;G*8_Bx*7pH0M2tG^R+AJJ%vo;zN-S>B+r4uxzd^VAg}Wz#(!e!#c!t{a?3pq z&2uw~=I|MvmK#*`-E%|QtA4KhR9w@& z82G{D09C||__MAsyo^4en!vqto773X1+?==+$w3P_{nisJIJ=nRg>k}Y4*XkkK(9; zHPEbBk8`R#Bl&}C0+$s15Tn`@Iufvxx<1$tte!Uq{o%CGkJ}C{n5;Qa{23ges_)$< zbiv=-9xDyeKXS(7My?BZ9K7WUDBBjP-3Ike?+-hts9BirJw*>f+QNIqzUs<&bAG>~ z9j+9e9U|rV(gM#sX+J2mXSg5k)7l2=(Oe`@hTKE;B9ZJ>{xqi350)B>E76c}TD=!o zrJKcGSKO6NL^J8n=q}wnUWgyV=6Y{%S3T?06E%tGF65vyx)@?&iVmq4Fy5GB-ahU> zkvh-_U?knhRg)k!-RS_+NxY`-r+Ld4cv!g$H68)HM?{YMS8S!fz+O_;j5l(LL~UI| z*HHEelFaV`mU3rup^UD27kC~Wpe<%T;d?wY5IA-WQU`7;rySGadb)kcV#wf%jz8$` zr5mROnO)#^&;S;wYTH#>E%3_oiTgJ<61e3^QJ%5gbJ^ils!y)N#WwvS`X52bbOB1y zq4p&3DtKP*Fk{4hc?FDM+alv#UFlqi68j7H(01Ntn%QUz;Xgrs`fNw3u{JmNfc|%| zC(_^j#%EO*X2$Bab91qO9A2;{yc_=;^LTd0L^wfrjyC}A_gwavxucv8Xl%2}e2yNT z-gE`~5BNzpm5){>x>R&uwuDI$=c`8}w}hVba&;-6q;$j6)E|+y@+@Z~eV_zi68xu-g%TIH!v@Ave>U*eehkfN93nbv@G$@`m|;~tHKF|BJi(9^NB zs6On9eFdCx=O{)?y8QmgMqrR)uKt#$9^OTr%`}vG8BduDz+>wdb&g4h z`5Uf|-U~yyD&GS@t-C|NDM~9gOUX#JCI^}2V1#+x14XjlRE9wvY$u+}QX$I!Q z;0$*RD~XGVFU!=MGWqCb!=_L};^@%P+)JWY9* zx{LaSof97y)m48h%uqDfuLIs812}^1?d``R2##Fg=Hu^#%k&29jh@nNwx{X-DY#`Kfa~Vn<$f9+<+HLTt{yf;pSp84 z+|F~jZH9V7(R}?Jr~#77*9J0gBAgO_74*OYZOp1?n%lw`$7sbl z?lPXu)Io2!s-#cc$*AV&IB$fv1$+nC>ipLo%>TE!L44s(K{k6D=XFQ?;@Kj-Z>q1i zo#*btxFgBG*l`Ltj&yS_#(HsP_sw>_VlOG~M2}|L13#e(t~Ao%h*fQbp}ga|Me04U z4jBgbLRJ9H@IlB~wMX|QYYixQo~x%bJGmJcg?qVg&VO|43oZ(?^IqoH&>^79+rXia z-Q5{TIdWS!73Pp;wnfnMyhGe(COwB^79s0Y3$U@}Tok)lF*@&&YA|HdScKWa7G*2PhuC+F8-Ue~gl5QW?I+*`_|sVi5>RdB zS^gmK!V3y@fJ5rN>?3Hkr+^vicm&$_hCzZjf`5`N9fiPXcAlIF{arYt_$zNR{EGRc zJmcM$HI^UC|5NabNs!r)Hv~13h)z;1N20-INCbK%F2(j7N#o~f-*QKslen#1UF<@U z#{Ld|-x>htp$ z#%Khn>xd$n>My}5SlrGDim`MGvc#3;&SmFwYXz(An-*sJaKAGy^dTCdI*83yCMhn8 z_g$Nq>73x1$<^ia(1zU87!AvC$HZR9)aVB&;abqXJrc|9gRX;9Y<^&W_DpUI+E;r5 z{^dpaT96gJ0<+9|AI?XK>rG@=()g=s7=fY4h53=XjDFnLmGE4U>aPd{ga-^n>7nyCA8j|0mRaFbc}PQ zdJK1$uhs4>^MmOibSoId?a?(wKO%M9(OfGz{ho$6g(qsIu#A@lM9xa&6jLCyif-T$ z&@fc)hS4eB3$97Jk+R#ZkEXV2JFvsW#$45AK!pGUt#Yjdlf-+p3(*PBTvzxe#Sa8f z6)FzUUD)37R-g$m19reY^Y<}FwObua*sX2ecunmS9jkN~??O?tPZA453KJhC4vD52WMvIPiXbbjF*kZmDmZ)#c8#UJy!_Z0Ihj@P{?7D?I9V$m7 z{eSQ-xUo!8b95Y9qG)0EXWt`tZtS1xaW*=JpXl1B)(Ra&F)a`M5BRw_LWgOpojMS7NfgN zc|i;2WO#qca~pqfC{s;;5x%HjC`Q`fIi7>l4U+R4S4TYGB<{#+xS0;y)a+@#i?9q)qF$(lf3m(6I7!_Q>z9|0q%+^q zV%k;^f@`WrgI(OvfZ-S2H>cs%wbT@|liME2C>h5C)u^IMkpfMQl_8^~} z*R+3VYH{_MnRx|pER&{&Z06Vl{4S`1zv=jpAF1Ah$H*AoiR#UIhL+;q0xi_PqjMlT zn8y{mW&nSJuZe@gHYWx0To#kTK6md0754Q^Hya5KLjH(Pi@z(~Gvwy?upiHx$i{j;9HZ5KsaD&Y+t-Xw;IZ zhKLXEC@uq5BU@u{<6uYS0l$6jiv}m;F@BE zGEL+GR_36YIvJI4q|Bq+iU-&btpUuje}Z*%H{E83HJ$@sptG54c8{V6Y$)}B|57hu zOS%5a%kWgtw70%YyQ~9FV>5*~v`F8^xfiRN z_V#?`%arG3!f|`eIht;p!^mei=OniaZGjpcHl!ZAOA9;p^9kr3O|eAKV;!%+z2Hk% zxvr<<5`EaQ9qsP|(d@Y8c?#LT^OL(GpQ9hq;~2Zj z&Ngz-RK18#$8H}8^9A7Pm}Afs>_~ABA;r-_dIb+dwxh=xrLGa+0VWn_I6c^M_?m0B z>YF>qRmbtm;`#f8nkOO9~BWGSh}`q}$@0>6#;;+EL!gK#{VvXqLUS<@LCankw$R{c;rUD966j z%@msa7FZL{7*-F~^1K)FpQbu@72Bg_T9@RGRP;}RTE62Lwc^3)>X~cuQ;%&rMV&hlqIw_x zMAiHBma4U3ZT0u|yQ$jq8irS2dqSQ1XU4(VU@Y~&Sr@22xA`h!#GCMwjL7gUvTeA- z+;!BAUTn(wE%9lwjdq7KYX&L*HZirK>ngQASR?J=9uKv}a-h25n}bwTx`6^?7EyB+ zBvYsFjR^nD`J0M~*dBhEW1--iBPpQ&?^Kr~c~sl8ol@sXb}BN~lJ@0lHg(SUH`Vx7 z7iw*qCj5NMn()L92z9&F8tP?)C;a?r-8Ap#$<#UPW2#lCS(@|9dTRRZThvGFfGs34P z^rx7%u~gCI1ypsbW>nKF7pT^WA5u$}riTAJevVqv;Y8T8eF_C_Tu${}l}|k)N~@zx ze^TS@6R4J72UAz(`>3p8ms5Alg2Uapm#JALd#Iu4Z((ZwdaBK{Zqy@fzwp9I3&LCO zt_x4w-KWxUX%dzGs9)+8m6KYuLtWkDkvny})k=+cg*Xq zc*}+{;otUFQ}B_Llu7%8)QS^x!c=(*wPtmj@TyU<;n~j)P=!6uQyH(1q~?A)Kz-C4 zuimomB2~DsA7$P(FueBf!C^FZzwDmD((XL^o2v2O{IFjWmpXr}n#!o*re>FKuTI}l zNu6xAn0it*zWS(wplaT5rzSbL@YOe)t4H)YReh%sPW_scnK;$PrEbn9D* zm-=U$ol5MnhkD;&RM`Judw5CZ;S}B0^{ERiQW%j3c!nIlLZx)-N)^TIrO@`()cJ-N zsR_*l>e|T-)FE@UseK9wcdFo6ESdYVT2U7WubuTwV~ZUM(`L-9qatemoxZ&D|^@ z55#-OltA5JH-0^S(VuT7$;Os3h5|ts)M?KLHL#Bu410r%5F3^*M4Hr+mb;F&6D#;0 z+%se))DZ2Yk7EmbJCV=$lE4DbUUnKgfKVYd#dG3IhuvF0zAbl680OS)ZB&D8J!P)? z3bX;Yv6x3cfcwF_NM(FC^scbMzQg;}bCdnaucqg*Z^1Z8iHt`6SGvm=;~#?=%AW_P z;FFE#0+dDNH~D&y3Nl5u1t?ni@XzEPbdt_NCY978oB8re_tHLMywPm99k|BtGewm$ zA&AU1wxG9T2L1NnLd*ZWy$oxR46y4ICy{5a-zRpgW)d2BLPniFhZF z&5qzW@PX%lbZT&e-xGWYT>(}FKAKVjr$d!S$n;+zGL%<3o8){QO*75gyp{yMWTCl4 z2jRXz8{-?i8PdQy(6CD+kVb)CNN@ZlRt(%SY!#A)@o~8ffYszi#wxju#8zw$-cq~S zc>s+le(6inzV)=z+-+UnwhTKg>_pnSBTcnPSjr+#5LcwG!2wpl)5dR>zV5CYYK=T8 zDK+Rdn*tMz-++50Vrb0$^2`q&5yz{CnU-rKp<0HALJEDRY)R_Ss@W9>%l`{?Bs*4h zO3hCpDxa1Fob@Zar9lav>W-xg{by79!F8(t=u}p^JFRwQLY2pIq@pBsO!bJvfN8qa zH#C!45X>b0NQTQMm;bO{GUugiD_0^S*g1J}O25j*VSD05LS0o?I>A5CKQHqqScn4& zUsHyp+L!_E8t5T@HDAXm$hU>b>a7Bzo93QCkL8*{8SeVpMO=|%lz%y^!=T^ z*jU9pW;g)gUA#V54edvL6YRaOJ^R9+$FGsil3`-E^jVtag^dsWenS;&^X?(jJZ5yg zyDO=Q8-n-t`n`*Bqi>M0e_Jb?Le!LQL6u-TfEHWJDN=)XvWUQ8nYj2vO!MkROuCD6 zC?ox_7)ou^DVtX*iT0%rzITH*DRlFdXn(A#2CC>I?CEie1`7` z4zgHfi_ybCFsL?YLmiTrNi9Oz-Z;^h^vdfM*-EWxw_&I$9869+9~zNVgOZ5}oSb3|HG3g)8%jEZ^*=e^DlB>rB zSNY9`oPeTwU15-WFB4qr%_4(txKxMS?l;DQ-SZKoE}UmT}=-MiqUK-Ar#?X zWf>PLPiW@*WqL054c3P)WB+lli3n&oS?)9B&rk>mt{-i_T;fo-4G=y8-3Pul%p`Pv zjBuA`>aEh(@I^Gj-IVq1TZ z??Wy1Kcu@tYhpTk33`uP;PZCBWuh@FY6Kp}KN;#6iu7mM*~l&SH1t9DAHGdJ$lJv4 zYIQ}~r)+;&Cp0cAr)*()*OD2Q$L1~4Duv4IPrL;tgkG4PrlZD4Pi@0mJ+O0 zy1Km7FB-OxALTXbQsof#e~3-kB~&<*Wkxd(^lD$t&B05dKeRh^pRw9SJ?L{>k>?iQ z*m=qrfn20l6Iq&F4nLD27V@nCp4i4ECmbf;VMh`hr5rH~=DlSdDxZ-3OyjM8l*L%n z6Q)>iB^}21BzKV>T9;I)O8-o}TIME}TdyX3O6(piE$u?IQGFnWf*rX5d5_4ey8EOh z$dPI;Cm<9FfzQ4u=(vBV|1@0-Z3V}=c6uMtx5?qYAO3krwYd_nS7Jp*>7Us9sK)Cy z@yU6g;$OL^;G(uzY{a%O_E){+FSd*Hx>-7 z13n^I6F5y*k{bc!AuN+(JM@O-t?=3W=HJ!vY`4I-IPOb2#us zw24_GMp)+s_W4Nf?+MRQy+6D3D0nGUX*pGx(y?C2mcZbIq@;PFnZ#-4ZpB8+N#RUN zuV74aeCRZOD`C6T#(D$)&v3A0OVE>y_>vNqm7JEHXEjO#+a#$ zCC6PC%(rNT>11g%r1Nz0wneRmo%$K-bS5jXA-X%%Me1N`;(H_|;Zf{#F@vmQOfSF5 z-6&cU*8f8&re92FoinJI;# z{Xwz&~10Oo2U*yQ(TOTrX%k#IXiRrX7qYEmQ6p8tV8 z^p``kEZfU>CTvRVWG!>-E5&8c{U!f6>|gPjJw0G>Ic=o4%FsDbif=0E1!j2`z_I2Y zQct+I;e=nO8Z3@6y>CB{aao*V6LPw*Wnc`kBX*c>I!1a|7<2qq{VHN0`oK?m=g>RM zQKn4MP8P`?hhNJ(+rwXBoPsxXPVgszj2# zM4*o*?EOt#$VEuq%@p2>>CG0(`Q&DfapWMhISskYWXrfsgjN32l;|PQS0U2}&uUno z#kh$)XO6}jBXdn1Fp&PnXNZ@{BgC4}WYZBrO(MS6ffS*(TvWJi*lFG&XHGggr@0!5 zNigZy&YnlR7bWs->Akk|bR@DJX%v4{=I2WM61WlbUN^x}$a9fhZBaVaO( z82z0b7v!!+dw64<9@Y!L3<%Z`$fV2lPYiom3$Y#F7uAUz$&C?5>({`w0%=SPamFz~ z*G(uuCL>XaU8;U7u_l{BgA#)NP14W6GwVR~okS!u$F$4wJ^{{D<<0Sq2>y#Ocx^Bjk7I_(X1)l# zA34k&0Pb*YgMguizEMK+VB^651fBQ~+q(3DHP;s%-86&?z4ZU8pZEdnfMFnXj+qZ{ z3)C*DXN+dXdUpC8KwZO-T+Wf_ZLIhy_pQ(J8|j|hA$?=xBD!A*&hO$OF&a(upA@#0 zJoQg9Rs1otP?i4m3z?f?bxf-4|u9#{+aKQNj#IJenrhld_H>4Y|gr z2{ry#*tMIbZC^qsEMRjw#PoMf z1;N@4Be>q2XP!bV$LkWF$a?kx+B$47=`bJNsI&u~izgE$+?qg3XRh#ud%+sO2T&3A z9&u21stKPXoYz(--%IW|Ep`L_Of}Y=@^-|oD)$;Mz;N&YYOwcCxrlU=FO!hKZ;-jF z#{ojVk@E0&s2=DKD2{EeY3YB+sF-2c1h$wD>i(830uRG)$N)OWy`*JdPj5{fCgA-6 z#^LS!zr8f5M|vxMr(S@QaF=vG#FI`56VlJB5>)-f%J@V4Hk^k|0?e3CJHGI@#5oy* zRrj|kn2ogN>gd-i1h^BI&Erfgy^-#&{t#np*_oS9H-(2Hr&{z36}zvYFVp6$7dYog z()ua*4Bcd9k}l8F*0H^~9n}S?LX~-kV};@?Y^**UuHZJ6u0)4pOL#Fjh%d3fFbyNJ zkZ<^U;-hgW(>EO8_XoNJO8Dn^wpl5D2|NVq68*8E-t+2hf$r!cz!O-xJB5144)E>5 zEBR}AC)C@;PzHxA@kgAiLL_BKxw{*HCT-IDY1#U$@gZIcz%Lu?DW&T+WHcpbb}RzfuK48i@n&hqd41BmC|NZ+?2`|6~6@OD^$zZ6bGMB!`8?${O2N8P$o zTUiV1i}3xj=f*#ck}Jj&yWJfn|z;_A)@7K02X@sl(QU_ksh!zgl=Wr7s!j?RnuFLUwETPQTgr zQR!#1G!|&uKnS7IC*|tDp&o1x|O)Fqp53d?jWOJCHX%4-qt0gnEdj zY<#GIeS<8;pJCRZMjp?*g+`&tctf8$&|Uq;v4QI#j=}Z?9)QiTcb?$@s7v;Il=g~l z)g9!G>VSVZzf{?n4=DOB z9mO6o3+4CqUlIyolYSHuBF6wHfSv5Il37q~Pt@HW%bQfpQH zWv!i0@E@N3fo*uUF2gjy+a3`EckRFNM(S*phQBB7k_bOJ>Bl)7=wsN3)fVa!JH^MT zyKtBA8T;kCLo^}2D~<< zUJUC`;QC80fGunsEYuPKL)fq)jEOOM zF0_?7srnH(=lyNp=UL@t@oC0=NRis7%J=_dbF{bIeWiZU z9V(Ft;9*yNxR+3m9GUOrS22CPhuq&d3*4G*1~ zz^^xbGyqg{d=Bwe7!|H4T^JgTR@KEc*ZdlKJ0^Ev9PSK3I&D84tQ)VKiSCtE#W#u5 zv9p>MxW&|f`cGJ;cBH3ZJ@{{!iJgYshMnY4*%i2-IFqjzeudu?Zc0?39Nmv#CparK zH3G;`Axo`borayJLc=eA_l*Ahq~I7Ygg;QZ_y(Fki8sKPvHHYlmLS z)skbi7k<>}P47pRn$BPY5iD@of-`0=g54pr@p<@dvqC=AG_BwTTu!y*K??~#&(Pqi zvrB-&Q%Z};( z)Xf$4dk56Lg>w}#4Kw!&+`vs_DBj0`hXP>E?Z?2n)Z{20T0K1tKm zK(p<=XME$~9Q|_MOr8mh7n}I2sjJdFTS?4;pRtQWRk^wQ`WXUV+Bb(89~_QR+!oJU z@2S8N@gDpRxvLlbo#=MPhJq6q!KUc@LahZA3mRGY3O3T8h23$XlAW+M9ATc(gS0Ny z9eGn{XMG`+gjDz&(<0c6z3ukPe#ifmHS^ApXMsojDT))y-~=(rtgkaBNu9_V>hYu_IB`o1;92H+N)bA)i>z{z)$&64^%BC83UX zsP_SzC?pWiBY$UX!bb<^;7hTthK9&Ua}oRy=@MDm;uA6sXczo1R4I}h-WwFdVZ1wB zq|1?bdOINhjaoj1OL-6F}w+QG>s90nSMdH|TVggtTU4!iO2$CN?7cImG z3+})qDjv3C>xv8L7wA!77uJzW5XWqm+WjjbP+ag2pog)xK=1%zhZZ|`>Imj_#_}?QknPaw+7fxeuk0FpB2&-w2Cot%&?@Gm zff0g(Z{VEB@8|a!W*SONFHJ`+U75Z>WBnl$!p95CsmsU=vcG?e=Mz)c)rq9IL-0ke z6|fITMw$k;%Fc1^1+%dgW8{8PL!_pDO?U}1fqc$POSS-lx2pP`IGV-tznZe(_Ix|_ zY2Y!eFjT?Y2b0yO;dP}D92fW>TZNlV+;;Q29_(0AuMg{=hT2j&@CSCOFhT#pKT1ps zbqeS)ntBGfR3V}ow#c(e+Z`+s_Zd+GO-wP>iOlC2QwPaN_%FAUyaz48rkFRR%LBVj z4yKBEQOi-@E3i%vqRG)>iG(o_d(0V#!K@0O?KdK8utTQW>Pu8@;v-v~KZ2``tqq09 zR7QnN<7yGN`5|}-+Jw}Z8X1=(_XF+u65TTt#aq#XATXpgP6z5cZ>vwUW7YqOCVfKM zZ&kTofL0)*#jCn(lT(`Q&X&5N5m&i?A+9W09n29n;eN6~SdZqhL*cM-ma3`$8?l1x z2^ZmWh1XnH(~iJL{ZZpsV>5O;pT%0S-b^#vVcgE_;e6Brc%E<%ZAx`xYj|zCN$6;J zZlEY-fRs&%A-|CGGytX%I#0U4C*LG2;3UB+e8^!n4O;!lqV`(3biNlQVGGxJ( z(iW{FaZ2CTPzYz4ay-k7>%Hd#Nx?6=0b+&ztf2?76-g@6@fvA$+qASh;*^M-FZcsI z$y{a*iGQLtP!<@5UqZ~ndB#G3p7_WcdrwyioQ$<#ry44Np+%KFDVo913*n%lwy_3l zWECxY0oTzj;#YCHHYQm)-s^k76VmN?8D^1ax}>lT*=ous4zpd5Rs1^suY#7rD|{bk z3>Dyc<_6N8+l*ZDb}4$qK1aLJy+{L9lWaz&=s{U2|6fv;vbS=bYNYN;vk!p9SF06^ zPp5t|Mf#wsiL-9YSW3;bmareZ>F&$4rT$?bu)&ri;kf|h>MpIvR zj@;9P^&VSokNAsUQOvg*zq*aQ!8marN8=iLrybRxGA@Z2A8xv2*fR9CO zpegVtU2A5Bn{*@uf=C=J9Dw~*U~hU)-DH_Fpk!cK9bgH4Qu=4jN!!V1h^ z_KCjXO3XuyAwzX!SF8wY3-{y4du!-7YVYeIu0Piid*|B9_4YM|$z~%tRx%Ws0$Y$A zR?3?v?E!1$|C=gdU1<~FU$BYFVLCX|R_}o8 z?<`n{=F&8M8LiHjunGJ@a|%kuGA@N8xuKSFcJ{Q(ZkN0N{A%T$Z; z@4gL83bhAZ%M=TMHw)OPUF!J(plB=QlEN(2a`>8Vr>cRgz{|-#yRE88u4|qU^V~TZ zY~!N9Iw=x)oU6mzGWwx!^Mk=U%xvvU=??m5u%Z4h|=Hyv5}Us8cSjh#$=fX_+%h+Ow(J&0_j6Y-+J zYg+C5wzZ+ z@-!X&|1klbwEZLNf?F9UJW+{IKF`4f6LKA8=sedJER{``W`~U|=OV2m1(+G>>uDKD zqsYL9h|93SJc!7F$Ei1(kMN{9qO(%Bz^~A^=)Uxo5k8^>M-!ukL7JGuq=6kWPBP6-ywHW<@^+`J61Kl zCozvd!|%h21Ae5I#9A)Mf5L6e8zJ?lXPHJz`?5Cdgb*6)i7rQ2cBOh=;2E7Motsj% zgZY!tE+zs05R@jr>OAa+{v%exuMM3>-eY&TkHKyHKxUqyxryhSaQj2SR@V)+g5A@7jq5 zITnvrC9=sYWE&xpQrln+mFZ7%E5W5i72yC@$t^H1nUx1qxxPYL{u0eCtT)T^|MSE4 zNrAS7p#O==;Ctdv;qL*XvCaOouEuW1J|nAPcenB(=Snt%XK9^oEjuprOW{CwPbh}^na4p-A?-? z|5M_1StQcBXpXTwoXIUFo&qm~$L4aPSG0@7rtAXuDE(YM+OCOq4kLj)(~szRrbjXh zkI(ppWm(Row>BqQIuwo*E`c*CS>Z0isj?t4ya}(y9~d2oLdY?f227~WR4&#GebJwR z%M4M&WieSi8l0$|X6%ps0aTce6I)fYxJi+uAj}rx8>OvpBVjr}Hd7XC&J7A~G}=r{ z@z2KN)Nf&W#z5u|O%C7Ga5MOdiVK&R_5oIRiDnv52U)0m<1J2|soNm&jDngdIF`}1 zP`eaG*Oa$`>V-xH?Z4-mr}3x1rB>9LFQ?o*wIh9odm7uCy@2*I>|<3#KZt^ValaIk zodbnV`ltAOqHc)Pe>VVFecZ~X2@BEh@d+Ts&ms)y0_}CYF8v(I2FFXgyPf)zAq#Yh zY@@%gy_uSZZ9%iJFEEJh752ap4jl4I1}Gzy@)q9y}z|G8zHhTjT~4lJfN z_-m&g0x#w7&{sF4v;EnMjQZhGLVr8zZ@>>N?OOC$dcMZmh6Y&pfYmnL2}ndUpMjWS zh%tV{T%^FzDa?g%aa-{qe?5G6X;-5`7a#f&y@^gDU;`w4!7&3A_NbpT0q$)^FYja1 zL}F5L7^$h>D1vkYpgFe_wex#3hZ1jufAyKf5oiZ74;JCY{58Bg;M8qqItC5|cVHL8 znb<@FAnjSYhOUYv-As59WXzM>xW9$+0ld4HyRZ zDCz9~Ec>Qa%Imw*=%>EP$~z! zdjN*mj;xLw&$+pJz#+zgE`s~%@1SlWk*UF7fyd(y__yklniTgv9?@MSv-m~GDfI_L zNgIiVc;Db}GZMzgms}Tw62-BQGY)a7dCo$)6~qGn%#s72O|YR zN@|S6VV_eHk!~3c!QtgMQmP69f0g-&?-b@Z8v*yB7RqhlM&C2Q#_nR}5{3IHj>A#r zKYu>i6G!lYd_DOurbVC(YoY5AT2b6Yf0#LlU)G-^dpjGz@Ac`_Z@8m+G&%;$H~ywp z@;vqi1CT&&cyuuf?z+k8`A`PNw(1jh-%1sY@%ivHNZIp{>TgF|`QSFO>ziYQGWpxXnhrWPr$sZY8?F0rIQ-C;bGRAzYy9 z!g6pUn+^7)`!Mw#8-d==nu=Q>&pniklq>d4fWNSvJ%iNa$;0l8+5vV7Y66dxTZJBt zU;3ly2o4m@!JhCfEVJkqALB%Eg<+PmE8dk-Xhun7`Xj3EYIt!eE{x&Y$y1?U{8xC6yCiAgzIf&`qEi#_KyEMcj%triKpeCQ+d%iK z|5BB?B9eAKu$$sW%Dx6Ww?xg$m^R$%O`w+@y{$xS$1GTX7#Kh zV?OqWu-n+%ye)bXdlp$E&WoNF`en|>t*URoYG5;fFI1USJW_SH->+GVL|Hx$$V@vi{jaSRiu+a-#fOnLD$h^ke29xGkJySrC219y4CEs-int z933DIJ-nD0l0CV+zwLh&wP=TBD7Rg*8%*bx8;(nKx07)5!`mV|GGgMSGAh{0-3&Nm zS%c{_4~R=FzigwJ*S3$r2Bp75n~;fXbStw)z~_ja(I7WKzMWmCnSpaezwndftKb#m zj!?4sH-3Zv290N*7UmkCazl(S(O-CV>@W7F|1YF7dOyFEe(9KQ3K-r7{(^&e0pG@V zSRe=(z}Zf|zmcZUKkEKu7jTF>8LMz!ggrtxrbKhY)n00!n}c$&Cf+1(2boL1&@a{= z#5ROavxkg3)K5b@0{Pe+!yAGPo?_msD{~vrNqlX_CVVGfNa?PPbXC*;FxFQ9tqfe? zw~JLn@8L|tyc9RnE7Y1=;6EKsS4A}cx}V$KhTZzhisAfG`C8PYTcLR9zbDbj3hkx*jyk8Z@^&CQfB++WfDnHV}bl4?bS z9U@JDSS_>9yvpE`Z1QzWFUZE5eiM5o651H+s=$%-LF^aljJXa^$8tkUm=s?MHBPw) z@4$TkT}XSw5n(aDzvLG@n^?zxkP-=Z$wda9{$?oU`e0`Mmcxxow`M%e{FaXozKVPn zE})c8!`>mkBjzB;Cm9+A4d6MpFCiCI7%w?su9fb@GXkHGJ0`o3#8okQ_z7k}pDqkw zE+psz4MRF(1M-eEcl&HCBA%?G1=I{R0ki3A0oy9srUq_6Bt0hGyO1(Sjw3Lsc0R1AJ)e3+%Tgw2c2fh z!7jT~`3dms(l7XR80F5<13VY8sOu7Vlzi^t>mI{a1ghcdxhKeP0mNG=<0Bo>$4pQ0 zu;*~_Dq}T{Mql|(U`xRLd@YR-r%L;dPLgtH*~NdrgFsXnQ11t}z}e_!c~7c$asxq4 zk7TFmGx(-^EMUGT(etlv5qX{KY{=AIL8q41klFknLh*dN@C)=WxFwMd9x=>9CTaU= z>j?{~5vKFR5M&9yCt@MD1!VfMLSOQ2a6huwuuH5di9#PUMTnf!>%Gf2Nv>A zgw?c^SJPb*P6db29?d_@*5Dh%3_}lOA%7?Ft+pSiF+OzthIa>labE5KEHDG`%C1HD zKBhfq^#2W~VB63sQZ{|9PYxZT@}SPh$AF&s6VTG%p^yIF-ivAVrPkj;MOa>0xs;m( z-j$C+k7^&;D|5{ppIrBH*C3ODnS1X@)}4IWA?gqK0iEd2((G}~fq;^B(#gY8 z|EeFUr&2S~IYQ6W404m|8~gw1zvcQH;5hg;+Yj!an&93^cVZeHoGQ85k1%I+E9gOD z8z`N8!6l-@bU)dniX3tdbx}p}%~ieAN6PS6+cWp9tvZji=3UG@RC(RkvhoEJDt3|a z*^kLM+sex4V$ zw)IVcL#tF*Y+1X^<=-kVu{moB<;7#p*&J*4TiJF$%F9<3R5*H;#QL3?m{qNIx6HNu zR#k-23o*RWuGloix{9OmlVf6~yx5`MzsnnBG_!T7tv@s)cTCol-koBK&ZDh6A2H=M z{(79%=x*;=*(b)jZdFo6k0F<1?Qa!Ztyy)fH$y#bhtd-&#NU-Hga7+!OEFbD^tx?9 zET>z)iihD?){U3J@^_1C$J!Kwww%HtHr57}?>X|Y;=#kNnZw)FtAHj3ZM#1*<B-Pkz7}POl-TZWZ=KR?mGc|8I+L{@UXZ>CKpIHBJ zG|LQksMx#?ub8Fctm@r$tc&Nov<*rTt@A7{>--{G+%8w1)ZRuQjnsxZNT8Ao~dS@-TSWrHCJ!VUt zR$hLOUtUq~e1n*)y3FQYxHjf{|0SlxzF2Fndl`K|JBG&2539*>nc{pj9O{DXU3v zUhMN;gRg)JX)};L2X#b&5jD*v+XHi{6HG>80sog~z~TN_nDhmUm}Wt%HlFRwzHK#$ z<9IGum3vcqw`{U_hgfcj67lBirn*=T3wX+v!$X@RhpQ>4W)sP#Y!Q`>UcxMRF1%j38BSyB3nvUcg^8J;4r|Hu zs4b%l7N`6ROH~6=Sw{FY&gI~h! zMjEML^1OFCxg%qA$>_q5#zciL)P%1s?#DB9$%dogB%OvV!rn@|xx22)+-(HV?WB(D zo3-+g2CgR|IogRq@}sHqc7>7m@LiTKG&&=>?u` zTs=6;n1HMGTlBBPC%7q=oT819>b#VuGCwwcKxz|r;EhdlwKa=d?H{3VniqQekjbDA zwgwMldu`ey^Rqg6CXf?-)xd%Do0j_n+bsiFnwlgTtLE|5s2@bj)*tkKNj+qUOhJo? zo465=H(?pe!9e-B=me3?Z$n*7>Qo5h>!CRHAI2_CYDTL?E=RsVZHel7tKl)$K$$52 zD^E_oZTc$n90uq0J{%NWN@=U!7wwWq}KvQsIWEDx}o zhOO9Ld{{UvY9bBY&Fu&MUmXSBKU^J{hSofp#rH`+R`FbsTGEkTE&E@_NcUyuEM%#w zbMXr0|7bYUJW?v^4VQ`c#A>2*IK>#uXc}8?u3L6EE46G~up@UUv_c%rxx^CHQ$|5O zb#&m86r)uK1N}44f~$f#P!)G2Pap3AcYPD?{9$jRS(&EN)QjNoMb8j0S-0Ko3a@nM zO3ba>+Rem!<1gbnU54@t_z9{-enG-I8vhL)MiQ_*=sxuVZ04VrIgr2Ng^?CWJ$MV! z5a@6C>!{$yVFwQkcH@U(5Z^S z|8PH${n!s`aI_?%h>S1mAhkUHj+|wViZe@BAsMFgaywP^z(Z&q+DG-sIUl-_m&NQ< zov>>|hPK!JLxIXr9p5A@szK2jKFO9UR5kDA&Y0TS?}@w39f`@&gIOku4+&U%%q^S4 zx96L&r?{Wd7ebf77nsHR5FRl$`ZUI*Jd+i(e@lrqHTVQV4LT;|Nne`OC5Pr{`M+93!R9n&6yj|e=qV zb|F4IgW<5g+#4tx_zTE`+9LlzM*=s62i!X9p33Rfh={(nki&Q3T>Cv>C5BfmlD{y$ z^mGHhE9;tzCp=`7kFX<@6{W5MbIwJ)HMq&#&3sTGEB=-Zgk{$+TeV5_+hNT zQY|K70i-NqEF;^yV8kHatfT4`wrF_&>u|U0rs%WJ~{>>ndg* z2K}9|$HWk>yar9h^&7pb7j_#K85j3@JZvYM@^b4qz((IKC?vKPXW&-Ujh zNOy*B%o=uexF7Qe_}%+1=`Kth;DLd`e2az8agV3^s3WA2TBEFFxErZv%BA}!)%1XQ zg#2v*2(5^2hFK4;Nb@+wOY*0N95HH&@*Tw2Otz6S?m+eUN#L;g2Rus1DeCS#V3J@K z*wW}R&(vr;j_^y~KJQ(i7TGYk-PBy#1rKK)!q2I>>^q;DZp)F%~Jw)bwIz z61?!62!gMC9TaOxw_XHT?HZyJvOvw~9KdsSwr3n;+5b0_j&wKNKra}tt4lx`usilD ze8ti-oF%r(n#MH`lmM492f$iwb7_ZpUa^I4z!+WW;ufd_ev8@X|EbGjR<^LH(<7sd zr$cyfw4s^rIQuGZ4b_Bs3pZx+%&m|z(^>Aa@KVE-VMOh+TH(=LgK*DGOW>cKzU%>I zqr~IHr0^>Luc*#<170N@HVnwB72V_4Sib4baBURdEeS&1awq*>A`Rs^vwftu#@<_| zL8>o>%YENfqjh)d2%1@j?aU^umac-@>EI;8aFHM2TC(fODb%XKY~~j|-l#{bo8AT0 z(U}5|%S3|em-g8`&HTW*RN^%kfm4C=hAYTDxSurnJQqA47#X@IE;ta?9|UZ=!mJfo zS89V~WE`h=XKv7y>V~slH=lqHT3$gXiFasY?poruJ- zRhaW!Gp1f-IMWB&Z(4~lrnhKkT^6PQicoX;r_yQVTZoHNGV2TWKV-E*Z=Q#ZBBnCh z&`|9q`ln3^$7v(fpN`GWmGow3C#T+W%5_nZz)n+t4(%ra9*TqmvPiLMv{o?wkIhA3 zsx?xcaZF+lM>ljNql~C#T}3G=mRwDmzc6@^encHIo}+vF+qF61Ri}`?0}fB-79!oY zEBcpMFZ43T;=>~b;{fwjd{C_~xj2P73sIZUXs&O;`RtaC^w45)CRk|hc?SLU&=sHe^dxyN+|ws9xnEbFnO zgR=iIR2O?2DE|;(BJh9^nmJTG)j@DIL;x4T)xclqQ1vVzL6@0S%WzLw47RmY zO6w{9h%N|InpXv5wVUB;=t%II`x-UMSD)R+%qf5N5K8=HYB(d!mQF68n@ZQ+1|vX*g2k87K$T zDZXi{UwbR36co<|dK=%%QcFh26q>K-JS-cB`KR1=_LHuPJ2$z<;gs%+yLCOgv17N` z{PWe?qepl0pTxz1sYDGT-ZY1ysCnS#puTin;1{jV3Q8`Vo8%|`*Xa3Df5off&TIvR znvNS@83thGe4UcU|k%%{KW0q5ympeGzuq8kD_}SPy%T z&Nd~tURtU&eMFW=HWGiB&cq^csks^37C6HVcK9Mm%s_CJDFS%O_F7(Z8txiBplb#V zmp+~hHj=#p)v?Vit825lhU=fwgA=7}WuQ{)#H0JIkJu;Z0AY_imupCEu*sw2q&dVO zXoWk(RB_ZLM?iBk#*3+pOlzH^p!n=>)%NAAt6k}-bkG-aJw zZFE7!t?XiJqdJqywsbyO{`%<0@@?@QtoSoDwtJ{5%LT;Q!s@5x@4+3`aZFp;Yxplu z3dvL-Nwd^W|9o=wzInQT)27ISg;n5dik+Hc?thhd$y%~&yr@_LIH+~%r~8(%M?E(s z3L@&>C_{lUpo(hUVwyIC)X;tDH*QR`JoTr1jIS!R*8Z`?MrpMxxE0z)p5u=C9vt~k zU!iL5`62)1o|gtt#Lj8HsZ;^*eP_6FZ?#fgDdB+Z0(8SsbHx%>Gk7Nwg5sEG?yK=F z)!Myxbuutjjmk2;W%QVW5mHzMMM_Au+v5whFtU95c}D|Xz_A$#@4f?1^pyZkFsjm1 z<8-;qTktkAO`Sn5FFvco$l~M{nlxsPdp0-;yx};jZszVoVN9IVhH9$r<56qV!9B`J zsf6x!;z*<=xG8>xVzY*$9@u-!TFd%7hDjMmPF1V}!G^5Ze1T@L`<7>?vOuThzHIO8 zqmj9VTfMExL+LBQAWjUxAc%taUR8eyF z>s}ewgc_$hTey;Pc}V+f>ZpU%O?4%^zZIH&BRmh(df5xL1*qX)3C$%FnSSnGKoZzn zC5WSy9_5}P?gOI<;7AlHb zOO*mA1JI{Ts8+-pxII-&49HXq07YrpXiGmt@z( zk&dr2AMk*@t#l|g=`mo3)Mfyo1o{)`fYO|w;IQXk>U!bCfCBJP!?c)+@@CV14_Gt~ zzUnR2H3GMh(fnHeg=7bKymxy3O)2>evw&_4QY3SDk2~5Z3rl|3^~wFcbzqGv2>4Xz z)KkH4xmz6!uvuLlcESbpby;=kH%y1WYPxtUNxQ^e@&rZ6oDE(k3ynCrQ4& zFWxQc?#0WBM}bG73*;R4ByAJVzd*91_62-sZ4rO=$c;ExLNcRiKS6q>o zyBfBtM}8^^10MS@kBO>NFceUzVIKfi_Ng>QfDV}r-`Ap^c-=AiU9$C-bZ{&$*^G-}v8!e@7O zPbfVrvx@&t)N`&#A#n)$r$rsNu3iqEd-w6~An=5EdER5sNyc`wJu)YT3h)6)%T z^EF?A2EZ#vz$=nxymgdgfphZu?0NFAR|XGus;TYCLzS&ls{%j0{plttwO!XB*y+@i zroZ*Z=ug^pU>ZClxwYEsiKLzOg6dC6@v5sp1=$eZd-0%)JM+_M8PH;@6Z?E4Zg0KHZ7eLcyf z^a`5xjL>v-RMRB+1I6d53#3F4(7vXE_Wu4}jy!0#x~Fs6zW=~D-BoZC(7#}_dYa=0 zbrhi84efoB4|+f{9tQH4sFNk~m{R+Pf1$m-LM+Ia|rG3HoN7vP|ygIk>vRL=K?Wgnm%^(}WI{Y~|iybr(h=cKJnH?vz* ztAPB*qu4E5D)MWqYJ=UqcjP}@BfNq5B1VwkkbYUbKQ4VdRfn`e+MO0hg?+u^yRVJJ zEZYKAmQHCmwFaOb-CNrpSk7$rCCFx_6acQ4R-`d_mENJ6<8G}g_KfmxCz*m}_9{~2 zt4iVz@|VsIG;;a<^Zgq&H@NEZBo9)2RLQw_$-9FGJ^y)511h>q+lBGaI=YqSl=QSm z$iS}J$p8Bk-&MJzCR`>mDHXKybQiiM;Lw(Pj*=!t3DtnA1DqwZmFBdKGNr1ocBwbl zHHk^yxDdqs1N^&nInrD9&^x&3v-(QI3SYm}x;qDYnnJzp37QzY!+G02$YJx(O>(#> zD$#ug+^CsU+%W~BLS!r1S9v}1oH`6}?mv79@ZsV(=K^3AHR-?!@{+C@NEWG7chWg> z5c#rt9Q`6KQ^n~({cPq>W-j$ou^cv2tnOGEWj_eN@vN5E-OF92z+F`}U#wUwQPM^h zRi_Ev6V3Rhnh6(sF9Al)S2`6O;=ieR z1GV-1ghx^;?*eoXdqlI$JBm4WU>KP|PA$svJdD2#WH`#yP1rT+vka+C0iMw5S_QpL z+0NC1yr_w$&Xaf3l@(~zYc=hP2f5?H`VJXzCvO2*&ASQgMxF*PAt(1XMP5Op96$6a z{v+(n+(JrS>?Id#l#Vj;mE?vP!+eJODa!3%6bn2Lxwa%GQQ6k%ey6u<*@T?OKlR`6 zn|2@4Y;kT*IpnVg<~dp+1*%phRrC|Q4c!U$b&e*wN*&tIACC5knk)+JHO2un67oFc5%tEzidh^Cu@B6lk)tN*h%DxRTws5+}% zt}#%ZWv0TnMb}mL3N7FfN0zQ*s=@IJT%tXoxK^TY6uGu|eeQV$*OLuyrRo&D=bc`~l>R0mCv(YsPnC5NZ`P)CilcehA`?a@)^J$7ArNpz6s$Nv)9^mey z$g)@UAM=@+v)*?HCKM4GEtul&SePO=DrfDx0@e3ka5ikU0lt#{C6$(h7JZ?6sXb*6 zspYWG&okR37JY(ag?A~?n$-E8yDQtfve!H>S^>^mfXAm|9KPSc3#g*d8^773L^K+) zq=t5E;u8CLVE2aYB%-P5@ugRV?vs(VkCNvA!`wC%vM;S(U$2b3NTST&t}eyb;d>?X zz`3fW-u(FgWJ3Uh>O@+7-{z8)WIg4wRx0%=*-UjUiBF>?SF4Yy->J8#3H4!jsr!*aGX6Z zpMpLGN~^~xuWtW8iq0~+iTrKDxO+=ok~A_dks74|io3hJF7D3a(BgH8XIvr` zq{ZEB;SYp5XL(fSvlixhgeP35&u$@aHo4Y2_1<1+RL(nAdE1?YXbg$t# z!Uy&P(Q5bU*mm&$xEk1XLO`a9i&02hlP%TG!s47Kp#tQy{w6e;+Qr=h>tlzh#^4h+ zicrH1nD#CsaaSZf)8Xap9&W3<3*CaM%T$VI613iC#HLu8t8Jg9SGyaFZIM!90a{8s zoPTTPg0OU$ni4x3TVcCU$??tnc z`NiF|6PWw_2?K_X$L7GJbVjx@>O!MX9hHdG1KvqlxhI^v=}p=~$2)c$bqmtM9(*k0 z;4Fo4;=fXYt{c?YtJd2Qucw)?kP3KDGk?SDoFS0eHkJP9{S?Xj8lm-tZY9$Vb zfo>JJUW`F{fzO~_#kqOyJ>!Hkbhea*rMk+Q=X5tcLp|M*!`0^&+pYk0Kn99N_8Gcy zA2|b62SPB8eio7pd@J5Vn>pUz0PaYx=3awc^6w~Kidu$wM24-3exSUk^?m8e-wsd* zUjnWcpxQ4^hvB0(IpKdo8|CgeSZU_0p__VK>Pq({dT?}a_6o3?$LwAy$5zC4KssxM zm{G_QLnm)*Y_hPLvO*(Vr#*k{X7mRz)$m&!5VzLef=&=}^8l!YnkGsfL^L=M%3L z2hnAAoa;nfCX@m{WP0ghkWpL$wpVQkzbrp=?{}SP7RTioh6}wLa^B4528|yaSa^;7 zyJZ==3j6>^sJ}AP$&Kh`s=rvnb=%p7nh#;p43V|f0M7Dv(J(!atl!wrvCqChfZ40w zZMqO)j6#fx)qkgY02h$u_C$CW)u`YCsAE|}k^7`Uht)^U0YepT?^p2=^^&@b*O89f z-MpD=p3oWWN|k5_YJYPNphNCf@F}G+?Pseqg+;xc!=ygk1trfeMa2s*sSoT(ppHA4 zR{9oF5!6!?;GQ5I8pR~Q; z?z{&%lTcGL+)Cw-d4)Z%Y35 z&g5caS88iwH)30}+PIB$4fdU3qQ?W&MQ^x5bzxA>uJ>qH!$fKc62fL8jg=>l#g5Hz z5nO0{LS-?_4Y&1$&=?}xHdN{*W-Czd?>Q}rD5M&j(*!WfTgYGaUT13o?Z87)7v;&Sy=^Yij7~rhI23BFje!4Xh5{*uz40Tom7*Oj zLO&{Z@eUB`a*|q&3*83H!t6T?2y%)eo`T5GPm*IijTLBH1RK6)_spbm!3u_qJwTo%^&_%0(}ePN2E=A=FHyCx1&J zfxo3{05z0TOSF*B_h!{job3X=<6Bcn-p`I*-~_&xXBD`Dct=#aF695M-Oap*_JTF3 z>ct%rTE)b14xl6a6dvrgQ0-u(8?P39q-K2ZQCJ8OsUb+{`v*UK+t~N~i3#$RmRj$1( zBur<}H6;e?V0H=j5sk##c-zDqz?qIsa3kyvbq0LFwFMXPU!{TQH~s^11ei~jpoH;X zkSom8r$Aq%xk4O}lwg8yYOTNx{+Oh5g}Y(l0#(QR09_~^1(tzZv^$~J(i^TN@yCbm80?ZVU+2Vzmb5eiBGU3I4z*S@-;L=8;50MLT7pZknwW7)1-?kF` zgj)|+*C*i@rT2~q?4oW7b%l$dXcx?hip}LBb)DI2|BaUMFxyoMH5%X;_@FwJnL#;( zN@R_crFdH!iWejD4T^uB^l~b z7vdo5qo$17j;>{%H+Ac}!hN&5G5Zw}|e{T+x3(wwAQD zucteLf7q>&p$akmJdblt=oj!8wiWw~_c5aji|7f!dic4E!d}tmLq(tgFDYKUiK-;U zn^Km)1)0v&6+WsuM_Xt}JQ}w#e=?kDx5G)EZ{8~kT{2nwwom|Q>Z|6sZJz49_dNa1 z1%*d)+qFHYZ{AF?3Dpp+WJ)|*ekxRl($SwIuQO%tyUcVq>FE(W$BqJM!`p@B3qWHvF2Z{UQr46pH^@b8k*_Px*WhBcM9{^T}w5CzUDd3 z^#nI+|3_O37jUZ#uS)(G;6IL+;s|d$@));VI>#}9 zmMlOA8K>x~i50FE)LDEG^@+*yOrUoQwZL?Qf$~)4E+bpXJW4PE*F7D@=brnD;jBBq zhIt#a1)6}wdkJ7GJzU#YRC6629rAWkKd^uJx^eB$2t>zxkL-fJ);EKay={fNR6aT( z|9^bP^+kFoctmFxraDImZ`tmuBxE3zt(ccUPQ@meCgTIJe((+l%4N~Vs5W34^d_MV ze;4sOmBl7k0h*~Nh}wSIpo-m#*y%eFK`Mr8s&MFIpgn9qbc4d}*a|#QjE8XM)E&Df!X`hD=fjbud7 zoy+Y&dC;Db(S;ABZ=N~qdBy456fC0p?nni$@(-{Jo{88DB#Y`FbxE@a>5Fs*nyKn* z+=`p@he&!aXm;hj%n`7&Tpe_6!eL|uH`sBL6TPOGu_{Ed#<;Z(t^vKClBsk}E6pfj zk$XP95}lymh~WHJKrv<%OyGwI$Fv(~;`CRZbvr4fA6`;7n1sv(x zYhMP+>LPnMUR$Ha>M$SFPhIVy!Pq{-FSw81jqSuGWSSR262ZL$OFO}3&W6-vdmHx& z@ij=Bj--#T?`qh~JuIw+iuB?5B))^^ z8r>hg6l&Dh)U>B6T#XTu7)Gq*ZCq*G8~w8InJ5UY;F|NPLR`)~Wtw%Rp-N`p3_n=!)+ufi_&1{_^vH4QoUWDoGu~NR9oCn8 zgjxU@Cf?J^1(F*Si z>eimS`d~z{SO$*(PBXRSu4H@cLey;Tm1l+*Qvia=A0~nMic(Z`W3A3v*IDPxmR9 zj{erx;39ROi6#KSKc#M*mE9YP9!)59YN=w{eSdlR;8Nge5uS0}7PUgP1&J>MEA zQE@uGVEc+Enw&8_L>K5y^gzMLy@W=4TG4-ynmD5|8AR}cel5El4N=S?U$Nng9qHz* zt;`KhIr@7IMCZs^!d?6uMN<7Vr;)R6iUYCP>MKwib%E4`3#V`}F5+2CG`E#IO+|C% z;Ao;sdcps#O<>-O{h;3J6VkQsRC~l0JVmZ1B7|yK+FHtq6PxN={FR&Z< zK{rPm0u5#l5q=Sv3&ZLItzUJQNb$~x=m&B|Bj>hcuF6X;SPTmR^sfLN4gxY|;;I5Pg zKG4TwRray;A2dcr0vcNZ-W!WU-Zk4nT_q%@HGdBGiuL5Fj2h{j(+F*%m~#4ZpWU^o zXsDTMBsjo8ARjpo+?WfO$Kl(#8thA804xhD!Jlv|U~^NMp5`IJcT{iWb~sC|b=5WK zy&btNGzD+x3gCZ%>ihzrJEb_a+=bjntQBkm%ZM=kJAF9&G?)b~W@-be+Q+~S)mnvk zG=#Z{+I8DvdxCA4g~%5eqz*fmaa)if_{AuXslPh)BX`b%M&6gmRhLZS#v1tNeVH2~KE_qazC zJpj(TBIph!SH0`F%=qdNEtsdMOP|HHcHNHA*xD**tOJNk8wahDcsPn!rWiZ2wKIrq z@LwP&4RzFXPJ?=aF?3t^ZZ5vXRn-Z=Z`aDXnx^g!Y@W1&^`S>%=6UOIi{O6HI9<5+ zt-*yXqtodEXR4Jm4#5tg9b`MyqaHH-n_{*?Z{)Sa4 z`=;?Yg)z#~{Bh=Vc0ebqeA-_kA0O^*V)6;4Hqz*(rk z5UwNGLz)*R3I8ZA-T#Wd8Pc#pzCqdyfD|s-_oC(4Euu4gLw5#zq>aChRet3(5b|FG6s{dnQ{GGnawk_S>#bFKOW@|Y8E2f0S$3dkP{Y@ z$%XfUdFVRNHsmKe2|vs}#2ieX@K`*i(0RQ?0s4ep8$T4kPi(j4LEGu)d<9&aCUq9A zN*IG8zz3y2{19wvfTdRwAf4W!g>iQH-LNUB^q%em}sIu+;)<>$Re9>*j$@+g+V9DX-+K)cu5BMwAc+1@~7 zfEa8P5QToBMk7}eu9j>p{tF)qb_L!GH-(dEE0tN97nE>Ift}oc&{!(Qc-c_{$#Qj~ z8Gem49$4miE5^BOidnlW9D(lUHPlb04^mDO>J3~R8peG^*3;u9(4A$-avTr`MQZVM z{x7i^B@oTnFP-)M$N&p^0DNL&Nw}!6t-WYmQnGL#x8%ak}L|y#7Op z1(u)}SWVz&A)(|E3y?KxOxZ5^-2~7UcuCvQsh*7tgn9JC9hYL$D7c7ZpWEWSuRsCX zXYD&+zQo5+gqdvZoT8Gsu%M(=DOy7(aEA5_d)j$V6Ap_%zBm4;{()aq8+EZ zQ6H3`OJB4qca+XaCDSUp1{g;_13u+Whd zlY_7#CLH!qoGzZffo$`9b%e0bY@yMAz}4D4o;l1#LpLe{djz)jrt&J=8}G%qYn)yE zfUD1L3D2{tdVT3Uv_qb$6hdd7!S>o~=h zi+;wRq7Ci;<99Jy;1p8N7Hf!1n5n(R-9dhFN7z5by!(IgK2fS; zMj)p(=ZQSFwX?TiFY0X{XgkN1qjfy-UNQ7CuXk6_hYaET#;pUn%WNH>pLYQ;o)}C$ zNT^}!tZU%x=(GnN z-i7{>YfuH6E#P=?JDc-_nDMrvT^ZDI?>giHa;~IA-z$HWVo9$SI>Iy7 zmXqn*b7jDXwne%XNj4*@ z`U!dT=mfZMX~An3s`#$wu&=$<6{6mG_^bCkR2y?4Yv2I#m37$;(kb*d8n&yD$<$P~ z*b!4ONApyCpWUjpf{!|-POwU>fQu+Pjq-5($oen2k+)v8VGbHYQrp>JSF zLK5~}G z;;D|_cy@dv{8NIFMU_eCuh<{De7Y>|sAhvW)sw=;HS;7);;fp^G(s+gU%R>rqp=Ln zL6?uKON>OC6s=>1=?3C+46op;?oQf&<9wlm=ra9CB~>*_+5rz|o@j4*R>9M~r@8M; zg}PW+h}Mg0PPPIzIs0Kxps(yPZ6l)zTwUccGyr3?j?UMZ)sW9a7jX5P4?rh+c5_+Yn(AJdR@{ILa!iTB z)t@|FKtDYP5tTk>Tkv|qLVZ`NtBWHBNLfOEU^zQP{?~QU_mh6brMY52HSmucaJ>v2 z0{_r_&}fA1LWrPpAJIS|b}cLk6dT9zsTW-123LaGE43rcLi zq2JI0a4kRIg-}**46qG+Nn?f<+&x_#tOmFUxU7!Ro$|Ctm$+lOvfMe2aX>GAIamn1 zLi-8FO1`uxW~1C9K%;szx%jE}DiZ+**yc)dS?tjnDYpfj0DW|J1Zx_udG_dLxVq}H z=+EA0uTf7SpT$mKG5W>R13aO-rkh6hGbBMb0Hgk(_N8)?KFTiw9(Yw!NJ*hbNxC4J z+FQT?Ph0qal2vSknzdPu&Qx{bfcFB5(A)Vz#0vK`?3Nvk8wea_2kQyAFH;~ig?c!z zcM z1Of}(g6^O`2S^Bc$$xLTNI56%DOhhv#8$y8xn7zDZnad6ug<@5-M016Zi$`B_O!Fi z(!5!EWAmlZMdyC>U~wE%ZZ~^tVa>vL*cLd2`+*Eo&a&6;xhgW*0!fZ zc@C4I`|^A-2WL7B5`%z_mm-jDHPXEnEfGWH%HK zi)j{NggQHB#WQFzlE}_=rHMDtM9tXvUG!o;#a&6^+)QpRI#IaDji5K>khzq9K zP<=aODE&hG0D6F7ZmdNXe%=uVZsgOIStM2159OG2^dNLpsKL*mo$NEr3_LaVV3(8I z>50q-tPWSg0LXFD27l&e&~H5(;73YkrLOk4(#jEu3;J3@Zsa}YY$1i!Qor@Q+EG`3 zoo?=&P5`etd>q$OeCwFa?d8WYE#SkNQ&NGi9^ZgGj^=2G z!mTww@E2GMg`GPIwTX*p-R^rtA2~`F2_*YW>h3t2kR(oHk9rdV+oIDPY}hxzO-;Zt z?QSZWb7OxBJzN*f8J(r&y(T0y7kHt^z;6I1ZRHhcYJW1vSJFsZdW> z0gntmxM2jSq+-H66^7wO&!XR`78ia^(6aY!dEN(5f8|;K0C0%gj`;Y6?0FZ^Oh+|G z4P0leMM#)>4|FWfip=m9M26`)0+@D$Qx4O(rU4@`3o?po3G^_8K`S)Z_G|_6-6!Mj z>CwV6IuEIG>;*djHFX|kXE+fyK>fWz8s%M7xhoG?2(BpdQYJbldy;lM&_PI0Vc-a; zOTmNO)S_JdOwBR1&T*M}qU0b)*w(wMQT1HaRNsKv^m5g=1lg0$@)TCo7(a|g+NaRt zA}7+(x?j3B+!^%+c3R0ZB!X>9-_y2i)kpM^(luU=3>G#@My&ssa zR0g^(dPnTT*Sw&#-obJerDRMcqoDx?^d5UUBEu-79Kbg7NIeaG9$|VuExJKIH zo^RUnlFi%(7s@ro4rp_c|3N%=P4q*@km*~iiq7E2?M~{C*d#uVnj<#wTKId0h7w4B z^}P4?gVkWm+5?SNxS^{f{u}@8wbN^Lv%G4kHssQ8fP@d3H z0~=7ZuI~St$IlI0=*^C9ppB1pZ>$9vK zqyATU#Hz7UHOE`cSL?Rv(x%*+-5|?q-*d{E8U9b}#XFi{(MQ-s?^FPYoEIFE6?K7t!!DG z_4s+t+I09>>zP_Rtka%Pb=E>ZDs%B!pgnnKnUmuj~89lOgq;G&fNti*4nvr??1 z)~&Zr-Lctv@4&9~KBcXyZt=#-4by5L@OM~HHKT>5%C+lhdRfa8>FW*oRr}K?S$F(i zZOuM6$|^KDZf&fuvcCJY!@6qcChPQnJ6lC?p|yPIBx|d`UFnxyE3EBj)vMZY$7dbA zu&?!Q&1dQ4z0c{5r$4lQyzZ{7d4GfT%c#>zeFA)_;1ovZkE6WF3<@(`q!nv^E7NRDI_5RP9)~xQY+$ST$Fct#5M%RZi;Kv#RbK z!kTkuVfx$`%dL&p=2RX|im!T8+O*1SS!%uH2v~n?-e!HCG_R_9{WL4s^R%_zC4+T< zaMSvu;#T_6#1`qss3Pl?lucG|8>_WRBG7utg~!(6rCnN2m@QgAt&F#x)6}Vae0Oja z{jS8?c6&W*t5&lr+svzNeKO%x`mT9v(plSx%A8Ji>%Uj)v}tt+vVK~##QO8_I_n_O zUo|o#Ieox?W$7#b9Z|J&yWRRH`*u}}#6H&hXAV?)Z~sV7d%dxeFmEX}mCfH*T5%9r zAU6zlub4nGV|P9lqu zyS@^39)3VRX#eVpF^=-js%pKSe2 zK?;u^= zdoxj=Rz1DL{_E+1)G_Hb{T`7nzr=4!GnGeHo@S1rAiCFgn!gHN(b9nvO85SQ{vA?X zY#==^N-O(dG>BDPasUKbbUOT(pDyRPiQMqSTxca%jSCH2Bc5@cqY4vDf(q&-4&t}T zFXjIHii9lova&5k{cQw{JghfrpOTw}YwG*%G0+`sjMxaZ ze=n4YM*@E{E@F?dn3za(Vz{8evWn81yTsCD6Z1ZWqP{Enm)E^}yE|0+A9=<1ff#LS zX4;8961pKB%!4GYa|xC%YYM*QZ}It%&Lj#?fU|rj1CLoGzUY5o9&!g)sQOVmJx6~ubyCfmfR;XUN zC}ZYzXqp+Ygq2-L=@>}nn+0p?ilqs>U0_R|B8|-v^h_v=I^tY}XSmnfKg)=36?*{S zy~o59;1e^zn=14G!r2j6Z~8D@Ml1@9LzAU4b8V#sdky{9#0T!dq;7Ap6`x5C6LXPo z#4u^SZ>2VuPbLBcZ=RA_KhlbY@q@sDqQRW$|70ZbcXE!5_opUar06d0D2(U>XlEN!giv zHFBfkgRuvH&>RX(5*8bsTtjGx6lU}rO!97NjtR%6(sdCMP#ZH% zHIf>a1ew%sG&P;p-^c@Nd zOmKe=J|uO%a#xyPsYeGMgEh&#fEL{XcF#U1P9yq<;{I5DJt7pCZh`!WXqKo?7PqEC?>xuwcbQbs%2FsK3G zGUf@k_yy7@)DSZd3FASaJ=l{71-r5Z+z@0;={m|m48%XlhuA#LJIH}eWTzyqMsEb5 zq`AJ~7Mtab<)Tqb_SoAnm?17nnGm=yEy9iTRJo10*8h{d18XXd5g(ahOmBTMN}227 zZ;MV!z0u31g86|0Et^VZ2c`(0rMi;4Yyw8QJaP?FdT@xCp}75K+v~cw8Mg+uF)Qpl zwU5o?M7QyZV+7eh*oAthPc{7`wNk&MGJ;RQT3lBjAdivT1=mpjm`?|fiW7b3m0sUW zb6;N(yaJzw-glpGvI41KnkqE%H?hrmm{S>Zm|x60)JN7Z&jNcI<_HCW3ktjX1u}zr zU~q}c)th|lghjqcyh$R1+zRfXmzp|C-Jo9NOUpcS(3OFXz;E&xBfCCwqruvxH-PE3 zkrcy+8)C8dzB|6l^mZbXXeL!)fAog}peKaLG`^G|c5uN7f{JeowdPw&-E2zwPfPm-U^?x(@adp`p|5yNV&q%IthGV!ts&snrXn9(BgD=z}YO>9^QSqBZ`ImuL zcq#~)XUQ!|ocqlj!d6Rn^l8OSX_Dw>406kbt>jp4tkeYWOL^eKhR?hUs7v(KcBJz0 z_2NlpvhkRyg5E{-3?3^y!UvF1=1{gJ-kwP?jgPv9b|6nfvwa@|d8pkOqHIJgTn1;w z*9F4RCj4sTk~|MtL&gi+yt4x{q@je)|JbA>3&~A#Bk3|4Mx05y>RYDtR(p_xg!;ad z{xbb;tfAOnf3u(^Uf+};XIgg2okWlPuT&3Fs4LREK+V7*|9)_X9+EC5UP1rHdnfXi zGZ-0iBcRY_<3g|;@v_uS@q?})&zj=P4^aD2A$UtVC|p+>5&M)G;~Dgl^P?!nhnd3( zyK#c~Jv&rRrB*5Qh>3C*vC=qRXm4pN;;JcJH?sl5q~3^^xq%L4vndl1$~K0NU_X#Q z-n;lgvXgXx7!$S`&mi-_G48oaqb?a|3BlY^WMrRizUh@5iqpz=?JfFmU^p3$CLSo} z2K&-lUSG3C4NltAX`ty`}v@b3jb$xBo}- zguSClsp2;GtUQ+NEq^I14X&tYs7++QB1_1>l<(e0PHCC@YG$1aoW<{(m3tdf3Ed#+hjj7uOsN~_Uc#QenX++WNf@e3GZ9O0`+9G9ko8Q3uY zbUcakn+{=@jDk3yOAQiaFFZn?Yg!Ghx1}RCRa?4gloM#i#f#-qbcwTQjT zqtUA;GN+SmLZkVp+be!P??DB#`+_pOl!DqL_YB`IM}zu^BY~lNBF+; zhqxis?y_Qx6At-ej2HZK%nOCV!YKbQbGu+iCfS%~DU=Pq8{!}u;a6Zj^Lk5&6f4a@ zHUT5-UC9?z|~=m?vvCdcALh*5EXJmU^s+2%9MrJCM!IyLgVhFd2>xM0(i>WU59o%$k zOzc-^1Kx;#Dr$7?p=Cn8hmC2zzgnv-a__!ai-Slmg#(|HT3CwN!3R2!;}~P&lceqy zodRDhn^RUwt$m2)yu`I$h$l+t16RwnGQR&9_lkZmAXqaaEj0n0+IvJam@8(&6O>Q= zQvj6jO5@-@u|e&6aWz$lY9+7qhp%9daVCOwj5ELBx9`cRSSC;O&PVsRdZ(F?7${pq zJoI}jX6-Mm=;wPx{0aKauZ4-_yAo>!ccjcK1p z3o9n=@2%wbE~N}lrph)2Zt!Eu4aTV8med!*8{^5=dE`S=8{`@OE$PgDIr#%pk<_em ze>rRXU&_tkK0YM)I5^tph0->2R`6e+9kVJ# zu@Op({RZAR$%Ch$*@5k>)l!Cy!#g13kQDs|d2iezCD#ZBe!JK0epoWn(B8h+w^~YY zFDlArACd>)4ZvA-UvVHmDz=Omj?KYh*bCZBHqJ;B2Ve_9OMRg2WJk|#YP0`8|6H=U z0C^9v61~f^MmmP}alXJ?p%MPk<`&WzpqjCkdptQKXe)11hVW_r4(6#r(xQ_02WaxC zc``TB{F7MW%(84vJZRb_+(&C9@`>aCwpTT0w1{@(nURjKgin-4s<@_)ZS>m zt7;yZ!GGhoP3zY$dVy2 zA1C|``8UBH!fw?~9!hOzsfkgRA;_uR& z#6D)l)(Dgnqk{_)l+P}(Qa2F$LWfa%nYr|C-B8Jb+(06Q(fZoxb<;5K9FrEBDm5e( zoH}1DERik=lku-)Ipl=n<>p0y@Hcvq7^-STzw=EtT8JcRo$o=yS-Q1gBy`g2Ks{rM z2*`gCtc;s`BQ51x#3(c=umGuTs%cc^9`id z;A4QKu_^MW_(y7w>VmtW>f6%lfrY&2*ce4zjHSnZh zC#Jc7M(TL~o6=%&U-|PS-tj~}=-JAl{+o!2?CM)-Dzo&Ho+ASTBEBQc4lEAtkl&aG zn;Y`lz#Mom78}%?y870Wvx2qBJI1XdNL*4nWRK)2{*l5G@sns|i-E&JE-@jH3icLe zdcy1!yxu!gcU34AUKFiFhw0a9x}byk9{wcwm}-u`l3NA-#aoDGrANzF6S!iNA_Fwt z$$Tdv!hBZBG;F1g1_}e+yQ7UpCc_5h6ENt=QKZj z>yq|EBgyiz3^~9&B69QxiaMAclFt(>;ali?-wmu#M2gN@Qjw-I#ZD*3@U^&a!7fq= z9thSo|H8KfJ4uz2l~RV7q&Fr9d>4FXxD&nD*+5t%+!Wn%O}r_U>I-oFuoX;CJjHzh zuj?vxbRiyyQ_QWv`nI)us{5RF0 z(}K07siMWw0V$=2J29LE-7m;RG}if1{w z&^QULAxS24AeHtSrgHxv@6Dr-8_-wzjW`VZ2;}R!BVy?;sBO`A>Zqy}Ih*q#gXnaa zX6n)Xscv+pr-@ti*5l?Ex0V0a9feYfLbikFXu?n6IT8mLoagukhWA*d;k*iymnbIL zm#WWF9iR{QJ0Z-Oz-6ON-5|fgu!vj8xx8VFmcpe;zy`$T-3Zj;K0}ug1Cqeoa~D#z z2*P}cE%O|qWBjhr|HSVUP1KGlM<#l0ycb>oW71Xc8qv(^gemZ2q$M)XsFq{M-ui2jD3mKWJB{?V`bt=#f0!qIGk_+V@;jtX@M~T zo*b>rowk%#!`oW!+lOfG3dQJGXs*9b^1TXK%22E{sPTxLB4^PL*l~f=SYy7#(L_mW zpFtnM0NYr%&{#)a0Ym_I<)u)h)JJ|xH1LW=^%Wm+IHmDq$xXR*gGy*>jI-1u)&+-| z8k;D>4vr*DU^`)oX|A!k*f8M3-cmylGsb3X<@@A)!YdC#ai%VoAxu;brM&5M$sxHF z@|E}=dOI*XF&cj)W|N(C`4(UHRw~+j8h-;t`T7Xgur#%Uk1$&Nw+(xO4Rv?W@6xQm zAojWC9$duo%;Ug!>6?IX4Bs6*MZF=ro63FJ%uS?BciDU&{)_(uc9ufq0_J~#CB$~+ zBvgP7;CzW1+$^ez&qxeEHP93}ne-Bhmv@n!$rYv;i^d1Lzlv`azWPn@lhNn7$XNrs zebYRXWv}p_8Ui-~9p-s>Rw-gXA3GSS#JeOlH4cS6{0XpT35xfG&k5FK*z~kKP8A?^ z$!A5#JSotHYwo)sW-&}Mq^p+l(-I2|FN-vt3FK4Vp$SPteQGv~m>~~JI_(_=wkOAv zxq`<(8tRv_37I128hw~wY)#ZJbHGE0r$8UeQpL3MUNIG#gm=sre43_Md?Po848Bcd zl=O#n(tEw7#uVa>oDaVzUT`VKrAUTRVRsc~NPiIlXuqL@Wt!qXn&BJoE+OX2p!^=x zkZpqXEKiAp!f0O*=p_VNJd27}8sp!{5$G(xwd|L!$mm7~bIAfqz>v47Kz!#r!=u3r zt-%ZjE2Vw5z35vS6pHy_@-%3bD>QL|&t;rXyg}Z8*QE@>hn~+HMl>rQkf=d7LPqqI zwAC~ld!=dsM0yS*7~Tg)&|2&cXdd{}gXptNL*38CC77R7lY>lu=)K%8^-TPbd>03h zmfT5Xzx#^2CZIA*K$o!3>9)`i^l#M9+Q386y^zDgf3CJdcebJ6w%fg#+?|4}%IR%T z;d3_N{tJr-D)=&VgYmlVD;F2=n0Kk@Kv(fNDI20fqQP$pGwTp`7MtezE|tJ)zPm}r zTQLWqjp#xlfc~W|^*j?yShhA+O6C72A$VUdf_x7&fHK4s-*mJLI4K!?j|Cml0e%Vr z_#1IE*O7Tl&L=g{7N3?LVGhD`rGMlb^nU@Z)ZYEjHriCc^f%m>t}CtPp8O}48&jC{iw-e0AkH#%__;+}$r%2+(lVKerIIkURGvsSI9}ej9)Lo@kzu(}<;9m|y88V~vRGhPp&;IfK-b>xl7k6yap2@r~u$ z&?BlZnT>T~dqB5^D)@vr$sECaw}{5jgu}+C#$#eja~{4xSSjb}_IkhDhRUMlH+(7j z1vG_!#*ZMXBbR)`g`dnj=^FCVM1#HYt=JPnr&a}a69+uC93ALQeizW$eg!^7ec<{9 zbjGGqQ;g)th#$#8+JuC&@%5Fx{b9#yG*3T(-+?M^3GNKv)7>kDx3*3_Aru1cR+9*G zT6Ob-(u+nveljgJ;7gvdHza9IQaeA+?Awp`(xiZ_VXlwQWNjgZ1%DDLRoG&Dj&05%K&8DiNgwBXdn| zLMUAg`Gp7Zj}x}R&X{V(-CTF%GB!(p5>3-C1X88(u|=%T_(#OWXGA@Jn8#fD(wJcp zILP1Dn1PHpR^YBq@x|E5p1KQ#+@yTc4Xhw!*)kgTYxRP05YDwX6 zQz&0pp0vrw$%D%t_#*$0qO*)|B755~PI0HyEi##m&qV6M-50ka#TSRgb#Zqn)ig4b zjAtVC;_fWEz{0{Gw*?kizcj!0 zz0;Eh#JF{9;2GKj+)V!`>qA#RoRUcEL%J7>iLd0*nxXz(axLpeEY`FFN)P07W3aFI z2l-4ePOdQjMc;$MbFK?x#h%fxAf4-(YCT%SZ{lBKe|iV8yYREYR;FU8x9r0T$PwH< zFw94bAK{_0iL;o#`d%Y%k-0>5Tw`&x7(lkb`JVjQaiBJcuCGM zZz81Vzu6aJy7_tIHBwi6m;Q-#Rtefqfj7+qlw4^Q`#?Hn9Uf3SPwHMH%$JRIFBxn) z2P_jusq~?3@&_`Bo}t_}%(8S=hWQ^^<{EF6LZ(X7J@YN!31zzVf$IJgau)bAAVJX%U93ODzaGst^*UwOQF9u)7pN6swz7uy|xu#Zv&6eqRCm#`DSLD&)L0MG<@ zV>ph~;LiE&(ogxS(m>l4zlRs_2nLWHD4lejtmKd546-(91b+jnZ{(B$r6V*VxC-8; z5dJ8+xpAv8Oac8neQU{Af+nP4en-9|ermUUTm2bOZ*QI(Ryz-|hG^&)M+rF3KgiP@ z+pUcANYW|e06tS6>3u+cMz^{zQh#F8((wcyjxQF!`T8fE^W8=3@bQo$Ws^J5 z?P@cnn~(`F=BEHl*cF~KclZ2No?^MasZF4^!{9p$jK!{&w3PaQ0r6M3IX2bb-?Egr z$Gt50`tO%0XRJB@8f zon|`nA#DMPvfW^Tr;Do$pXi3zFyn9hoBWgFRvC=%3S8!|v9c#sd&2#_y%Acy+9q&W?5U6?lR9X0VzvNGeXxy#?lRX$<<8BL)Xfi`B~;1Uo=%I z{)p=##3*T&%Vcdcg?aRal z5^IPCT!~UkdCITGS0pYIGi(0NMVR};FZH*SJmsRlyHuZu;CkpfVNKCDjw9OV{$DnI zhB!sf4O1 z*RZ`lZk=mK(J7!f&^S1x?4|a)wUzjv>Rhj|q+6_hU+LYD&O9i^US5)%P%tBl=G5figvQLW6^u@9$sW-@gZH3PEk#LyU(%lEWcW}y%Kd^tPvCw+71^E{0 z&7oeMbel-GW-^IrzVR83Qn%rMVOeYgT_lchI<7Xg%6|}UWzJL%+!~BccNtPGEiGl7`N_!eLxsJJ;2p+b`9SgN( zKO1l18_4bkjG4$__QS*@!#!63`pZjjFiIrWXLZ6_em&gLfT}m)<=|pZ2d`HQDUv+j ze_LnA>n7=4ce$&W#nU{nPkg|}={(qJD18G;T0`r02N_BJ&<3-t#a2bf>6zq#w^-{?O3 z3*(XEZw@y-iqLDai>4;dO)gW0mR#ef1!my~O%awjcA&XQ`NLoUFHLI@{M}eTxR%(S z`8Yi(Wm526MLjA#xXU_7^HOOaI^@%+8|fYRdT5RC)!bUTg2bz{(P_{TYP~fY63sy$ zjW@AQbzGC{V?Ls;D5g+miuiyPDeo-#rFVc<@HWHINdpH?0oTNu6c<`PfY(%W0Iq=OG>smmU?WS=o^X64P2BzD%nOd@QXOb z{DS*#4WKT4gzpNz$n-EY)UtvcQ5d3EIsc^(20Gz{w2NrtYUp!P735zeBK5?M8URzv zpw?d7+X04KbCo)?-Mh>3uho=NkJ#+}jO@m`!`tMR z=t{JhbK#FXUntPGo*JeX$vMGFX_{7p|3^lmk?MXYQ~9Idp2@=2l5hJD5nHjT%16yk zX*$(M>f^UqH^=ng9|UJ{`VnMWmjzI&j$($FP+6vNL_O2hmoB5gY zJ4#~c3f3`sVtU`Iuc=V*QrSPMAG~crxH==#OA>HG@FUCh`{ICT(XdzOhrI_I^QXb4 z*hbBbxSrnGQY}{|or3wXhEb73XLL#Ntvnj(LW_YxRD!>KsK4Luzh2Z3Hi;ND84ghQ z1HTdxj(AIaa0;TPfkU3!RH-yYK2H^MedK?gU7+vAX)4M23VQ?iMlB{H z(cI`(L|1N@9L;XQq9IfKOt}pkhwUWaaOr}YHPO7IwqmH<%Q!|oxh}@m6R$zP^N{i! z{*F(z?8Kt|Y3wxYPhT&6Tk|XTB+((c@F6$>-NX}_WfF8^mE^1KKSQHfQA(oa9=ZlLU|+-^=9Zysrk3(t7^B|SzW8ScX?<wzbX$`P3X;VXKH=13a=%m^ZJyGikAw^lOYqmoobTo74rNs z%0BaF|1j+Z`Vb|+9&C}=F8L?6l%2|7!pzDp_&1fV^wWF?8!OtlSHvN5AvPB9a+Ta7 zB+Pl7PWA$>#duH`U#hO%`T&v``0-!b-7Bp_B88 zK2s&6RUR6`Eu!b5Z}BF~Lv9Cg(^KaA%Y2p8diyH73W|N3kPF8DS!C&!b}ReE6vK{y z-iTW6q3Iv`kFx_Wp41qtD(oZ12&vI~ToY-P z3s2u_mID6bnW`TFHgZj8rzyX1i^XuLDxxbJjjd2xNwuL3@L!q={tVYN84n&M1HS*I z9CdY6YM_ts=h78v1bLUNvQ(;fl9v7t=?xIZyO;V}?lSiAYY||Ig$nWH-2zsV;R>nvIPO1k`kC z4R}*91gW=4HXt+&{pahXI>|fu7AT+Lss0FYf`6-F2t8dcgjXqp@xxpj0%MA?XLu+2 zw6Oro@_iR33Hz1yNZ*33*amE~FUWlI6p(F8UAQ*Azqzuaweg!37P8qmOQFGv93^VU ze+mf}jq3LOQ`v3_%6&yqy5;|eyg-j;hngCszA;}(`Q7iN067tQZ5m9>S5NT_|8i)e zxgYihGg1_JJhxnWF7+UM&_QksR7odswJbB-Td+R_m*Vl?P(H6d1|4!w6zV{m-Ji%X zd5LoZ(h*z)O=gWDm;=ihF^ym2M`O}lf83QP5+N2D(l$sh~y%eg{FSo#>N}ifdxK3a~Na5}fS}2d4 zle@txa&3SX#w-n>)RV{H6X7G`XayyTJ(O#R$LGxw$}sf6P1Z{N$Ik{PfpWsR!ih1* zpl+a(TFM#X|JGX-BU=VvVFCCSveWcHzAKC5HgQhiAKB}F7Pv?|0)^xfp#$;;eJuY4 z=7pha`Ms65TJHTi97f zy)lMst*?c3jq6|B0!h&S=I{VTKuppJkdYhX1C&0VFCp1~8y|(@(p*C?@egn)J_q{8 zH{DbN>m=27I@lWOo3uc@jNP}x{%-yVYOS3ic0yA$v#bfos*r55lnhrA)Dn(41S2<1 zxz--Z(Is771!N+7)2=0v4x!h~8Z=t_m+x0+mParBj~akzu@B4usjaY0e9L*vU0SAMSKQsm zTKrW2z}{g$gf)gnQZm7rZsgbv1bUi2gKZL%nP>PHbflOW-5z-k5t?UEDPfoO;6y$- z|AVmGKTQI;@ZDGB?pS*z1G|ew8Wy?^QPpA%_zmBXzRCM8=K$x`dgDymC`#*f0HYDB-VqPQO?LHRkI|HoCDvnK49sf z=>LT{QF@}}d+G}OiUg;*vW@!g8;hWFBR ziTRe*?O(=zRGJAzs`2^e!tfgVlY-%As7kJ_wj7A0`^l6k1&X2Yw zb~bbYCM8ZH6Zth{E#DE9O41l}#)lugX58sSViysIF=?Awx_yPQ2ZJ2us zi|Euc^T_y>nMJD}XB^J4RorUd!PY~bVawiqJR=2&Dtq-ua(bK7aTS-+H`@C4*i|hK z>`~>-+gLU2`QECYBc;l1{k68YQ^!`_)%LVKq7PJ`8kKAF{XAPa_IjZD+Mp)Y?Q4I} z9C>O`)rx&`rRzv;MT$J#HhHnv*7sD+>OaP}txl}4S2mVMWmNq($~FicUezLEjO{Ah zuk!Wf*|q~05;I!Bld6X%4YyTCkFhPWq*rC~NXEXrPSvvKL;9Yp4cedEoL_l-lgBni z7;IC&UaHowY+OAtj;X*$cC)QLtgSrv-*Q{ala9=YZZ|Wp&#bEYa3H*@;>+iXZAQG> zT&=O~{CXv`cMaPap)sH?=wu|L&s|xEHGpiZ}GK(6AR}Jj-Fmv;{y45STzs&5j0@^;h`;+Oh;98=ioD!lTB=OCrr((d2>sZW7XBF zUE62crj7oPdA10w*m}xb6@R(N)}ldSW!7PbZGdTKW%JC28D~d4s=Br%-?nQ}Q``Lq zpQ;}9`%&G4SWqpE&#fvwmz4QI>|H$(xKMSdrJ?FwdU18-)+V-Y^0%tBPhxEaKr7pY z1N*CO54u(p$)@VwWtOU5^^$sH_fEOO9k^1b#h{ znZLTKa#!h(%w@-G+I*TDiQkKdp(%t8yIQnO^IPE^e;bF>*EX%D|6XV_#{|NIJmmu- z)qLDUWj^(LtWK_l^*wac*)_l1y-nO!a9XgSIn!C{%2mCjW{EzQOz{Wj1mL z8k8{UWuRpk*IgOLv^Jhl4*JqC$r=ftx0Hfa=55KP zlO$*|-RrP2O}zJ|P8lYGa4T zmiC3-dOvD7ia|7fY6MBydit2u)K?2c!LwaM`TanO7I?g>H$C*aqzd|K=HdO3;C(KG6grRtIVy^GLA5b`TTRPfv zF*vj2hyMcbnS81l#Z9ix;7)sQ5Q2M?=Ou{HIrfnVnrz`qcS6)>%TKDm6%Q@<|0a!- z=4&qbqbxQ_<~OExXQ%k@;hAbGzXZRpO*eG2w(+@(T6ylv``n%VUqeDtMhSoTabOp{ zEtHGi)kce3Igq!DRa_L_ooqt93W9;h!TM~?uUsYnSf9;C zdEPKxB4ZpqO}n8z)DxzOXFPj_fuK56GKLvxKchr3|j%q2iJqfrU!Aa!|ez zTmrLrr(k7Tj;}-TocPE?fVHIM@FV{ZqE%p$Ia29LZc1B}4rNpaMwX|MLzAo0dRaTs zjj>qww6uF6bLa>DU*sI$An#6J888s3HXlP?Vfr)yZb~&FXrdiv5d{lxrr6cv4?4D5S@9l!DQ_FwD8xdq^DEmKi7xPCd!W$=jG2|^zKw^d=bc4?wYhYZMgqO;>Xy_2Gsh=dfoI0dh8pFX<6;T4Gt4%6-uKQ znHMJiR$eGHhUz=x&62ho<*tlN`8PR<*h&$Brj{F;QJ(bF4OG_77p7O{B;t+M0d_EU zg&vmgZ?l-~i1kXeW$Xx{msF5!R^A zs~sg}KE3~0>QYjlVhZjF+Wm~uIoM12o4S#_(Gte5Pktv&#dMZY*p>rtO(}_|L-&18 zO0NZ*ht`yN?H2_jep2WK_LV03Yq4`fm{8*H?+;pMiwl$IL^Z@}A*qlKn+=S#ZlQG6 z6Q$)!{nB%6BzB5!o#8Y8%qUNRJ!jJo;Nwb~n~LDubXV%GQA}OqlaZEq7~Dqu>PzJt zn=hG+)}2Umx&m(9aY}M&r=CaB>Et7SR`x94SoXf8G_7w%R%PSN_Z1bw>9R5DBe3;> z`np)SC61f#;>{>Ft(8Dw7cydkr`VOLgny;~W=a=m84<1Bo1qEZPy42xx{VDG)lm9Q zEHh)N>&fSaWb%8$d*@NlUGT1Ja=BOTC4LG_V||uVf-<*}M?zBz-ZOt-o1j1ZyRi1s z$I^SLBd7_!T2}=L^}05!5{@&a_*sC0b1DOOB?ju@-@dWsH)`?p07Z0T-&hJIAxAO*p{=;?S5q6^#Cw;;Kjd%Srx{=4!Pze*e| z)hFN27^ZeTCzJNjSY>c&q5KkSBJVbbna-I0R$j1eO&dgiS+_JTJ)0Qr+w5+qFN%1h zvdCAtk7951-_b_g3LPirr?(A7WZ3wocmrj!G((v}Oj4WBR(y{uN)Yw4;4u0yw9(Wj zb!U0Jvl@z0c%j;r@4Z6SjHpXJiLDj$%jQPOBZB&}m$n9#!$N9y4y%=V$fuHZwEfAu zO8vBK>Nl!O@|eIHIS=WP+570k%r{CO@uGEvLNbr(|4gk-8>z)YqA*2CX1j4y9buN5 z)_{4VlA`RF^z05e2Ff(vN}huacf7*pX$KW9VQ1R29C`K$Gy#=R46xf<+qsy~TCazu zn-X9zvK0A3JU36rN~Ay2?kVH34W@>~5^68&5?`b~(lsq@#hO)bPMNU~Fh4J}cCS z>XrtkMW?-1zG^=ZF=gY-f11p}TxbY&#PE;%JnnQiQkMf$?R&Ae1tmfSJHz+UcSu?- z&f}u_JHiX#haLfz#T$`nCfu+{cMJO^logy3cxviJdC0j^5MOTi2N>pV=)It6pSII@ z6{^Ypf)ym)!P<&l(EZF5|4?%Y_?>d2sSs-2W;#>l$ZQ$Wp!^Iv3jWS+jCWcGE0@rT zs7fBx&1JF!J(y!&V=Bb$)4y=Vf&C@YJ#c@xr@M81Lb3B18kf+g%;0|>tbe3M>E6nT z9lm4^wmk<%1*TgJC0$L^O1`5y{y$H;bBDBEv(0F-e0T+ zZnyU|)C@}{?RYzHg!2yQ$^Vlbtr|#4U5;mjxY##OUjT4e9e58k#Bmmg6E~;K^9>-b5iEZPSl8Uq_svA7w)2%)r=zb~H{2f^;3iSh8%&|@VbP!*Kb9qTROFQsNEAuuk~l!8r} z=Be5kt=HbPXd%^4zsFuPc1qC&;g95+Vo$j9;dk(HaRp8~5o#dv70$vAC;QnCu2CYM zUsm3&th2)So)k`jt|~_Cg?!&Ml00bXD!o_cC>tO-(A)PRz9lhUajTP`?9^VqL_QWi zCnn2-15(Ny=&eO0rr^GjuUoS2wq=xEYW zeM|#@H_mBrf2Zj2CkCziQ+lT4;F5K2G6&aA-k<)_a1L*d>4`*ZcZ?z_OBT4n)gIe| zafC`T-*hmInG&8!)FoeA>CAS?$>a?~Q~V{*mi%CIQ*`N5;zj@+VG9tF&S0&j`Gj(Riq z9Hnh%*(`3aoJ)ylz9}b^7xI_rgYWSV=wtkpUqI@xTCy#4l@4PMC35T~xx$p~ z7{ylVQp-|>C81PH37Tv^Cf&eJiFL}Kl6#X`YLeQ}NmsUTV@w(>oa$o+ynov(`V!{B6{=Ozmh-MJM(BR--9Y{d8u_%8k|6t491$Se7*M zbNU~nYuWm@T2#4iqcL;b-F=yOm%3GsGnmb`YhrbcZrw8n{M**nBEBFqBj-_7u6T}| zBj#(a(=kFvsX0GacS!2)7)3}C?lv`qyF>@N2600+k`28CJRLuvZpGV(zsgsJ_<={ZFIP3Q1?bmfOIF-pkh+j$88=6{DU9Wlu@}ysQj7$Xhg^?{!(mZGfA4K~C$Kwy z5zmA^LO1n?5f&f7_wc^;9MOs%t1HcLO})+z7d9i?XvKJfIjYqHoB7>bDYp!H5F77} zh~Fx_hmX^%*gPba9;d{x(f+Q|*w`u!r(vOFzCIhot{+(BndcjZUBKu03fNJu8wvk; zS!j^{RIU?k&K%#`&G#!ZkIIm`;r)x2(bFM**GnYPaoF<&Yau>opNe1E|M0eQ7~2s# zBBR^^U^{pjON%5m<FweOC{|oaYy|4-F3OjT%Z;ZZSIa)O zpS>%;3^W-wqkYv3@?&8ex7V~>+F`h+wA2g`BauGXKp(5^=4=d4qq;(2^ml!wTx;hq zMe7p#O8;^b$+s;Uu$vqorTyl$jvr7T-$%5rvcot9_G=pJJ7HYnEPkKifAj=7)jNsR z26VgtY7OhepX3ZN4!W$dtKQlUSO6 zWcs}4o3KvYWV`{+(KN;{;msUQFb3JpRU1bO?Zqq3M#Kg_7di!=7I>J8i_kA_J(c}z znBiW?jwK=s5&AOJUh*Y+ycjy3+{65dtyay$^_8v(+1Pk#zBt_VN&IAgs>?>}a9!~v z+Do~?J@_%-alS2J1UhoBj6LPrY<}Tc;SXd3Pim&)5%#@gg(ev}f^S0*AYRv`V3V;G zb`5Yy`JBk#_W0B-OcS5iND29Hy|_AD3w8|n(}1gX)%AQloIyI}7eMI1X{HYL4kC~r z&_HsmYI^+zR1BpEi8_QjSaT}QI4k$6IM8zoi}fBNF5=e^;?R1-I-d)=4XuIqU`O$} z#wSjS*o`hUDV zoRWXkv6~JE;cSW~533>Q4Bv`f`mNj|c0o?0cuKkjwIZ7F{bNgX*NJ*yD%2Go#dfpb zVa&v8xG(cTJO+*9T5@`NA9)6>vQHsKjpErGH54Wf=zPl7oR0Ea!Ym*v&`2_U&!u5H$e?} zkYB-fh8E&q)m~ps`vzenTvpJIof$L2^8s3wpH8}pqhzbTO6Vu|@Z|Yi*e^l`k86$~ z_mEtmzk3B)pMT+v!>%MQ2V1kL_;Tt%{t9tqY_2?28;7*f=L3teZg7F_mHPf3n6Md% zR9hO$*g|eEznZ+MsX%KOix@L{#<@T=lTEO<;#_xguz%7T?iF^{8Dw|)ENlQy6Zfe8 z*nUVueOEElvzYCMlHfj!6dMVn4Q1jHs=vM$eH3d4&&!QPo^$^>`sTe>SpvoQ*W#Jt zFSrJK!|%Y_7_MU^IiLD9fq{|uC2W~vuI4zl4r<^CS1re_v`>(+uz~6)LiBP3@!Vn3 zxbePZU>N+GUO)l`PX)yB7x$iRD)!_C7A)}AH|-Y=VV5=TqHJ>=-BfrppHs9BS(Wes z8)-BeyBJr3L&$cHZ=BkPh87?jq%y}z?py9>?NqUj9>&^Zdkj4_MI4(~Y;d{jVWWgF z@CLgS(h29>O_dgIpT?)AeBz4FAWv8T{RsJ_GQ&@d*El7s_y|r_yIdgtDW*=;a9@3+ zPnyP+Ck1gQ^2%{YJmNX0?$%|WBtp7D&;W`fx?>aIulSFHjZE1@Q>G~sE?k0t!sl^F z*@PZacGK(Re#*PV32F@ef7~s88!N>>biURtcc!xAq2*);t=-THZHxOMA9hB3M;!Ai zK2#p*7x@uIw|w2Hg_PFx9iM`+a<%7*+HDKR>*1;n*LYuh6Ub2}i&Ki$pigL*@>UF? zt;lTYJGMYe0}4M!yIX`!jm3Z1v7YAyi~W!;X>S`kU~t?FxkbRj)^uGrRT6*k*Qpik zcrp>WLc@w5jYqE%RnQqVnd{(wki)Ts>?Cp{_yoEI_C^Qee~KEz^~BN8N2S#DH|3Sj zimPNB)j#RE>oo>*L#TRKCrb3#=~MnmK08|tH^F`yk_0WW$;4@TSZtVTG-T`9Z=xo& zfNY6gA@pcX&rj1RA|I;8&ZxHeo!kW~L3t_nK}#X8TnufO%Spd60v>DFZ%ic1)x=~= z(-6WeqOOj53g}D{VlKZ1f2r-rsP8$kfo2f<1q*2V`M(g7afRs{vQ7Ghs4C>pli+Xo zk=+BOM}}Hp_u7+_ERqfT0pxHC;mD*P&>rnXpc~}0H!)skuD~q)3Hk^Q)9oXA$M%66 zql{oH#@V@QFGoT~V($V z`o7MFwxRP34_Wm;SWmHYz+`THl4@af4E2T;n73NtG0Y<;*YGSjMmm$%k(p(@gLE{u zfpKjoxi>H#s-iy_=0&b|7shSV+B|1G)l8DU%#E`ETLN5+J;60~J%zG0+2kH$N3aj1 zMYQn}_}Kfgjn^}SeUNzB*vZpXY|Gc@2Qudp=Wu6SkLmZ|mxvUo3HCuho!Q9_U_Kkt z;0EkpNpImMc!^%Uo5Qm;%Z;7gB9q8og16`f!`U7dno-oO)oM>m-}0o((RFlRoV$^& z@jVfO{~CSIF`nrJH=tKI_G{WlJ%v2>`<}SMaY$3p%QrVJg=185=$>3f=&S<~SnVKj z`^Kw2t*6@Ka3|tBu18im8nJuhdBaU)5kE>b;`~{#G3pfj!qbRV4C7!udK4JTdYK&N zEo|f;Ltf@Ed<9ySGsQC+{Gja#o%JphFX1lKcPzv|;ltLEzc=QdpnWiT4nM+iOwI}Tsu@=}CN%3q0avg%}x@VQ9kteFi z37|}t;WOjHw>h-U$7w=^Zyg zb=6p)KH>!^hpJ`2ARbleic#V}329Py)r!$we55M|qh&WVO^n;CH_S@9>3)v&0|maJ z<0p2LonqYSjS_xvD-*AQy)e6K3e<3&>GK*gq77tMS=INULvSRt+GvF)7=*p+`I$^D z97<>rwGZ?117il|^~Bm2Pt}Hd1(gngd0%Uff(MIVBK3vsj17Bf=#I7E>ZmV`Kw(P3 zFI>JD38hD`=JRBL{sXWH-{7LCfpoUm6`Rj2WR9}sLObDCd>9&PdW$~k93`*y8a4ai z^KgHDrFgQatnj`Vz4JOS5WNX@z|JSgU>a6RkJo04Ga(9kScGAJ;0ROTY?j!<(3lJ9 zcBwo@AA4TEkMCZv)6f|kj(u{!R{5Rdq*uZ}&Q^x&3EAji`jVCxPPp{kT;Y4pJMp@q z1r*ZMr2j7JiKU}`*l*nMB*K3g_}e{2E@#d{TM92G>=Vzyv+V;lli7M|q9+HbWd8!% zv*YkP;C}t^gj8=T_7_{|J&m1aZz74zpUfn_lbA{Mwyy)b7DvW)!)haKl=k?1{-tLV z6dnKCP}X*^&>Q){4%a21Wvqv8f@bhN@L08jltIo#picPqwtEXv+z-CCVYgJl zJ%l*%KkO3N(moiQM|#wJ6R7ADPr0q&0<3}NxwwS+3N}Kj#aGUD`8^#jzPo6Jwqc(+ zCzvE1M2ne3y1qn0^{_qsJ@zv4B)_dgwaxp6i3yyLpR%)`u$dOn#qkf_MHpVB5sjcjv-MUb&9RzxDJCU(&kUh@- zDmRbbtC<~J22VuYyreqnG}0@s4Sm_QLW~6#2?InI)LLrjSW6y&XU6r0*ZOX>Ylxhc zrvPE_b-A0f2D?EX1+UAy2ZPc-41jKATcH)LT<{dpQ~2Jo2&!dV!$*mG#UBjfrFCnC z*Lw9Ab<+Cg-aOx>ga+b1uM=y(>!Er(eJvF82feHYr`~GDa;=h%dHZ-afwK#|@jK*p z;Fg0!fLGeXz%J=7GzYEWeC(~wPvAX7E>{z&E&VC>!$&pJP`GsNHEIsOC4?xJV;cU!mQ-f3SJZDHFegT-`d z1r0*Ee4N;y$d8T4J4f7%Eb?4qkI;R@Hq<4Wwl_j?(F`wl{%pQn=)?V1wBJb^7s51W zF+f_{kS)C9jYC z%Sft3>r34dUh-K&xY$%oO4>owm>w9en}L=JdtGx%&8O<+0$ zb~k#D>(2^+g`H+U2-V@W(sYA~t!8g|AF-F*i+v;2onS-Gi9ZqdVlm#&Og*nocaHz} z5bqkIpXdl0OzJyeDLxymfga>P61m3dNJ(^&w+I*c`gDr8OJ3`2;pk@X%R0Gk1$`pR zp%$2(8;k9YoM(sBSzHqPSid|Wj@^&V*{5S`ijTZ=nC9$Ncd=ncVja&sbSN!J-Pt{w ziMr>SzU)X(5(bMuf##l{;;Vhh1di|2jKFxbCXpN)hUT$T^gUf3U6?ckd=4!r9A}51 zL-Y@920VnxQ_XktqYoH6V-t}PS^;j?#)0%Go`=4Y(D-xq;oNEYnR*)ULWkHl)8FL5 zz+cYYalg_|@ECg}b~&$rhlQ(#HPBP+p!yjVs^0BShG=oSyvzPRr)5z+!x}?)T)1kr z9v}_inniZE4`n06e}{3dtMGyIa(|)g?f2ww2Fx(odz}sE{w_Z48YNFCq{PD*?CavX z>l!1i6+UTtcVfvv>(j0R{YQ;k4!@gOfW9o0H}vnYuB|2)n+Z7u3#BAE_C; z2rOcEOGAMKme=kByT#svbFpW{Tw)nu!&b(h0CsQ}oKMi>&~d{ryHf-|`?qk49Vyn0 zdP-;MKzJ@l>GyFVbO+W8TF;L0jw~?o*~oL3i0Dy0{e+uZCpo}7(aMV z=KdQ$Kn{oIAZxMLVn=bTx_drkWL=2ow#x<`Dd-isRBiu_Vz(J&`b#9LiP{fGWx0D*rDxP*aB-4&*}r(IwIWU%YW7bVvLX z>S7wMeux*;G;Ef!gzQGHA{5_fT_d##GDa`K^T~O-x$;?jJ_GreaLdGR+(EVb-#+Op zdPAAcMHKCV;-~>&DLhCm3<7Kd{|6ory)H&>P5q|q5L9n1wPA(qSg0=j(0*8YMinw^rI%4n*uD4w-CRBz655_rqS@c0KZ7-W@A)XQ8vGZC za%96toLQKrg_mhHSkI(Cbyrp9849(PwBT9oD)l78^+&+%sMp<)IS$vxcDh@k zE%vQ>zU{x=34P> zX1Jj~IK-n($a1F?gU)o=rt4f# zL#JaKvXx*6qJbZ6Ru})E$D&OTQc31ButN`jUk3A)Dx&vogJAXQwa3kaV7{v@I zegd8WD!~{yMi&A+=vdctp@7QqvHUPKA3G3RuKk{q;F!bDCLBDb3j;22Zr5B9;D_L) z^m1`9`doMlGu&%;3RH)+!CBOn{ipF&NDn+m0ax_E?%w0TAv`Sg*QNrV3`6+B6spj6v zO~@tHIouMOliP^fi#IlX^z^Vxht9>X$3rSB^9)6S?84r|`3PKezfoOB&)^Q`3!;66Q*Z0Ot~%#7QiN9gE+BwiC{3 zrvuO7GH0Zkp#MWSjHMd(GpmF(&Nc)IZWQ*>&hR>$U zIXft^Ym3UkfiaTjr?ZyvJHdHFVhk`1n#~P|%?KRdojA|zA<7J`IgtIFG%P+MYJ5UJ zs7Q4RzGgqrCl6@QFvoxH0`G3jnCMk)&pYyK=z?Glff#-VPO&`V1tPKU^dr6^ez1NW zGL(83Td4<<>R~##hH%H5Pt;IT4Szd$Ak~wtd7ZnBGf7+Z;m#-Y^4wm4`cesBhqU#? zz^fv9sScDVM#CvgU00m1wsV@-X8(-7H>P@8I8J*ArVG~?Zml0@T%}*-9S>iFYw9|; z0kC#b=Xemk4a7jxS%u}LPK-vkBV3QR0MYo}E(bf9HbbwN1!%Oi9&3wT#M_Vq;XZr? zyntT??&H&q!;EXh<3;U(Zki5;UT{0u4POy=35}pXAY81CB*5kL3r{@#8M_H|Cfw@8 zOUsXBS!fni!X&^iy}gYc;7qdlx?!de}CaZX}GMTd_G{hPE-;5bjLtiGEOBZ~?T@+m*0zafnO& z3fJ}f_-uTJRP4m$9yp#mlmDrSt{JVe4^r@%L=oH<`2lBxO$~2|FhZTTN>Rwf_#w;) zVihEkoz*PX6Cqji+%-vjkgykjf?VO+$XE7PCN(oy-Rn`eHVyiy{?t`?u(t`-Z<3&efALm_}w21i=BD7_(Gq7dc5A|E4i=E}_bD3iHuFh>z z&~8Fk7%|-C;)sso#ljiV9$ngwEPOlCf<{O${g@biaGRWjWg#`>Led)DR{qMqL*Izm z*iFL(hhKdWhQpmWXYnBH6440UAoVI5F6bQZd>|jm*GEc_0oZXvCO*v@BYSw8U4!=_ zVQej6#`=gP)7^+dxA7A7Z*e-hS+@)63p~s#mZOTDa#`XR;h{`B`oT-c)^Lmeqv$OB zqDtRBY=p}MySuv;J2BQk5P@NsI9)SCD1x;*Y2)0-}C+j z@L`5?p8I~j-|Iqe+q!B~Ei7CVdd56pIuko~ML`|lm#E#;j*j8nLI*rm?_rzvUL>jg|jUqPzzmhpaUe@U4aKXoBS|U4<(XE2xHi^{527 zwJ;&j33p0}@B}|Zt(7}rL-92AE4o?j$$u5CJcT@j;^AWAPmwV{m*=`~$i0nUVrRwv zsZmJ6)m6$|Dg)P968J4xi8&Q(2V#3Asjzup4z5CW+^{R7&>zkFCZ{AWHDA(0|Ys_fh`l zz7G2?`38ZLHT%RmXjW)P85Hu6i*O4h5uBi%v8NHIphNsod?$TJddR&*ZV0=vQsg36 zmy48^vN!2#WOrSs!Xo8Et2>^S9767<`!}q|_pQ@T+9_NlGSOQgMZW{fL1Wxev4OrR z9OwH&CVK|}-{=K+1E2-EfH@*;$;)I61;@nuv}Ejy-9~nz1Msn?aHtkqotYBx5qcJJ z4{dAkdHP~7a|AAiPoqh`8NA(>2`-OwS$25Nfm2*z{7HDLyQ6>KOycRb_f&QFN9&zDSC%mjO|r6B zyGt!c2FbmM6V}D@aA|%{vTEfDiNEFPT1URWU}0AlOa)tTyOE>d0DzIh(WOEm+!?q< zd_}AJLbMHJmKoz~kae)LC>VR2?uoTRs>+lr4&A}@C7*&5gud)F`WC)leQO#Ond`bs zzrjynUEq2if;fr2;Ks55x{$A;Wpco{NrhLCWyV-KQ%=RZOL63dHL8W(qr z{$IqqxKVObZPvX3ju*m=B+vqGD0k--=}|Pv^*6nv7keC<5%!0yS#6n}ST|xkf1NoC zhM*9ZLQG{}F&mKIKq)YX-N{YiqlqPk2x_%|Rm(2QTJJqF2{miC#J|ivOn)#>wYjz% zebRjxbUPjJK=PXqM~Y;*{8Zi`2YFmPiM<9MN;OeVCT!30+q#DfN8DkC^FVE-PTXtx zywwDB#5}@S*GzVp^bKjKTrp_mro8IneNP#&L0-wEXs@t8fp#>=ZNPTHNk%eyvap!F zW^`%vh&%=Y5OgpR9oxzM8VeB{%BkSjoHXFFRF7kX4b7@zd%$pZIB)amh1&)Fgc88A zYdSsILkU^vDJcwG4_esIasa=Ep64^^P*C*_Mo$1|FvYi!JL^f->5NN^P1v=3nL7`E z5glbZfRWe@LmVvwPff7z7c$Lsl-Wc(ou`CdNT&ClF?UyA97d9Wy7D8~hZWLokv?(S zCz3ss4(0y{DT&%<0;DbAOXCUgGrZ0HnLQQV&%58W69q4oP=EgWzNkXhF==)x7dhm%=s4Aj_%F&^7K}!NW)AW zx#~i=t4rZYW+u>5^GQ$e?%>jdBoDycW6GiR-fB!cdM1+vbQJ$*?5p|n=9(uJG~mY5 zKcIamLsQI20TFh<%}tLBf5~0=FyXOuN}Ixy2|v@YRbNxj`~U^oxiO(sQ&Xgj2)$(Xs{H0H%{^?p(f-|B7xQ4hvJA!?~aA2km6CJ7zMv zgw=Yrp7GwjQCGA*&I|q^*oTM!ME-LAC&Lu@qH~?`E`LHij80S$bcb#lo5&{@kUSI! zl@Ieybr(Dp$&RScZp?4(?jrW$2aBiZXPy*!TDVb`o!5n7z#60%^GWa{7!1p4{H$m~ zduEJhyW+HbloIe^8dK%1n8ki3`k|@nVc?ll1{#p*P+R_Q!CtryX$f67bq78Ii?tgs zkQgQ`#O4AA;D3M+;eOL&@MKNVs9Sc@}b1w@Xs57(T}RO^ww#Y+9J=sjWrKLTBcR+p3bbd8by2fKpa z4vwe(hg^lKq|~HRRPxURk(iq9@hx%Q)&)Bc2$?U1=eXdBm>Fo=mKg3 zcNG7_aGME@q9-}T;7^?^SNy5ibB%V{2k|ZA)8^%;b0AXZU^M zX47EM?n%VYq7CUbWFxE=S2NH^9>cVOkw9N|df|L^DKc8yEyN>pnj{F7=3}fzIsq&2 zL*#RL0{4rUCEe#(&uYyVd{h47dq~wmkBhP31!$MMo%bMIgFto9;Ko{Vq`sd6dMKda zi|Z;rH(oCIF180zEoEfan9hHQsp2tk8C{KO;;+IE2ya0&Q`bTVw6}U2VjK15@ov#l||I4)NT0QbF<)oAh-=)y;lqK}g?B-t>Z*Ap4&Nc^087k+`6 zfFXv%NV!N+D0?3Ip}#4-#l|r;sn^VM4{1H@vny@gTaYHg4E=F9O!F4@VzSBEOicb_ zpeGsyNb)7uQnWAo0!l$9i6!V1us?iCwVDW-7A@rK zAv{!%XyMB+4s>tl=Ubh86XcgqG7KbVWm@5_VVCg!$W&xK(N9Y89)jx{M)RxO{w7^a zKC!!4LQF&3yN;5>h;?F`D+^iyY{vhLD~CG?8R!nUjC_Mm(c8i!v>n(ha5qv-mr6Wi zmcip=+6c!{2`ltoH4VlZlA+4yUA=|0*e-l0urAWqJq{nvRwoyt&A|t9OSB;qyZ;WG z1QepDOm|{BvaQ(zcvp0Wu`BTcXaUSoDkv|Pou#F(jLUF5x`inm*g?qN7^Y1`eJF+a zLaw4xggWp?(;@h@_FkV5O+{sxu3C1nO=Y8R5ShU4hYqudI9|@87HDURuILVGkk|#E z8ImdYH?9EsYXozHbQ{UQ`>|*Il9)*T!-kqVDy>bcU1N>)w5C=Tp!hoj&3uSwuFytf zNPT2_JMXX+_?Uusa*;6_-c0Vo4gw#=8B7aHed>~>0r?NO$&(B@<#szhk?pZ@=3>wf zrur@lBS~%BPrl&1rhU*2d8kqy_6uQfJ&lull}PfmRG%XD{LSbqKtf$%L3AZn!}?w5 zM&A-By^W3YhLZ=l!9)YG4t&NJ&);L(@?pLzIW4TSmH%P$#2!c^Z8I<)G~h*9Fwsb~ z>~TcBL5$i{P6xDvKgkNdC6UFn@Oiuo$zml<^!rzk6MSv>ZLuq@ zXY>%&5AGTL0B<6!$(}Q?uXVd=28C33V_TZ{)enb1_ov9 z-#{`yFE~#S$QSTlG+QI|l~JNERPYluESn`tWAC;_9}z#{1T>aCZFq+FC#PFZV-wg6 zDO}8xig-Ton_VUyS4LZl=*LPw@(wpaBWq6QFUXS$cdCT^1lkb!!rTl$4wVT>$Omu= zv6>%+eee&H^GTIW1$lq0CD})SciCtj_9X=z>Lp_b^o-I5`63)bb(Y)mBk3G+n5qZO zid#gTFb0f^taI&7`6?mvG4kvDbKsxkdo44^x(E6%M?QvDs^L)-b%sx}v{BZ=4%CDX zvq*t(`3(^e@7sFqAGOC3Y#8?&y{_?ar;7ifGxLT)xh{!)&308T)Hv$A428Se`JWf8 zmx`g8SbsxzcV);CD5_wqcdwxtS0l#Z?ZE0Jw<#USMT*V8#Y(s<6OX)wQ)RcO5qFTu zGma3tiGw{iks|X~p~S@_Vq6Q)Fz0BE#QB5iA!NqJ$$j~BU_NwCIEd^QPWjeCc|a&q z%rD{g(k7-3R2?~O%HR%H5$US_(W=ZfgA>oJB;E?V0*wNMtu?sZxb( zVKZ80F>j22@TzcM>7rDp7scKnj?tFTDZEq5%_u@OYa>}ifP!UMJM(t8L$nf0*)Lp~ zuRFS&G9&MCfUHB*W*YE3JBuuq_wz#u+*vM`W2=mVG&fyuFcj|!+qGowTyDC19Z>?{ z^iQpsxGY<2VZyz923bS=EB~i^=G#M#mpNf6ki>oE z3*-gnNlGaGO*$!lAv%ir+6iwByFKm%8Y{mvZ&#n76^M#n)tFZhpYGI!&5#rDN-Wzm zQSu0V&=~$8KRPfE`A$V>=dfl%8{(q0MJ|>GQ+W}6V&cK3!fP~!xWKrGHO}SKLg#1K zS^kj$P;l-&JfA-wT~3@W>}_Z$y+eD82T^_AYNnf9o1kdXe4P>Tru+gdO9*8LL{3!( zd|OcQqR$jZ!-Js}_o5nCv<}(nsCsvmDbXJfrZ|^0aa>*1r0omqUyiSLk{n8C4F|ugb;W@v z-^)XOSFiX-IZ|>irH|TaRPwkSFDyXI@92!MPRxE<5PhiN0x5dN-}y6;ZYAl~32& zgdbb%Sh0FwY0kYRhqk*a93ATa+s5H?I54$w#jkR=Bc#v13Y~RQMZ0Df9Nw*S91CA+ zp3wQ79G81ID!gfD9iLa^q+DzzrnC$3IrjZpcc=@#z0F8Vvx*N@o|S)_)3ZEKd!oZ% z*s$%>d)X;(YrZW%+0mFXr$?-#K~!r;?F67=&6c(mRkC)bT)#2Y5%H!?1v7nO#gNzy z64llcAObI&oQcp-GR35;9zcTb-Z{w!!dekqN9KJS&rq& z2g@(TuXB6`b~z$X?x{GD`=z|~^Qn%mLbr;@bp?)?xoav)zuk6BKX%hG_srIc?(>E? zemS?ML{Ds8k(JQX5%xRVQRhwhA!2C0W3^Fr9AP3V?q^P}$lY?sk>%X%Sb7iDe%|zo z&qHZPlTZ1M&Yqr*yW0TAs}A4GtNx5~q&UCj=?Z#WhGW1i&e462*%9Gj9j}Z-D|VK5b98G^E9G7PP8HXi zt*_{Qq^@Jfw94|VlmU(jzh*jirAM`u+~p2W5pzgMa5-8v!78esyY5&N)4xLQky{@5 zu(YCr{;H})=lxOe#6X?}w+{B-qsH=xVk$_C|KOY`_2GZVAF%F;uMCC;PXQ0Do?wc# zsnS5(glD3If@_F6QVQKzGI%aZneNN5MP3+tAL%X6LakT@@B^Ku^f!IQld(&g(R_qd z%GnH6WSDU9&&^(>xA~B<2>+pPi#7lY15XU+<(kyLVjBn($K#EQclejYPbauQ4lZD&MOxOq?w)iyKC)2e%UGP*r@Fy9w4=e#RWi>y7MY_W*rKr?&bZ zk*CI?(5_Zacc!!&TY@%5p2odFzsvE$31FwRn*f1jW)t62c#N9ai-In4qoGjfXW6{J zq4FJ`?3!o!ju1>Zua`$#cFT5RDDbZ(pL)U;fTGPHDd0?TN8AJL-m4$Ant4X_R3=*8 z@-@NlRm4*NGB`?~FOO3!8qM^3!B(gl`L}S6e&tRUt5}kx*}maeC&I6sDojJ;GvoZB z_zdmts4v}QUr9kN?`9a+ZT2id>e8ElYe2wTOKY^XH-3wk{3mQ8}O9q1^UF7$-|C(FjS-PseBLd79z?EG>jl?h@N9 zZKxf@B6_tNpL>I-5om=U0$LCvW-zq>PlN7~5danEPJTcL|k1>*ShTKY4)iR=q7fP{+sI zP$5fyYN>A-K1)4^d{$cJ)*&`ZO|@xgH3zBN#YV6ycQgJ_`g;buFS2Rq%)kdElV>0? zBh&Dn&7gmw!wZtx?$|`WHaw1j3z7^DZ4dPpxk@ZD^}@30L&6R0H?|+1Ap3+ocD-kf z?^2dd^aWqaC+R3ur?3H0CT$J0E=AMeLTh{Si|-Hlx;RO&lf%@z+;j4@kV!0%6BtmN zIp-UA33GM8?ka1fD_|Msk&N6Y^d3Pl^|XxJ9nVK|4JqH!%Q8tGLF~hNTN4~AEeNDoZX$}W zkuWT%VteEE3!~|C>JaIp*Z3>0&pLYCx~4?O{l>*gavFW=}}Hv(@XTn%ub7TLnc-7;89< zPLK_Br#uTZ64H6+p&J6XV%OqS?oYX;ay2m4z7lgP7yPy0{Rey&4}QezkgwT4_=hTM z1HHv!*=;>we`Kw&x&0x^I{K#-6a_9wgh8{+zwlya0T=*P*JNk`x|W-PsWK={i2Gm) z0kgsOv7UYZk~N(5OkwmU!%cpXz6hv8K8_p3K9R?%>Eft}*|vYES|wdeu9J@olEF5? z)$%Fjs_!1uLpI3+gr4>-XqvZD)d8Q;byftgC|O*RqrGKP@l2VF*2^2k2U=&ghO!cG zV6FuZLT4$X-P^n$sSFDuMS@1w<4b7JySczu(Fk0Xy0*Y*d4!}h_ad^y{{S65QmRY4 z{h!dwd_O79a*>FnJ1WnpF5qkIz2`6bsZlkZibc^g#8hw~eHi#GaIj7}<_^)J%5Uoj z^fS_)e@}dpI-?_Wa}AXALVJQFKr_t`xH)90yEEE=5dt4_UkJO=oV|-TrsLKY7j0I{P^wbFk+s1>&m$`U_aeH0zlElG1i6OT zTf{+Qi-wYuu!i=aN9q7^}<&&p;VDHRa_dX=@!+hxP%zfDLXU zd&nB--w4YtL87#V5E>FLml1Ac9oa&?rSuAC{9)8IuH3x_Uyja?1Lap}A46AkjoPH} zv(OM7flQD;Ny!3F_3(w0yEHrT0_YZW9UmLWq(=(LN-DNh93jLA?Z5>z#z)|9R3Gs? zs)sxXUrfG-L)}B!mKqnglQ)(%x_ig9_N|gfTL)WRKmqy)YX*HFW&tGyN%JS&T|BL}pZ3MmD|1tY#?VD>oohaGjmJ$r!fwaXp54ea?>-h_ zZgU9=RXQdg-QNWVG%x6 z{_Ku~8KOd)T^e)OkSX4Hk~SNq#*qp9S-h+4Bx|_;=9_q%f%ljse6P73F}_nL-+T@a;ks6gy<;%C`ToX@!hc&qOe+#Q|nyzCa$hSt!6 zHO3;Wfp=^E=)j6d6W>l=CMDa`k-cyxx+gAa&rJMb*uX$1djKgFmgCJONts9-(cF0v z(l8K$ZP9nRGcH(dC!A7h_}2*^paN)_^g-L})Rt+avE`U~w3Z{P0w0F{_PL`l^W37k zW)Z50&>g8x~z59%}w@HfNh=R#&jk!5rR3^i$Vc zYL?umKKe7;oQbdBdQxdo>DM;Gfp|uLXpfCA?ib(1ene1_`GGWhUQ%;=?E@cdsOF={ zY~8XX!ph2T4mNI+ZM{Sx)H*doO$%;U&zH12*fQY(T$cDKVL;JcWt&`u+jhW~fLe|o zSbd-|c~Eo6pb1$ClS}QEKIDA$3g%lQ#~j zCG6pcc(4Fz8M_S756W+3sK4TW)^`q zZ4wy`wlS}4X<;i+SnIua25yGV&AL^XY|KU4>`e!Hp;P$0{1D){`J~5fxbMC0?!EhS zT&TVgu&JmuTjpPYr1|?nYngrOa{G1DlEg~j8+4tj6hf5+KsSFTRHoMRm8!R~7nW^B zn?$F*OYocXQ2c)34_{{1M0B{RsaVHAvOh<+rWAh*e)<)i0I}W<6 zwon+OA{PgnY3%kzP#Sq$=^%UrmsknwKl~tQLts7HE^!1qU!E9DMi6|BVQzeNL1dti z{^TxcDgV&h-yIW(=aVu!16Kdv)^l3_dazqZ)+U_dX}F6u1Zp6UC08JaG9ZKlz&78sjK zRR!K_^tBF@dkp+S<%lIhSpdG)JfqEYH;FW%3ci6VMDxuN#7E_!P>47JnZKEDFd zY_!C?+;x>5NF{6N&T6Z!Mcd@fGzPd_K2U9hApO)CpEHF{2=9aqt z$VII$-IthzMRB{O9Pkdc1=~%3LjTs7449$2a6(qm0CI!)KpfX6OWm!fC5Blp8~tm% zH@QB&}XYa zH#B@g&PUEimb+){8c&bpTA?q1Hhfp6xw~a-rL&M;>3L-S|K!*r;8flR-8-O(v7>pK zaR9)HZGnuUZIn})z;6t+6OzgGYOK{ud?)_1ej*OB1BEq2A+dR_4q#ZIQR#+>g z9p1BO9@(V%C9yLxvc+j>C%=f#A$~LAs?kUJrl7UhE<&ZXj)iB_t@WvCY$~x98LoAv zCxg%V%k)P2tFId7M=^Qdj?EOQPAae>tIh9B7mQ)j1>Z;S>sSi<$2(I5v~Nf=^E2R? z56@g}@EKWnSk{7&qsGp_oZK071NH^$fB+wX`~WYLPuX<-4F8b64E2Nk=oSUC4=v=0 zd@sumCRzzky_1X^?K6cg=yf$u>|+JAOyT>$CzSLD1xEcqR4TKJJD``5O~FDc&2nAo z&pj$`p`IeFg)8N&Y-^cL%(PVQt7B^vdqs2ZPA)oc9gk;Ft?`Hadew%1hnl-z0WXAKeT{^3Cm8mFAFUeyTyk>pj^C3Oj!FT0VQNLTG` z*~C2EdWWyBEna_LL$dZWY5OpxrL86uLkzQoBO~x( z6z(gRzhb-2iO^L-+}DU-3S${3FsC*eY(G5)utA_4M_usrUV=e+Ab z=_1h;e20KaP2OuA$%lGXxw)#55sRAYz6S_P2(Zt8SBQr`flWaEGpR`Fv>>$5&KB`rzltR z!442AHD?jZVG$ND?V z&sC#5#Qj?7MjY{b)sx^%>o|UZ{HN`yEs1#|NbY#Ds-+&LkgtVY0Md>6GP1P;bAt12M`WeMWP!zidWm|0 z^bzW7Zt2F>Pg;8C5xG?NOPURZ=?|(qxVcb0Zmq{xm`D8Fe^AVln{zqXWA7J3mR!NF zXM14H`Mp}lTjEa!9yv=*y}<5zKsd;+$2-{Uf!9ENxM7`OsR`4*m2@r6qkL9w zKrZHxWhpumi^fgC_StoLoKHn|@(EHcx)NB#wnge1OMt6jByR`Zz=Nv0UESHS;Ca`T z{kt0bqi+^W6?YpZF&|rWWEyx=0UJ6$_gSzH z{?gNz?}2nu8an~JpK4{FdD;YDdbbfvEs@X;Kgqm92MT|}WkSKeMT8YD6#9$vm2={o z=x5MTVU}qN-9Va*56iq6wH^E<_Kk9(AiLB#7F!J`N5x=tOh@u&U_vmQ^QpziX5Z() zF)gKCjekmXEWQcCLT^i(Kp%91pA0l75?cKNkihG}aH)y40TM#!HH$@kf4a6!=*uUn z0-+~6+CEFRfQtA?Be7C=L!Tgc(P0@~G>sU{|81RN9)^l0s^pVWA8jRE^XPb_cxu5D z{y87w?|@_>-E@z^=iFZei{faa>zJQ$M+m#+yLe3%u|;85iH{n~qqF=~d1ec@I)VLS zg_LO-4P2I!h^GjzHd6L9$s_us>m`X?=k}shVFB*I{Q;Vg z21Rva9>)=>!aNs`C8v4at z#xlY0K9(Koy$W=|j|0Dm#>COMspbaAFS)1dtvN~K!A&Hm__ks>(IbS*EKH1+Fpta= zWJBbXmR{{)_U-P(BnSXKGpZ^_Vsr7|1)aG8rfl>*Ez{FP2rQwKScBI7%A`AiLx2sg z;m%gRmU|9?DPj+~QV!<}rM1HPz;qJUd|76^HL)INNk4WEJEmJh$Y@=4J2i$}%RP~U zL~pqz(pZf5{{_qy&JopwLv**RFnN>sVtx~sf^Ev{B+Qn^zA^1o> z2kFb?=hPMA#0qWF)uE(w8aEiI%GCge({<5Rp7o}kTHe>p7v%lsQ#~vA!kF$tD`5-2 z9I2;yjPFKzou{1jl~2?~qMEIreBIw#4Y5w-Z)pzoP0%8^K4zr`26xGw=#A=X_6#|M zoFT2Frus0Tvv4_jzxZCw_9ozCu{=nYI`dZH5>QPe+0BvXFcJJf@r`&@f5Ri8>nwjt z*+dNE!9%6KuCLK&KpOIKqL%X=2sJ7^>blPrffZ0~Eo-_nw>bh~XOM%w{2kNKFfarh z!nZ&d*9##k8bIg_ZQG?8)7@{~cuNbM#Kv&Fj4gp2{)eYX`Xp*iGU0%HBeEmf3G1Po zfGg;CdZ9DhoRN3kM1umHOVo-PAwSf4-IhX8XvUYzFjvBcF@Fa#w8ZW3z-QzjvP0qM zE7qF2*W^AT$<){h`btc3q~d46YTDe`UfQT;6MSKMVNd@I8IY2hcSK9~bTyW*7Q1JE zLvvdh1zN`(heC~Yuu|bYf7Lul?r4l4O#EeYf3_PPL$%A5fh3_R-btKoPD9u2U(YNr z{D_f(R?0TKo!T-mInXaqQx5lEU{;f|d^Ue8rs)rhWIk1R&g!je3%cxFR(wcqt68#o z$i2Ajl(&ALKvEdGM+b$2oAFygkVvuiQa;*R%OYyP&Z)Bue~2%U+m`y$eflqAHqlGH zBXIJWz;|c?_etGq`e7~fwk0rREF_|zqu!yPJXKJfDveaMc8|}z&1iCz@rNK@gUHL=OBu;%lmw{p`$>tT%SKnpK%+qC5_%%9sOImCH|Cuqj#W*^eflc+?Gf^ z#NfH9d2GtO>roJSp>d3SQnI_H)LiI*u0&>m>-TBRxGBBZ6q`3+FGQi=5eJ}U3Tj>-+{RP1>4fu}Hg zm9{0XN)MO*)1Ddy-W$MV_)OF%VN=)x;4xx1_jGrLF*ut&SJ0gHg46l%{MN`!cn;To zSE@H&jE`-E-i&W=>6^G#v?pZqrQn93YPG0K5?b26p@Myht$Y4HLx%*5t308fs+RSQ zA00Sh@8VA~P4!oa|D6yQJY00g)6qK_(ZNrE{d6d-%jx0Oz_d(%c_G^_GDK_uOr{4T zUo$#tH}^GxXY!gbo9Ls~g0tu)^nBqj<{Mqbdlm>3_M0xa-$-1|ZF@TaVki18FgmD2H+q1r6= zm$-!qC$3*U`CKcHG{MOCpZfQTcn{Ph;OCG`O0|#=|h#S=6z_g+d@>cn=ue~?VvPrIjwZsj9 z3GrsFZ`4=4%)A%t@iT)bt!KEO!ja;Q%6aOP13+@zJgqT)yJ)Z09&Rf&^gx9%!LxyP@JaI?%s@UN3hDafJTjS{4U9MZ zYf2=FTQJ_nz@DhddWFxRY2y>5i5cFz+FcAhgpP}YOl4V5!}2||oLzSx)@6cKnBz!4 zdOq^Nw^5r1or(~cs3r@g=o}2=?f7-ji%kYD*pSJ zf$JfSfJ2peMbm*)l*e%Jh5w83u(tVd{E+BQi(7hb|88sxL545%K_y zEODQ{%0v8f$^f*7Lj1?j9|8$|G(7d(Gr@%;`KDMubQkl0|JyrIy00IL|4TmeOcCxC zW;p8#Z(Sv>U-SrcCGf%bFc#B6k>jwb^cb|Ub4~bLub9*@h>k?ZR<=Sy7CYVAB)@w)xn_Da>4*wv&J~kn$C@`O;$Qc35_7Cu0 z8WrzH|0}K}CI|iOus}2VE7cFb8@NpB~H*IH1u;N{RHkcVwT zOU8tmpxe1qOlh2hgs`o|4(}Kb!%p2E#Q!C#^A$`N`c>|0{Yd}wlKRxZ;v15t%Lx3C z4q?rMP03EOl?>276Rstsl+IuxpzQc3*i7rmlJ@>xVvd|y+=H-dyrxIV(e?(^;e@}m zLHHAUS`5MK7oW1FD=n?nwg1x*=JCW#3=jOFy;;B6^#mz}31fv^!D5fFmHLO1H>D=7 z!2aHrp$U^p)&w?TrHX~>q<-NiTC<5@5fUu69fmrW+*2A*Fa0x+RKmgbwvJWG)FS;# zqC&nF-x6#TA1k_=RhWvihXLE%KG0-5@9g9@nU6(vaGlC>Ba_|xB7Ykv`$jZuv8L+U>7hH)RMNw9jAxm#$R9%0Se-}j zZVy~BVt77}BR%1|fWk(a9u@T4KMFbsEN6xTjac3{kDY-%ENsnvg(G33JYA!XZ3j=d z`$_ZUKBgJIFUVl}DtegjX?exAVaoY=P!8TLW;*qiJLkjYYC;v{p0_So%nstM^v>vh zbTy!T{tVAkW3uNF+$i_=I#=9wc7>bPI(xT)6vQRnb`1bO8Ipi!#>&vA*fm?JJ*LD9 z{G=8ze*~#K**%dsY~6)SAa-aLjRcK=4p@Dp4fI;3XieI2KtJCoCBbI(x0KVf`eLz2 zr9~kx%b+ftOn{boG>8WB+VZdN4z=HFnl63pxO8r(C49;J@$>fbcZe&petv9cqYa%I_i;h;#5>gu{>k zJJGpZU3oBaGVVX_CS+hf=5AxED%F+x_(`<`co2V!+WmHQx$V7mx}_d97p_C3N(|+S zPW7p_Zc;a*s@TW8i>jkO5qeTu#)X?g)Y5X@`;9ZPu3$shMsx#jLH2^n?%M!F>*NXE zrMQfCgS!cokS=eAY9MoTbu_O*oOctUL{z8kDC$0ryh8R+FTn@_qo1%nOgZKZUoSF- ztm?0?r3f9~)4CQ2Nnhd`J4K<4DAR8+Kdbdw zZfD_^Bykb#HBBMxzygcFw$buEr^p66ocIq&cK3_*K@;UdbYE^ew}I&AGx{x7(fG(z z4rH)ypc*t-YA{j z^Go~z`@14@^Kj?3wC}ArE}QpEgu#hB+gI=7wC?I=&VA*&e|jx33fC$%EoY_~*!zk@mzou{>u3 z)=~$NX(b?C%-&?_%n0HYFbE+lWbj*n#0lpQY*gaMK8b)rBgdN5Xxku|RX7T$s&^E>KQtt66#g)#1ShZoxhA z!=&t8<4I;VF!$hGbg|GKU(d}Zr{lcP6VtIf@f`*h?2UED_eq_|C{pL`$6fR6;(y>* z*aYn-<1ukOy@R^bd<}Tceo5WeYD2e!$1x?w?WSv}23wW=gFGbm5<7dNag9)))SL8d z6<{E`3=Ls^)Ai$;QY_{!dWV(J8Tbowp1Yl^n^Yu^#%3T_=@M5Cz`Hu-^)Wi5>N)oq zE0|@O5$tkzLwGMR$@L*;lYWK0QZvgn!d~Gja&!@fo31m?7QUtALHIh_0zE0*MIsA7x?&YA5=z@i5g$X2hP1*< z;}r`v{kOe&{a*c-bn zcCvjZa7K(&Z{5)jPz}_&{T9>``VuC4!;M$9+oQi0MrFU&X3*1(NAkwU9*$^d80|r# z8bv_#T<{-PMf^v1hnUQqeFfDEe@AjrEfdChR`JK-UBV{pZ|!|{lduY1g9e2o80k7I z<6s<_WmHMDcu!{}Vj>!Xiae3o%=J~TH;K$zk5yLlZHC%H4j-bQp=Km=nbpu*&o-$K zf@W*kfyiMpnr_ZFK_*)Ii*{v#_qXRMzY%q_zr|AFy>6^&HM!0wF(%v%z8B|lOIfV& zMRpR|ADe4fZ9k|$LPNbHa<%XRTf<-DUb;B=Uc@Lh9c>NNC~gtc1k09R>sI-XNRyWO z(rzPH%DWfUwl3T354Nh=(HioweQ?i{D(cVOt*0!yig%Si(zAerW$R7b%7>nsY)y|K zts$#w1eDFO`gYU_K!4x)_b|x{*nwVO*V#?|k73Z37F8iK91>0Z!YF!Uq4&Lb9 zC~X!0tNfGww6#ghqu_tG1Hq3~XIrOWkISPQ^++$>-YM9p?Ro2`+C77hH>XSkJS&#sl_d2Lq(kNp@O zd_LoAd5d3=_09aJYh6nlP7lNkVAIsf!HP!@gN-1CCb^4p=`=x!GMOSEMzYUJe z-*!j+Au%Zr)05}*ihY?us1h2&bU3; zx@5U^&EJLA=gp(6jgQW@Hqx|DAGh!yYqyr0g4+(Qv(~J<7d+SZcpCjz?+R>2--@CA z9|prunJaGI27@=%?8%#7RSv$&Jr*1=;9+{`*;eJ%N4_qPo!X?L^~JU6ZRSr)8~NdG z@b##FDh|Mpg2>(%!QcFzw6VpP(p!Gqk{36|=3DEWC@pWT4U{vBu3NopG{JkH&ZPYkTc-Qf^wACDdTZV;Ed8I) zU95}OAFzg>KALvo_x>O=wpCfkomuH2-CqU^O)JWKh7s2BC*E0u&qr8G|E1GE$qwu0 zKL^Td#{CtXmo+6=v-8-Bsugk8h(D`?OEzr?E_1cER_)R;_*B|nR-xvtnr2^0KbI+{ z=_EPzO5(V*Q;Vz>rxqB}58Q8D;mx&KyE=BIU;H@Nnv?sz{OQnE)>vD;)~}y@Pv740 zTzY`caN+q1xwha8K0>J?tdqZ}c};|wl;1n;Q#=@-roRi1=kM#kc>~IMu%7^U>w>d9 zhf~qylh$#i$BQaUPn7m&i-m@k-3k9uJ&Fp1_!3NYCRPp1H}A*in|9c0v8ROAtczs5^Ai}qwHbY?DFo#2~?*;0hsSN4Q-S*j%xmPJX` z6G!V$Ahms40{1+diZ1!}(t3HWc@A{gKMtEC!~o`vjwSEp3h}+fBD9U zXWdQUTd`l+=+!vZ@ky|9(&7;G)yCUqh{9J3{5lkZB#tBHeI z_UiHB^w7u}07?DO)<5tPT&mLEXK2r%*~A&bpISThR%&TxN#O@uxJ@>vGrRLeKm^xI zm6FRiNXf>N$}${N`6yx(Zlyku>BPTeFO?Qx2hzNch_*#(*fCf2ZYtiY_VEVq=cn>k?*B~p0PkTri-N*sH-%=-$Ts>^{vR% z=0_!}J}FD-e#Q3K`_+8Wf+m(JKa0-uu|joag)13j4#eQSiqlO~i-+?Yv1;slplj}3 zV>g>2<%MmFZckzN+~3K=q$jC;689BvFgfsz!a;Bz-Cxj)8EjlqoutMwr}_ zATlY?#xj*2FWq&#P3sUWE~}YhZJpnCRNI{dYk{aVI_lo zxe{FwV&2c+EQSJYHG_on-nVk4y_02#*dJVpBG5Uu#31uw?uL9Dpqg%h+#BkT7}W&q zd*Cyans*rJ2OdD5V`t$fu{riseOU7v*(+lA74nc1LezM!sL_ zUS=k)os3!J2_u2Sk{_X0%(Nc1Za2d}c$CJCTA;@X06~Mxall~Z6>;ICk0BJ7XdklQ! z|0dN4#QD0)gumGLlFJX=DJ?Bh==#YgQ$kWtTV7j&RJiC5j1~t_CVgm3lJo~X!-bWV zsDE#ozZN-Mz8~2R{F*sHt5)xM({dZrZL~*PSX7h5BTw~`E{6AY{{vrY?wC0CH64;X zH1Gq>N(ADgs3>By`IYMm+&Q(FP8PpA0van+U7V@!A2%<_3hnj+DlPqBp3OegS6&pF z+~3?*w*uWNCh&0qC-$r8L}?;3ArL8TDcTwP4WTx`0F`kM+|5&yM6eP0LsMJ+` ztaFN5iZ8^4x-aS!U=Pu#;=XyFIJBf$$x7&S;b8ukFrjoL8cUo`YDdOF`@D<1Nz|pp z?&gu+4FQJPR+3jFl#WX3o%$y^8h`DspEx}6D_(BdB^^;f-#|129|#=~jyOI=xm<1e zcZ?u>cDRK97{4%GvjuypoJUjA8wby(Ehv0LzjNKvUgcE~xpSrCo^u{pv~DI}w=AQ$ zUr}Szd{eK)UMXYBLeh?=*h-d=bLAAMi|?qh5J=7H3Vz7iXiL%MoBqJn(Z5&&-y<z4vCW6@q?6NZ)-;<{fl~MEL&X~_L3{fy~D7P5MA||=6Y1$Icc+~ zNwVRSynWRS!5DFgh%!Ey`gHY} zbY_vRUvv{`hPF1+J`q-~$&U;_6K->nOl`I1O^oa8|Kn7%7Kua9`SLEYGSI^(r^JXy zvtsze;vl7gutypxFZ7+4U)4LqB*70Vh8~@C{BtL{eVxvay4>jawu@VN3d7{AqWOt) z$(j5!WP&;oNoPcKnd;YFRld(a@cW7Qz3B1V%g%cGwH5B_#!IBv`5bLw+h^smndFkldM;rt1(`0N+;8% z(z232MGAXJ>Oltx)VxOy)i+kBVLS8MP?`+Jw1HilK&!4C$ET2bC{v=R;ty6qG2ij zh?C_8-uXa0kC9cF=cL_oAz^D`JIiL<31y{aChp|}+v8COV!-m-@K}!YN()bH`h`8l7=dc4((0;?^Yt{ATBB@(b6pXnOLiz!Ra|(p4G6 z4K~y;e3m=Mdy{J8=TjFJ4J)c*{Z#%$NJ#BhT2cfDcJjIWqS8a81`)ZB1{?CU zm==hlP0GQ*MyYVyZz)=A?~Y|u{*iRPrI5CYJx;sv_r6u&0T=BHFTW3*cAS&zxR=TG z=yG|#QX}`3!WJD+?i2|H{q!w27w-EPFDbn1xa8E@(hPA>FXVspzl0Ojg14~^n6vfU zv~|{N3N30-e6)C= zT@ts;=lt;&CNQ(8FZj*}`ZktK4kQ$AES=^{Obt&uliUKd#kUy zZ;EfS6zck9KW@q;o0|56CVVU06#CEH9<3>^E9uU)Oc)P~M%m4JQoQ+KN7dN^yM7cr z@MJ));K7gzy)sVi)iAA%0bBaQpB{5?PLDLZEExCN}Z6Lm2RnbA1 zQ^LyE`o<^1r8oQ%Uny4df>7IUGgkQ4<0XzAM3YwetIp%|1wobMzbG0 zM)kGKm)(M&-zyV7gYOvF!&;-fn-ZVcLm~8Yol6Xz!0vq8vTSaV^Ej!EO(ApXF<6(P zH2bN*Vedrok#~{46N!=c z^|myr=+L+{rTNnx*6MkStqsc>R+LT~Ztc3s7F;-XZg5Pq)z+ z!K}Rv(&l(37;9RbWHqco;n(;OKqE2qQer+n+KkCni$dihY%miDd}5C&V^Ea%gok@k z;xFMZVUjQ#GwI)hLzT`p1Q9mTwJbq>f4D|K%YFC zUCI4O$lzfd5w?LbnuE+TC5!>2Lu&W87kgS^_?cOq1LM){f{es?Uh>6YJ?kw1p-kP*v-!qFQjhS`}{+-=ArSz%B- zCa1?Nvzbur?AEHsx zHTns5%#s7&@QhP=E@@PbQo%=x8FZHB9^6gs!QUg-N5>*}QHb<|b385RPg1#L0w3Zw zeu93M{M>bz9LV=)4RX3E5;{XJ^7b^1(uR<)`5B&3{zu**GzPi=0qDO}Gu?kAP7iT6 zk-CZp(Gq_@xjSm0H}EsX)zS-12>3<6T58`a0c}W}5HrZv!WC*1z8a}ZoTrXZSNY)y z8Nyfcp}dsL<42$|-bj`9P$5i3BbjS_q5rP1Ne;#4&=Y;F@dip4>YU&)^yHSCGD!!P z?`!6WVq1yrxJtBHi&3OaSfV+Cm6$nxjkbg-k)c3{rKd1dX~$fI7x|a?8VFU8M$}|; zSKo1BuTY&`6Z=wEQy9f$IzfIGGRho+?S*zpxIiew$aJB%w{!eyQyTt=Aopf49xWUUk@8Xs#O)LYM+uSYlx~5rFYXc;7x~gm5|4c$WZSH|d56R!;xx}w>?UR> zHjp+rL>WeJ^&ErhieK?}Via>`|88M0npRkoayv!j=xoagISf9G zPlC^rF_sAGw}1u~pu7FeK|x=Ptz#G1dm;w&NH+;LpteIREH)vWuqb}cEv!Mmg*|nR zQF?%<^$UHY$lKumpi^;)2E;oWF``jenHb660}fJ~Ofdh)v`9eU;qVZ^0$-r|A!&R! zA=w9-uY30ds)-#oWVp)cp2!tnXZdE{8TKUm7dM>@)p_J+#7IX^ zaJsLlj_w2#!FSc{j$O#aQBks)T8S(qqQo428M%``#b^>S=m**yYe$x|C&W@#Lv1v- zlrIuSz7l)Hp97}jez+a_LKrKIjrm77E06NEb1xN^nu@4~a(nWyK1?|VC&rJ~oBVmI z)#RgemnkOq0#2l*Vn_c)^Am44vV^}}#Y8pf2z?i69l%t|VF1yi54`^f3BFES>y(n7m{{9_6$M#VY$ww=6_<35ig8d*96}gN@b3_ z>lXS@TVWO7431&CX}0mAcUb&3>@G}6pNS+Sl=% zq%ng;>3EdD@}XPg8|AoZ8+Jo@gN{(zWlv>iV1a}sf(*8cdCJ6KiEz!d z4k%Y=|z9%eq!y`)*< z5Gho^OtUOoJ)guu=C+oF0utR*q*=S+if^R%xLn6mV1_m?6vqlZiH#N2 znPD74P8K_=B=Va6nPOvqcR^8~_YhMYK7%}9MEK$G=)#`D9{LL=U~VdgJOd>AziIcl z$6DH&>*)>(@BN!e8I1E}pho)~i3|H9&LqQ#?ZmGHDdD*ER?Hz0<&C&T{6zOn>VjQ# z?<0KbKJ5y=7T-$I1lCg5xF)9BY!$kb(#c&}gz0A$ZBnby8pKnxJ$s+A0s5bO40~z@ zVy~O}I3_}v-zFV|uBq36s|Jj`W%|c*jGhiXrDm`L#PQ(VNRXK5>*G6t4v<&NB78%s zCCycOauca1{tMhbAztpxa##v^QYB@;fi%Pc-xTKoQRW8Hau77)&RPhE3g&Ux81q{x zvB(iP1vK#$;{!lUD5Ex!8;Es4YO&GOAn=unblkyC;nm63j4wcdH3N-M`!OUv_8lWa1o6Ed7Ol zZ$87NiZcudHou6zNp zL(S;gE&zLg%V-U3zP1|+b4M)MF?I-o5v?O2s%N zB}ntAzc@Df0prGo=M57cXgZM(bt|cWs3w|-cI-TSuzZqeCT75EbJuXZ@KN7PjuuL> zt+p}v6RH{WH##j-)*Z?o!VZh7Clt8nvt979{MK{!Ccfo!rRgXdAouA z*by|~Ce&?NIUx)wz^CE==2wUNa0}r>?qB)s(Z>+wmYKcSF{n(8L%$n37Mj#Ns!kWA zr)_qz3q5~(AIAPgy2ECHMaCo9SGV^B=NDW+UdD(Fz^7A@oQ8_$TN@8UE8uU=_2e<= zIrgviK+IO5H#*k6!%IlO-mjiGjSe<3c)rY})Bnr8!191KKS!Ndd>0qOQ_yCGw>=K& zfE-5K@{aQ#SrDxO_0zrMtNQL3Xsl<>MQ)w>1NhR_4%x={;y1DrrL|I|<^W$EwL6MH zRJ=t!b&g_0cpdA3e;6wroN*@lQ4HXpJ!84UC?orc*L;0hL*7Ad7RE4sJj=x5H?gO} z5%DG85VP}J9W#jy*hAqV9@})0dqu*;*i`@xpVXe=E&xV-oXYVxYF=XMtH$GUFFfBp zmwCuWlk*E22o(jS?x~|a_>g%;R`RFp8F4$M?*zp!_t%MJTrZ$++&ccc>JGWY&$sU` zh+ZCEFDE__TR%blRN z+(FwG)ubTObK@SU)ULBaCY8%*#1-OvoZ*ewX})RfHom57jjo2&7qt-U>2{I@9mDCo z5;v8osxQE|*%nYrOtu;osYWfslZ=Jl-}GFinRleIrlQa%=^f$#<+@iB{dT=_hjgC3>7um;KIcu9`;{6m*v z5x$Y=PbJS-*V8S21K!i!+;|i$|B8j{^rYi}#n zE_9BaE^a`-@ll?&axRj~;J$k#q*@avI7b>Epxr%bz<=5(VqN(p3bW0ZwCmX6US;`*9h`+$PU%Pdxqmz(sNQD?e-d5Ax&*h8y6QDyi z#W#~`Y3gb&k)PC^jy3L9f?p*sh~9os40>42`H|w|7LV`M}Rqy*3jmeW3#%=p-1H+r~{GW*}qfuVOs6RsF5a^0$_D_6PnLyN_K1 zLzyGsKjbE{A(+Cp5sm`8RyohnKbA+yMle@K~N*t z&wg3xA$Wzg{0{QIFa$7Bsq*PP-uQH0=2xR7SR0I%Hi0`ii*S+dr(Xc>Cwocv-1G2a z-3#Dr;Ym#_kDHUE|H1M6SkH;Pm-|AQ4cG{@J9Y{NVQb^h5Me74mJ6})35F%FU=HOA zk*w4aidWo_```+G9ehd*5GEC3Y!v)}50MQ_YGKd)OT42%uW^}>10C4bp004e!ylvN z$OfSql~h>D9_Po=-E|GTAld0&7JK#SX+h)^wFUrj1xb2?Zcev``%$3OrM_&pe^Z-|QZEywHUkC%i&tFi*)Zd>l4PKayP^+uU1$tOB^GuW~cS ztg;VZBcp)5;BfgAJjXP)XEn~hRA6C1fU$eoNRz%|x zQ1B7w|U`Zt-j2 z-(WAeH~E|FhWnrh?-x%9UIlGXU=x0#$KeC{L+E^Pi3os&h90hgLU*UeTLnMNa*CVi zX79)6=8ui7>zbu}m(qzkU@SYyvsOO?Gr{i#)x|3;(ts@Wn&T{q0PAsi< z>E1wq{)zXrzc00vswtc^l_0zv3;!>-J-Er?rB*pRd7H(2(oIMAD|7gVE(ty8zK<>O zlQN9eqapEvIG)@c*bd`hGjVzRb$FHgwdoBzj{Ftqq=b^E#e4h8!eB#^YIy?lfO(}#pW_?*!%oaur|3=-7-so-zulsX<}`2hvJ!Pm;4oL!HU4-vP(}gok$HG$YLGGyU8u1W{h6MBU373Ofa*rG6uXW$;LgUlE1gZ>#HA;Z8+ z=wKZMol3X`73=f7n7cWBNjDA}=f%AJg&*8Xdnr(h|QHOz8T#Z z0xgsHv*^UcA3}4c+*E}dX3hd*+$p*MmgT;>1BcJX(>M=-5? z_xbA#u3Yd^;AW{lvW0jnbeCqy9bI+MlSqgw4SN8ldU{d0mfNOIn%9=ehJL^mEX*SN znfUSeCI`&TwSQDNd<>8x&%q1TtGq+47^{kP%{swGv}9+?H=&=>drEyz`f5<60xXpv zt%#MCi_2~YH;aN`+JdZB)g9fxh`B6%e@ zz}MF~m_J3-E-EF?qsOsIXcNhapOsAGPGP{lMoC`&W#NbYuXp+Ame70IFCO!4Hta*U z(9dDL*qsg`d$?Yshb54D6gSAVlyV~{ean!J<~(YNbe3yq#!R8uPs)Snr332xeWB|X zh6)9aC59pT60D)4t(r(13SwMG{Q{H|v+=rMpKNgJ|soRdVbQJ7;qp2#iWXsVTS|s)(_7O_R+=$MGM}q!+zYQngs=!8FtlSTL zDjLaE!osYoHVqSo-REo3?V+fawZOAnAKeT!JD&-)#pf_f!TyFpx=XZ9wGZ`1Vc3kH z#&)2ihz!FI$SgEMrvaTMuT9~e%N>kw^y}>#sqWZx)QukU4vZa(R#goKLHGc_5bc28 zqz3#F^IGRvv>(^j z{SLVR|BVIX&w$^#!CYg;?We-GOK$Xx(A|)M6_ICJWs3XAZu}ztD<6KK67?Go@}sG% z@^X0_G>GU(?^BcIIp#;WM>ym~(8kJDcANbbzE`>*(NiNTub@igE;SUK$&O}fNibTO zOBA-oK-GWwnGV%vQsLM};W__`C6J%wcdUo#gI25CB?_|C&0GUdD!836A6rOl&mAXd!1=IBEv0M5yP3M$ zNh}2!p(=(o#(Zvz+(K01MeT43wO>;*u~OC2k;}PtV{}Ih*X3V@zsVG@RenqRkTkO)Pzz4|{$9y1pBt?j~GRUZ0n;4y~qenbnH)^uO)r@oJ+%-8kzC9^pgr1bK|#!Ws*QxVvGK z_)pwkwvlL(t^zmc*{A_+?mrLU&@aX!uhH-K?{*&L_rONOO<_FW1xtz9%32jYxr>U2 zrm0+^YVub8llX>8a=jpnSyHD_!UBIRb-@tylzA-n06D?e;D^iW5r%!CP{L>9E9fGZ zVEzwsP5@jJ@H9+z_Y>wvR`7NX`HggdsDoNJ{Jnv8iD^|G@l8ZkgNU*NXK-VIUDk#;oK)-3%^`Y^%-14 zq=soZ)`pG(W(xb+Ww;mli7%k417_fFbgTafU`4tZj=Q5l1AZl@s(Uv+4ejK4iG{*3 z_+GUtF?t>tBZThI609TJ(GyU&f&teAG8g24F~UegC+NLkIZ&+nz*nFI=m%HwKfyTc zE3!eeR#U9K;GTp$=LNi#v=#{^S0KaDvv^O{i5+Pxj@$u$0u||I+;Fj&X^)xQd%#n~ zbg`e|GM|}Whiym<5f+4P!}H?%i`g8lZ?1od$C5kMzWLSIO=K>-*jMHm4OSCuT)x0d zt+6omgpdiPsl@GU%!bS?V85- zW#7BQyT_g(A`KD`UC1H zq2@m^`SK$&L(cI`l=~2Wvukj`G}3${wlmTh+9O>6{xyt5BfSt?mFlW~O0v;;@=kvw zDYE^2J4_-ru<#<3l)nZud5sCYc|KGEx8%Zv5!eKhp!f1b;6TDjR$vGC48U5xGyjuz zn4)a`sdfHn^8hJVKC4`kPZ2apIFo&E$VxO+87=lHtc4tgBTQ%M7}G=^u*?<-xjmO+ z`D{|;(GERdl@8(pxzyhk&$QK5?L{eaR8oyZFSwyNUkX9ip+4j?RTmlT$H>~8+6RoT zL3Q)jL!UC^Onoegfo@_;<17DY{v_2w%=I5CT8BQyugR@6BZ!u!xB5MNPhT7sN5;8D zUo)6;B?~hESot6R)se(CPq-tmaBq}e;X&?((lsXNn@LyowaLB8{KFH%-&BMt%hDCh zWF{~^@?S?H*hU%3s69SJA(oNzd>@PM*;6D)n4*m-su|d5yNoo)n&=y=XNV3&6mP=p zzA@Yw_`UQH&q70WgY-VzYI!4W(Im>*hH8N5evUcy?fDis@2!d#kn89~bS_cA{S|U@Q`5Ms?2kC3>X!(;dpEMgc zs{5W{;I4g6VwZ&RV@MJ`LYusQAw}qS;kk2_GzuC5oe~&0TIs_-ab)9J6fd2W|B-%) z8+}KBU%Y@>+0(F3z29D3e=rwXO75hFcs2A+wrN6DZXCKw=>?&)mGi+ftB}r>)?02edYSirzj9y3-dv{eGomU zgbPFD0(BPDTmAs+EoL;zxR-f>qAJn8Lw++koNfu!6he6~+k_Q!uejo3u;vqtr()OG zip3D@5&YdgC;pMKdQADA5#T}92tEdOI@fWJ^OIb^^~da2;OX=N?PT&Q|IPK8ujD7# z_JCKt{pl^LA^y943@mBpB&bPhwMVhr`vTa&UeAw*cNutoIblv$!j>>&M zj7i>i;5wDPKcVoh>lnPna}zjXbf|=S7+ocdreA6^vp0I}`s*=f0|JCOYC4-5@;%eQ zdN2hQqF+2OfH(G_%LD8NPQnce-?{T_9l@!&hmH9$|AXtQjOkn8e6EW)U0=gqjEr=T z)(7zwd?(~P+Me5pj)?ge+8#69g&7>KH`qe(8nW0M3NOg?s?Vj^In6V!pp75{mw{<6 z#@)uaSZZf%t=bbVsP>V@&ewXax`UjCqFhUE3HudahHX&3V2tJ@P>+p9(sb2@4#Ie& zk^Q2&md0^6GS5q|rRmTHdZpM5OqU;nvrz|s+#Z#+m*@pvK>AWm3V*}X@~_(K%VxLT z_LM#=UNCkRanbC?J;T^7p6N&>-iAAb{t=qWW-d$cvgwZTDogXJJ_LMWY)=da{5wv; zU)ZvUTHc#G9(czkM0&Q!C#6yB)4f%(Pip$HCg#k4gkXA|aDu%ok8sa`zbjKT5y%iA zfG_eUK*>N7sTVTE*CLD0cT4=1{MV|(m^UQp7uY9Ynu6YZGC#C%j^{eroj;zvJhzLy z*RTQxnaRphl#Ny2xu37chm;YQqVG^>p%e6BrR{%q{JXDL6w zB*z5OqP)uT@ryINs2wOorEuSM{X|ZXE_H_VT}=dJaP!1_>{=!u z_tbCn_{5OtN&Gdw3tmY))79`5V<(BBe4{42g^p4m=s)5aG*a3RshEGbvBS-`^Gt9( z4$p_8BFC$ncV4NSu2i02#pFK61HL{o$2pZ&^dI>z@Kj$A?4q44BTaXcpHM(PBC4Gm zew}b1Z=uaF`h{7s8B`mIK^c0N=?~hDyrN&to5`R2aeja+o3w#f`N67-BSoXpe1e<9 z^`$5<5&aiip5tJvm}{93N4j%5wLJb9(O>!?X8=QpzR*u9PHg2fd2SNJh&P%;Qd@o( zz8I{pY@%;_lEreghr5oxp3f@{65QZJ1msi6eN0W&FpDsg(He1N{+om^=vM~PZ*Jc_U1^vGZlJRqHGEycE*_QZCvU@8-Dph}`7teGk@QY( z1V0Hri`C~)V+SOXZ^U2no)dct`^+P-Tw^1;uezreMfP$v1x}Dd5T~-7F&dZY_TV65 zK$nR6aJlmT89y5?M_PeqoGC>U%;LI9rIvnId-v_g$XMH8a@Y?e-rwjoyWSbSwv1 zyqR}_o)JE?>k&a(>K2uyQakvKXi`$#v1n`eQ87%Y2Ddj{W|nyL-ZDM|ONh&-<|8-J z``RnUrxJ_)kE(90OQ`c_{vmNrzW`~E-c|Ym)967$zC7Bl=F6q}fx zCz7@BM@|0GyV-%@CSWPZxx?H?V^$#dc74ZF(bv!_BVy;M?TCu@tKQnsaP&eXy4=&Qzy#=S^+W>55uodz`UJaL5Z(e{shGJ&|Y zxGEv#??VBmF}5EX2TeqB^q8ScDdzejj)-}0`kT6h_5}c)I;k9b;TQo8$A7~|;o8tTZFAuZFpNKqj&Jr|*UNZL zGYX=)oiQ%&DM%!zL2K3Sv0qDTBGC!%_IMQ50gn=XAr7DzTg2t+c4!{J+5{G@4`KFa zhNpJM*gBfd8RDIW6UIsOA|{JXVt;@dVKSVetwC%69-`yLRCEM%N#x^S@I#zh`V2o` zDCP5o0@q}{$UlP8;c4-I;GycN_FCZ-R+K>V#;x6ZQmr&SpT7t|hin;6t*EH{)&K@7N%n9c{+9fN4%~D0)*o<85+|`O4Tp{M_#CBB zf0Z1~ILYPubMSd^6zPWNDF>K4Vsm7&JfFXUmLlasI$r8tM4linV;G$e$v_+IG2a=R z!nMSolBp_rbgS?mP6=B2UiL0Ww2)y;b=8u}UX@Yhtg1`15U;dgLjZHUp3?n)e_y~jOB~fYNL))=6Pf~KqPxZ@G@XA_0#jD z@FN)M%QCw)US&Mg#L~ z0eFrrmY*=Kp-y6BX?y)0=u-&>kK^~aTcW?XOfdPjD0<5yJWssiPV)A!*s(Fn1}P|w zP;^Wf)l066^(Kzs+f5@qpZMxnHL|hNQ?4-0l;-%JW7o*#-1pD>v~BC6IP-tg$?ozb;Ce*zgO#(In--cb0e*Ir8b@pTrXS$w4{0(-?M{F z`SKNCRrnn?7PVmg=vs!+>JIT?WqwRw?>s&VR#f-Q|0y~P_9oIU3}eL>3RI}K(aCrw zY3h=eLUDK3#ogWAwJu4MNyam=8l@@jEbg+n!{QF#eE%RgDMJ&7%@ zN@OD20@ow`xeE6cw52#inhSelk1)StZ$SaVQf3ZbgQR1ZCHBcam6q-Uopa3)M?!P8 zD1Ql=ginr@96{<_sE{bIHbb{VkLcbg4$@qcs6JwMLu%MiIG%b;$3dCmIolS~RlyQF z9$$lw;C6B4Tp*Nfx~Ka9v93tCsdz>`5A#=VbmZDy=tyvw@IxP_T>~$`Uxl0zLk-VR zJ=ez_A@M(dIQQfCB0J$dxxd(aBF=Erl_>rc@3}&SA>zLBBkWhlWubB0P;MHwn=6iM ztTSU;Pb`nH(Li?XJnoZA=WT0?6AEJHWLWu>L{Ioa0m09y`!<7 zxX+CV_t~Y+Lg!&`kS8s^6f-#*=zm}vJsxa;uDg386$Bq*8t{JJ`%Z^=l^6u~5PHCN z?^*dx><##k^|CqoOm2T{17-@>Kszul7+uQc7#85c5;O3dceHmAT;{yx%qj>qTrKj~ zjzAvMgRx;^0V*Rb`U@2U91)0#D%6*7824C*mdl|5#7p8@>=XS3C`~t%%|d4q^6I~# z$@*fDh)pL}drG8iVoqF<=N$f%=ne@^3$7qf>4u=ihFBd|`jdka9wAue7I2YHzp5Xz zPdI}2@czpw#31(|OR4rK?BaJq-wl79q4Fv2(@`3t9OX3iNw0A6KR?=-)(Nk@9No-%2E7nr^r8Vn9jl=h~(|W4}#4?J!sR z(H;e9ho@7o?cFacjT5IlOqbu0`*s}bd?4bVZ?!*m_VDfU+w1e6JlS{**>7Cc z{omR>NKYeLs;z~8Ta2e>?DcIdN%mEo>uX$cx~0#yYo)L4kd4NAkD`2ShPV$*dk7ix za!`syNU)0ee+T>D^3vM;)tPY2s| z^sWCk$TzQF%7ID18ejaojgkOjym4xBd*8^rBYbaDv(o8fuyeKiXxcZ>+gv)BYssN13&U7)Z&FE>&+5h{%q#i$NOQR7V zy4zbjB`9@j3Cu(i~5fm ztT3}0aj&wzJQaFH6{=^e1F<94gQ5lp%jUs79o_gc_5gCrk`2*hhLjK82RCtmVxx0@ z)KGS#c$&OO*26DInN{Nwqu3KYE>v4@=(cK3DqWbjs5c;LX7fLbS<_-BiHgQ|Ic4UZ z@@4kT_#zGl@37y*S-OtW7D+mi#A%T_iEtW14pVe9Y>eGb)H8gUuUApWXuSS|XPLFhoOOd@IIFmW7g zP6L+--J*YyPGTP2B(IKHXT2rcVVl;XYmC3*fV!u>hEhdZfMmK^SqgPQmRm5K`%1HP zIGRzqADma#$-2ch4(eh404!t10-5DIRKxjl>)i@3b}V4Utdx-kjB zG5#95h|9!w($&(A*h>3$a3-+A+=i>?xE7tNxhea?9ib8ATGVOTHE_D+rEm`coh&Yw zvgp@kogJ63U)nSFHi}M(Bb~RM{T1VgKzh5O4IN3pHay};-41V^dpe?2pVYT>^(K?7 zs6F5QmC}%_qpM1ms9uwo!B^m4ikP++fxZ-}ULCOyOx3)nCt6I<{GFjPiPvAWQdy*| zfE$#qSB7Z2sQz;_RWubFU@N^txbx^~^#sTwv8D$q7L(WLz3?aeEf`R=p32}_L9y^L zWS}ln2!)mt8g<9g*Vb2T3VFdkl3mJNf@XDmnHkayic0tHT%*#f}d1z^=%CY-{yKBoKpD zNh%a>4g&(LvEV>y1MoJLL0*Cc@Guo4A62}XU53(70ryc8C_AS4%H{x@L9J~kIA4g+ z&xeIbS5Pi}8H|*-vV+`Y08qULW`LF0Yx)S*N7_CqLM8STAS--n3&xY!Yjmc1C^(Ml z!1g9%A%JaeeN#~^?n1B6$7CpYCOES zY+RH%|E0N?7L^<=wL($&7kDo;5%O?Niwp62RO<+@eFOJirA7wHw%Hq7>PHXP>Ch%r zm%L9@hv-kH28wi%g#UBdoIO!2Rww>PES5FWw~w=Hqjb9+U(1FO^WYOgXzUYDs^hFU z6W$C}y1RRHp5^FNL!9IQX~w7S+fLWPjj-`RZ;s`Ef$xy>phU~nRNE=*JJ7)N;?mg{ z8kwWMy@2jbrNF)LUeaviR=m=^|VX*k9sNC5~-N2Ox zZPxWr_mOU5nYy3iR?j5IZSP2T2L_LyhAgD^$1QO6)*X^=Uhnuz&Oy#Dd^x+32tz^5 zBe_VPS5(sLut&OAu0qlVU!m&ZxkR3QjZo@ZnLmW@iZ|nq$m*5d!Q0?7Rll)ynyFAO z&`34d)Z9_y?!`_Lqr!D!f7J%e;{2}oBr#3v=_A;O*749I(WPB&Ue3QkOFY}ygxKL| zXMuB2!g~0PzO8;2Ya%Ag2BDikxn#Kpu8?IDdQu+bw z#+@t)`pja9|ji>@*EqRe!?ckn`wtbp@#Y-8hHF+>yGJjU8OaSQjkOI}ZUYs2 z!JE61sEsiObC3L4>jKcNrHjmWRjJmariKdKN|S{KIxDXe6&kvNpT+JiXW6sm&sEyJ@p+EA+rL2fho2!Ku-xNZ9C- z@lc@3Ub4WpRuO~_i5$$8hIFDt@q}ue?6qMHwHb^+8#-KIHI}Tx&V-7Zz1UbLQFR zfi^NaSSFfs6feu3M0RgGDfEUUw0%{!Af}z7VMKg!_o8}X0bmonk!B097?Mcfrs0ZF z#Xkz-iaT4C=I^%gYMI>%ECA0aKy`qo9rn?5mYR+pQ9;lk%Lk?pB4>uW7D*@U<1YpbAbW$JMB!mA5?CC$@Mkd!5(Tnz%^zc{V#FK)f69(JQn4WJGhOa zv9_T&O?A!wiGInppkt8lR43~lE2WCEMWU_^Cu znJ0t9gMSAW9G6npB>Nt`cRJMe^T~Nn7cF}IC08rYANh;QDI%%aVc#uT0gjOuKsR_(-Gj>9ekYO* z{Y7Q~ZFrk~jWW%8RMXzn-l8@q!+WASoBl?l$W~+?gsTn)X4oi|ADC}%1+CIZo8SC2yD^*N7cqu3X@~Hs*O&U#N}ItzhZYX zt4m_>ZjH5ZYV@Xdb8NOa#C4Wy0ALaz}QKP9ftOX#h3c7F!XqT6~2@IQuF4;EU~vd~@vzdlUFPGmZU7q55?DkJ5U0 zknqJWpxxO!*lR8j{HiL2!pjxtWMY{Kj=msG)y_w*#<%OrxS#w+Lln15hwAmZp|;RU zD_=_{-YN`YNrf75* zt9H7CNMW~b6u(J&Z}tQF>aOZ7@;0t&Vkj}6-|K!W5#+b&_d8E=SF!!}-+~42z%4*K z>$c?mQ2fBcaDx4eR$xnlS!593aAtgA0jEKO3UD!)Tr z#5lg2ZiS+$E}K1%^G><81=9lS09QzRV2AR?qobM5*aUkQW&r+B^$##l*I86a9YHgC zP?*m*hBigii2vzI#RanG*gw*CUm_77-qjq&J;;B`v=Q#*&4tYPNUkk&5?spE*>qwk z@`o4)^@j8DTefuje&7!@OnroZraA_{G+kq!=2i>qsR{CBe1WDHbPT_t>J^QUWhIch z!15<5-2x$l$szPN&Hy!JMu*pMYp8zaadc#1jy%^s*m5?fKj~$Lf|Jzf0AOqb3 zT$LZNN{VAelS zZYNnZULk!IUC6J(5-P?a0zKHL#AMw==M?@m(ii@aH?U-LgE*RabhR83=X>mv7O?s|+ zr)X49(mb*qp(pU?8L>?w9&LM~|E_bp^OS$<`w=_wQSLp&JYBv^ua6~c<-Pfyx|UQ3 zx|A)%&I%vE5$+0>rCu^m_xVsO}Aik*a~ZVW-YbJjzV#o z&1?rU1NmVuf+tZ&=`8e~L?G)&`Z1qXZ;|Iz5}Cj_V|tqpn<&mI=ef&#xOka1(lNku zaS)lthnkvO>tv^)`CLPyA6m*AwFf1#`*?7s(O_W`R zR^l<9F6D!Ti{x%>JdJRF&@J>w&&1-GFIr^#f;)r;zI3-H}^OXs7 z9I%g6mW_d?mh~t}Qd|$%0GG%pm;fO2Zu^6Mx1f!%E9jEVSXQPg3LQh;SLE|!(8kci zz^!Z*bqHII_D2#e`=I2u;YDNU<6Ne-6V%mKV3szl?9GUby!PB!aAL_}vXdsBpM_gZ za;u(Z@ILU~IZAyV?t%5OZ^Y%`)+P~jDfhs>7A=cepc>D04SmhDRNtl}*w*+_ z`E%|!7ff4t7adHGDD;vA@Ixfa-oTc^_>ud`!+8@_%cxnQ1J$FUwoEf{BG*SwS#H1& zBDO$psLc`ei<2S`aXZ1EXcB!(XY*vKbBOJ(MdEdRbez&K1%Chmj@J6I?$t!LVY%26 zpT|vx<^X5&hVTm`i#WTh#IXx&BiboH&L5j5q-ru9yJN;ci=&%T4(2IyLh}U*MZc>1 z+XJ*a7=Pj#UcXS}_6q*&STuzH9`O%gFW z`JT6q1I{@fIk!OwgWq{>5xWz|QL@BC5UqXUX-9k!URCblqg^kBV$qZEH75q!0hE z99P2;rJEfA@C?g+p*t7~eMgfdpU_gc9j-*yNt<_<#OYd!e+stMU22;khT;Lv57a}j znV}1{UYv@)1FDee_Q|2GnI8^6T)|hO|C*z~FS=}GG%VL%5MGeGtwG`p)3whpHxq=l zJ^41&AsPnbu^(_1SIwrD>O=h^aom;L9t)8_*F7fR2>00koSo6e*5jCt|5>!0h~z_= zW9o6>9bvR)2mVzzMLehNK%S=>u-_aB@MUbhID(i62CyvgQhFN)+paR>k!s~8D4pAj zwn4N^KK2m0iR@+Gt!f45K%Rn|>=Lwice!#FN1LroH1Mk|9lQ(Nwp8KSQ7w>H>;>w9 zCKr5(s<^f2L!h|CSh8QZratUxVteS`B|lt|9ow-KcN}y-@EkX!2tlz)&QfPwl9qZb zQMgpDP)8qEM@(x7(v5TUEq)x^)Nq9osNu?NvMqI&JqYhtBu3Y;%aE>65%Ws9N0DnD zuh1~zY5*+<#z%jrf1&Ywh-@PH7xvb%Nwpc8puSGYklQ2!ywY^Dyi#lfg1`~VUB!or zW`r-c<>#~9WwlnlQypDgFTWv}74x*xdwIIiRaeRexde^htek=S~pVwhY;e+%1G z@F(hz+TW_OE{jpBWTiWbCz^8G7*x$-8bvKH>R8<0?66c6+$z{wx?UxV>Rs3)V6N&_ zWM;`w>ugn)vcGP%dXX$w5{TRZ2>T#UWZXV$j5X6S)AW^Nk#OdB*$(YebUtyF2!f;R zPRVi6i|>lgM^j@5!dYT4KHIvD`_8=;{?%-UztT71NyK_hTh)H^^w@*wfVknb+0!1R z;O0mL7X-iHk7{R#Jqv$u8BrCw*+_W&uvS&5AEPLmMUSuzSK4Cssc%r_Dw;Q=hirq9 zOem~ui9%xup@xNjWBv&khu)wb1LN)WI9_!}4ngc z!R**dO}6KuSG2V;Q-vi_0r)#0Sd~N!;nvE#AtGDK%hcCtK#Zm)&@MU~Tk8yQ)+ z+qw?s-+~W{M?2IwDvryh2VM0Jb$o_Ik3;NbxX%@PFwZha41sw{z#z|joz|&|T_vu9 zKkF)x7UEUiE~JI_llWJ(kn@*#oEXfn1tt|XfEREl!Q-4p*{Qe-@Va=JY9KXHSookv2MXG&&5Q1}8d_@Kf4)WpvOFtAFud;x~m`ta69rf5(1w zNZuvF2|Z=%#Ae8HBspP&c!pmtRO@Dtb3AK__552p6Kj(*&e1F?#Q9qIh2{&tke|Bu z<#XsG#A0nl^Sj7;DA4&s+UEsT({xL8y|o*oIf8Qk8(^P*qLVgB5m`a0?K=B;1mpkI_Ee_%%bxkT)>iW53Z29p= z-1$U5$(Gv}pDuLFTT32be!(X98MeK)2&UcboPRunRWdOV!)&Lp9e5+|oo6Py+cB9K z=xC%p%>4#k&H^LD8N%z0VNBtm=!rO$&M(zl$VlT4hZy>|aNBdcEOswK2c)fdwCP@3;$FU{i{c{h zdL!Usf6r-Z9a<6D$~9H%F8U4Hbr|>EK94HmT$~SE0Xgk7vbE63iPQ&8INb^yZ#jiq z3-$qH%Lb_L~Gc(k;K#;hN>(P4cb!C)G^V5x81L z(SB?R6f0yE#lmTn7pW02+u`7jY9Bp-8^Aqv470|F4~S--)2{JY?;>eKKxnT^EE#Dy z%KPgQ3}YQVrQOBH&LQwHcZs7n+ZJjB!TLiQ zbQ}G}IovSE+mADGrJj6Tt;kAju5Cg6#VrZ3YG3HyG4tgow5=RB%|2=ZyA9iEzgT1? z8%cTbBVDv2pMd%KN=B9GTwpzDw@6H{_vjbNy|M3izJE=Ab6Fc)m#) zjQ>j#!4sJF$|EWsJ*h}lH$W4agA%uW3VMyb29C{FMm$xk$Zkv6NcPo;RzrhmV zxD79h47dkWt6^0G_`9+ghV>BQbDZScfv*b`+-U5Cs|Qhu4npeal>}&8!+fMRs8bb9 z@g%Ooei$FY@zww^LG>5g8mWUj>t^%2B&*OwY3ed-?^w0gK}NUXCg6Q_-%K;4PVZ~Y zlpVOeiFy+5)=c0Nsi9gCeJm8(H^Z;NXUr{J=|3ASES}3B1D1IkIO1&0-G})MN4(ID z9Vx*C7K>BFXWlIp_dUJreFCl|Joa?)T4ZMnS`wmrgi{^M>UxSZ^~dCtEkF~5`2&5p zL11Tfpy?HflC+|62xM(iW>Q!!M5zp1N_B*K*aom)%f<=SavrGwpJ@gZh0A-`l5h=| zNzdflvW<#&+h5vdtBw{0@CK*}r$o17iTot!M%YI#OBKOSxAcP_QF}F4ZQJAh)I@>X~RKKV#l>2nufRRO%JD1$yMMJYExNcG0co%r-CV} z=(0UgQUTg}9dVHZrKw>iD~7FsqS1Hqm(W%FcWM*qUz(*nMeU93Z0n;Hm|T?xeVBjJ zEEo2pzaTp(#DtQU_q4%&)Z{I->-ih&>C z)RHXbRf%XHO1V^prDLGRXyV>6_OZoi{-Dw{^;BDX%@%Br*$Q6>p2h?KRnT*IEq%4D znpzBvu=w!-^b~O~@S4oj_U5_^GSw>x)kFB$s1~}u%J9kgjUHA!RQ;Ta#@;l0&5yB{wqo#jinYiFfSc zsCirq;yHgg7SeY^j=Etk&DlNbKO)MYCUWssy3X7n$(eT-9L&FCZ;23ZU?Rl>S%kV{%!eA&M7e#O#@*24O(W2E*O?)RjpCF$nMIS_Gn-z zuxxvXDiqF8?X)XP7NZxL0U)xYsdXB1!CPX{4D1EDQTxdzZR*08(8~@-o5%D9USn#;mO1~{jxp~g zdIAjB2nL8>LU+6?8p+Mj`e8TFm-f+WhlkRiw7vygK`K-$L`W9O4ce{532_Cs7~cS- zQ~BJP;zVLJzR13iZe|EJEO2&(60wWGW#lX~)pQt7v8_U3WTKQVrbIm}^0RE_D&&8W z8}L!mmd!EsSMDd&NwwB=4xY*vF-rqZAuAT+bCOAQ{ zCpgD`7du9`5v~eRlnC~q>V$4;dw5%M0kKmxO*k){AQsVgL79j-zUof9rr8QC<+7XL zQH6(^54^O*GuOx+#B+oTEeD=A^_9A+g42E4F3(4Mk`@C#0l?Uu~DpV8IyN2vE+2F!-qklD7;Y`%7- zJgZD*>JoWald5{c%~8c;Q^;q|n-)3mQmw|`VuK|w`y@^sIh;O8J#xMSS`!&eDR$F7KvSPyCaOBV^`<> zOSN;Rkh3fg;Tf`*_EVSvt_RJP5|OjW9N~m&udY~YFP&TVN;i$$iv1uKhNpQF3VU&w zv#qlN4FENq5&TD(Z*_3>$#3=<{C#mHHjB6jl(}@8GE+Z&t$nu6XSD~-D|CSA8r25QMq;>+h|5{{{rVoeZAk2|2jr08-N!iYT9wNPK`4gglXI)WZv%O&Zas)R^$%z zN_lIpf|y08NLl0{+jrCtc|)zC-aD?F9_1U&1EO+e_s~<+BTS|l1y1x1EemjMfqUpq z6|E(Xqt~22T@!^hwiO=G`3F*Xf5(^OkJT#fl+%M9%oz}$1KKzZ2N5KN_e2}y1R~sKyx%uI45j#>;;aRE=#AnacE83Olfx1 zhuA7U5h%zQBcU71;ik#p~vS$%9>HpA6^b-0f`%h@9V@PBQB&grmC^|*e)`n0x@Q=Mj zr8jjef>d3&Xw6EoEvpl>%xK#<%w!#<@&G^ObBm(rjPM8aX1OM28FdKQY2U7DQI?8* zR|Z=5*fUUr>VoE+Z4NLD=)m7alf#+-W=+C&|I!>_6ZO(ASHA;Qs%!A$s0mPeWRdcy zvpe_Emd5olztqf&oKH|Kwg79j|B7yNFYW}jUpf_Ni6`=StP8i5Pj@HL%XM1371Yw% znu*hoHiPsS>KJmx>#tf3{c)}0?D}Q;+PGN#B<*R3N|z?$>fiDH;_%jhrz5l+->OxO>TzVgW9dY3&SW~)YM8u~90=HSH)Y<*h%pYtg{M4CBG z}My$oEpg)?3_kF&X$EyDj=V2J;nMc12gFFB&aO z(KQ9r2o9W%M(dKLCw88DtEYo!prb;!%^fp9dVTk08ZK; zBi~AWU<~%H>=?Vmw9#^P?{+gl45e!%8}UDu-y~)UC|PEmL9bR)mO}7M*-_;jd=+N4 zhJm%Y-847oBHL`qt1z9stm*=$(Ag1Qt!CR9O*-yF`kL$K zcZBBRizFIm2WvCVB}|aN2fQ>Q^&tc@8+*nrFNr2*smhUx^3J-Qx@gxKVQ9R;QD%_G zH7uVRHx1q7P$Y<+X;?7xMtE4>$Td_*Av(quV_(H(;vHvJ{8xUMzL&&34^~X_Mp>nW zX7QZl2slohCEjB#3^T=rY)5exSOVWFO0X5U9J*-UoQ!RIYgwA8%*n09f-5PP4I|(P4ErevOqV9_Vw!vw1Of=L{ z86FuiMzLJ>Rp=`*SB4ONcw||=bU_&qyPi2^GEfWIReC>hRPb6nRGY^v##S?L#N)2J z$VPFBfkZAlFFE|FpJah{L@WfKbH6DMvX+8dumy09^OpWHHz@YFyPY8vY9_Rn4Dy4@ z@Ys{?R@%GhEinS=6IsHU#cFI7d>Zp(itZiveJfv7dund443cV@G24w9E$f>4!u5N6 zof{tToqymoMr;|A@#n_hb@Qi?#=B*O2R<*@Z9M)h-^heK^tI7+t~)MY;Y(s4q*uYTt*VpS4{tlw`Ok^iJ~yb*ts_IbYz<>9rxt zEE#h{zNIbh_On(c)e@g5BYjWD{grX%i7umapU>$DA@hxN)pg_kA)9OW|J9?m{Si3B zcQK~!p88sv@=`P3+qp-4L;YU*FzjgUg{syWuRiAZxNMV8f9Ztpz;W7FIO$yNrk5{` zcSAP&<^}p^{6ob1P7iux)ZJuiBXj!srggX0+zfr<`!N;Du!NW-FGyBKK=hKdbK6(d z)kXeDf7I(ldSbaDy=ml4!plm!LE#Xzk@S+**qCP$CUe~SN<{f+q2(z z`9z9w=y$B{@b9Z>Z+4h7ilfIGT}vjXFKCDQzMbVWGO)+KO9gz!w1_(g!0tPJb?Nmo z=BVtx#QxunP4gEV=)dS@+7a1^^f?o$1I-tXNLS8#UHe$bJ230KSQG!rXY7)HBQ3to zc;mGb;VC_Dw(|9R6;U_(L{56=lgHBQHSn)(VQggdypQtjO)}J}M|Vn>U%qE--?g^( z%ILPWz4oLTSO1V@c(e}29*uTB`nsTX}iR?ISH?VDrNTb|XP zecCo-!I0bOck2u3&H=-H?=M}hl|65raR5Tn-CRlfpE(;3JReoD~$<_{W6Z%HujBDPBa!hJL($_bTzbRx?I za-t=Hq^@`7c>hVKxY1~lX9Tg9YgzFqpV0N*e^sJ)cBTR57PpEIVg^!)Tstnrunr7? z0`<>nTf%KfjZ8GO5&GbN9LRfk>zjeHBRWI zJ)blCSxUE+4z-#Q*~ zXQM8XxyiAviTaTAfu3oqd%70D5V}dqN9%9t$EerNh2 z?rP=lO1!KpF`^<#Uyg1M{pPiL7P`|iC%OZYYck}9GgW(HkDC77zal=`wQGM1{XG3S zSAYHi5agLtIouQEEf(w^t38_cW9Pb-<5jM`hHsqS&?oC|;^oZh>Y&Vj>MAqkRV~^j z$DWj^E%#z?>yl%SY3mWw_4k=?s%(plZDKbEf6ziu3u2=jpr@E_0sAY~$S0^2{MxN-*HHTk+VrE;o4vX*-q2nVziGln}h zaSge}V6rMqtWl(szJw;~?_7_h3a&biud9mOMcg@n#g6u3aSIhIGOjs)*S6#Dh&K!f zRN^h6PT3o#HrTw4k4z`ZX4UpL)HkX!_L28%qbq+Xgt!c@Mt@yuV2(*{NfetVB=utR zYc{4T-7CFWHITR$JYJJg zaNbZoqh}{1P)2f^u3D+l%}1BJUiwa_#?^*b{%3G_nbh*+uZa&5IHGf02p@^3ianf* zsDQNLKprMmph|cS@qhaCH4l|0q_J`8sMb-aGCV z?$1g4wD%Lk84KD!!EA_?xnsL0HYMlBUFJqfd$Xfm#~lZ0ro3Nljc0OYyun;q6z`$7 zC$2@U($;JT25vA2_?fWW+qZ%whE{d(lFp-^bFQx;{{frC+4?4QOH`Mv_A?>AR~ZU^2Rd==0;Hauc1(#$Yko1B}%gem$u+HTvHBvU2pwxs-mR-{}? z@+NMkE+!o&7;RCsyu3WNM|HQ99&u}JT~ZR_gFL>}&v72Z63kR z)pN>wR<279atW?Kb)&r7x%*U}!$QCFHjVowPN^Ch|5ekh{Bw(^qB`Bkor9zJ4`5Tn zSwk;kx`By%5?7dTDE>kG7Nk2-y8m`^MB-Z?v%jD`C!G(Ei4Toy?>*_LNkwbUd)gQ` zs5_Rgv)3=T@%_LK=`rP~x>C1ZuP~%lXL$>%%FFK&SLq4Z6f{|a<|0M^7_y7fwM&FX z6;0xPdDTfP-3t@f*bZyjIyS)jZBd%mnp6go%}N}~jZMsVbi`H=V^ixhQJ()&5O_~{ zDDZ=C5&fo&N;pCJ$}Yt2v95%F^Vj2BxxRYm_>_q(J+_&!QN9#*f@$9crK)83g} zmNQ56|70^UZQ*Ah!k(y&eBVi551*gP2zgb3#85gIXfzE+B+pT=AyU} zo_3xyBIee*76f_OM~mT3~-8H4+ zWw#LVwOS$`SFJ{2U1#lRe8HX|Z@7MRQeoo5>Jh$aH7aSQB8pS7OG)>!^_JU-o~Y;c zQQlwVO!91bf8eLJX$7Gf?7iZh?o?HtjO!l%vg!j*#Ifb|D?3*P8apPfPt|0+Ndq#+ z#-G-{kNX(gT^|~E%TN(puIpW~oF2knOm684<)5Wo_cTMbRfVo=o+C8OZHESB~;>eOI6|t=bOle@kQKV zt{+iP-bNciBuUA~G%d^x(&kcMoU^Ic>P_A$Dl7P}TWgx9mWYE}Udx>+bWYs z*M1EuC>lT)!M|H;w3l%mmI-HjCPYj|UM2U47|$g-!$a0cOv0U&Yh)9=i&AehA1c13 zmC&QhRD3UPaC%@k$=7+7+FQ{|K3>r;!42%uG*0XuRm#xT-8D}Q(+(KPmxhV$c2p^< zddp^$)3N_RS5z(XTl*H$VCR#k*Z9(IdpaaMDenHfXL@%Tj)`rdtKvNIVPbefqf}q~@RYckBXQqrj~m9rsqtA_v!-Pi6;?7+OXg2cks#F?2h z3n+qMVe7Ty8rSab@3rgdcm4o=SYBt&d7jVneBPfYP2O`sk$a_nrDT!U2a1nHFNyg# zwy!=*ITyWxIuln<`A4bJ?4}s3eBbG(P)*;(-EF##u2!Jf5|!YH_Aqo?Y$UI9Q@Q|b z`=W$yI~q5w5!=7Gq&%k4v*g5md56EISdSDo^fjrkZ$0j6zHev6w&KM9cD%K(ENoYN zbH~Wy__)pSRhtAg8>lBK21a+=c{|>??{IN&eDto{#mh_Pl>TUXKJIq$+~z~#x9^x- zR%zF_=vw7RZINM5lltu1%f1yi#f^=a0m$OXS{16IbQr9)qhV>aZ9CbcdRENHQfd3a zy?4t;mFai2(31jS0tsyG%s$S+e^yc#hE?6{b9Sh6vdT&1}|YXUbYN1`#1c&y}jm?PsL5L zyRp@AGyLc0bm!uBOehN2Rg+upypw3OWky<+QsN;IvaMlaDPoH{7a{APF?D0OvWRzZ+5w2XfsQ~)PpKkzi@$& z7!?q;P<8Hg?m1r3Fd=dWS=KZ@v`OT?;DY}4b2<%7XJKvI-lS7h%g8>ls;M#@T{3xl z-nRVMVHMuksD_*N><$TNa-msBtVJhZ&Ti6EtGr!q;%Tva8=X+&U|%Vwf-DN!8Co2f z6cc+csVP@&+sLB#yPkluva~3$a}OWHR+JmLtcJx&hl>8#9kercSF^HhJ7daBWpj(i z?W|d{t2nL6m|aTwz-AG}ElR}LWuP>2I_Tq_mb9Ed#ZM}&yCHL?dV6@{-)!_?C1{#Yn$9a zR~rv0s2tt6=;rPnT$Q-Oy~f7Z8yoh2PDs%cmg38%Y^$Ac-~v{`c0KlREGB5FWl?DoltXA-u@{n3Kit7pFMY08D{qoXolE?xysEVDBU%TL7N+x}== zlSYWsGxAsxcgeRkJxo?n`rG$rSSEzIbZ1L9s=n)0WkDcGc`pJ_tEin8(qdE4Ct zEPh?IBG;#aq_!c|n zZr>Sqs-S9d|Ac}Ar^7oX*PDtR_^WJB@}_a~lHI4gEk8suEjJE&vcHo%E&2AFSIIxx z1sz=C->PL$(ENj6&(1k;^y^3u=E2L{+y z9Qf5mZV@rXl|0wJ;NaiGI~-WkqW1ph!?A;D)94=e^=s6Rn7DFHNsD{zi~%Wm}CRGKLkZ#u|gcj?ZC3Gy+vSN$$B z%b&Dj{A5qx;0NaIrhdL=%m{Qzt76||&4W+Xdb(%Nr}Uxy!CZ7g?(X`aEODn%8T>eI zv@{B?%G(sdtM%+(_%6y5ZH+h%>{b3i{ldRXOQ@-QkDQ)pnLNko&-ucdqc?oG?;pWvZFkWBgh~1P_`kppNuzoh%fqhZ z?qH8#My58ZPF$ovfMqCNtgU?kyJf$^eDuTG+jSft36^So;63W7xlgS-+ZbMt zUE-eYn}LSrykTqLoxm8_7i$T)J}A2P6pq@ zFd0k>SIwWrL!Ld%6OY19lLj$W;3gS5BobE(E!BvAI^+E)>EidP^{Jc2=b;-NYt6kZ zZMm7=_U_rP9e9}R_IinT1w}+8^O-fXDPXuKssV0aT+mdVCKl4~ye+9%Wi2;NiZi9N zBHYE_g`bqA>}#$yQ_a?e{pPyhc@EEK)zo$>ujGz!j?xs>^?nPVherAQvhRv^Fp6(u zT0-BX)-e&!%wNWKhF5v%1$XH|RGfPj^;P(Y+}Jd^Ci1AU-Y|wjy945t z*;yC}!Y3Pt+HQ(oal109V4pIUt;+1mKE&h^TbN4nJE@+jA--?@KBc`5#eE}N@Wllk zh2`L#nUF2QHLMID%u86|mOE?~pg8uDG#QLMdR3mA>md`ztyy3?|Z_iRk zFZq?0EBDn5ngWvPt(ZOX6*n9E&DHYyfxkfv+gV&HUsnRe>h?M)TRuxPAbO)^WT7l^ z;j*IMp!!Sa=o0L`T$jRyQQWuu<)GH4Mf_=^$QmY`BcKo`f7YsNN8JzP9w9ryU+M$4 zY4qEOp;!UkWn&$795tM2Sh+LoB`_2&Kj-_X7WR5}2%o}SWESINgt^vI(6O*LUxnX^ z1R-2M@e6%3#cN8FpfR=;!Z`%lZ73ae-U7JQg)Qp;$f)HF!`H#Yz!YvMr_mR|3h5ht z6OW=!=fNb*3pMBNRjvq3ys4*GIV6gei1HHX|#w z3SMac(sM04OfSf0Y7hDoXajqjbMmT7sm?T!>Y=&9e^eq9BQGUFna=k1 z4MK>A-a!n3cf!^(A;{meg&78~pa@7PCxo@ic5fx$N@gFG%5Gv)nZKBewp-Y2)BtN? zylCEMIR{nSP60FjCsb#=+CM?b%mb_*-`dvGSMDvvN~loeqa85SSshO&W0@M_MWHp{ zT}u)_i5;X%%ryH_>{Elz;F;!Rds8w|nGFDntiW`{%Dr8^h1n3WLF7001$K?%VYlXt z@CoAC$b;-vVz{PJg!xIqSWEScCUgp24)uAaPOtqE@R|Of(mFq$xrZXnqw>2FEBF%2 z8BnfOb&Y51lTKKiBeO~3HF22OgZ!lc+pdtQwsNeXAc_rCYsh4%XQ=T&$xVA}>&bvNY>2r=K(y9wbL&c?JHA3A;_@!Q!Y?jYOBQ z2yQ<3Xttu|TshWV-lID?8q>WphHDa*?EB~(gI5#mcze7?!QO)1^fYY;>Pa@{wyJvx zD;^6ibez_N=mU#fqAbP|v1{-RVkj!t zV^U^l1#FzVOgXM6W;~=K-6N?rz5&LiAkf`dTTNWD=2@lSRoV(lk&DRr(v(>xQD zuWVI>5f^M}vKPMhUf`z#mb^yS5;nuv;2^zM>BN2~d}uPFY*Qm+03O7NT&HSbk$v@#5?i2Wq8YrH~=KLwn0D4P?{AyGwpHO(x6lMBp z4WJ|CSx zPt9fDa3_3|$<~;p6>C<7IC~kUI(XKP-=W27r>Ml7UQ(i`n#s@hFZG*N)z-4f z;Sl};!-$R46GLaMKb)v67e>~$D%0R+Q=(@nof^6x+hGXy+<@cZFX4jAtW2X*ra5@psG?)TVBWt(78Ct5;$^$Onlf^5h6ex_qo z9lj`Xp65MCQ6t%VV5@YE?IxV}d_$9^N%SFVfclIK47K&@=ZV_9h_qN7bWnrRiq^+tt zo(`itp|^~0tmhVXq>Mg7`VyQATRvGGSh4#S=VOQR3>=U_xD)XiV z)A(r|ha6Z#?{Nr*MG-f&CHOpIbM9|8Q;D=DQn$becoqA|4Mn`KKJhG{5kGBcYnyLA zM|X+{2Pecdy4@;?>tWi;hf03TyqqKA7Pd8L;_L5NMt{y8NvxN{u$?;XTJm7p4fc*N zmv?w~S>B?F)@j^mW+M1RJ+>@xZUIMxfAa3rGr4$<%Z>lWw+4X{Dmupd)a#C2*S;z`Rq(XeIm($j%%O8WX_-10A{f7yj~%btI8gx z%gCeI@$?{Om+P`?05{hWK;ilxCoi3pW~y`LA5@V2nm3heDfiK8SktH&euj4(Q4>}b zjLLK6bk;~dO(VVcsh)Cu_&uXFJ(}5vO(PkDllaH|#4{93^!x4f67rD zx{=s`{on$(XyGgHG}AZfZ{G!N2>k9EZhQczvaH<9B!#!*7Rfy$_VazvFsw*l5J|)@ zV~^Mj3Agt`??P(PqTG$EBVX_=SASdgsG*!s7%Y5(|H%`X&fIO{Jri1(uGF!ld4u3C zFvVa3bGba&Q+aI)qwR3>x>yt}W}|h)B6YD%%NYbdi6-)nJP)ySlGe#L0SdUk+|F&L z@sKxEd*vefB>W*CrizqO4fRGGB%hQWzUh9sRk5WP(}TY|HVC_2C%2_dc4M-*WY+r+YYM) zoptSJ^V!jIFv!l$&?bADsteF%e7>K?2dcpi^hD7DX8n7PJEPp;&a0H$z3DM zg{T{*77eyVpbCC5YDNg%6jUC zPEW`q8?)6n72y^&3|3KcTy0RX9E6%UtK|*yYG_ktp}Y=tXS#VFL_DXmVHf!`N`i;k zB=w3AMZ1^=*lfc-A;I-ZUhg4&%gh^;yXbb zs0^41{7kupGK;M$T|l#f1^c-I8M`d2MxWHuS6gjq`Qd3zk1u@89(6xee)9hO9Z<(p zmfIg&VVb~Y<|a6LTV}x9;Jo>~P2>7{UIt$!{lGl=g-H`Xa~nb`!*kaAY)w2KY=DpD ztdKe!A)dfaN$u4oL>`|aUSmC(mDuj=I5eKF;^;)bgBGkaI|rM|rb_3L1_$++sDoQugZnpXyBQ z99m-1_gddY^u;EUSG8^Wbl@*Gh`taML~f^W*e~w_2gSHtj@$2P1Qz7YwvRwV*{ysv zmd#4%eDapf+e}j|>nv`g1ujq^$*?g}2b2`Li1JWd*$6CzneGb2s%ypJ^}Nfp9-sqS zt$1y2`z6p8^#+Uhdd`B(K_LXVj!VjS@*_7;HnM*1iQXP!9p4RRo7BSAn%VK_Onmn;;5Z}rwiw@;+M)OEN%$0cAm763FZa_H!OF;^Ehi)*xNyFs zxwpM?%bY~A*g8~Op2V3w{x*()Zv*iU9`Er-e8>@5AS@A6yOr(iL;U)w0R7g8-@%AhRK zXQJA!iiW?o9KscJRmsTzpITGv5ZQaeJ4&btdy23L{TBX)I1tg zWoQ`8*Kmzv|G@ec)Fnc(A2x>)i5;absFM2|oUab&4$z^fbJ$>h2J_u{T@5F;D4S7B zwtU@nc`su`L$$}`O|O0foBtthLpyzjdZE`@Q-VJ#7hpd4NHta!N2K$y_LewEqSHVT5wju3_F|t-rlf6JD31Q53 zrhV1Fx$*dqi2h17^Hu(`GA=)E-Dn@}dA9LS%mUsLPpO5*HR=gC5iOS2G?+*{VD7kY z8J30}$5ya;*gEsSI;*mQ($v#4ykA5YdK>7U{{d>=*G#@!13Te)x&iwYE_ctS-b+#B zAtKwbMQLxnR=qF3UJfR{$Y!Z7ww&;%MpFy){$qjnhOjz3GW9oZU2;^;dr}>D&+eE2t`O@f5)$ z7Ar@f??giQ2W2hwOfkuwct-r-8jnV6%c+&VAUH?#)hZ;)aX<2OSbMBko&{Q3t7{fI zT)zv~ZcSBpt8a+6idWuXGvn3pvECDmAJa?9p&PT)U{jLk_v&5l66Q77&n`Df_-Se{ zHl3R78deBFps&IaMBYHz!d^C(9t2()=Xe%}3?Vm55;fa+Q2CL!ocFncQ73bI5KTPC zGU*iG68J{x=h*J9nR80Z1>M}3JlK>7yk@l1cw z$uu*K1RB=ITu(+10Et9P8? z8M%TSChu$TM#x$|4gZR!aAf!g0%=VO-}#EYnlL!%3elDO;4aH*gdPY(!zZv?=n-tXkj7cq<;-JvPB@^% z33=)x63_}bMU`?ROy5;Ts{%iOKGqH9URf{kX2fgt3cq$`53YoWmrrxAu#H?g8EWn< z@3x5cA;M&0vmjEPZR^SE67H<&Y+yYg$Dwae1obDvX@`a9JDhss89%M%C7;90D? z{DL(V{Keg}Rn-RL)5T!y8(mX;OLn3TdB0UIvks&}cnT{OXK1U0CG~l#1>Qnwi!Q61 zh+WhHz?kzJ*ojZx&AyIoJ#ctcCF?aDqSf3UC5BEyXUU_g+c%wk>D{#D9AfY?wZHTa znaQMa1F1fG)N4=PK<{!7^wv>N3TN|L+G^@v!ACH{@tsqhE7eJMo~fy%8}8BR{7bRA zDaH`Y|3ax5!{t@fB65%6kZprCT^Y|3SWEICdCS)&GoFvb+k0o2-+`UJ>KMw404qgK zY-+7ao~FmrgHbHDm~E!~VOpuS!y4PJV*lctxL;0{dh@A><%)!d9y`qU2o%1>hgX(Q zc?WWf+>^ioxEs}C8%1=p{vp)n)`2>P5%g-?AHE~uyQ5xN&%(*_IC&xAS$W3e1z!nE z*59_utXscX1``*hByA=07eZJ#WVd!he<~}OK-(^|H&HpK6nLzE*dN1XrX&0qZ$s>z zn*~TVsM%s$ast0DWFRV}-hd@q8@3!k^8@UcPMb@K?3mXtW+9MPhx&KA{7q z@#S!~`7xNnT%io|YSaYw#P(~mwF}foqK9yvd`~vv9!i77#%eSeYj36Q6|Zn+YqU<3 z^0*S&;n@4oI(g?a*3#?vEBqwNs4M9$v0Guc$a(5uBgkDX_lkQ0oQo5Y{w2(1P* z>r@aEaV-BIbQt{!j9i$-MlH0HEKj!sgSeG^C--9k7X$sL!RNk_L>?U{-a`p8lfT!n zoEj#dm7}zVVn)C~Z!}%Yx)iqO@@xpr5EBbKi<6bBSVnkVTrE&6ADCN$zvm8TuwD@j zzF~RmX;0W-Vpw5K;HBGYJRj}*SBQ0w$(h42VwCn39D}Xso8)0=RZdHdU{`i!ZVXML z0Q1niXK;SuYfoJ^l&q$)QoQnK)^6hs(-*cMw*}q8NWHTcj)`Q|PhYcS??eng4>*O_k*^zK}d@kMzaCTFhQRGrz<> z#BjKitqE!bKZMh=3$fKkFFMOx#Py+}a5)_6YH7SJ@AOrmFP8H{i22oqF&3np4(HHa zcCTrPSVvevEkmvCEU%M7WN0pO#jta+DCtw)Xm!3Yq;M5FN=zWnfu;O-F_yk3cg<_B z3^u0{CzT%fd&7V9d_-wMay0c$?yOeNOEq=yIXpwdw0vDz~Dep4wn z&Am+96%t3SrV+>^jF>`VQu z{tvc>kCiN{tqP@D!cTz9w-rKhvO^-5o5{!<;5c2?JuX-D)RigWn3`jmW2w`i0j979 zm_qjuKH5sSe_R7Jx6b4H<{4e@h%(MPgqmc0%Jro`n}1Ss;5mZTMqs~CJzd`(MNSnn zWuDE?2_?*y+o3h|o0!0!DVlqVP=}3UvbDd+N`VMyn_&`-e64vXRRr{ z*7ceWU^O+J6x7T3Lhnds8D|8go}cO}Z3@~cw-z^w3*?n_L*OuVSMu8cj(7=^-3=%kd)=^<^UK|Y0KL|=zhJgp%e4^QlHV(B!}GD- z)boHd)IZ`DXyZ?br|556TVYxJMq#tBlGX@*^^DL8(Mhn;U5mZo+Ca|~o13bj_w+BO zjQ*FL;*BQvGfmYpB9ZtA@$ zw@{yB^%NdByGR>hE8NdlgY|cIr*AU@ox7>SW?$|&YYWCd!YO|>e?%SV$@p9C`{rJh z3G2v>;_HgRRBqT$VIbYGV6U7_2ZB<^Wp})3w7or5AMfre0IAw!y_L&)$LvR)5)Lu%gC~(GS{tc^+XkwlN@6GQ9v|!-2#fhkz82OO z=okIkq(jlcCYUHMv8==IX-(J^>7{R-IRGpZc;GJ&hf(hN(51f+Hfah|$r58M$)Boq zQND4JRCDl9TfmRXd+sGdoF)g_CJ*7Bt5Md6_+`{DvV^tp19fimHF=Wc1bg5yvJy8R zpQn3{>tKh8Oz(T|bt+r@FJf*^JNeP(Q$&n)3zg4Q7S?Cii8y5X=E_6{VU)+sEO#GP zE(_@dgJOIq)i3q_%%(#7V2SeOyt-!n&MqJ`&yc$myr52qu8^>IWFP(z{7-7G^(PK; zEu>x&AnPkJ%2H)JIh@`n7ifLhK%3Rn5Y7eXT!mt+HiSJZe-U1*Rai!;t6nCyqEJk> zUpLj#ZWi_f58crP#l#+50P|_LVkcTCy~MMwpQx`#V*87muvgeZysJJLvY>ioRbQO` zJHt^4T@iL!e9Bc}-PUzNUuie$iT@A%Wh^%xhBhkE{7BkJrjsk>S!!3ZC7bA*sx~qA zVEgd%g@@c&t+MGZR*iWQ5lh7&pKB!>C(pPwJ`oGx99tW@R<1vn&L5CBsJ~T? zK57XnF!43C6!tSrfqV6LY>R!U@s3G>qu^0-i#OkK-W+c$r5~IBRZjBNB}LZkYx!SH z9NW};n~ejQbP}K99H8xEaa61%Xl^n=NzNOrnn}&G3X8(~sw?yWpT$xdnCu%DIagI( zhs<;a5$<&zK{L4$HBel~2C_TV`Ie&&zliEWExA3ahE0Gu#73zR>-PHPww7B;x8Z%A z%jL)1lAUyaKA8JOyJ)}2NVc0P+O^7>xMh}agX-azD4#$dIEQT^dJt3yF`(_A)!zwMpBiY(|tjv(&4r<-`$4hJba-9B=^Pyt4A53DPLr?N}c-uM|2 z68@&9+$iayWthB2*+6*<`hrB7%Z8*OLswOFy~jvs+B12Gb&e}R*Vt;o^NLHpEy-Y% zn1hjEIuos6yewrnDwKP8U-=Ho2C8W_dCxm1bSB=j!0TY;#q zY&fh?{{e1gt{R2Ti6c-VYKo<9sl~qnTd6ZjQ}=Qq({&s)64Gf=t?6oGXpFZpHP*j~ ziOdj{!SnKV@NTV#hoj|Nkw-L^B?Wg&~$?fL|& zK6(LEp;_ow;FcFh9+1Dv*#$$%8rUo?(^9DW)sM<`kqu6E-L<^RFDXnn&oO5augUe^ zzJ*RXp|C_g#V?Td6H}BFb427ChP9S~HsljfQ=OjO7$$&*o7ai6O@}djy>6=)bG$mW zKq|Km^Q@4D<5Pj%UQZsWjMui}dF*bXraq0h;W}=tU|!P+-jU8)@Hc(bYl68(f3>%@ zH+`4t$}d%owh&^CV3n8AZHfPpZQXXUIcMMl=qA(!-#vLBI!D9@W>R${J?t5>9Gj~B zFJg_iA+#v9u^hga_?sxEKDko;mZA#yO^wHQSZ^w7=c+osQtTP_@BZr(|BrNXjEBfjvfjRNshQ zR9W-s0PtDFSg+!{XEH;VX$1CP)>UqM!D(z67LT3P>D#M`dYVb+0n}hV7XshC@Fev! zGbi^c%yu#Id%8pRAz>O^z@~cIn(q+l=B|!6@H$mVdje$ek^Gl_pgAoagdyA(wYfY5 zQ8Gtmv4PZX-p_Vu>uhC4cqv_*J1cjg>a&L2=-^1#KkR(-K#`}03Ab#osJ?Ki)z5WO znQsj+UxBOmVOU$sN#cOm2IXPhd`&bOEVngd*WsJ_AaGRQ`|k$qsHe7ja!sZRty85lr$u!Z9k3f-q^v;`#30rw4+`!=jdm1=TydNwT8Jle zzfjAWSZbg9vT|NHq;IAiYG9$J!spmg*EdTSX^X`a*@c`f^&`??!@O2v&w8_gQ~Vp| zG}wn-llPG=GiJ)$$#P1hYlERmxoIuGmD)>=Q?IaH)X~&qqM!46*1u$g*50ukUYAD@ zSBMw%e)=5c09CbzmNk5QL^d@cq$BC(W@O(YQsi6AXYK{lR(CYp!#jG=BC*G zq+9T5=mU2cgqq8_hvI9c9n2KFQb*V-peA}Do;4xq54xK5l@uk;L6TXsc;vbz?SC6NFTv0`Vf1bm%#^E<~sxTyH^%! z%nfWP?xG2*JC;Hf8S?PamNV2uPYD}gnZ_&XRdR@+}S!1}f z>`#1etbVtnv~3WVXNt?PPu8E_QOYk*Q*Ed{)e=suVlV)` zPUBbNIto|b5rI?>Z3H_K{6gKSbCC<-2wp(l)HLm>HBa?pPf_==-OO!}!&kM(2wf~2 zT!ndA`ULx8_8e29HBAe)HF3lURp3No0$q=)hNi>G+&U`Bz0UMbd}pZ*H&`DiFpDQV zOl=7GI+9M$g?N^#F5Rz*^{Qr;1wBQTb0+5A8y3306{~(x}6;pti3_JySZcAY>orIt_7<^i=F6rg2Vb zqST(hPDe-$=|APMhD>gfp`SWweH}PmE{22n9=@jNG3CRTSdH8bJ;Q|ulk|RXZlr!2 z_)5VG;t`c;o(CDaOkw0~fC*D%AlNo;(AH4fge}Ty{Z(DY%v6v}f|j;2%QqVEo;t}~ z4|=0gERXaylN$liUI8><)CjGea8LQ4_nFQnc+AX#*S$~qL%x9#Zo_I9$Cg`C0TR;W zMwWaUXtUJ6>FRu@wT~GS!>NPvM7bI?VzDTIy0K1nPNgPW{`OAf*4Tr<6R@1TBn@C&VIO6No@Q@{RWZKT>-At) zWpqeS^WI_Y>wZR8aC$B(R(oR-Gu?EA2A~Sy!PY3*U}LxO>ySb?JzpJ0b}KqdRhaWp zR%B;wn5A*<3+Xz$of|+{^Ik%*uLeP=8%Kl?E z^5ba=Kj*9`uaS-#Ut>M6TA+q417XTQ4^|MYobzTVqdk>n3YJivK&JXb>t<>J&wxxh zR=DJy3w@xGD(Ri}Am0@b&R&+Jji2mLi}fTLRPr|6!!lgo9N48wzM;Zexh+0WXr1v^ z8Eu^>><3q{dA4)%Ekn6{BI~Gotpc=D*d{r zTRs4Dz>j4ah7QD0^nq>yzf?ws%BH=(Q2=)+hL%q zm5;6(Y1=INtSd^Hf>s8ni{r686_6=>KW)DF9G6i$v6BU; zq|YA?l6zwlr9iIO7o_ykzS;tX)vz3N0A-s;KsyTbb@$X^`%!b{llXkXi5?1(+384t zP0-B(f9ww?+RDNo0_)j{-uN1bH_$PawIQRM(g;WMz21_DkqV;dI2)p7w@ zjVKTX;iBA?shKs6YUn;uXA<3w2)n$8Pe%alNs?Ja-9QeYo>ySJm>l@$eB zm}XiZ;vXc#O+saPCt&$P<-Ssr8m>I#l2}SA5>ymN{pBrk*EV*Rduy{I)_}pek9c6| z!xAjP*Rzb|N3q7pszMd&iZl=1urHQNOl~-z2n1IwrMByiawZ*-`s>n3I)+uHF3Cjb zNG6u=iTT^Y%q@wK2u>V@E<`kgD_pmLM^!fOWzz9mQW304~d+o6#KJFz6OV*#l!YA-IP37MHdaB|itZJCQ9L04q zb!Eqa<9g~}7Wjm(hG}NOW}!x*EakYKMf8Eb$-Z=_31_{b{89RPZVgGpQ;8SY74xLL zzd;_>-EGacfFWpq{bH1iSD;eQCa!^Ily4O*)BYAeD*u{>aF5ko+{7G{Yjd@LK0F|G zFFQVFGs3^Ys=fziB4Uq?kFZ*1ZN1JvAmgc?rT}q0);fPI^ayMCedbBrb?zYUwnXD8 z;GWo1on(Dz`{+5V^L?DQ@9al%vOHcoB+VClqIh!<8K<6gT%~FD8TnFdM>AL$UFt2; zlGPf*0ArLZAGF0v<;T_p?4TS0qv_h71k~B>&q_kH)RWuKVPS3gN6dd(ku;px$97!z zRm>CL2$^Jxt_OCbTG_4=_vmHf6|WUuu zNo%zIZWF5HzD2%Nl2C@A!P!b(U_fs7QMsH*)%ro9pos4bSCVFUiXW#{C_nIH#8+zy zubKsdq1vE)QzH4dXtO?FJB-Ru9;)YqBft|7VLwO=R4T9=_DjDTiA99GP!l%IJ-lvXJ7-wo-;L7c#?JfMO{7HRa z2ROaN1;bV64yG11M|@?ip|!-j5OLNujyG5$QNm5dF43K}*qm(a6Ft>qB`lE!TR(X% z{8qOrbwNt;{YO`~SL5yUCE^Koz%>)p(<_;N)?ofPH6Cbm3$Nt z>Pt!d`aFN~p?8B?Tdjp6h#IyRzJ{8=d$3`oZ4lonyq9$sKTPQC2w;C=8M@nPGI^8< zlh4Q_MK` zuT%&6plJhLOzv?!Q@c3B}(usQ`5A@9E-|3Us!A=NI zn0j!Sw^+LmPs{a*AUfT01Vp3J*#Fe#Y%kkcm_t#{mqc5bsXX!4mh)`G)sNrnE-VV7)g-r1}Q%EEr*;`FSZ?Vz{rh^b&^uuGJ?5x22&Dp2Un zwI%{#eLP9NA`I0l+jZu#zRUF8LdYh`Q$T&%Nbj~dQ0?$!bqn!TNHpBAMg-NQA7nh1 zub>gQkzRo1)4vomtgS2X9od>xq_7+um42FAV6C}s@_uxHpN6_nAxax6UF{{b)E05C zK`^z;BwCNTrYbqy8%H7g2a_eg4-Hc4v-`Yj=vKxH#5%Jkdo3r3vr4vas4La>)7+h7 ztg(zM_@&r{pDDJ${D{Wn-!O{`p^xPyV?lwD-sjv!c@~@EPSIY4cT-wgQs|!1Z{0nl z6R-jLe{|2Q(~M!uJHyZk>b8CnuUk?V;vccFKGb^?7( zf5mrV&%>V5Lb6bQ-Oq|I&^Tqr9`j6|o`J5^mG=yryV5jArBI z=FkA$C`;PpDFg}JSgM0IL0F}3r_z{jXqJAGMk;;jJ$M&|q?Tcr)TSWO_7SxeOW_4R z4co-dRwmGlaEu;^KDkJFp_`|l7PP?19m_#8?jV=s_yhAZ`Q`oR{+vJC=f|H#_i?-a zziGqnM3t4Aaz}l8aWDTBrC>Jt85}SBoU@pJ*@yOFcDFFsHGw~nXR%uJB*J{*oN~3G z9hgg>r3yV>HX0p(93Ag=vm@}o)aBH5qK5NfTv0_@OoHD_a`~c$Bo?SD!keJmqoscXOo>a_}|$AvSJg|a_T&ZW9{T^)G^<1%;zdJH}b9m+d`^?jouE< z&Qv?!0kOB3NY1Cuv0cgW_+N$>E;;OA*cbW%8!LWroMMNO4ZPu64OEEM6C>H9@S&#$ zO`6-#HN*toPTKK~aIotauFtQ*L9Dfq?nJTK&`h3o+$60GO* zQ7B4rFU@VIZ8ZJEe;`H~o-lW9Pq}XS^_iWTCe#V2i(=XJo;AW;`s@EGIt%tD^0y1) zw8e_F1?p|Q@r))DX>nM*u(-RkSdqouOG}H?8*e-_lbK{9#oZkiSmeLByS{n9K&~s9 z%*^?n=Q-!jh(2?95U(z?sci=PIG-q9T$5vQ@Q(718JcHEquF>kN0q(AJktEb2xJL` zFZGYj1qt!BISI{JZ;ccA9XQ=ht2=j3-)kJzW`F8AW3SAS?0c+@S$^J;4BWITnm)_Y zpQ>W(GfcnJ@>zA+7Y%22PwsffgquHGuOHZZ@OD|CEzY@`eI*TGWmF|(PPEQqzYxd_ z(cqq8NBLCCYi?@Ror;N>@jR$TQB`HlHy=$YF^BklQ!fbG^A_ctVN?8c>y!NlYWwWo zQ1QS0^$B-aJB?#9XPTcyKVvDkpCQose&UWO!E}E0#2j<+HOv2>b-o*jSGx;gbqBYdKe$^3d>} z^_>rK?Iq8%C#8I{Jk%G~zT)%|%w=8W{^AVD*i{wNc}MND;&$xk#`5T1d;{+MN`+xI zk7M4fe}7;CJ7QYUcoKVVH8yx`@4#>Qo+90(9%fzp)JI*gMUoLMUoLy6_e_q}?ZDi+j zY*FoAlf8n^tuoc>-x+7=KNyS^4fuU4-WiK(K(@GcM%BbBX_Zt|nrX=l8Uz_=bvITl zyN(m89hPa&IBR@+@RKFqTvGj#gS7n>%?PH|uFK{aL+L-WmRmlt=F~?032R^0?%G$i zi%gf|7P4}Z`Jyq|=b{e8xg@jsg2^DbzzTF;z+P`?Z*0X|XWUw$6*&0@!9v4(cH;wg z%s)(DGTK#$;}>RwQEG2;^_-fR+AsWrTGJE7Sp_L7?rr@8 zp42oZ#az0Gmz28QbjVOqQB--3o2YNZ?^gFSV`S|a)47^snQ`SCvKo|iH?A@a&2G&J zRXh|7(kHXKR}VDLagb zH{MJime!MXKc%GNEoZc$!Zas)vml>WXPU$qJeb+TJ`{)f3iz);mLtKEDjA-JE(mU1LhGTE;tP%Hq8=zsZO-^*8u3 zn^>;v9fpJH*IB=-N~;^Qo}1$IQI(R|%Fs3A1@~O1M8WQ;m*`46%KnpGt^Xb^mFUr~ zm7!tg!)y_+cb%u$!VT$XXWlnQj9U_SRpEj!Tw!}%`>r)uN`ld+f5UP{8f;4q&y^rqwm+pl;XLoNM}NcwxFEC`pgkKM)*Exo>^n@w;WEc@TS>VU{7W z!w1uG(ZkYZ1`WTm{6+P$@_YOvo!VtP6ZB~hO&=2Fk=_Rk!s8Jjhtj z&9!*<{ZZYY^|elKp2WVy{?7Yz>vg@yV&M?Zr=JclKe+FxkikC=8wDH7v3$20eu+oZCDv#N?Yvk&%U?J_Lk)I|4eGR+Nl%&ip} z&P4Mm!}$leA#+B`UrBdN-&228pR0*wWn1?1|EwwDjcxzi0JB6XbECxLIjkc7cEQ(b zqj?sGF#jrFmO84sDX&At7WSUZNvxEb>zT{UfGv`(ijvrF)qF}lQ2j2`#Zq>R&v=*p zKKmi-QCUo7OXI9Gj(IkxAE&yaEMs+bch0-44f@H}wFzc^W~cYHV3i_s3cG2lAv0ur zSq5iyYd@B=FGXSa%Y25{zw?Qj!tA%aeAXZVlI}_F!{2#ufAy3MVeM$1y=~VrYqVj# zvDv}8Xx4KM8Fw0^u8AOjFKB2f=$|Yz53FmV|DE`alg_%sIb^P=onV${UN%52a%;P% zPu0g|4D4`@r?3V%i}}Tt2PG>Ew=I{gy{aV{t$B;MOX>dsb}vm5ZYnosdn#OfUg`z5v!+MdTfH>vMY}_+0k+DTohhfyrR=us z7bO!#nKltGS8#$gY}0^@DJfL?GxM~ZCRr7RAyolQp4C}i2d(A)9 zTMY>r#kIY;|8n69AU)6g-gubvQ?dIDP?AG=+-+p5vs2|Ur$ZB9H zN{zS7;g%V;W-Lg1#i=+Ls@}i}8B(gJnH-sud3xRuJ;?vab#YoXDCPaE@aZ!|=Q*WS zeaeR9wAnk>lo##wD9~rtHPk=W3!*&|>>7iyExW$viFr!ZKc@7IdV?yesO}rFj6B|M z)7*?p*;}&;tLW(2vq*m+^R?-Qb!P(z)09ulY_}sP;j*E5p0{Da%nUOH8a# zuxWiAtJpZTVly{nijC$3bi5>99LJFZ>$Mq!vYS?SH9t12NXxHo65Wsh%}JbHywa3S zh39hq;%8S~snKMes9eVOnkQ7p8iyK_tWQnP_$O*^SM`W`IS2Ox*(#$a?G=ffq1qa=J$qUP30pYGu91 zzfoakz2&wPOy~CP9GB9}6kQe-jxx$p_Z&Y)$@@X^!v&8J-uS|8AFJ=C(yqSy|&889c+^rZG*-XE+X+zSW{ekm$f)E$7ru=Sx!~Gt@QG?8CF|z1^B(kQy!wB{V^8}WHx2yWJ^-SH6j<*G`s(V#!@3PS{(|2NAXRzU{O{4rzS?#r2X^gXg?WEEKQs}^u>^KMu!WKZEvY-$&rF}CHF zvrku1ycyX~^~v19<~PRViZ|7NN4v0JWVB=tPyZ;;XZPgaG#QH^&AYb{CLwbQW|+E0`FXrF)g zsf`1Ug!ZMCXuBVLi*=r`K|7CV7W&(l9@)IusC{tbVPw0}8Xhhl5?D)#;K2 z;jwe$b*WGv?YgG@w0oA;gpbr^g|rV3Yc7uItKBj3XQ=RYiVi$_4()P_rL#524EF*_ z?f!3T!dY3bw5F0ax}kko+86UaYs%)Q=%g)M>5iQ^roAcN71EQ}T;28MC$&Fc#cCDeJ(`+TZ6a;X-Ph*HK<(T=FN8MS%hCB% z1GQ@sx~mJC?bF(2R&9^QyR}0L62gZbHqy0jCk|iw-8jK+m8 z0a(`(o)TtuHPjW{stg}ISr)z*NQg-O{HRU!b!PiQ(+UJRF^heLUbYeO54 zHPICQ-qO*>r%8U#(oIrO+KvY)$)c`+upC; z+$rnl4nRlKV_kJD1SjTzpwOs3V|tSDn|^e>ogld-+Z1k?8LTd1GC8`GE@U z#BZrudh4)o=<_bEeW)-z<4Qo=(+g^SpS9XIQC!61=QG0{AMMp_7(YroYv)~U-koLI zqbELVUoIP~QN8hprjHw@83}iagiSTt8O18?N0vif_NGwtA*Vy+>DNixZK-@s%`POe zrYRC>bZpYz9qZP7oBca{_F;2fUvEp@z?48x&^*av*J)Jr}e97D?Qoc+W9*}T5 zTmT#o=k8e)KI-6Wr8kPT{_%;C{TD;w`AM0&oSJo-7LoyxW6(8igU83!ra3plho|?@ z<_sLB?G!&$wcuGRUC*M;;g-gfNKwaHP40kXox|8gJ7nE8_24H8?YVb{wMonW3xBvC zr+Whp3b$$BQg_x|p*`Dvs%EsPU8M9|iB?}H4^Pb6qGhCm!-1YhwGZWO!^)bq;neJg z+5;(Fv~PYFY5Q{wA@82!;g}`=h2O<>i`*>A*8Uo_S+_Rm*0y^Y3C){uLffp(b@kT& z3d3ZdUE1mkT_Rh0o5L88q?@oMPCI7Z6>W9LH` zpK520pQ_DHm>+sPmeG3lA>mG^OT+y>-q8Nr<+#SGCA4pEuMf8?zZ&kirb1g#Hb-0X zdrO)E;YGW9H_1{JQ;62!1L!)k zhxZWd=|x?%q6pe7?qG#D59_WHKMDzrv6QJ|`; z7yDSGyFKf=x2JETcRd{^xd+wcv?FY|M6k%)+>-`25hr+G@_I>6dr~BSqHkm|!ZQ9X z#{!`S2+JEeObRg&&)b@Fz)O3Nl;wjq!AfU4-upUIxl#b zZU83PO@P3wA&AKEcg311&4fVDf(%4|QxOvD_mTT}EZi-9ThSPA?wbnd73NDW0R@VY zzyS6hk;&RbxLmpiTA(PCH$m^9`Ig)8397f?v}cU8w?0c!6YxT`J2;!JhZ+gPgeYqv*+xFmu^+uz6XU-tk5MFh`Z!w>8;RGzI*Nj28Iwt? zautZl50lev z|N52#Mb?Md2*BWm0&cnsc?Tuc@kn>*9|*uIfE$)>ws>@&(631F6+(;QCT55A3EtA4 z=^sxIre}zbP#2K}?un}Y{5s^bG|zbrs>F7xCdsEL_fX>~AztAI`DdbB#8z;eYdTj|d}MiGbeH{fmV8RrF}hCLQ}{D#+p%`g zfFn;1P0PKd{Ee@4B6Rr3jzclR?b!dirg#18$oyPVvpBL8zhmJX@~cCt>iWMUV|{lc z6YC$#X4n|uu?%+ADhMCnUYt>cH&rR1u{lrcGN2xON^w;_+NF^lB=~MZ(OltlWuZ&q zRgQ+@o)tuty+%q#qLqrX&iRxSO1GUOLG+dAln7=80Ni_!YQsN3-GdxL08|0H0ky0Z zd_eRW6)C&JyWFwxMp<_RXQqH1WSI9Q`Wv?U)ZR4!$OG-Ofa}2gblkre_=twxFY&(k zo4VG%R1+4|@uxuF$R)0Jz8LT+bOIIwZB^a8W?-Om0FWzA1%1>4#^cX}{!>k(q}8CT zi+_c($T?Z!_DzD1z=7RW(Yy7wQ~Q?JUq&wSTgC@{N^ zgD%fv*=BEvPi}hzlnP|*Ea?*QaB!x5s;xcoCQ7Cn2n4+Ot_pcK_zzL7e~s%HurB92 zaGba%DfPaP_W=KP^@8W{Z`x-=g-&sJ?h->r}2yy0Syvk4%g7KJ-p@ zUo%SAMxzPyBJY9rveEDbrdjTGtOf9^{(o3+UDMoy;G83i0ugd3gp+Ch+qvq99(kb2 zW;ScHBaN^%UJYV`UprqQ$IvuIPa(&RG7)@3S=Z1oI$8N$@&+FTwU%-NBmA?dYxHAL zCn^?SC`~6nD6^<#w3__QJfd`pRw@m%N$N!ebOU6rI@!+g3<38mhe&+5P_1F&WpjPE zu*=AAAVj#3xiE#ccm2h*sl79HWUoRg_lhqWVYOM7l1Jp*CWV@J8Aq zs6lWS@>(>Proay9a(@>i0A$vs!wRY^a||>IG&JrPN{q44a3%9PO^7EUOz2Y zsAFlaX;(80LK9trA|LG>$5c~?s72~tx&c7|tD>7mp2m-2j!8}7PE6CTMYbILfipSs z(SN$5Te%QFrdwSPM%wz5@I~$qR6(s2?JPP}^T_!-(3l!Q=CMAJAAMh?^L&&2tHF_w zNBICWQ|Z2Xut>F(Jc11Nmmm@-1sseo@SFfQ$@N$&I$U(jyeyO`PgCvk90V)8LcERT zfea1~CMu+VNSk};^7bk_u}>pJCcw|#_1ch9UA0jikK_0}_GT2wIUJ}$J}c(hBcg`~ zpJ8)|BAP8spj$cls!yW3bP?Kt8LCM3Z3exvlRyu7LQn-a_jWcTi(SnWf98yI{U`rG{L#`> z_sRO7YdFyr_V~uiR}l-`XFb257sz*O8=FsEr@IKK3((W}xC9;WH?tN{#rCN{9 zQT{=Eu6<$sX8&ayRl^Hbf<5VwG?a6J^r-v5E>9EhY9I$W1G|ZS&_JjZd|cNZQv2%b zMxkTTa+@jm7)=XYl5#1bs5Q=U*=Paz)fTjLgU=#mXf3dxae%F1fhL*m!)&sC#}VjB zfU&HmdXgI*wajZY#y^dCsoZ1luC9|G$9IZ1-?{Y3|m^AZbv z0iWWo4gPe-(VLa$sVTO{nmZVy8O6*&ihu>$4U9*1Op>L31^)IOgB@sN*L-@7Jy!LW z*#hb)oq7lcDD#n6iU@296jBH1LfWG?(=2ck^8wh3vP@UNuAUPJ1}?!W@Mh1>dRd>-bNEP)VRE9?U-+`Ja zeY}r>avs91RbuBdY%j7=d7t7T_mpw2_f>E0hkXt3OrKuKroD)YcTgAza)fHpbRk3> zW5!~SqoY(Ow2XSh{NqW~FB5Jc8wnRWzXG4-=XmkbKaktdI%zYBO}>_z2npphbeg{< zs#;Btw0tTT2>|E(sdblD3EdnOv_blDy&7dSjQ>dMvktuBJiOs&n zbS1JWa9lpeGAoGKci68euOQ3do!(K*{{qv)6R^(AEZJLpEp9{1@E|gRlFTUTB$-4e z1l90apbX->WQYc-b`NpssYmPv&_Y51_N78V5V@|Qvuluf(&qq*@jcamo$SE0CQmqL zBMu0r6R1T}};soHY){Ny7IrFkV(4Rd@q{ZFp-(ys;c66 zNh$4<{Bs@sTs{5GS=q2fc2ZGoE=E$-DfDpPC*d9CXw@%Z6xM|5OrKTq)bDa$AnkPp z{{E77L52MoJ(;@U>yNeftp=|N8bm1u3G_L!Cs0^l>A5aRhPDQ;a5~}nzG8-tCQ&;= z-AM$gCi8*ITt{xp$hgoLtkyDYImdAmczT3zOZ;~`4xxadjW8VA?Xs>oB;V{7Qm;jYq?Q!o4rH8i zpCV7#gWL>V1x7+nWFIgfcmmi-!vxFtX;qvs?lU-iO1nZg_+-F?Y@@}IuV=uGlY;CL0A>g~2z zTf2s1D!)zMTOyUk*`EP-@l!+=bKlVe5;7{P0r8&@^4ygF;R1|&;);JYEOeig>+n({ zRrU#SyA}#JiY7^0s|LE(Bd6&9-1B64UY2XCY@+I+4d!ES1E^w}`UHH@Iz7-=c$~0lw)*XhkzEZR3tyw#s=p#1)W^d|u^R1nSg2}_HIGd3d{R>CV&(iui5H-Hc*KS0q_bp` zJMIfqkRON~tS$1HmFHbp8{^YRUz!r3k+zT05#%)0UH&q-DcBq0kyiYF1c7tG9RCC; zLf@vG(#gJD<3veE|2jC#%nhF+X%Nr+wli7253838sg~EbrZ6eeL|?pknWPa=*i?I&E+35;2dVw&D(q z4}PYY=ep`CAPV$lGRFQJIu+%b%6KnrouMkcxxx?i!MlSkB!dV(sl$t0^FdH3ww1fB zip_kJaJuw!-DV&O=p;T>LFeee%ZftZID8C*md83?=vlHQq6J0dgD-$b>%gNr2_MkM=^ppDdHqBm9TT|s}Q6gJ9tf@rW$}L%)cHJPkCfrj_g^Rj;^DOu(D?A)HjP&PTC%%y?bTtf!y2(Dsn~6mp3M!`NGCtQ!q7&2=9Eg~Z6^3o- zTV#g|2jspU)Oc4N9Ikmslw$c&8r%T5QhE)Yp<0R_6JN;biSLp=CR-T-)H{5c-BP|2 zme>V|l#WAoim$*=gf|p7a-LIFXnVnD*<{Zkdo@u__i}YWTSEN<$4M-QwEv>2tfwh& zAeCwl&vToU4m?b(gh9p+&t%-o_Kok0;zGtpZ(pi0ld5`v?x&_pZ}|(6{d7xB7sY*X4AKUk;2DYh zB(6vi@_>UYdKBtKj{}zwmd+2L@9T!czf)tjfg_b5HeL_(%E2?p>dL0rXP&mMd&GllkI4>%0d#FC^u zhR(oNLnG8T(AZqL&xQU%O5~gEeCRG-i@B+R>P6IXU=~i2!voi_X-p}4%72QCWtL%= zuv6p^d?^wE8#%Wq4!}3~53DV%9pO*P1_;Nzm>K46B0NuVlzi-OU@p#RX8A@4kEiCsP! zb6MmJrVx{m#p+FRyJ|142B(HPqlxe!XJ^em__(H^p7LKst}-*_BB==eE&ZS88L~{h zGms5UmA_XdMwaJ1V{(ZOG9%#(Eydsa`bt}19qdWDa&0EuLA~Ac(ACC2*4bHc7iH5= zWQn+;K3}yOZpXZ~&*u%$t`sCz%%J`U4RJ0Z7qE2p&(5uqd}y77k@Eb_uq9Y`5QI;o z1aaA)FUiJ7DiU4K<*Sf?oEOlg!80`#!uLp=`kG&**bLrKEh0Y4|M9E^hPv5a9uODk zlPk24B4lu$`vMX)eJs^VxR;H48C zt5jp5S0fa`qv6z$1ts zS#y98_}Z~v^w!&iM#yKLM5zV;EaOWdbS3#)#SIJyi4|IvOqmqb9UYm6y8D56^pN_q z`83st$@VV=n;{ET8Z-k65_^H3^f=iF_$^YUX^bAI5yGcT8(=YGA(xWBpkFRg(8w%T zA66fe{SSDT)6u&n>fp5y5B5HikEIn*zN|m74M+gCb#5ojAcm6nyd%NW;-}zGpWhYZ zUZc=cLD?Vz^kqxR@K=f@5aXCizM)P!w>knjH}PAJ4z69EN}#!3fIFy*?#uEV^14h9 z2BZ^Zez05hYDYVlMApRq(%VB{3G)HJNLP0zX9w9$zB=lX&lY8SuK+Ib2J)r%yZj$} zsk}Z~`f*Tx-4`cM$oLf7YwCaD34120`Sc>#46paBF{u2b zK$5lcM?51j%rnW~MauQ{ z!j^cuQX{cRL`-m%e~_z)c^&BBpFvrvV?KdAfNXcIMueJ)IRjKKMuZ@`qbNzW0t(17 z)dR1FnjXBZY6+hqdP5t)o_@wyhQ3xpf&&=hG+<97uU)kN4`F$@KsAg!p)7FJqs<-V z@K?nm#cx+wT8aGyl?Nswr&M2wnPd}m8x@oLoDqhPXSD{e(k{duy5`q1|1m?M2(Zzc zN?$``B>Db7q1Kd{CaG!4^59-iXS5KxA?OH|69%rFxvMIYsOi@keLM-scuil=Gi50) zX2y9}(g6E3T?C((4U@-{gNe}ujTC79l*h^vvBSQD(KkswecSOl2cizsGm-O1#C5^D zSvk}56Ist_>1WgHV>`6VX$>HE@C2?;jC-$1B0I;Gnd- zYa4L{=vv(z>8{Qs|H7~0@314vNKhAJh!Wjo&1cL9OI!^jJNzZeBH|tO&6^3n zcMYd)nj(}A5>#1eC3%r{BO?Q=slU)AoZ(Tg^rz-N$_?JODAlE^MfT^qFZ4$eK-#IJ zLpy{EHlXLxTj^Q0ZDAMK13ifh3mh>lRbCD9VW_kt&_kF*atl&V_YL!IP z#&wl$gW$*v=`a5u@MLTT|EOxKVv7`$yz@47y^%JAM@Tl2u(u_&X&*i{ocJmY^A1BrHYMULFQlXU$mNk7#Rrv zYtq^_MhVj`%$uNJ;3%n@xBpqQj{-kK495qF zYW%U%1JrG(tuY|k!mk7h`Q=KRccY@KBG%@Po^d94Vr)mrJn2s95K$QjiAH&@!Rveh zT1_NFt^GIjtNa%P*StQcgXc6|NF2iBC2{mOpw!dC-3^(EDX@9yn7UeK4z&~x(-m-n zs*`6ZbSHQj{@~A5wS(%x>za4SL*jtyN9YKBiTVU~gk4yzs^0xJB2c|CtPHfGb0IHy zjOc=PhI+a!*f^*|)(OSlWQW&r0@$E`$!P*_cdo;iy827%Wv{uUssTqkJ4y5Tus+FcQJl5r)WwR9xti6vy>WOy=N``~=`i*e`F+vyHn=pI5&*>6$t#%-Do8F2R zQI%Oh;I3kD(Cm(}=>kHcgdVNz==~?K*DD~_;hRD;)oFqF&|ksD-UP`x>LK&NGYWX_ z^0+)a8^X5T^NyCx00)Uj5lzS&Jk*ozSV?t6^KGe|YxYKZ#uA9#rI_tt8j@2T%ps#M>Ef z@BS}qdh&3nQQ(E(BXWeEB^=_7kPV63^2;Eu?CTOx`viNU)7FN(WT1_wxy=PmaZ-vH zq669m>*;A^c_CrDuS*{yAHZs2KJpYgz!M;?u~x2Qp3laQ%o#}$P%es>l&H?KTGDH! z^-3$cnN)yhl*5VL?0Mz_x-YClexNbRr;;zA4^Q^bmA##Iv9>NfsUQjRU606x9XZ(_$~N))Qk%cY(Ff#eaJ&{yxD% zBqRB!(yUqLps^moZt4O&R~__+G{30-X-^6K$|j*HUZpod_s#xHr{k}tM1jIc8{Ste zP4(IOfC}m^@K*+8!71KEYa^nOYa6&(w^~hv?zoSItJQPqCb}`69>h;qhUyyD?XaA< zL%o4cD{FO41I_(i&>O0Ep$=O75a3wG^oC<}D-}b-bL*4nRw|KZzwIfI3eIK>c+dJi z^pJ>NcZ5;-9+%HFF1B9+p+lpBO|j#MoenB0fa%a+_hI@rG9T-gvJzg07tznnr@=TZ8sdy_e>onec;-!yU&F4!MGU zth94CY5P%JDi!!k)QYpndVzTWgFnIRMTSy6pp2XiV1Mfl z#Sp7X+0eHTI+`;Ez3P{QvZ01V?4dc~T9nLPuT0Z6-MeeCg6v#=V z?9xqi5B9>ebkz~hOhE(1=DGw$j5W~{>zzZjX5MH5jFIlD+vUD#`y@<*hp8z-Dxa-u zOiv6hmh@!)(j=$^>dC?z%reM}d9cQ=p-5{529$uvIY3O3Bx+54ca1|@kmJZ{vM$a?vds#K^sQvH zsi9vD6o9MCZb)MkgVD=@*$M%T_-lO+QpX?-kiFz$YA!Vc>_>u%UvM*uV{FW5qWWvJzb&7^xPu9kkb@H?L&~8;6oBn#*>$* zo>?JrjQ_0tIIzms5M2!{z$ygXiC$rD$^dYH#xwmiMo$V z;mg88&@YRJ=qPcx*!2&3+cQ&2d-enUsCM`T*g)k{3yB@735>2z!6c#Wg3B%WfEG=m z79e|-I`Lp;+J39=7csT2jZckz1QMiHe5A6M*o@W_8$vsM66Tp;G||(Z?1s_dK14h~ zBJm$ov>>IPEy8@_lxK(Zk@&d%E58kn5`VgfI&k)FDQ(WgWzy69BH{h4cF+Og3E2i%%Ym~&jaJ?)Ub9szvRHQay2^ID!~c~aNFw_AEo7UOzn7ZSyEo_M3Yme}Im2#>0uQx$f&BUSWQMF7|cciuhv=Hg2FuHi~ z9_cx9V_n3I(?(%W$pz0CVzYOSzm@tUBm_EvmmST)IC5j>om4k-bK)4(lirPNHEDud zb5?@y+#@{kQmgB$Ge)ry;v<(8?L7Z^|Mp!~ZBy1tmm?&6iChbB^5mESda1XUZ4de{ zTR=VJJp!EGN7;i+QUxcv+i}Jf=j$wgpcny}Br?2-OX3=a&H+tzhsYJcC6!LOj#9cz z=rz}dDD5{!wL&&Rd{g`eZz>x_xMb%7jhI|{1toKKx8jQbfe5~UveN>=8PA){SCX(| zVla<*=U6B<;WvCd{6o%pcQT#k-9SwuH^Uv>e}T=BEKtL1<~r?$;E$b3Wa-X>?jKeE zlSmwFZ<4vAKc>?^08chcJRhuM%~$kO$oNG-XDcq5EWyM%iethTR26wrbO*>pO~4}i zZ}S>y3i47vBFeLm!__$*$kZ9}7HCmDtDn)sB^Q7kn)Q zad-#j-|z?~mznD?z%1b_)E9m#u`v*!$NK`Bpy&?U2>4Imi8Yln~Z??mSeP#4=;8N8JOfRVqzL3Oua((Np{Q(%u1}`whz()6Pex4@cy+r6ecU=x| zit2Cv1GK$*q-7?41{n)<0$|x4D+txXv!R375BFK{ByvsK+~1Wv?ra}&5|}HD9Yv?r zO~HrbU!f1uzad0q@cbhw2($``Q2-VCq~7{IJi4gk9&oq*-YZ})v59c^OUuAHen zQoah>Pn?qj=2cWvr4WDZ@3&jYBoGl$Nlgi~K|B@#HAqxUUKP|4*}P9&m+)Vq7TAsU zB_0WDv&_^2{DJ2_!cJW&ZDg;BBKx+($E-3~3v2;j0JVzJinic*2sid}Y(RcXTLN8> z73mv6f|%*uOC@KXbDt(*-c-uu<3lolBYh{_12xBM#YW4*KwsNG#9m(y?GSsODaW!D2otLWHo-B`P(xMn1^j99RV6kM9EO8YC*72DrbF1 z7TC5L&%tRaCGaa*C*T_W33y}a6~eF@q!+&z{mm>FjkU$Oc2jGR*I;zoDLPKn(;N6f z`gGs~@R#F}yN$XrWLCCQua#ohB&)!KQq!X+$-%A}R9Ep}d``%27%XcOqS0d{qw!!} zn3Fz%wlytAzK{fRiOEN%0@ocv%!&W2dZB!Zv*|W?7X{0|4{aGfShqZ+3ZCKYRDMS@ zg9i{ZG8(I}X~Wa~eb5^D7oxdw9GZom5kx1IjUe2CCF#|3|$tuAdU6cVsk88!x8yC{10rQ23BTrW-{gU0N)%p8|)JGOiu@O zTNJf|JNj~Ip;9Kb!KSlSbp0wdw`pkDS8x8ZN7Mc@_LL5S~@ zIobB1qD*&N1QZX!mx+~>rS3Q7B8|vUrBkP28fyVylXkXRNpz=t;bg2g-ojrU`YhW- zoY0KXa@7Tq=njdGJk&M&nf5pl58UHO-kJ@VFRJXE)icnw^58@$lEH7 zZ#gNY)zE8GbNN77zT=c)hnZnkR&@koc^3e8CkD7_nnP-=I;63;t#XsHk9RHD8+jJK zECNG=^B(FrS~tj{5&~ko z9LQNn7NJJ~8?e%AaXytibe*EkdGC3?aay}Pq8;we(QmiRb<^dtOmnZ5%oA>bZuw-y zJ?VRE6bo)UA}X&*)hB`^9_wZySI7w3^&{WupVHWY@%L0a4&3ZF1ivyU;YP zaU60l!JM)cA**drfF;QCha_rlbL&$u&bh|l8j+%j;3%{U<0M?rM=#@8>F-0gp&DRc ziPyM=Jfi#>h!lg~rJ#j^@m0W8|2=Fu_6eG%yQVp!jAI_y%2mIl--F8$l{OXd2 zs;|M8s>4)w<|FYl-Du7b~o~Ynnx)~k^hcsC-RBRAh)U& zstwR%Xb0?aG;m);<`Y4=gAxj!_&3@KXceUJ{>j`1hTw8UNl#Js^9=X=5?*lj1V$-? zh*b86XoesTosGN{@0Z*}c2Fz5e@p(MEdUf)2t0A01(sxPs6)uvNR@j%F&iJC$d+N? z3R(s3_Eo{_MQgwp@FGZs50rKSwDvCO5;p@*LW<#od^Yt_RtHQZ732fSF7X1~oLU^6 z*)CUHz-wK{y(`6=89bzRpQm`5bPUAw$l^#!*x^&-3K@hfqzuZN+2?=*%4_usMo)c2 zQx_jU`cG1^Gt*y-TSCrBX1;bK! zGj0*)3cD={jW(rzkoO}6U}fYcyW6Y*nVw{E9~T6_qh4q@-cV48jp8YTY4dx=ssmiD zwEfoIN(lQ1P9cnX;qOQU@#p{-NbxUZ9YL481gWk=fw$UyPbvRPQH1~K{iqjom7tY; z<{rt*bU?0=ek1fxAdA@UHgFfm-;~SskhtR*EcSOF2qkM>)N8`Yz(uUPV_qJQN?0u{ znL6Ts;HD`P-AlPERaZt>Doa14Y4TXa$}9`C2Oct$te=!9qIqzwqJdTIU)8njF-p>| zy4s4R`KNY8U5R%k29Ph&lWL6R4fQ{15!^%GB&^Vy$Z@`pg^~UdOmSdRQQbhMen&#c z*;Y}k7uv4>$KFj61uK4+*{qkxL~^5kgP!VdC#@Gkl!qLzqiQ%&nlpshsj} z^nYZ!`@ge|Z85Go{A{v6eLLr~WrO+|ybqJaPo<9(4UEfv0B(ia$)Di^@GtDBYD%ew z{D8*bldKV3GI0T$nHMX6^+oXu$qPhN64ZJVTd_5XQQ_!7lUErv%~+ zybkTauQDBizj;qWlO+&p&*$Kew6^eTDh;V7{Yw(QAxKBA+kqV9rL_=mibQ#zYm=2# zaw92~Xotq5AlWo^xUaIbUp^(j;+uLNxOWkUl{DWHLNIhwXQb^|Jl6?Haoafu;zbq< zE4@kRYN;+@0@j!-Hrp-kKwI*Mgjm#)^4T)Z_s2CYe=7c2uE{40QFs(mH)%ETO+6re z2m5n-p)2MCNl(o7ye9|GB7fKqY*|+_@>gP#WfJ__G*I0tRZ;G-XRtO>8`ea>$LE7f zl2y2?dRpvE2EobrX3tSMne_>Y`BlXt(*dy#-&^jKT8u9z+d$LcI|c*g(i$S&wawz< zz!|OsHiCwQ1n40$!e0_@65K$*`b~W?Ge2lGmzPX>Pw!y+6rrDIapmu6_g*@&+R58vgs>w=ku_u6jrlrGT2M)Wb3Y4b;9$8feVb zA}2#-U8ALzL^=2vR+o)y9G$YwoZ(L4>LX)Fo^6lyPSOxtKFF0u<`R|3!RRq;9+xKn z(KZO9u?@mD@H)HBe4d;QW=UOA+9Ff@eenpl#hT|vpJ$G`au!FtymZM3jURkg_jVcs5GcU{T*XUe?cn2?=Wp!o;J@kNsA1`r47=X zJFj^9`%jZjdkK|YGCMFMxK-OGtgzjs)^g*WXF@MSaYd!U0@bX1W~`~H_zALwf1R{Y zd7}IkY8z}Qce9NxI&9k%GogrXDTin2ZNgvlPjWZ;QpnX}FeZpneyp1Sq=P8>S(KeVBCVb-gyx1Tl{|$Si#sfv zQtj@eP-Vm$GdgZPt9tGibQ2@RHe9LrA^1D6f?*bKn0Y+5ukgSl<~|WByMK{gybh%f zIn{Cx9hdl!Ed|9%$Ax_1E1(FRR2b3=`Xmg*q$E+#hL4P4+wYQq~s#E zgv6?6AS~UnUza%QQ8yl%YGeukkFSc`Cx;3@>u%{_kqEW3ioe z#yIZ*Y06&YS3r|vJomU6oVIQmOJ2AQ~Tzv}PH3doGpr zKW!HATYO}#Bdf7& zUvQOvEZw%94AfDkF*mtuVnydAYKAh)_9|3?YM@uO7ZA;`1;l4v)dJWuVjEUIXb(J= z>lSabRTWm*STX{5oL-CX>|ad$M_nab2SyT;ENQe=JK-4O-={{ZH#mo0k6xN)ayaqw z{*u5=b3dfm(G#mehxHF?1ylD_fs78;%-$0^Nc3`kNWvX0=_rZyUa+mxsuKS@Y%8j2 z8%sCwrwSk01ZB^DcODP(5Pm#cj)UfssSD&x!|E2w>JR=(d+m)W;9rA;I zUt0#0QoCd6YBTbSA3^K=?FAJAogkP1j6fYDfC4YhI_zIP1r3BbUfFdt;hVD>*5k_w& ztOR!)YN1A8zpV36FLFV)V85m2=uu>k4%-gv@7%LY!#pP7EbtmPI5wzRs1KY@ zO-84nKYcaKefi#j8bF4)1RqQq*GpxB_gi5-N@d$B)0J(zPU45q3hLp!)zp09iE=Tu zJ@$|^hJ#yvdlRV3C1tYjvDi63r{Fppz?&9ECj13Pcvcx` z`<{Y#$XfSZw7JJ{z47M&JJeemF6SXp;wEVm@JpzkG%UMy)_p_k-LBy@;WGau2}}Jd z22%IsOa(9U`%;Hn-Xq1n2oY`I0e|U7_(-iL*$oHaFGAzg$5MIas!ATJl&~1}VZ*fA zN(Z7Qb(#Mu9mV=;lcA2xDt~V=Lx)o?=yhz*xo7e|+e)_&Ur)~3}<5O zZJ8=$b?6*fp#0(vGEegF1N-D2?i$KcrW;=ueF9yQ8hgi(F8>edD!LD8qDA}uC~@vF zz*6kJJR@Z`8%aO(Uh_}EaKS8`CQm3|lvfHBKh)Kd|6933Y}HQ0e1w{aOMNGGi2NV< znHKRFd3P|ZR^vLUNBzfq6_g8xj^P5^0c?*Q&O58NTEOt4E`j;UE$~f&`=$*3=fhZ2bi>?lwR+eA--Q&jYK;RrTTxF&6lTa?|@Tq1Bt8NtuJN~q&%Yl%TxdE?{X z0HZL?TPT(jBG_@@V7Lu1*3d&NvK|I^2p5q=-w-k6t({%b)g9{yviTU0<*h0WFM3E_ z%kSa;6=>r48+fB9VN2;awN>b}T&^fDtsK#jIBUNVrXxV09m<7S^PhKEAm{ZBuC04h{qmCX~6DvA4I!T-G~jr(t2Cl zZuAFVk<0>SQLSu0?7svdQJSg*j!F<4O0?L|JeRv zo|01nvZ2s8xE)5f0Cmi;`3E=2UO>f~F6)!2Wi-U}RI}x5B}eE&Eka5QJ|;x8PI?!= znC|WVB5szFu&_M}vNKn;7J5Jap}CtL+gK!4Aggmb1V3Ukka+ZCL49o@@rVYMI`%Kk z+}a8Qojr-KvQ@=YFjXNm>5Zx)-UO_x<$?sW-#Y*DcD@-&lLtTsO`#J;Bt5*aU>!i-RW)VUZ!7jyr;f# zebf{BdB#DFWQO8*)MZL@Y=-x@E6Wwg^FXikMcN!HQz)wsb}b+xQ?p&Kkr~7?V=q+- z)eSUbN-2>_f-56=g(O+tY6GAKfv4Cb_?UY=))iky48ms<$4O_P2ig?}{VTZnaEjJG zpCO(j*AuI$d8C|t7AbE{L+8YzrlCFuQL{jn<_bTB4(>`so?J(3NARxw!mGHkcm#Zp z{O0=sU-5KRBzU^{8FEx9_D*sAXL(-OK>H#+wSIQj2i%||Wrm^Zd#07Pt^tE;HTwzc z+$6>~F20XGk4hnvk$Lb5YNW43?#5>Tlc}r3V6-~giHu3Ctgf=P6Jc^4NJ5zDJU$2E zq<7{bOH|%8e6V<~$y4eqnnW%l&+rF8XYdV?C$@IR>+hLH%wzly%AnCj8@WIDFIiK| zn}-q)WSwpat>y=bH^DT+#f1g>I~Kx1%1HfE{5`rgl~&Y?&flidU$aL9Zl|SSmFc~X z+sFfXk~>*4IYt}!Zxer4AWMz#uQ%L{j>=_-T1{1A;Rl)%FvTUQ4h^d4b?*v`c<_g0gfVi!!htDbv4JBSShNGxJ0l)EwcD2IuLU z{por!K1_{r)G8^fZw@Cx_3a+J8nBRSMU!?BzM$2U1!r%09{QEbVL%-!s-|q^A=gmb zxIlqc2U%pVhCh}DaJ8YK;v1}tm>&NbYk@Gv-QsQRZR7}5NRCn6XkTQdP?e8Y$3X#P z3v!sg%?`yfeZ1vXi%;qz5&ORwNiS;H6`m|5Hepr;#wxux1&S1mjv}k`N9E~Elzq27TbRa= zVt#2qQMVxn|K%xvhP&8VUx8 z_RSV3E$FjqJ`_oL625S^$wq+^LSRO=oRaZCI*i&(4OkXRW-f*1$u&ZWbX`siuS#2{ zq}tA;bqss#?}8`8H-n{|+5WZ8>fv-}UW?^P1JW2+3~niMwESgzSN!kc6upzJvV9Tr z*%_<0*XR}-1%{5+EO@*8-%aY8;M`)hn?VliFsYM@mZ!~sVesn zp5s1Yt}BkhR)hPsZ6YpwhWBH~h-*f@Whj=NLhZ`09!RAN$#%?-aGI@jpj?!THiSIAE4K%gtKMRca#Hb-PD#1Qp~>zFc<{UG#By%xNuwh7;J+su3AF-=G;&R*nJX(T$u0n6yVBf)H99_rSrB)kglj}e&{a*F@DFcz%h`kw>U zY-D5VqCeRe3#fV@-HTVly!Joz5=YP4rPTQ;X|^U(UwWQgGgQe|;x7-_@)j1{CQqQ{ zur^2?f4qwMPNRd6)*d%nj{l(SHarsx(WjoH+)J%4HW56pWW(D%4fH~w6Y&B|kvcQY zHIXV**JX}|HWpNuoJ6VS5)orNQFu=I?yGC({8JTzi3AtP1@@ytlAO&%J9aoPmyENS zv9~@R$vazN@d%yJz;L6qy*fZ^U~MB8qfJqX_A;2rgoO*ij`m!nOiH9}yKqsv z!j&t&=1KsT`yP>LDrN5ydc)le3<^{7F5;1M*6w~rorHxQ%tZ0DqgGKTTi8A()Y{Og ztqOh!KSWEEisTXRFdfpygeoexyvag=@w`{aX)qie^iF#J(1_wb0Z9FX4-Jh~Ul-Yp zE|RjAr>~Kv#eT^eF-ln$?4vbQ!7 zUnM?Jwj9!sNO>jlsqjknG@=i2+4q!a$CVR~f&=(YY*kmWSWCktyv>g2!-+;ixLr9Ki3EvmJnN$ca$nS%GN5*j(U>5QPI1QLUe&;yqI6en$ zbTAtG%a@i?KfOx$XxO2%bfxeDS1bIG;lF!Neri!E9H(~{C?+7EahAuko&9p>IUA(M z5v7Sy_Eln@ttb4CbR{q@FjNKUWg?I7#}m>fivaW8zL}buzQ_JHez@(Hc+Uxv7kB?K zaGT}+?aFkymi{;LTl$xZA!m}^_4#0|)E@mJ%vTOlm2mN!D6)DBX zn`-e9dO6EFSC9Nk=)A%nMjpu*Vo;!hG(^dk=1cL!IORUGh$;t!r8!bnTWiMc_^iyw zkCKWw0k|$SFuY|O{WHX%ZM-^2{fLjJXW*X$>#XmId&>I2r@(QtCbUnxh@bXfbQIGg z^t;R-ei?*;Zqo)=oOqp8b173{QpnX5H5H5!FmSeSgQXQvUI?gXiASlm4P9mn>jI#= zx5lBC_!GQ3w;eiRJ)d&hz0zQ9{9-y8$<;>QA9NTRw}&_i#=+Z@N`#TNq5N!|Ax=P6 zr6)B#Zx$F2EXLas4q7(6Qbigmgh`993fnhj8o2@+m{LaBC;X#i2&jJ-+rzudH&*`u z1>n7hiV`EC9dr(Jg9ez#Ejb zy60X;?Bz1CdgfKct^76W$HLY?`(|x`8{}-F3v-LzDZN)~p`(S-NJl9{{qCJq=pjn< zY<-;I);p!Pg5!uOZU$LQ#LFY0>qs@lpEyT3fGt%rxw7Ocu$C)-8Z3fQqYa%ID&ZUUJe*O*+2 ztlBrPpsY|9&-3kqO1bXB(@lddrM<`FSHzdaj9!(S>se`$gkwsy*MW@De762_S=&4Q zh*A#-68Qx;co?57AE8e&4rgiG#eAFmUizV2l3vBCGDk!O6fTRNz;q{8$zSl5n4)e} zR_IMrza{k-i#%S+Lv%1s;=PJl*hc?LOtwer5o#h~O3FqSFinjfbCF{|VO5fao&Fib zQ~xc)nOD=6uO1A3fqLTEWIOq~++4CCL#0RZOm401n)Df2>T80W(ueEi^m0NDRm5-h z_Cv|Ls;cH3OLAo zWvft2bXiM_ZLL3%egY@upoiH#g{wFM(B*t=H`1-y9`L8|N0^TG1X$4(v4TluBu9x19D~QM0&s zLmN$`?iTkXG!=4{bJ%7DFnv+)10E?v7(NH?ympLUhfnuylGdv z?Xa+vnHPqGEyCArQs^H&Rd)o=I&TNA1Y-FU^1_n4ND}j&m`Kc5-Uv~eo$cZsY};mk ziq00=NxcJU+$3xabv)FS`QxpQoCM3E??o$DK3M3V<)*b_y(hiEHCONDsEPJa9`W&7 zo~D@IZ$2sAJ=lt~Y7jn=`175re@-O=mdFni? zlHt#W-Y+fCDKt6MsNf}fQLRILcPvt0!60Ln+<{-xZ9LK`FndCy^hO~!yqmb@uWjj~ zzp+hW9;MlZp1>&OzI-vz7Wsfi*kjS>)Iy*bz{!dD$b(tzRov#k?0jwy300gf=0^IH z7Tw4PB}2lk17`w_Bny!Rm~79C?5?sjsw5O&5E^IH*8Z3KE70cf-oW?3=5Ql*Ge1qL z$TjgbpgK@J)hD5H;?)*q!mN6X_#{5GPxNgi4?;7=Z%WV&M$I9P5Xq>mTzeLoT^e!^A*tKvLf-y1!arH zz=_EAF5LPJ{>zjCv_-CiC-Al2XVP6)cK$@Hqg6K?0TX>o*q3s$v>KhIoEJLL>&QK> zVy-!!nZJ`7fwoV%&mRV8;R@Bxv@USOx0fgpw*ooP%%qD_o~0~93k%rqx<@Wc#;Oy| z>cKnKzr@>wFFxA3#1^s7BexPR!VmBwp^5xM7{;Y=2v<@4tQI5x(`)*FAvUB6JkB2} zHT4g5ordQGM*}mHcdMuDeT0)rM&NVuJf$j}mDG;0_(7dC^n{O1+P<;O81xu2K&)yT z!+#tLVkZv*3-;UjIYOQQcmW6#i+C zaP(up;ctQ!^!fl0`i);EjOuScV(S~e;h%}`U}y$GMU?=4TPI1Bb7t^n1Gl1 zkT*&BLcbte`6~gnsSoa<_730{DmwcTQJWo!A>I|>Rkj)ut?iPcETheO^PjeVC~6Uj7gxC_zKVoUyoX&N|QTxr~Cvy;YS<9!c>d-xKx zE-}{J04K1;(h6d!D=z;)(gdX8{t{s;@(}3-jDcn$cHZRbuHqN zGxIRA6q%cRC3{M}OI^69DQ56*c$F`|0S&IRECQN(4~hTT2D3GoU-n$O zo1>v`B$rF9wEd$04aJDPO^UOs_{b4$9~X=d$C}>;4zx@r-XPgXU$Ru_1W=b4DU8+s zvYB(s1$)?H-9zQ9v_#b%sBK4e2j5DFN=qjckHqWgW9c{zO$3oOSq4_WiYZ}ni%1mm6*wXb-l^Gg39bP@U4&%(+@Z;%}WTUmd-u}C#qb06W37p zk~}6k7yJpGz(}@}Z)0i;*Ah!}$t5HiX)Imv=I${vM3yhSeyE;zq%cxG2t*KNteNz_ z(5_HTtp@)XyqI=JsX~uRtE3;%wLpjPP5TBr>0IQi8g3#)1=_%@<)qR3jS9vFW)(H@ zll)U{x}!&niR=a8r;*0DFimg_IB*Fa>wuLo$lEH~jqePMN?L9o%kd_A{nl6)AyO;C z=jauzt&t%8fp3XSM*)2vw1zm4+ZmdQOcAb=7SiS>eU;T;++w7Sf0ppxH5Q8zt^~XL z2N2IiDej0+S}!Ix!$;k+(8TphFJq^mMVd4Gmp_a6MI{=^bxFax&?@ItWLg07_ry5F z0UnoTK?k)4Dil1U+*OVPi`CcWc}W56xEhDJY?+q7nDs^l+@(JE2V`#3Jb#kDH&D0m z3A(EIDxMryV(!eDtdCRY=XYmg^H2Kr!}Z~ndFSAK&H_aN3)nSVx_fM{*+R5uG5a8{ zo_7M=Atgarn=Av1#1|=(gb48A&Sl&+0?@JYISsH=4vJc&OdFH=0|ZIQBwspUPpp|b4ToPp-2TyO9pR1cd1y2LuUE%-a~ zD{Qdal2Tu&C*0+7gvv^hyvWE^Oc27(YD6t$fxWtQSHR5+{(X2~moI1}&-o$di2W*7 zOFbC>G;qegWoLQc1owD*6bA%O+Q6bby)M1ScG=)wP2?K>KPDmkS{=vOu&2SH!Rt~5 zKakdgT1tMBr~0ScY-%h~g*-`im7?tld2Z?^6-T1XYTH42A3jxUkUO8MhOSU=XZI#& zkuUNLUX?1NE=P+DhjGxChm`U!2`!d48QOyZ>REk4Vr_mH)Qf&Z4O18T!&)7tjsKo~ zgD@(T%Phx2hN68M5kss|e)zWN*+n+m9YjOVo!^Vv3e%KqWGq`JySd}H+5&Sx7lH}t z>C(G!qFB?vPTK+53(x6os9YjIL=kcOW);OT)BGp23oI;c;5#98*h5IVcYVSa_ zW=32YaxuGb|K5BQPBu05ph&>TT^gABH~W|A7rG?DS2)}ihRPPcNo-fR7~IK!^;p^d zz)9bE%hudWrZR<7%w8k2uny8GzfieE?$oClbyh5B!ZZdmt@d!;6dQDmu1lBXZBiF% z3`tUv3Za;SEBF7$d?21;xBRvB4WVs%7E!1#R5#gXBE9T)u(qMb0bZT2+!l8sYyD&B zx%rp!Y5!rk4U3CAxFNV6cZQnD{O4<=HBRd1+G`$StK^>8=nEPLZcDIRF9yo_J|P+g zNj;Hn`F3-4^$f8UnTpLTyh-@o@6~_d+L<&IBcxkKr3|JbJk!L<@EYkZ@raL;zZ8wL z-R3sY8!S)AF8YLUZlF)-CArwYE-*{~g3ZQyI-bI%{5PGwO3X?y{g{41k1u|WzST!F zk^aicP{R+o+qT#Kjb665hcwy9uj%cas^4~4oYBQQv|U6KHLOJkryFkS=j8GONe0vR z0Cba9UpaCw+Zt~h8ZEpB>dRGystm~(H$LtFeaY58SU{IXrvSU)ueKQi89J(5C+{k> z{1lm^yms5v`^aT%3(+Pp2``iSKs?U<1`lG#T!lm|zD{aCGrlkRdCI>)FW4E-X4%}O`3k}1BRX|>SIy#y%-`pN735~v;a#4`@q9p^Cl4W#RJ z(rwOeISKcNb>tOM#8xfr>s#tO&h8Um!6U$OU?D)FS^1=8N6MAdgIEfCmPnA-qUU{; zq2tVUYCpM1BQ!y&g{G6+!S2x39GiX%Y9f>;0&WWFi!?F23mYm{p;OAo#0+?jv@h>R zC|e#zJS3jjAt;Vnt&YXghy~Cgsf-}yu0LH;ZGOxce#k(MJm zzK!~VR^qbdjt0W>8qcA(vt5X@uFI}kVlTvqVpcxOS-?6aXh{u}Wm?ldDNbK3HIWaS z2MIV0P!{5Y#jA*NLVAaA4egiGBQ?c2#@4Om9c5w))356ajpFT!9@)-&_ZkTZpuAoj zFLc8e$H$1N{wpMd#D*T1Y!AmM>yswplN67#Tk#uelP+n6*nsd7X(bvX#OYn=v;2JW zymyhgbLt_hmBqmt(qiIA>RNQL&&CUExIv8rvBFH1NiNUrgIZxV)l%{XH+5jS z@Lg$^J5l(SJPKU^7Gg~ck>q`r73k-bN9al)rEF(Dm@cPQpbnEsq%}N1e{1ABw*)5; z9ULQ|2rkL7$c{j-oF9X?rLVy@wqK6^@(=rP;&`Abvp{WPD}^0#GU2rZ#(Y5vm@Y`) z(D%>|^`Wz^w96UDv{DNtd!Vzk8M_&A>wD}El!n1FfrDrX6~#FH!_u5PtEm8W+~D;R zu@k;-iVlKIzF5acBKHCYyF`8?pON?U+x~;Wms~%_LR}2Sl_AgBSS6QbeV3Gda>2>|9=*H9^wjQ#GS%ge9^4n$Vx;oiocD5I6%q1sQXwsX6 zt1~|8Ewz9%D=C`KJ5_Pff72b%g~*xkL~>*Bjy_r6te+sK>U}9fjSm%ohxi5HFK!Ld z7AXt7Gp%v8f?a$vQE_j&Fa><;z1r|LyI1K?l{P<6qJ42nJ}}=qHPg?ok|tS~njoSZ zJ6Qe#fA@?Nx+7)L+AhN_kU;QrQCe6nJU1|-b#NPIULj2EqZK1JK?gJaf{Oa4Q*nSp z{b75l57VpDZRspq&rsZ&)?{I5AyJ}|6vLJBh1Gu?H}oZ;Hi0~{j;#^X%L7mudLKF# zFmQV1E_h8EtGS4(%^WF%unS^o>0HVU?3>={&@KNv>>OF*D#Clq5qccgot-AiglLYy zc(Sg@v+=kr7Mf?JzUS)e333&}n_mq$rOXzdYMWE%f!)20xo5^-9ixBaJEAv}yIaiU zS0|tI3*UwXyn?$C^NgNNYX`TQ2IOri?CP1S^(TM$0QNJ{mHbKs{5u7Kwi+qLtGGP+ z3DM6!UDiUq9TCbxbR)ijgpeR*6G!=9lO34Thyb6={ziPU@3cxTr)8j8MMaVI1%gtR z+-vg4qiqb8U~_0)>0QQ3KLzwlJ}Hg{eyacZV~Igp47rQi?^s9oC;l)Kb%2N?C#a{C zX1;2M5Al1j2lGF+`fJ7Xz0pBJ0^6&cJ(h3HM zzp(kpI=wY@K6}0PvtXvwS*xY2z`yNx8Z)H2=)Hx}D346d*D1j~0y!e~#7gTsfhJH* zwPir7Vdgt0z2mBL<-Hd?0ilIx1?v@T!ePY30^W0wEIx5(7nl(lYH0olcZY6cgP8ochS#7oZd%VOg7MDu>l)Yv~i|G&F-5PwY{D^Go=y#4hAE^$dAqFx0)(CjMnu98raNfMgi{ zsttiwXc;02yRJdSxT%Z*870D-OdG{pWF$R{ES5BaNl?beCVK;EPFwZvG z&_e2TdGxuw6R8d#$*Y;L9;L*iVtF=0IlO8SHxxDL9a)+ z%1x6dil^jK?2n{{$Zd2fJdT=9wN$&?BFH1mBtMmR+eiRxDSq@PfK8C<20M8{*&f?g zssQ5l&P^6sGw{`xt-MWsA{Vk*sH(p;TuzgTWqb&^lQ7KvP8mzm$Z)Q0auKdr{}9zE zx89YW2K){8)qMOh155NM!^Ev(z5FHq<3xAvxtzgq;69P^eiFVKhi>wPzL`WDpU>bDGPi9c-*LUQW711$h%no_%RA6FPw5C|fVa#iq2=H-wK2w+_rqlY zfBy3wk8{CzFB%e^Tq#25s`&ieIJCr4mCVe4NG*m}0&|k88%RKPkuoN`EEk@}RPyZ= z+5qFD+E{n6jb$x4S!zvh!A;)z?riW=egz0fcrBG7pP7d@djgo*Km1~3s@11tko%0; z4*PwFgid+6fH}XiG{CeqZ&faDo$mdcU4Kx`vnum}tzsuIi@(EP#p)v6IG+ZK{X;H% zw(6h{G;2nA<$v8geNJm0hR|oYop`Kv;eGDOD2+EoKkKCma?sAyP5)z0q-~ME8cF%~ z+eXvtv(6>-0$TVw5_Qx^^2Wff#Q9qi_81-te=U42c28Z*p2JXZB8e#tKv&*6V74X0 zvKt*$IA3V2wDk11jtAD}b{1|Ytrl~I{pgL9a_s+v>DVYf1{aJK{T;L#GfRHu9KnwH zUO-a{TcXpIgWiq!S_H;kVdvQ+$U!awC~a|A79I4PD=4;v%Hk}`9=?VpOA zEjXXvR(sQZg*I|OzI(7OJqIlb{3~I=Fl-1CPn1)COUZa8 zb3m1e2s#z9Q+M|?wwH281&8aM*bkN$@B!ixa~a&N$LdeyqvUB@XJrB1i(X@7@!Zyr z67}uXO^;&lDM6}-ltv8_%aB8u?{s#sj8eN{veE{-b)YIg54|D_aumBp90DBmmtkhH zGbn;@o?E26MlKcqUDO&Wr{q}%1J zV%0zw<)}U1hj5j&J;HL|WVSiImE4Ey)YgN`jdby@a&4(D>DAWza*<+wd8nPT!Io9*hj}rokUtiEXyJ%@PMLFxbs#;Rtt6Z047ss=n7)Ar(QP$W3`o2!^j2t{zkz zL|UnT-M6G0h{;q_H!h_<%xGQd#unOshj_@8r{uMHF5AzB( zt?qB?D1L*@F!D#{t)NvEm$P1hrPl`{#>oOV9P$joeePE2OVcKE2{0repC zpuq{p_^B5{$(h#Bzb|EHyOW)q3oXvjqd~4v$D|6*loO9a7hZP@G{00oZCA?1(9YXU z!rX%7(CfUz!R~+04L9&ka&CV($FclC14s0=N}=oZMkFMzyF1_ z7gqb82AQF5M{0(r6_~@qp;#w5S+);&_RiV+LXK12jHVr3bT8DpcUR{lASS%DS}o`C zDNK0Gy5%AGk~j1N+!=cIoDcmhTpD;c9tdyU(c4iguY2gTcV0GE_;Zu>YtG}QDW zRPXDTKsHD?)sd;8CKpVB9wimRlUfaS21>d+XKXb)Yi}1F6(ip`{^v-u$K85tpA^^A z@yN{BhrDg*tOLypf9N&ASr3~SQkn0;mZeUFX8z?5zPYqI-0W5)?@_ryMPZTOH*~mkQ5oS;rA!j|q%_3WmbEu5XyKfGc$%|e$Ic;N z)CcCm=yJ{nH&!~YKPe9VZ^M=lRv{_qk8KtXGEW?zs*Db270(M#6qHcor0ULr8ODsj zu4&;VmS>K}M;nH8oN?r=uO4nj)^@zyygJMqCGRVRY4#SPA~Er{UtyH!HytQw9E zYfWj=_N3sB(oaI0vbF@@&Y$f##B>cc>weJrXr>%mb$F2@@6Mh;o9{tKpV51rpA&}$ z1|4|kz!FP`s?>fTm=5F4%62?7cB~RQl*k3EsQsLCdpBcdRqGV)ZZ8uKeR=8l;8+*( zWh2hn?8)GoH)yzbjTfOmZ+kjAOnMM(zH@`)*Xi9seNfo z>RHD5ZFFAfPi?a!z4pxzGQDMpA02YsbnXs46t6hWS+9_jp#RKo>`k#l-nujd&n5H8 z<-~0=Otkiu51bGV1zzVaBi0q}+k;VqQuBbWLb%oI~!z`k*tV(&|U~vT`3h zkc#S+kqy*c

x9eWTWs^N}Aa49*qn5dYBo=`1u0D?t-@Yw|*2ljSY^J9RfToa`Bo z1~y}B(^}hGQ?-k_xg&E{Ve68ilBxsw@TRVGA*8q2n;!o+xd_U-oBRrb4 z+f*N&g(l`_v1?o{3r80MR{NnyYzkfri84}t@(bee8rl*|Q#|0S72CT--nC27+(HA?Z~?lxGF!L7L4k3&nn zjhvGljkHOdR45b%7;KQi8JxkL!r*@{a+BvtF3z*h-rxSdwbrYH#QFME{{()ew5;qq z)!Y|HtThzZ+;E5qvEEsgLn{x)pZ1!3EfYF=kC)Gi?c_#0?JJ{QwC5`85tq0bUX(WL zT$M}3kcvY|P!}qtUF!7M%=q1YmG~q5H~mz}$hWO%ie!+Dq5pD?`pD>e*hKy!yfJ)Q z+yQNnfAW+mbaU>GOB7yKP4Xr9Ue(U3n_YdjN7ylA7432HKV3%89%o}_fBA5+J(2;8 z)_Dw#Dt_fWFl=D2#dl3wLYk>gU>18Sc9$T;1nRT2yUj~2UG-b=>-K!z%+foqp}N1t zO7};WaQCh}>K^8l%HK+je9x-~dCU}>Qse4f?oB#XF(#p1Os}eOi3_1|UL>)3*#XZ# zSg3OW6#(2T9}qu3Msg)+GP`V#LAs`}c+CNW)zjGhHDO)shO*AFP3&C7Xw6c8O53FP z2jpY>73VwrB-qA%2&!}DIX6Z(=q3?Lour*bH1|0D1h5Qcma}|w z;|2L*X_)nP+m?n$rIT4hti}1e;hUs!D^jz|F#k+$H90PEHJTf7DLGmDl{ujeaXv*h zK=PO+{4%jHxd~7~75jB8A3Zau1G?FjXPnmzU!5pxjt%45IbmYCZnuL>fb{!?c`*eB zi`2<|%M)4+r;hgfxK{h*{%ocfTCJ;eOe!gHCCL9*=+QvvJBeDOzg5-jZBF7 zsr?p!RZxi*z9;;6$&-AveI-iK0ggX)f@ zrdCE85>h7`xSHIQ(+La22I)xtXsNzpJYVdrEst>Ip;7VY-3|J-+8uuVaU*|H5toWp z$!ZRCdFSh2GaxnAg6Nbf`T%UE15?0V%q}ICDB`V~AJ4EpYnO51P@Tqj4{1W>qw!|6j(UwmV4}tH< z#W7Qfcc!xV4lxg@zEomNjB`XxM|Y}Iia+a~Uw%qo>=@5I)5c(1ZSM_{q)%F@f1;69 zKxI(nO0h<2;>?O^Tz5*>C}H>U7Qu7kl==hYHoRU+L|a!LiN9*$3BbEn zx3qS5>@!XCBWH-)_GwkYWk0Zh^dVYY8&5&bAKH1b=Zu9JB1S7}g889dNFE5^rsl^!TE$Q#6qxO9D}{@bA(O_-mQ z_{cNf`$zH|x1#F3p{u8@wSi{W2-SkTh(vw(uAJO zcjHgM5%)oCN@_pCSBWzzFIj}?{fZT-RzJ$c-YzHek+2$KK^CG zEa{fBZ+x}L+wY~F(gnthEFa=K6xX}vaqU>YR(FS2kx*Y_h`p1dP4*@9N{voPOtB`7 zFjP?~F@Gj3O(;#?T&a%BH{hNlvE!Il8cDkxUPYzEFN2@QD&2ZrSI0Q-5a$}_f1W+= zQx$-AcS0?Hg_S4Ww|~^^ix1%Lr~GoBqV4gAec_JT3Ek22($v_FJ~E`cd%Tw=mwAG{ z3;dL?2YyOOSoMJFOR>@AT*YVCIoDvHPv5*Op{lo`Tsze}R3FMM_Nx!FnC}TsBL{%> zvES4^xP`6h2;{T$1CAElaXT}pnX^JvEZnR?v#`B2ugOUl`T}t|p z{izt?nd9EzD&uc?9D08*+S3Hjt^hp&?(gL-T>~_A)y4XKq%ZDMT$hyo=m^iD^7oD| z&QIp$_I5@4bY+Cnc{E{&zZ;Mp`y__a_t4qmBVwkCLy`g$FFRc&Go|j%B5bas)O9NQ zO8Ks`E2+)XmL(Hiu5}64H&1??VsKB_eIFtDm~ z^xv^c@d}Y0Gf7vVov1yoTOR*U%(%D*x+Ai_e1G4choEscpiCg*&+>G*% zx<4Ebow|ak3Ztt8hN>dG3sW|DC%9)P>-BEua?c$@Q!iDs%Fx#3_V(~PDw3VS37z!H zq_MsZRr9^~^}h3qEEIDkxvzec zrlny`{KB}%st#TyH?7R!v-^F9&H2$%pyzL!T+^}WBx8tQD^7^oqphdq*4{*lw$^$W zx=sv+?jxh2m>%Q3iu&<2qZ37gSoVYZ7B`xAA_^ozSHeyv!NqHrj@ zyVDF#%CMuXf4PRJE<00`ka)XlM|FLS&iB}`Rx>5(jrWdI?>_Bb=mb3*{p!w-WpVCw z&qPlr!yN6^*tWV6&$eUU9#uMl*@1UVF+@Q?nuj!pUEOc6IHg=-#`w@taKi}yb9hHH?XyYPDoAIsDEMjr>c`Tuxh{KnE$_iSkn{E#+`7cB}T{Z<<2IZ z^pjBHQajm)cq0?*oe``8zQI-bcIc*gKlsz|@rqfNL9q>BI9X+wiOuEo$g}7{{Og3p z_&4u!Ll~OlJg&Kcta5*^B>X&}HuYT+s$G}8U|hVpHTAiAC-^E(s#zAJkDpq#N}5I; zLIZ6(^tGvHSdwxHEE-_&-aNJaT7&67R$T&osstb31D3ez_jJpuuE8)2_%e}m^!5LK6O?+SR z(L1xMvEiZiQp)Nyb!wy3OmR=yReLDY6eI~k|5iUKCeZfO*UR;%BO=+2Px9O-YvYM5 z>dIZf6SY?j4f^Mf7Hn{dvG5o%O6x<737N4kh>J0wH9+ym%4QWo`eb?m_Qdd2doZPz zRC;>F-P7#yPSwmZjMJ`;yO`1{ae^zVCN7~x!Yc1t!>Mmd!ZqztNJE6Li1L0-@eDaFv0Dc{59qKBscFr>{1=e06!1t`CJ#!zHV}Pi&1?sRH&zwLGONxvOiZ^L6F8+Cb!1mm$?_ zDuMX^!czSSa6P#uX|mw78xz8~vI9TjU*(-Iwk7@n4lRMiVf=m5&EkN#w#q5IAz>!; z4m{~N%2gW2V|}=N(xTXli7N3&yyRXRd%oQ4Jy?FZd`?nN!#cmqeud_}_ij=@!z0g9 zeN&y!_s-eK2UcvX*yOGCj!v8$Hzc`B-1F4WK0mQG{bkjC9c2isIw+N+{i5vUV|0y$ zPi4I|Fn^0361SAxWPYQ4>3HBqwTs*Vp8t76eaGux#ZON?-?gfGecgiGmBPNDa>Yhgc?K*c(*;3~(?QOhNyU*ixy(o8eXT zp0QoGCAX>pjvCmh&VgkMiGP55WgoRoV>Qm;ZmrYj_UKXjvF29H@}$(%xRji2E0^aRX=%c3rBRC1Pj-)0$ZHJlDK1%6 zc1?%sK~yutIc_L4Uwj*P3Y+D;nc%alqYlKpBZF$b#?8Y{*Z5-RQia|@nzz_}A%f_K z?yWc(Gl-gAv0B%d-NwHtzkv>y57lS5cQ~ucHTlgqbEW#6KZ|;2#Vp5=_&(SR z&&(8crJ|Nf*Opr5b2+4WVmi>ERy(<8|3yhBI41587+Ce2LW$P;5iG8ape4WdhdVFPt=wt1G zSf&21c4TF!cZmK~#XYx1-;Vw4UYT;v6(Y6l(%sh13AiE++hWQqek4~*+p5+ioUgde z4RKE`dbfL5>Q7XZc^;tiVtyxD``tEeyq)~{hMYeOdyb}ExllVac5~9Tnl|p2 z2~T6Pk`fcs^=h7 z@4i)8=Tp`K*e937hYIx*gL!m-2?8{o4^^SAgqd1`YsQqfJ%QFK-rmANb3&L0^E* zc6E-qqU{(vJn3QOtunhOB*CWprI)*`+P=O)-kS*%d?~^y?6b$K$*IeVGNt(=jME$i zp|v~57NyPn(W>hy_l&Om-VSvcQ}{H+jnE^ecViB>Y|}sW&&ZW^5eKU3PD~wsc(rR^ zTJrw7w7gXvx?-Lob#>Z(T~ZoXb*X4wRX6qyUU$;5FKup%=(=n1oe!`2($-IG?{xUD zj^4US8N2Je0bf$z%xsZ%xB6fm7417*);cv!`Z_YLV$1!s@3$lC?7O(MuvT5tYS*9b zGBu*p;a$fwy1Yt9>tq{JQYY9tq}{mmsBYQdRcXNm_pI$~QRZwSh?CiNiv6k2P=xU( z9?GsY_pr;cdh8e0meT}ZwlpG^D%(W`WFMwG!V%7KVjr*$+5s%K{t`!-(~yIxmYK_I z_;_(1dz{GNslvYsIfUI&H{6>`9XJVT8J|j ziw=5faSRqnq^sKj%`JD?k?eg%1OQV%zz1Z#?U&hE+LK=ihf4^a%RSv&XB#2*)F`;a zY#ZVnk*Wx!g82!~8-i17B3htB@HMgnvIXH2OV%R8q`f$<2GPUhC{qp^YCcZnO8bx_ zgdO}{^$C4KpTq)6WN<@i0Wgu9hd#$uNC)-y@1^PGA4E~DX++}tW z6U!283DpBpYx*n9OiaN%f0q7N+QT0PJ>)phpEn@ml~wZB*s$Vt_!W2`a#oDu0j4$M z6go?AZSA(9!GiIub2q+0z% z8p&B`4dsHSmuS$*_HuMD(nwfIHUO!{Cdxs?Q{-P$zVQn8FLB4*ksF2_)~yiK+<7*d zx`zD5{KK~o-)5T!{cDQ?v#G`y3SU=O8h@Cl1V2T)i{ab`tC7KS%E<%VY=djT8Ns_>v-22_XC{uII! zmN{&D`%?#mq>$%?Fo2P@7vF&U(VoaSv7^}6@&>2z72FP6Yh`<(KhuMV;IqN|>^pRl z7=c;E8;sgAf=vWh+WQDSk+F^@yY3->C<>4%q#J8OK9YViO^8%{1oTNDq_-^3?cus{ z_mCy3To~d?38e4>Qv>uT{tyYxP0cb0>TM^A1M>zZ%w;su-1`~nsVX_~yTRW7i z47tYsg}N9gKMCxK{ACF!J%F^eb%t}fbV<%{#su{Upr~Y5Nv?A|QU`yMdeJS#@x*Fj zKeQA5NndA&VY~U&j%;EOqcj~r*W*8tH~yftdi4XzjH|3?MFPQrMboWX^kQxbjm#}^W9VkZQB#O&9pxb9He*x?tR!|yVRSZlUCa}I<8~v@c!(Uw zm)b{K8dw-Ms#{Vv$G+kazK3=nHi3N!LC6qwm>mUH)4vhT*)5ElZ$>v3Zgn0Bj7#X0$OkE1OhvX6%_T>u3iFxQT0%HS z(Ke|k{St90_Yeyef5?0o5cWd+CVqeh`9H0Z#^`P_kQisZDyDLU_+Z`#F2cu)JK0O} z!*DitQ(Q;Pw>JDQ5KC9Rjp~7wu#M0VeuXC3&PLBDTJ2{PM>%gR*E1kG1g|9j^ z8Y-6}SekDx+*IYzcdexGyEs33y`l%Hl@{m7G6=B*Us64i{$_6a{V(gJktMT$ z*5I$aPU0NyUC<`7+7d_joyy9a_LR=+Oa5yA-Cq8ZS zm9$&?%5};()?^K2(|MMd?)=5h_uESQT08M4i@s~ZP2Y%-*4;d6)QK&$z39>KQkKHr zVP}|?!Wwb6a&yVw$P?Qq`%T1Mu%gJK{=oOQ*-E;b!=#-W-WtyRz&c^ZQZL`g8ifAX zKOFwVpA-M&wm_|dQ{jWEVd6b;l%fYQ8HqzqX&wSIf;(Xq{8Dh8;6`2_I;GwuYh)f0 z+$>}YH!45f(%*5~N>obK5a$S2m+hzao|WEM!w@V!D#y~tTbgSXP}Gu5OyHn*xKBYho!P(owU>pZ-y@s z7I43XPP9$8b;a7`cau%Q_e3Wkx~L52vyvF;qogQpW@>{KmiC}8p;K%piFUv$Vl{oX z=&sp?_m4^y*7GB*wACux46G$4S(S8}P0JlacA4~iwlLR!i)hUDq<&%xqU$Z7IV-YM ztRgyEVYVSUmHtz?#PPtgN7D@p%kN9SMN@1A^hf@%>Sf-2Ad9HPZsXyR|Kx3qD8eQS zwZ>#f4P@H}s@&FmYxBbQ^k}3fUnylt9`U3Ng4YlkLOwAhKLzRyp>S9Dot3BenC#SQ zs}m39{_cF3+pB2KB~wlIGng>T zYGjq|535A3wQ~D7+a@Bo$R{S)YnW_Mk8c&%nY${_vF*?d%fIS~QkCr=_&f??+i4@E zu-&tL6t~*WtJ|Ypc`dJJE3q@SWPlIZjWvL~+Mgiap(o*LN(^al`vAO#XVV(wapxAM zaHARtflqLUp-|+iv51&ZeAn7nY^U5WwYSv6S+YIocWWV^sa%fTgQoBc%!%j;DH+^A zcgM{_->6uymOWZ@7rdU=4R~m#;D6|T@CoK<{&OxXXp(eX975Oe&BVygLEv<3LJ^3L zFuj(BaEJV7)P3}%SyJu5x>zG@%ZOX(J#jH2@xfST)W1L-{EP_0K%tqCiEUHA7S+n3 z1m^rfKfyWybJXKhm3}E)XW0vkB69F1Vl?^~J|Z7O$@wXQ!P?X6GmgNjk-=acJdQX^ zKEUP+r_I^s=iGR-mf0Z&SvM&r@aw70COPQ^itv_}rrhO-{>rJvXU%`2$LM+{g=@;s z2KFP9kPlD`g$mnhes8l#bD`(Pe9KI9gEAdEZn=PbBpAB~7$wwO(o3CaZ}f%Ms9vXO z7!#1XU0Ekx)D$4D@B`#PWCYlL+L4Us@Q}+>R5j0w3r!4h^yH9SeK|J z_Cpcvkrb&B`c0Yc3^uJSy$X(?C#Ws#6m^#Eq13=PC60<&)DhZ9?ulTDbV9(|<7sRc z`UKmLI*rdH`*HQ+0;CVpntv7kP`w^oB&teN$*}zhdsQ7FO~YGq>5h*M4Q!O)h2wn?W{#1T$2wd(mMmWu3{g>QMD0iE^g&F!CCm?=-Y*h zxWUd3Tzbi5?Mbjy+N5|BbP?*tT_+C_toZ=3mA!S^I&y@&W$n6gyYz(_r)eqN)U3i-?p|qAc7ouw zPN62zp{hIJG32G?3H(t7mh=V*cAzv(AtD?r>LQ&`Zt}M)3usU!IUP1s^2DjZDNKE7xOY5wS8Jr)+ z)KjZearR(xcnD236FB2lQ;8~sA57@@Ptph|*18QICZ?8tW)swJQN7|8;w-wNe2UD% zhJelRtN5*`F3L{Gc4?erLjM2cuMvfKtE8FrB;OhgksQcZ=m-BhQ6jdsj5RI;oWd~0 zFH0%i!nr_sn*{mG%qMZB)IT>eR%0`Rp;!ppLse*LDP`os@Yf6&^v1hsvqj#^q=hd62y&Sfj>SpBQ7+ z4}~q_NPHp~PYvcK6p!JSB3tO;G(_DdTM`z<7-q6^BH^Ed)4%OKmamXNN1?F`wE>WG zH1WVxNeXZuq!#o0V>-S-eu(+D3BYYG2VKi-jrfO+BF<-@1!?Q8A{ae_YN%P}-CP~A z9FBDiwEk5DScgPi=VsF(+TY1d%owm6O`~_vF-$k^H{=(bLr&-ZM|b13=zQr!^jYR7 z{6rb0Scc6*qMbI$q`C){^Syy+tf{%9wXJkY*x7s?cD7Z=E(nsQ7&f5?9dh`7w0&)uVG3Jpd! z2Itu>hSaLUv3O}53|W4#{VkKx2?9j?E1?!UTfq?aWHegahB$@hpwEPV#VNp3cs-Kp zkg!EU4sxBDQoP&3VJ||4@jD5UTP5aMgYci=pI87VVBbr>h;86q#D?&hOr-s&eIK}o z>uG9?E@EE`c|eF*W{eQCfMuq?0lDA=_XrijO6LNiJ1rN-5YHe2oMJyi^u?~2T4HO$ zy7@_k*ot`Xf$ z10*e~q_&89ey(kIBx~O;EJZfi)>HFLX;`reW-FY-v4zHF;*pX^_9Q$O(-hxCYS?X3 zd#+JjrfU*=Y-%+N3>1X34`cD9x-W6h%f+1K5g%%|sH zN6#>coD$N?Hz7;$wV_GY8G)_s6N%e49W00Mv&PQPN?vg*xp-R6IH9?0TYE7ykj3d@ zu^dlfvrs#=4W+D5Npn?1ehfdBO>>;Z=Bhg8c#%nLhiJ;)lD=uIXCj1ViowKPWP_{+ zWEKI^3?H-oiMJwFS>NIB(Uuk?sFlp@M^itvEB=h9nH+8@Pw*w`3a%MAifcpPU@7VV zFhFsO3nk_LJCHdt2YtZFMYSeEV1cW}w76e9!^~FSSFP4y$ldHbfahcJBxJg5Iy?*v z1ubGMH5E$}+lYg?iS~9FOu+o^l97>ph&a*5R*JGnyX-TmcP3avME@IM+7JBBJj4C_ zB*6l|6LUe)DzgniKBEhfBzvq;iLp5v^rrM#dzXqqI{|~>o6f%AKrS`{CconVB1 z7(p*tLH2`uB)Xa3VEZB4hK^DXfLF1^g*W5`avbWPSh5+aW#SVm%wPkeheM=??wiD!!Ekp~B~Xajje^;WcV9l;!A zAhuG>h3-nMDUBS%e6*C&A8j_|6;);Gr5djqXY8TC_yWyNF2f3qGWI4z?UwO4zt}>^|-saTe%7z?Oroo2z3vt{Ikq z4WZ&}-O*c?`@+9O1ZLC3<7xEoVls{SfA`sT+&B&yV>-a@rUS)l`$FYKDP3c>)pfpX zl!^K3(_9S~gB>%*6JwzE$WqH1MF*nB3<4j8R{k~h(DDJjtPI2+7M+k7wZr%ino90r zK64vkgZhqmgFa!OLujNX2n%c_91>5o0D2*Nj5h?B8EYSmV)QfmPhuAS4o|iu9ZU)z zC(aWtIv2C`%re!T$iD0YY}a0k6os`zZ9SqrG{-eQYV0 zVk{IhOM5#%acVTEzzn|U!wlUu&2~ipb2#x zSqWbRpQ97_A3y|g3E0B&7Fzs{|DbB1UN#H@psvI}B_cwgJJ?CYLwdB`MBYI=ksGNP z^9L|Ul=xp}Mto0{DCFb~@o;H|_6~R{s7iDf$Vx>li)hL9Lk}R!G=wOlv!yRcOXUPa zMgCOF?NbV~k@=eUR^yaStCGREvDSJ^nDGWZNi~K#f))uf zM+-il?Ey9-E+B`s`%90imgf1DAmnp=9r{P<18xo#MgGYh*Kmcm=*##9RiDUwb`!Ul z?ua|!>DJf$4)twRLIT8F?1+GH{XLW>QEEcycWfVNx6F^~FAUWj$D@I-;#2yhv?m(# z*B2*+snP@KBDWREP_J|R7rj)9HCZGxaYfymn8==C*4TEbBAFS;M%gY+u*h=RTt~al zdKJkgKj(GRo=0+(@zg$ID!vEED!own#?+48%0C4%Oyej}Ud#Q(wu3J4?FAdsOzGEn zkvFV1=QgaV^(t|XzDN(XblW%&J|gy3Oh#5p&A|DEzxITq73ga74a-HvIpu6RnCi*y zhfA!-nZDq9(ZOMNHZj0L$Pi;sg_I8l_>-gF$D@n%kRWm*L+04yuUE z7+4Z9&3r;JY*izQ~S*s;1K0ap_jIv_Iai<_6O3A9)^)NIoX-$=RA&{%ncR-*ba1%?GRvL z<`U+pQOq6VF02{R1oRU9z;Y!q*@6-2(xf8|BI+c$6G7y6DI z<-gX)#wut$-^O0fT_pRFEsZ}5{-bYNF2NzH)5XJ0CPa_i<8&gWo=2|Z{T_U}tYZhJ zzGMvhfm?0QM1Z2&=)O!V>?t~)xr1femvZM!+2mQ2 z(d{6q@96ey73+f9lF_suOxK}J(KOXF!f4(NFXnQfb`I3~fvsY)p`+1C?NDL_eMH?f zvO*r{`Y7&@mEn!}F2T3_oo>*Q#rEkefOpa7)~#e?)vJ;fyU)W)>2I!$xi_4}T|j^9 z{6ct!tg@$b8@NnMR z%!|QG>J{q5;tfa=%1e7O4OBg0PU7OSEBYnT&!TR)-3c zm%vTf2}Tcc0c}|vVQOnDb@Ws$L~b3(=X&8$XhHEfA&QACx&`>tsPRA17qaY(udwgW>bgRIf~CE->@X6KT(c^;iMxdXO~Fvee4vw6db8oYcDRSgw_?E z2hr@`@hX@gmhy8B_LQ9T3duBhJBdu)+B}&e;yD%{CI)g3hI9@G3r@ zix5MA?)aE&7rd67OWwe8WQ!dY-1P7|Vy$Jkq=*Vo3{ZvIzrb_Z0^o1cQgRiZPPCI8 z-15MV{8sBx;GeKKtAlNz|A2<)43X05WHuYELvp!pL@(l>j;G}VfZ^N%VUG${r4d)z zXf_&N5@%-4Z-yTVq;LQo zL7~V&x(etFAIIzYR)xo~htVAsCZw&hJEAB4u;J_yaydDWzK1N}7OJ-x?~6}3C%qaz zXnu zuBMx#PV)&l1dXF2$w#ai9ftp=+T@>_njHUWUs9X+)r=9EAhjS?u`Q%>;v4avU#i8- z|A14NUAmjnBl}8iZSl0IY1(x_5%Z4fDt-se#ee!cBY$BrWU75HECN<8&Dn!0@3G{**VOUd6$20I5b6t@}4RSacjpmT^`w)G}XQYXaFj-2z|@- zg8Asb>1}q#i+jjo2_(1k!?nX!nwR1PnZa+R-?3TXFw)7*+B05Mp;NHl-*qy!auaQiu zvD6%DBe$M8u2PDlgdp1=5k}yIy0>+ME8(C=xGXqrGeiR2RM1|Yr5;bMCua$%%o?gM zGFNJiv{yZ`G~(`y0|A@}QCz@ws%6A#_9Sve9L*hyI;BWw8ANYCq2$nsmag*C#f3VJ zq=4Qj4xp<^58qmzfiwdb0oQF>!Hz1$b5aXgBKpGgHgb8`VD%=jLDN*)WlBI=*z%(y z)Dh}^#9fSrZ!?3KX!xnW+E<7UUZg^y{=h@vyru@P$Il_{O$uVI5n*ma70gO=gzL8D zf8s>+wc3shcD`kIm(CT3lCSX-iW0?lxE=XF@RiNPc(Arnzw-{PEjScoWpkW9=Gv0e=)LxQX#ws7n&P)) zbYK|1*ZP|MkF94>TOM`_`vpnRh5Umuh4>X0`N42G@kD&fk0^0+fuIfSC?=wjK!c_M z+Jh~?Z`&8c4cKwVDgGOJ$l3rk(xPC2I$zpkdmzFP<$T55Wgi)5JC0#PqF?dXutk=+ z&>ps*NoM<>BFeuTt`{cADX}rSw^T*-v1*t-s1F!n&Hms2u#V(JF$6|fKWJA4`B&l} z>hYvf^`0cvOrx*dYc@mtjP$2^naC2Xdpjsw)Apo8^2n5BlIJlN%K`$>0P7&B#5^5-{gV9W^{$cD7FAS)+>CK+^(2O zOoS@KXMp_}hxsG_fm>VHkr>J+lh=uCw2y#+E5)_KK;QuP5=}uz5(tYxr)-lge}~J+ z3nnsht3^T|nOYm4vMN=3Bgjt2&Eo#ZPe?!UKTamEP^fK60unC+7)}r?#D631V^gpw ziv*>MZ6T-mBSk6qiUUk><{iaZA{x=VLzFd8HJx~1%MjnIq8Lfj5sbuISrgT#*u~;Y z@dx-zks#M&S>#1qjMNj041X#vu}34n)1}A~XaZv-wi@$y+9JPOpBqK`9ArSo!Hc*T zU>iVYXVAawdxehT|5RFGhvE%;i5Y|CIuQFy^f4C8ltSjtk-C0D2EV@81KpLUf60@WcF+DAliZ+^0u^w&=p$15KtynFMb%`vj(Uo-eJ8tJ^Kka5U02p2*tMHr4|cx z!xpVtX(>!Up)3b-yLmoJ*~TmqU-4AmXmj!1V$LiWO`-1Q#8S zr~dtTB$7o>vo9A@)t{{*))QG{yG~cj$J#E~{$d7LPuh+th8pjoL$DP{`oW$STaicC zL-WLW1};Ij~kA5`9pyiBaNl(13Ceuyk@ z3o%fs6^+DrX}sxxaRhXN2o^?h-#8Jzg>E6q1IXLxFrYuj zlTDy{@RaQ}QA>XS^0C#$J+z5<2=%472Q&o?VjGv-XEBW}zNkI- zO1Mql3t9n(hYeI_S#OJ&XaGxcEl{@#)A6O9uF=T@_v3ZlD0DmDktf9-=x~0jqnx~lS&1xOLmvxm&Q2jCi0RyS>q4Ye{$g;gGR#hL>B1}l zAaY6?loP=<(hzZ|IGW8=ykvN08QYY&NHjsa%M7Mj=s3BHkKwvNm!0e2i_$A=`~7>P z&cQ2DNgkj}C>U?s!gUjm2QAG9&BKf;^ox2mFhM;5$s{_I7)S}$7ykq|MD1t4MLsk) zs0_qU+b9;*4zo_DuZleiTL_U>5P!wb2Nr{)gq>DcxXUIKjsRz12m*ljIZJqpJznJ- zFdUwaPGwV}{pLchE5a$0kX7_Lb5h|DawQDxKLu3BSAcg zb>$aDo)99;yN%*CQY0ortdXs_3s)I6Rby zvvk3RpeKzz^0I)7;2kKI*kb;dzRt`p6vZ#>Rg{)JBL?C#(96i_h#_2sq__Q{d`{d1 zdl7Mh8tIbrnJ7ROTI>D2<267mw@ImCipBM8a*;>uj_l=c`aAL8*)nTdaW4pClkx6s z9J_%_uG1j}b3cS^6>c5{x% zv)D56tdKzz(;ZEZgj}f$|IF9`%-DsrjVNwo&(#bCPeBgF3@Jpk2%A(Bu_X+QRbgK& z{wV~S%%(wKkZ#x~^ZnBGaye8HT0?9Seh8EJ9H6xrNUqHtD!c_2fT`?gK1o_ow4Ix1 zE;pZ*-m4rcdEOXXI0P&9vMBi|@1^BCI!heL=@hH@82Y>wE2;&e5e8BDYrCeZV*j_F zX$5AJXOnMuhvv3vCpTTytaL`{ExIuh*1ik7lOBNV>+JEbr;W1pg2NWejfSU}48xwP zl))*89q(*5qP2h#%crg*r{Qh*_$ZpVN6xqY2E7&jhVOH)u?0$so+AE0DoW7oSEXBQ z)%d@v>#!3JhZnQ&qNC8OQjF=dVh_?6vZ%bkdb9>#qPSDUtB(AG*WE@?-J{w?Ri7thwK$~D<~yzj^cK7LS&>=Q&?=q z(TnEQP#e-j_^`gDoOnv7m@-y8 zEIcT9#et6MOw$_hy+q`7u$}@gvgnYCygp{}%p*WFix|EyoJ4n( zf5}?g5b-L9vsLhIYGFxBWQ}-J=@G+ozKLD|Q{2;}a5MP)z~R6?VS;8jyA%MN?bQ?& zCnm9Qcxp+4dN12T9)vH%(zLDltjOp76Xple80;^eh^nUrPG=1jei#0*?$d=(4Eap){zD?RF;=B^cYM{gqjbi4tKtc&c~c`4nBZN`r?zm>liQLrZ$ zB2t!a*kQCc*SOQY?9*I6nA%bcj)2n?ovt9$ja>Q zY?9(|(7>UG99lTsu3y|aIPC7Xzki_3YqQy%`MjU^GXYDQ#Aw|t_xOGE3*SLC(Kb<9 zlFZ{*G_9EVTs1GLI=Q*VY3_%!x-xV}b9=Si<4k#RVyHJ&%YGkC;43k1vtm8t^JlI# zGEWj$Ks_xMyp_4cpQZPphRS#@g!?lg6x2uh)-j!udXQ#*I@qfeTF-)I;d*KhrY3dW zKPs@gu-8B7+@v!o56 zhT&~6R2%q~B(am(dD*AnbXC)T;^}x8>9OeeO-wl^jX$i-QD>qQ_BlIHs>4*J8=7N{ z6!&mmfV=Cf&kX|0$-XdA-2l@0H!KS;5pnkApwhX=`f*~`>CvKXFBTD2Zx3fzs{u}=FBsl0bXwq5;BG)oylpW`bl9;Hd#KG0KT zG9PfG*jS}FodGZB-Zqt?@~n?}tW?4sbN7*p;a0ML#ha$*$^Aes{W|$G^HA&|_U3yV z9Hm1WN{7fTSq|G1oCQT+A@~#?FJYJUeGvTnXU=KxK# z9kLdhh*t+UD+84=sGjnIwlW9ptAxv`%ef|@M(l9+dwWoL#B7Rfh1*k)u@hDoibJ#% z=%sUq_^ZJRJ!8(q{jlN9=x`q8qJ`QjUD?+#tuu;3Q{D9zY83R2D8f;kJ_^X22OxjEckE<-gsK9n@n!Gb6Z>N~)5qmu6o>zj6jvvHs@HM-SNA=sbH$*`~yWTB?cm z#rieXj(-rFh#O2HSb~)~gx%O!lBQg^Pp#_)3H~`TDV?(ROo)fws+WQN^QFLgRT37>MCozqYiz8hSL z!@=kL%(x8-V=HdVt;1^2QNbQ9?hO1z))pM%QrjybnSV$7v|1pZ6}?*$7jqu>J}wEr zu>Pn2WK6JE{$=)2z=z%mpV24T1jef`^#qi}w&qvruh;+{oZF2U9ltBwJLR44mQV@x zQ?_L}t$oB~+>PF$nyo#kk@gj0KQJUPE8IJ%>U;cVZ)|2oVyY>VEYB2Zf6M#WiENpj z*VM{rIafZW4|Rg-Y~Pmi)R@T>Em^82d#-v{3+ehbFn1%Z=ln%2~op~sL%^*WA5;aPH9wW*h91-?WuAXj9U zT9+Gn{d6$LJ|y)3-36;BvT~tFJtwJ@AWf6170W91Z)6JiCMH8Cs~0lDFB8a-aPsaXTK6 zoy(eY<}+QjwabjVOKw=h#LYuV?(u=PTDo$W`C#PONtV+3LNF65ImJPdxgBAFC(L1b zlXk^Y60FaxAkG)&QSGeF$#U{1cWK7#r0GM7OY4O4fK4If5OZ0MRRVHf!}l|Cy~ihU zL8S?8w6?U-mKGX?pJfrCIagUwiAqd+aSVA1ZV~QM->5dmZ)~Nh3#g2&QaH;+?5E$@ z2!Biblb;EPsJr1Vs8D0@tm`8e@!>K4$kNU;-}H~1B`cY;4E8K zswmFKIdTTO+h5Um9*+v7W*<>#W>QK?loNWx#c-E5HW0Fjl0;)+G&hO6r>)rBfZypY zW;%kh!3(B2u@lUy`5pDQ3{cVoXW^}yDe7IOWMBbi(OJi7Zk3YD9)vv&ZY7(j49kjD zQ?BIpq|37PJ$}V-$P>rahU^&l2v1kyf}2flWiyeG@)f;>C+T~6eeiAau|`-M+U8Lk zVy>FKLMO4R!OA3aX6}%B9-Xp8AsIb@ONlscuDt>GliA8X;pS67LnJ?t3pE383`$o*5afN^p*^AnB%zfq-`Yo?lXv5*PZ72jwJ zn8kdOHdFb>y;$HagN+@}E&EYWlWd82f2>lNTP8Oce|$&1#KEYH>;sA0N8&a3o97?3 zEOWqpnDF`X%y-$}sVe+EI?Z%Arl)c*SYB$dsfd$=4}t~00Ts~{X^ye_ni}k35LMT+ zu2Y-X?rdXztL2`OVtUp6Gw-J2aggN>OFM?{2K6S zKd8xZ)2OAa&C^5K!Pg+#gI|bx)B{s7b~*S3w{l`!Ap8I=rES|^!sc=VVlgP^7z-*; zGnAcb12t7GZ3p%OWsF=U#29PEIaFM*I@wM8stnddDxUb$mYKnaD!;dPskxtRmG?JS zTA!kQWjk>FxHWiy>AsRgp9z*#R~k)=Xk%4*-SQeupk81qSk+b@F64@wDOv$oERD*K z4L1)?%ilo#s;9EwsVhRud_gzqopA&Hs7~^nVWM0n< z5^Kvjd|e|+8>BZ3P`)Efx^_*xs}8}NuxPrAS5QV?9(<%Ac8!>peb84@Nz+Qn&xDSl zFVuGFitrEUo>xKs$d)u#VvSfmxfGmZi}tQz2Dqy_i?Od@4SP%ejsk@D_79*HzHe7e zzEEe>8~mwUCoF+|%sb0g_8lFKp0Me9l)WA)&^_G4)IstZpR7LDZYmD7RVvIcE=-_l z(3g$DpK_n6J>xs@N5sqiV#sFSnoto{5Zbag-Ry5d6v8@!Lm)u1FC=9{HllF|Z;gDV9W9tv({&Q}^J_lT;2^gvSf1dxLsDs9=< z>J{Z8PA1y$^W;8SCcZJNAe+=q7^N;zHK~J;1%R_2vmj?t>~?LRvBLexjARhe zC#VLm8$?bN{Ak#-=HLTt9CH)QOPs~$>l?V1x(q*vKZSnig`5yd;s^7cwNwiXua*E^ zir+&lh`+^jK>d~Gj=P$|^hJ-vj)8LF-hQW}tNj2~)gTbp1fu zqhVvydU6$2GN!+HI0U2~YCXOf(UTvUGA3^Xn8({Je{(Mr{Jt|%3kHHgMj|<#P{C8% z^rSz`qk{*CC^-Itd@i9GD%HA)s;2uPqMXdw|4>m#d~k(Cb=k7 zBQKUsC%$+Z5uQvIEwnFzq?4Ac@PFh}?Wv+TbI7qtt&*y1--yM=JzbIOCe3E{sI}Oe zbQGHh7L)rzYt%h-U-_AMEhQdzhf7k&DulnS)eB}2rNWnm3*a>9t2z0$%CDly<|T^M zr}9hkBPb$FxTHQ(?ug8SRnkH=eDhEXc!k`<&_NiEF%Q(%shzzlHCPx-yuf#*kwSI8 zIrTHJ3LYSW?4S4!9OYie1?Xs{RyfQJr)C(OOjYnJ*Nu;(`uaXucKCW|M^S6|o*Ilx z)CBwwt%N6Y!+U=@G$>?W*&RQ(wnxQnR;j#niqetAF{avbgW6l{n=G73isxKOUkt8z z4xWXgn3VZATm1KwEW@6{5iuaRqKJkHiNo)rjBu6G~t6NA{Y!7|-<=4{rv3w!UwtKUB0A7!}VwTmOKL=ffxv%k=jC0RL)d<^Et5b2YGi=^zrC8!#PF32Ll&o8FlXDvuKija+6A z*B_r&8ml2xK`rq9G@7OdoYC4p{2S9^Y&1T|8g^6drQRsKv|3Mcz0uk3!$DBR9-Z6} zU8L=L?O0a-u1x3ptGViBqw}*sUx3aKAq=Z-#*OhrrVzXme{5pZ=FA&5SsiHf3`fF? z>H+dM@DGup0Qg2|EL>*FF&!+w-~@YdM{_Zr>ue<4G0ag%1z$_gc$-4?_2=O=bZ^@a z;jsR1;H~aVnJ3cva?r#6uaS1uhZ%W?^QPgc*bbedm06zst1{bdHO=1@3idbG5`KDH zaM374CF0K3c(6FLo|XxkfpKu4bWmBRpAvaATPZI0vd;Aho~p>Cw15yQZK?E##4HYt(H0iykZ*Z--&AAC+lK<4Y16|&`4sWmWMNK zdwuufwsT2x8@eMHZY0@DQ$xZFV}k6?huLUDg6p>#tU}x!XwlaRpc6p z&&4wa5C1UtEz#0o=w@;y=q0{&&Yx&ty!6)WVV}d=KpDeCI_jkP`<4DN60dK4~hI@BfzE&QpKjFe# zX8C^Cn_zPQ$Csn*VhM2-{VMJ+zLqtc|DXQCUBiI)HV1R#w%U4v_gWR4gdA*t+#x_x z$lu$3#8EBS0BoRl@{`?5{KtUNH53D(`3#9g=_m6!X8~H6_gPqIt|2GDQ!w8i6}kqF zkZ}wfZbO=>D&9*t!gcfYbC%U9ZjjsNTO~j9UI8t*Z0ZZn;(DubL=5>^_!B1S)%aJo z2P!9bu>9#SvUMWA!rwW7x*9#c+jiEuCv;u9#dKHa=>d6-xjq}k*U>tIQeY0>*^>k+ zp#Q;Qam#2TcRpC;ZEY8rkEENv35!wx`rfnaEi<`hvW2{4KOPEdDa53ttu`4~ruVZB zOAB~AYgmoFmdf6?rtw~l=HXoDE2)9s=J5N+)Ah;4@?!3IV1Dp^fHdzmrm@-TK9~=) z<+1#DP@mo-zd%-Fe_o54YA@w!B-oXZ*G(@}s=@JiCDT#gLgjlm)UdE?mmL=xhJNBf zAZR-(OcuX#L%9fdlj_H82KA)`alE&IXPLIh=(iJ!UcP1O5m!*!2#{VGFm~K3q-}0?_GOu3a=wRXBU0o~C3N zow{*qqdozdK|k)5so36iRZabVk@dlSYrFZp1Kb9W2a7$kBR4Kt#THeg&S}` z)(j-Cjj@2^R6dzoVH~ca9Hm<+8Jf#m+}tB~uCa$`2_E7@!8v#nJ6yaG`~t?}in<-= zxaZUO|%Oh`NThqIlm|?ig{J zHIy+x$Sv<^;HbjsYR#O3sFb(AG4D*(qsVmiFdAqG6sO|HC zvu+|=n2YF)6`}8ZUgPmkN$4Kha*`IF%AWM+VQJt{@s z&vTnkSq|Xi`OR(3&F@GTxeGL7N{ADMS?;6c%iu3`o4j;p7BSfU0bd6H8RWwmCWV*} zws1wBB4Rn+*W3fRLx+uLcoz2#oJcMYMl16HZE)imA_1!V068b&rFyo;++1H$71Rhk zs3{X~t6{yGXy0)9e9~51y;83x}m> zeh65A_R}?k7lumZD&?`?MHtzrHmd%;xpQRY3%8X-$=;)o9dp%_yfw{&)pr&!90M-c59q25|; zL9{4Kclp(*xHdz3;m$K=f2%7sA{tOXy|n*LW_2zs-A568Qe6c`nx8Av^DY|1Z7NoA zD!W9#CVPn$z%JD=r{R-Kz1>6fEeSu9FQ!bzD_|Y-7;_%i);X4Hm06n^$d6O@iAIY} z@e=>CjVuo3pzW(~veB;ij(kpE_&(L2U2Z>Z?Bo>KQd=a2NrOhNfdm;SEPUFv#SCvjw`z^(N#qypfEoNs+eb665BjwrF z+u=8Qy-mZI19-XQ(7UTcjqh&|zR4~@M`2Yq$9$Ac<$5_@DDMo?ceLJ9DaRf+a@F^A z&tO@OX1<}S^ikHOM+Z0BuNjkGm%^^-l1tMWT=RQQ%y%T7A$ zP*JH|xEVTvM%u6N?U_Md8%(TSgKrDklQdnzQcK9;NqjVRH0mV2!oAg`P$kPny_Ecy zTo=zbcdyC>`{~z+N{$L(o&6*E-qJL#P*_CWP4=Sr#8zB;?G-%asIT{91ufqoP4~-Z zK!l&j|L6Ui8v@75SJ{@niJV&!h(7!kX|_5Pj^$ib2{_&$q^H}O1p9{0>5mvw_=tZQ zWmpf59qTpw5f~JC_d)JExNQFo&Xso&m6;0oy|@dEaP)IbiQN;7V>jekwGrG*WlHD^ zI87nL?lK)_!=s!ERmmzjz+TjYg4b{&%R=&jzNzQYW zun!O3;wL4~vFh4VZZ9?4@hf{X@g`j`@{{ZzP)AmYjO^q2M<`@ph?0N^I4R`6_1-j?p3BS@%WlF zpDdMpjmsqBlY+27e2RP6o1<;cC)TG*2B_;?%^o8cgJs@l#?-bW^ki)PWwUPGCB_1Ge7h+-z6qZzM%%UM}a|3r+*H)O;^fh;QxeV%_qDj95~Kj2bF>C z?PI-3%KXiX*~ZD;)nml}ys7jbY@)D2Xp%P$pC)&B2eUo3QhIb)q9y7W+@RPUC6$#h z&I{t=)k8#<6s1qE(i3xVsML>cNjKE9#S`pBc00H(-&Ge~yEs@GG0vb<%Q zd6Eza=I6~u5?UkK__Az@yRul*a!EM?lauy1?h@y|9D0UYN)=l+zhKYovLt)(nj&Yxx{8@021U z$4aq_?4!_WxKcfa>#8~Ej8=(XY)K*Z*wWdr;J1|JIKWq=w&KfbBZDwK6zIoxXNsrn z;HNU>YV`F)YyTNt-M8Q%lW7oe6$KTxWfv%a;jQqRLi2vGJZ>Y=mAD%!z`qi!l>Tfi ze`@Cy+q50et?!6{w3hF|jKp&s{Rz!9PO1U!^XF^Dh&ELqS_Y1JdO54+bwZ7#i{eUi zH2xd6ME5KY8BsBVRPzVJ+tY*mrha1v@oAKwD`U!H$lz5tnY5)mfsC@!m+rgiuYp^E zzieaUhvj)#O1R9a$<5ZgP{6y5b?1fD0Cb>tDfiembot<2Djra%r*Z8RG=g6xkRCTG zj&~@j^gZymaEN!4U-TuQHtVL>F{hQ8WcQ?~xWB}1-U73gT5aFTM8{O+8j!zw2kQIa z8Dp9k%f)NgxEkOax?oy>n&zAid$>(Tt9YESn(k=r80-%jxN&pWgf*&NIpTAJ|MK?h z)tnGFwO{tX00+QnwoV{uT^X1{Hj*{nirS#(XiLxm8#qZO8)OrCmI=-yVrytncH|=b z6cuB8NDXXm`nN!FWg4&;bb`UiFv>cO~+kLGgFk zS(aHT%~Cr_>8KfOkvvWPB=>U6;OY|BZ69D9n1-iiuCSHxI`ky-Vs)inQ$L~H)|Rn< z33uUBUl)3OZXKovPm6>OEP8bwX}F)R`UBT zW0HGNq*fC)R~sTaxE9V3aq#FBVsHdY?t(ovku$&~PbRs4Y zsd%tv+Wi1HW~D@G+d9ciq!Q*px6*c`63TLN-I&W!`Vlgt9qJnJ@HYe`Wr` z!@|?lBVZnCryK0I{*9hx1&mK>YroCDpl-X12qrizr6-dr#+d)0ky-=G>`A$tUleM? zKZM?B1TljjPWMBMZ9UhSkLHWVj~6HI_#fJ+Pf=@#4W*mlxqqj$Tz#g@h#qSxiM%p} zD>5wu)${&V1aX3_o7c-LL5$Q(t^s;jLuOKIXZxBx754PUkVBO^kW1;zOlIxM4mpJ> z4@QNeq}IMtwyEZ>dF$23DNW@yrUof?W_ChH{5bT=KMH;lFIn%|FSv_%iyvCw+vcJ$ z_pj7Tr>QOR#c?^chu$fnns=V%HQquGkv_?*f)xToO=i56Jf=k0%jV(qA^JEy7M;aG zwW|LaU6Q&8ZLpd5R|cwg<&wfA9MWIf?-`tZMP(MaLL71s?Kov zgYsNiQGrB%_qfT-ZOU2Ye(o}Dg!?F}v*QSQMa*Goc0o2?6592z`n$ace^R@Rzp4&x zG+#-18Ys?391Hv!y$&x^O7TTtuvOqjSvElny$Ns7)99e#iOFZ?;vK?w*qtd5O9b{9 z6RQ@?VeV(r@}O7irQc&lhc+b%&oCc37Zuc$e}gsodFZb8 zMSagLRSNm%lsnu?E~VTMMo>k;o%X}9G_H@f;iB;C+)`SB*Z~jEKS2FuAH|V|<4NT+ zs22Q1XtDpJwYGVDk8Mw>m#T<=Pa4LghacPOB{!xHkrkv=J%b`~DZLneDZPX&z`O?} zJEsKJ>q()UobkbQ_yL~PTyli}E*+IKAXK{L-oq|`Tji^bhwZ5eWO42llgrLG-i88N zZO~)4*db^gSDAf?>L?49NqmB}Tga&f)=JVDX-A4=o?AL%_h0%JF>x0+r^ph6j>2FsXTlbkeBCGcf>ZC-; z+5UZ^BaqBi50>Hcz4gF;-%MLEem1;if8rg@_cFSTjo~WC73xVKM)*KbwwrV++q7aC zyIK5&+~GT>43)ZY5!SfVP;u^<5EaT5=TvUTFgPQ;pXqN4dLOU>YogXct8B9ok~RWG z8b;W+Jkav+xO`g5H>tl^)p*`4Qk2=;b{hd@g^*c`9K!TRwirAa);7(opvEH~0qpsg{?4*&Xw@E?+vTLf(#e}&5oG@$RmGH@f`k}b*Jk&WMi z_C$M`8)o(^0Ya9Cb9uZI?1QBQ0=TFIxEJxvHR_!9neME8O8Knz$I8x{`h2YeIZ_XL z539UX5g%p=N3Jvuz7)^m&Wzc%94X``sgBq)>s(OQ2jj}%59>zQb5|DVDq%gD{ts7_ z_ShHj`^n>66qq4Qp%yS>da8TXIC=T7%$iXEFy2R!ILT-4m{G!$44= zuqX^ir^&ydO*@4r+dB((=6CS}dDM5sCmJ5JVyKV*{oN4FRWImyZK)sG)J0V#HSq zHIV*JY7EX;N>h7@>EZe6W8nn94E8eqCkJQBE|iB* z3*G(Ycj6!1Phq|=mMYE>zA{_`ycu^vH5`q^03)0KGIuo^N?$|SpeIv}o(~7p*>RBR z6>4KTr_80+W#)O7G4t8-x!rI)TRDlPMzLR#DR&8Ds@hWP6MBM5#h-2`9)FNNjXs|;R*f@8#3GQI;9J%la<9`S|y?b?oBxitHUtN z805|}t%q-u`P8nZ+FGeMahf>@V)c03RjlM$pE5rgR6Gdc%$L9{^+0eWo5r@bUot=R zmd6^CNl0>6Y61VjFh}$ecAN8s@#f>yKJPAiDsYN-17FqSY!X>T6k}TBNj1D&Vu0K+ zOaINTWt5@n2)PWX1)q@nEqn1v-z7k$UiGUSt?$yFIHo3X(go0u9+BNf zuR=~$-zc%{F|-~PH@|@&>}4f)$|Bt%T~%M^y<+sd19~1uad99~E+p=;w-NAbayh0( z_<}VP)G;&KJ$1gKbJ^%36Uf%ej}~I}3TWDPBMYa08uqAha3Rq1M&I`aT+m%Im#5*k1G;<;62Qqghx;Y&AQK$Ghip&R&9ri6YD98xd);(JGa>0 zI&p+{+B!&T4w8~8TCiV2%atL@L>#3zq0T5&4If)y$-pCpg5;)gy;=hzzVWX z6!RYP&X>z^Te$qpnz<+a5&D|^HST26LGcF~k#hr(<|%R--T}VbPcZ+c{3kTf?Ao!M z-pTJX5gy=A^6#OC*zVcq`VydzU9KKgZbPU3SbYf2ZvW~yt-0+ztivsTnWqXb4L4CG zb)Hn6tH_TG+(wU(pQsJ4=-<>N_xY5obb@~Zxt=@4jF#u^h(cxI1DuMgarb;z+1A3} zmVud3xqqN>_&a)ToRzVN?vH)gPtTP6a24l4Be)8I0_nB4FH6{dH})65vYP_loiT7I z(xt^9n&X}2&|5T5_&qqBJ>|ct7CP4G%h}%z;I7d`J_RLO9vFL2C!NOc zv24Q^a%D9N*K~H3y5Jn^=->#hEnAX*MQiY@{6TA^)gyi*y5#BpI#N@tj|0^3a70hR($e$i;Gv|(PuM&7>U-rRo!_nGhP zL}jo1luQi1pdO*!ps5-ZpYs3dRoPq=rjsQ`o+J#HKKhR*ZO>k(_r)vowp)hz-e})B z7vxct;;5u$ zXYfM2S~!8G8MfOTLN*y59Id8sTAkqS4w@u<&HG)Opho$oW!=HC&XTwSCo;2)#`R%a zS4(k6UHn^e&E&h(gp}$@TkzYc>w#y|SooU1%zYr|VGUdjPf~w|!t`lP$&PX|%6ekA zEd$Q7b|!9EXWCPtB({?_$wBl9yRpmsNX!X8VG1m3)o0))G_zW8fqgeUgngI0pU;Zx z%s=AB05e})Yyx-Wj)t>WJ9o^=o8?e&wmp9P8nOb+HcQM+$1_Vt);4kicPnK*QBCg) z?(&+>;3kI4`=WT8;Jui=W!V_2j#l}b(dE4xp#pwb2HQ5t3-cn{F*nrC*KCOTg^KhD z+lTs}R6`ysy{4-Bx_M^lQ_0GX6Z9i|Pwa^rZd=2rwewTA3f+g0uqI87|!%Ob;KmUn@l z3v=Yg>T7vvu%X^gyvwiht`!dC=Er>#TY;KPtUsHr16y0(f%94dQ`6Ky&Sb{H<+usD zB==9AAKr`dv+X#9Pv~9vnt@StPx2;;S9+;AmQSXuY{ZeMX6c1ssr(#tiRqau8FX%v zx?aD8UJwBlNU|_k{z;5VWf`;5&1g(=bbPS9c^+3!Ys+n6gsu0jfx^$*k?g zLJPrJ;|-SMX)X1Zo6`y5m!LK=U0*HSvSu0y#0iVb-pcTemPfx(*jHES%baD)?-bl4 zpq0C0ag#_Ac)D!C=iUrCOU64S|Cp-%8z90T*U5Z5$N z(Zk{ns6M++zHYea3anG8+j>Q?fUT_i^v&!aYKGJU{w_YW>`#0Rqm-p;GkrWa(UQNZ zooP?-II4ngc&7zMqyS+27kdk~8}XmCKrk%LPy_3^x?n-greqW7XRTaygLoU}>x>?7 z&uq_8nNZpAQLD*2o0urY#LmP+=|)Oe>EZnde!_mi3ecC=MH)hmSCCRHA;>Pk&v-{w zI{RhtBFuoM+!=B!!LaJ%73Qt@Bv>3AskGp`V4p#_lvH(}i|dV}VaMD&)2K#3zR%# zLb=m++j5#I4R+vJY8C2@J^}6mge?vpFt<=XgRSPvTWe8s>6>7T;@xQr`km7pscL`5 zWLm*sQyKa^QR1AuHj8$MQ$0-ajTN@@_6BM z@EiOm`BQkewKv_5A57iXODO$vx`94y#6JStm7DT5!}k@f-!t4rF&hib9i{I`bzX9= zl*TxY)BU+sxmTGD^1Ymqp0fO#+!KxsM2mzwHn%+7eikJszM@IW&7`T7nOHW_=$n;N zTN|#r1ZK18nrEJtZjUhgm#!A7M>^_zbXen8W zSz*tE3seR|WBSuPb2WTH_vIHkbgbx%&gs|m?{d@JlH3?|Bir0RQs0l-+200(yxUR7 zF^_0wK7m%)KFY6Q0?`#W5m#stxS3fcSKQf0|4*yJyvmv5oSF4b{fF46o}oHO3Q8?~ zj((DILr$=*hR^7~gNOCEVp%?zc%R$^)2ko$b(MO?&k`1+1&rOXh8c*9=wjApOciDa zJK0k=yxbYhU(o(`=W%tZu};o^+)>HWnUu2b8&~}Sc%X%im5Ga-Z^7I-(c~D<7y1Hh zebfI;FW4t^W1)*?L0y!sOdI_R`X4BzKqcB!T1{0xtI0u!Hw0|@pCKDJTDzlM5v+`6 zoEs;y@x(dsfVe3NWB* zswOxJ#of|=rZV_K#o`L}{A#YRItA4chC_=n1Au%1_k%1(Z;89B{9$yNzZ=u9xWEp2n|2ZXqg1Dd zgZXM@;$7@=CI${N&tsOuSUAnFSd`R@^6o3ovXX?ZWTNk^e>CE$680e5iM%9jG8qUh z+{V2!?~ARJTT&fS?Jg?gIBob`mU=2Mv1Cg}8reV&NP{fv$UPebGb3YBQkdD;MTOL{80r5^K*mMbtn)SGHg@=Z#6PoLmWb-Ne^k(#KjaKy4l zakPFV7}59O(ni}ZE=Zx*g> zbH?&-;S#9V;>3Wxq}W1odkdLgxQX`m*vk(Iw+Xdj4)}fsZo}oodwNxpODP>&5mx4h za1n7!s0r7OsFL&-97*h=H}R*$k->-@C6-|rr3gbdo0-nfVCDu-hQ|c+L7~4DrZaueb zfE`5N$P)Q|O1kENCw-Td@3~#gvy8stAhD#lyV`qHCiOREv-2h%@82qvllEJVTl#7- z)MwC<%wV>&R&|#C4`~S=LIG`&nO0_pe$jV1|3{u;>iSw4pR=3#i{aloX$)42_d2k;?7GudNLNuH<1lj}J&5xg#;x>=Dab31kh-x1>4~oOQEi z=qG~-tdqPdG2|}IaFQw0gCBS@_b+|0ay;oXUy>C0$#g>Uns8liw9_Tk0)5OO*b|;I zY}&72QFW|Jm+guMbK(+b*@CB6^ zyJ*fJQ zUV3vj3Lj!;@|k?C?De+8Ts;fzACFqd*MlE8hRRf4X)l>1e921k(fe zx4CfVyriHHhh>R9V$b3_08X-p^9e;*;a_K0(G{jc(nx`pFM9qX--YL)EV$ISH@7HQ z4?M=b{UzwC(l1`xy4y|1$M7xHL-4KcBlfGk^FN~9^3Pmn@D6;CbH-W@^-yVhJE^_W z2KHuk`mE}ghB3Nr%C4QJlHz82w!cFDRmCaTjg%)w+ zxaNU(%s0cJs*@&6B%FaS4SQm$>cH+|nRqo=DwGwwQKxuandNxS?!c_^)IDl63unW8 zv;nn7^N|}j(wA{vwEW-*>G#cxZ5d3>t>c-8R$4Z)9A?;#3bqxXdZ?RNTGqutHf848ED+NWflb>2x@Y&&C}OSd3<1>@z#0U5@JnsX~{3RJC)?7dls zhemf4tJp>_%C+c!`VKJ@>4vV!=W2KsRu(St01TCQlHln+8cd~EjS^h7Ek8cr`p{rmIc?VsiPV)^{ zRuwx~s~dNKNeSR;U^4sOGev2jSGB%Qsmj+09fhV!&5#?ER;z$9LVb2W^Nu~LUV~F; zNz2e1=oh#oBlntvkBg1*5;Tz0iFWE|a0UH>N+}MpzTxj*V;yW(QF)>lmz~{9t;-Ie zDzUx!5}v7e4bj*)SU3jS`c@G&bdc>%OYMnVyRM`hJM&w~$lRte>8^hKu@47Iy0TY2 zb8V|pJ954EsDiUAXSxPF-_c}I&eX{JUk5g2+Ik9Nhxd2QFEKw-=W9>bx6dWhgv^4- zn-3FRA9~$!2w(y9=y$nY~&kL`WPA8WaD`_ZHY zmm=u$p~$h0+g!!>wRFu+-R^o{<%H|ch@MvUO1((gb-i3o>u-vj4?c}NX}r(1djNfG zz>?c(%^vt&ovt-MwxV$V(WO5uE~aQ0%JEs-YxJ&mOO8gc702fYjhUrs$|>hIyUlc z+SDxKZYJ=;J zqpi!|2d53p|Er*5^=AdGE(~`~S=>6ZuA)1#rt<8xQ?F}A4#V?dYgOc>D`!wk*UL=a^=ixS1-Mzj)hhh0 z;Ci@w+82Fqy_(c)ef#BaA;)V zV97OWU5ChBhdUBINs7eJpC8HFU*C1~e^UdD^vbL$vFg<-W_+{Y~0_an#Y>`Mc82J)7?8KWvMu^~v8{-FKw9eu{q-tSGfK z@^kAF*SyA$($XI9a(TNJMO=^6NTpI&B4yfDO7GC}d_f_zDRT17(#VCuzeY|x>yqAP z)>2og?CGwbZxdb0pY&=R4u3^7zizx9!BS|wut~B}gg6Bu}xb{_gRB&Z2PV3rXbHR+?Yz1ZOe@!DE zuZ)!2z9&+6V_hWqj^JANQ*&KPry@;Wlukd)wsp-u*3@Nvwb)hiVxcSVR0UVzaWO4- z$*0Kswf!QN*!`}{`)F6GbtR9D>etgX=25qnkKkQp`v%)~R-6 z0i#$mGUho&W((6?djS`PzZ8Fh2W|H|D`!+O{|GxX0sSNRU%(paV_#!zNGk+~^|MCp z^sh`${A}c!@gr0}y-U-Y_J+oego()=n;ONo%*xUrruxCb#w@M8!<>8gH(-U!IhE!oIBDFv3=7T2A?=@=I_^X9Awi4#tO$dwOsgbiKn6uUv?sad!q4D zTCUE4Bw26 zQippE6aR5ngnPEX@zY=#^qg4D>;sH_kvf~2;q5^>geB&~05^}28NRP|72;AA#w;a4 zJ3R|+pIpV@G}tCDhbt=Sc7s|@mG!Qr)j$bP>HS~9OzMsDkD)_0VI*1!nohNoOU6&f z-0*R_4rfXnA#IZXrw>cdR~M({Ii|BowwBV@^yyUh;HCl|ybIjZPLU6cf9+$ykPuC~ z9U=byYlc-H<7^haLhnLPgl<+5n~{kj-?aAj0mYrgi=Jn~9BDNc z$LygW5;^*f=&naqdsBP33Ro|A9b6uDD_@c-QYGa~1}*~FA}wgX2h96>spsUWwi(G& zu;EE#m6{3Hk<*F~e8pp!6X3NNUr@eiCVC&O1!`$4Ky13*5=*}`YAGLsZ^Y>to!BGA zBLWYIENz{X6|7(%8yJ$)DzuR7BVV@HcGgb+VN}t5L93&Qqqnh|=wak2^X;>}wVf5w zsYG%(EqSZAYOozEv2!ASMl}F^8oooOS@v=~UTk@#y!0N2=Hp-F)7CB6AM#tyjrEgU z&?wcW1l5l`Wtk{$5vQisu;%Sblh1jM*v<EK8*P0j*<3DHz3S>q_8KM ziQM-%*+*(0_RrKmwIkGf>|)+xu}`3utvX!JgWF@3nbHxZ2F2)Q0!ygbatrVR;k8Cg z`Or*eJYPjGX|Le_!k}0se3$OT+6dXKfGyS55Kdz<*4_$HP3UO}m)QSFnT{CtgHe=v zQCUOWCc6dltW9N&zocBlR`{>cIV>X=$kX(da*JRU?lJjZKSUP_r)XG`gv>i0O=f5f#$SQFaf+LAum*@!FWsIQ(7UJ+|mQ<@WNA8De_4W{l5fO$&8 zydXU+@JhR*V3B5_G5J?`E9!9GHlyrA;lBXOD*A-z_dcYEjy?`ln;$v1w;EkX9ay&z!C76Hf)Vj1KNa? z9=s2ZVAWugce1=+9!t~(L+El?D#~l)s3X8axul8yJh>Bg7o;>NQWQKFHWraG#D^L$xZc71O#JkW*L6ZKruRQ-zHG_WA0MBF3TgwHS*G|xsF;l&+ zys&+fPGd)G^OfezVUxwTNG)qg_kHC&+61DJdp6t{s!c!T#!%P9*9rNiTDhs>N@*zc zO`0SR7FS~r`6nq2wHMr9=ALK`WR6xR=_mDHLxtKTH&=T=$JK*{FQuZ?2Ss1#aY7$_ z6sUvNA}7j^oRae<=)m4JIhl6h3W1o+Bz-4VGDC%T$P>ZOU||TOO!j->ja2CXuI>m; zQOynBq-R33L*F(l*aW$}E;6=Oi_7ZG*D8HON2w?}}krLGI11D1J1Hg@@rR??~$q zqPe-3w{!A6uL*P!bf{Z$12qwDW_4RD%a2n6Oj|Y)TaOmQLoka!!~cR_9*kIjg<1r; z)X%budJk~E6m?7Z1FaC;rbkITL~Tbss%~VI;~djQTkCQLH){*{dXDGvs9^Qr6(gHJ zWi+yn(~GFG_8QJQp?a)YH&Igqn(`~|J~GMEOl-&Z_x2$ssK-2iQs1o&)G9i`=VA50 zA80jScXBPV%G(`XhbJc#7si4`mfUy}Tjg0CRd!gho2ETzMWvHe7F*BJiKSDwNcluh zG>2}3by3^;X(ku%h8S{|+FE4bMvm_8&z|bs>G)Tv9g!3w7QM{3ChF05xd(DntRFs^ zD{ZbWHNs5#A^9#+E#4*kU9?1g9Ww(TNaK`>cUO)mAFz!L#yiYTiXH5DhLmTZ%;S*+ ze?WamQT^yKvPpk&rbfP4=IdC+F4GD{`@-FVfBm9SB2mVn+NaVf6)l%1K`;1{@f*T zrN_f_pai`@tq7@54NxYz4_(Y#^(p3u2jVU1L-GLp z0*A*em(Jq>Y!{k`2dGMPT+t=D9<`dT22L9jEeF6hsj?pB1uAYpqOs{|!P9bIa9`-f z0=lE<5E~mWg4Hc;!Bw_T{2f%4Ux7mXUpWQT^PWoY%Jt+A80Fbx>{~sNIu?qzR|b^= z^Nn?$HuyDdBv*?1Mhw*#vPYo|$3^EZ#}3Dk5M^x4u&dRS*MTMURX(VHhx_3qx{Em> z>{6RbuZtd%UDyoj1^pWPr|>h1+h4J6$2Yd2`89|QcCyFUnWR2qcaSsFx?tViLLyiMaz!XVJ$>7bHT?#=nQCjM&4AC2Fn@OA|=s&o>Qi`;hpiJxr zS(CbX=JKbZiJ%qo4H{(Z!u*_rxbJ_+53{{0D6n*Bcnm2a3n{OVulzBjt$f(l9Gb#? zleR0P1Wq1=6bO0AUo4}eWd7QF>lePJ4hXHd+48d}2g1a65CX_V>nOZg)J1<8x=xw= zkHtIW8YFsz5?211biZh12xpPft|rx@=3j&cG|wrxZ;~fZgC`Q(RerYlHN(^ z6PN;=`)gC}*_)Ax;JyBg-5lv{AQ>~+p?>*Ml;F2kYyzfR>YCb689SFFA=@NgMG`<4DG%R8 z&y~*^-Sm>6IX=!eBd$CBKY9qWK>C(fMa@dQM4s{00Vm}m?LJ!Cs7nxzy7VXON#8JS z8;E;4QZ^@Q^Liis_2T7Izn%#=S>X27Us>XdB(EzdrcpSxXN> zj(B=%95t`-GrmxD60c$}N9p0MU`1a^?c;l}yNvJlA@FYY4;prKlHLSM&`IRdU>4z$ z{$y50XM^;kelSokzP!_H7}qCQ2&DqnK-!?xyd-be*m{a z>Owvxss>QX5#jA|am&zbV}E3J`X=?UHHjKx`;(cVE{HVHDo~4%23q$}RnB2=NF!jp z^I2{Seo*?kP+&nK`EpntRPaH~6Z0BM^k<sP;q&BkV! zConS-!gU?IQq*t z#@3+4idZ?T)|03f{u0<5jM48>C7I*oZRt1TQuxn<{n+eCG1)1w6WOV4;LF=zF_rLJ z`teXV<6m}DBw2JYt%J=}#W6nZf^szRY-kw!8(mZRmj2rQE_96UMBZUKfov%g4U!%C zO4LGRaqjQdG=CaC8~HtH1}y#!50HwRY$|-%)s#`@DKL z2xGUvcIqp3h?tx<$T3EOwLAG;gK23u$#eGe@*=tk^d4Uuc%2plGPQKNS7JGHtH3_? zB-K7#y2xo@}nZd!H>8;N;ohhXiI*siE@_)w_H#8uR;SphyMrqj+Bcl>DHye zQklY|&`D9nYRduJL;pjloo`;0R#t4C$sdWT)K;6i`RqQ&{(n=tLlx0#sae?Y!qBdK z=o$3Q>VQe^GTf0n5-0dq!joN-Ef1kwv90n;zKn0)Z;3K_2(E4P9ITCfC~R&iCDy_I zJrqh`SoQ>^>7tq1W+>W1``CV6fm4o_R_&8S07 zw+=^scMkz_?sa*6lww-Z>&h7+KEkge$>#cTBLcTYzkj7XksHRBHTN#;EPlY|l6AO# zoE;j8XUdT;MMH2U|v_{>Sf*`qN145Y;Bq z*?z-5+S)BVgCmUL%t&@4Svyp*xTVr}y5#+N$r<^P=(YMO= zGFC+f>Fv?&Y!0^EkieT7k25@;gg^_jvN{JsaxT#jXICl`v+<48N7F36n<*EqB&4~! zi5;PS{O-he(GK&SH_ki=>yHRLDpvug-05hxJRSTkt&U%ewxo~BoHmF^rT?_m$-Rk% zQnv(p6L*rYrhLG8#M5Z6{F7J%$svDM8c^%tJh-~H7Da@O+&|)9 zL^64syGwSCCSD|2#wx3|@xmc0#-~fOkP}kPq>tV(T3@Xt3=+51o~Ok6zrhJ$H9iKc zQua!*?0jD>sWdg*9)}t53c9V)op2J@Je7$Yx)Q&N;g|`5+QdiKAnPyih%ZN#1>Lc; zal`$`k&$p!wx%zE^uSuQ_b_qiq9@xxW=6DM7)(B6UgN!t19ElibGa720!VnB##@T| zlHYda=rctR3p*CbCD{qYFU)0W6hb)2VR9@ZP0obK25JQqHV#6k>B>dbNH2B8ki8}B zdzojH-?%~)8Ydm;@~g;$z=z;*e7rWp7%Fx^>kxq24)qi2K_!WrMyzwGwvtX~8{&oV zwrG!d!1{{pgwsefa)=g;Y70A3M}@BX26}^u<|=yo$5)rDSnv5JCQmeHP%&5;VzsnP zy+aq^^^g+cP;rerR(oarvU@O;kME*VlvU~?X%75V%q*x&TC|In&Pqfti_JyG1S9s{ zpch>)c+9-Zeh}TDWEv>_4$8$&C+pIi`CDn(n2fAAS#hOT@ zbC$8(pq)X`oaV*XI!8?aQD@kcM!YuZT;b@@@ZFPH|7oq~OM; zsC}rnAW!eFEd$H-Rn{PIyY8vY=_)p!TBCJT=i{n80U7C=T~MG`2NjvR?o6;jFRN^| zl-85&-q#VNQ$(h*1V1r;K zW3)6NP$)OT?&$ZlwFHQLXg%Vy=wrQAKqa{zQA8!#7s~tJ>jZ|t{j`>RxN&F^tVd^&b4(z522uF$cJ_?k9 z4oJD!@8GsSP9nKeF0Q>c$U5z~vIe_A z+*g(;t+*LT4f&lJlhSQlbXuFQ6mbsUR3!~7iS`zTY4_D;1=->(P{p^hPJ#ZRC!PZtjhIH%T1S(VN7y9xogrdX_0dF=@D$n;e3E|J`G8)U zmIMB>Ta4k6!Ny!I0rW_$8OmY{>3?)4BP5iMEKE9U&B|DAFnU?GJ2{hih&E_E#%{9y zfT|n6!F`M~Y`QWItY%m0|G|GF2k;ff^aHbXi9Y~d8_(?@=ov&w;w2d6NRgHWXGAJ0 z!^vUBO1cyHj{A*8p;huywgg);JWMN#{BBNcbWy%145sIT7#USkv~1Tmsj0A)P`Nf@ ze`q6pmXBuH16Vt0oRCL3j;Jez81+x{3#5ST&aQ+?Qa5N9eSlg-?I((2OS)o!O&g7d z{U<@Ed^7Y}pCiZZ!>#qP?xHL90ag!><%i0RlTs9z>noMt$y$q22fHtmtBT$rd%1Rj zBD||bBif=dQeW9kU!SrkS%813v($9ycl|xi z@n0w*>iq4kZIX6KI`+a#fT1Z21<8F=a^NabGmt`_lq=hRQJ3Vq(n`3NHU`odEmgyt z8}E@B%Kx-X^^4^cen5G#e_&#BsmMDHPs5JqHREomyNSniduoZj4VWpn^bZlbI0I6D zqozDCe1}|=_Jgk+bYg7qKjvO=yB^)K1(QPEm{?^!U6tA8oT9yUjHhP@c(?%-zrT)F zOYW^t3zny=r^N?fSVXc~;1aY%aj`9BD}Bqj>8Zsi-p1gbbQH>@hbN?DywH=_9chgU zYtYlQx!Mr>cjvdnReDL{f!NAEN-Ar;8SGQYJDQ5)h{@C=d>-H7z#4Y3>xM9axkM#U zC#Z3Ho!}X4xN@7FZ1>UG>F0@)^m8%W*+%$=9br=GZqx+(3-*ZT6TQ%p5X6d)+Uo_^ zd2*@lc+-qe4hQ&;s~Bm4w+&`dql}lKbj_x>I(R9l)2$r00?mqhhgyVJ*_Z5jguYZL zy1Fn=+3FZ#rNIQ^soy~zp?6EEYAbC7A6uzO z&Ey$T=4U)x7i%XRrt{n*!9y^g{iRn{mI#gD1okr?f@0axPy)c<_VMe;w(@@FUvY^q zWRjKV@n@iXu{qUGs&8{CQJR?I&Z(i)u9e%a^8f18$NxMC$a>?5? z>MGcWB*RJWZ1qFTb=Q001=Nj)k-OLhahU7x!rOIPdu!suxXycXGYWtM^yhDa@9wS6 zSzaqwKFnl`x35? z|1&WHDi9CnRi&PqyZbZ!j?|>amyqXN^}?yXmj%~S5Fv!khDyOzeSMV4>h9F>=54lE zbQ*kzzY)!mo37TdxZDNQ?f9pmF~!S`G9ZUuT0ERw6uIumHy%nKoM%CGt@yxq=f$m9 zT6`G7)W`|P+QW%qtK&zg!y&V=K)OJe5X%?a*+s^T(55Ub(!aP(GfMB_%!r)xlrNrT zUlu6vT(qh3OZ*nm$5x#x#kEJy!`WOdzBE1pIf3589-3Zx>;g;eAXVwUl0}KqX=wz! zJ#{QT5~a+Ih=@LAX+wv@^joSe=Y8+@>1UIo8p<|uZGqpwyG_N zCb%R5vz43fcG#OKoA`57CuoF#Av@K&(s&ao0*sCJOXaCHDOqYOXoW31S0$#PZLmq= zd0#WBlCT;3NS0rZ#2^oi0Ntpc;gT+cj@{EVyt=P;E`1=Fee%;KV}>?^Wl=reF+Uv-*`HvGlLk1}D{`Bu9Gpu>AvE}RkZQR{bm8A?B4KDxn6$K+!QbI5 zP>CKKZcI)#Ho#E;5uK?OIPZZt=GKAjEK7BumIi#}O=lVVmb8B9qd>o40kY2jdtvw3 zNnU&Ej6z40c^NjJg#TC6-~3n9Mb}cufsKQdQ|AD=qbbIZI<_EJlTYw~3L>5Q%nNrp#OqxJ3Nfpp7cQULZ{7W+gVCwsBu{XZIBB+d}a$(hkE;Uze!E~Wn` z-4>dgra+4dmLqi%9u@vnx}%M>cUVR29rfOD7Bs52pW0ElpQwy3k(1PI0h4`v$gPxd zUe36k`dhjy((S-@^Y!%QR8Mro!76E+LJge#Kv4YQxEY!Q+Nr&mp-P?Nr)hnig+vAV zdT2hFr&GoTM^$R0)#P|gEGYWGLYdXgtAjmo7dt<712sFiB={*IdI$Pw!b!0=RoJob97bbO$SYlQCl{%MA>Q+1B07x0?)!nz|>4S zW?OKIV_8^WcSmkUp%iJFU4Lt;QM?e(PHQUP&~c+9G)?Ov58&p563i=5*Y>aD8ot85 zlde&0^J?IQ{g*VJGRN->?J6M5V%|9t`L2S|iaM^qeh9M+n8YoS5 z<%b~ACWP-D1sdIz$Elr_-u^%Ne{r)pVjGt~$2z9A1^)m}@a>q>&={u3@|JGmFOA)Z zvRyW5m2IoZ>HJ_&2(|TI;xX!~`bvZ7)wW{og|^W9Q>j9AQ=UNcyiL46U=@@kv&kC# zF6%Y0i0P*ORXi`qO?OUv23WR`x-o&TWwvCny#GQf6~7UEC@DY!+f>EExsF7ELacW4@(_{ zfk*f|E2*9&c~au`qGjAK(~o$+E9zSH-!jkSKB^XcpLa3kG5;=}j^+g(^tG`ZNVLGU z{dwLJnAuhjoq+gE`^`%dOQv+gL{61HN>BOTifl`9oaa@nlt&g%?utAl`iZIiF+((6hR2a01&e)F8 z}=6)+=m9D^@Q8jeoQ<2JEf#_$=u6m zZ0ztZHa0tt9DD&k)vId(qI~2`R1 z{f+2Lv6+L7`4&?eV?V4*;kj(sXr&hf3)EX$N#+efQcbjG;V$HdxaTRKj3Rr@$aQ=* zwae3sjd4uXDm&MMYc`2V43`PST2A08RCaJHxRY=)G(c&hE-P+DF7p26xD`1jRRT|3 zc`55dX3uu@Fg=cenLCAzv?bE6plJOWIvCLD-bBb4jOD=bV6J}$_##vZo@b||4{TO3 zV-h+dYg=HeFg`GesupO(IwXLvrf*~&KVXSuIbIY8m_At@=rPU=deOmf`sS!wuWV$R zv5W1JzAilYLp(5~bf`$Os_-rD{d>y`f+(-IpizXP*)o!UYz^Oa-G@F^x)xNIqp zjbun`cSW)rY(gPHZJ@+xb))U4eQ8L=f;hg~{#W@Mp^mNz_&Vg`R;Q=8^v3+Skh6|L zFLB+>*3`k$d6`VSyzf+Ar@9@)WtM-CvmVX@!$H|>y6Uxh%SY?@b_}U$mEwnkFF0iH9sfLlG2b1v-Kf?Pd9dd_pC8F^JWkJ{5qmr}LaEFI!69sh z?|;c(r7XkB{tJ&Z#=wSFQ*Iw*!<)ieStL5O`@QHIT{HL_>tRb8N)+pTBzTri4}7t| zY;s%M!_i!$X$jBUrK*N!9NOg(g7t((KnNq_Qkx7p3 zWJP^~XSQP!dET`~`HLuredHDi4?&&i)NhM71^bBdu6^PeU<1E|&;HfiT>dJaErnfe zmnIRTIA2Y|1#-R9`1yKOSeM#B%wae}=ANuEJzMQ62Vev(^T){l6J|3qQh! z%?|yBZ(!aaSJd(dVb8RcB+ffO@#6wnp>AwNXa+nWm zyN(6m5IxXwo=}7owk&xG?XUiB%o9h4@#2u4Nlx|x2(7ebY{JYxHD2sdv{u!9* zIOy#bB*>~IjLaSqgsd#^aZ+FpY?=Q4VL$u)!__{O&$&ofw zVc#YM0~MrFVn1w2)X};D{v;_$lkGoTllWW6iu_B&e*XvMtJEDN;w@~Yp?|GEgcrU; z_$pwNaHg2)C!NIGr2P&3qoAoJ!iGH#+6Gdo>GlP|b;wh~u6+r14OYT88IJ=EQ|}nF zp=$CJ=M7Aw>gJ7ft_GL9PHenNG|r%SJuA?c?yQz_4J9YCONfnzJFN)ZQ>uuwY-!A6 z@~M1@S*#z2HNvKOsjkt4SBk$*4D$N*3e;>m8@dP%K!f2K-UXON`=Tt@Z{n5xvC%v7 zo-8O4-wJnYtawj%p%g(FB`BY5yeW>fNngkU?>zvs)r~<^JN~`RpPJ3~2HDu+g7ff6 zaK;k|Jfpjs>u5Uz15$nk>*^nMf1qmYHsF<ot5lA+7X2jmCQ`*(KDZJz zqnd%U!O}_vsYqj(4pfNF0*fR_-hz%%OjruFpy+#43s)hrQp%J3tJn~vw6CtY4Tj;L z`K^2B!;OpyJ`*ld{nh_Tebq9-(ehxmBSEgG5grIM$YFy$ZHX+Tnt8eR~G$44knx)2UCegV;%`E~NS%r;Q>?s4Z|WS&)FUcv8fC7-ja7OHZbvqE z6zlFUN=}m|nYWT%l2*y>p?s;Owwo%2)Fr=~>*8a*1F;yv9ABFI=J|o&bdL}!dd}NA znI~xr#jBuQgB@hLT1~Z_Dwn{d>~(MFJ?(WOyKWe zSEQr1qv%t0o9fYMqb!yu+>^fuKRp-mW%f9|ap7`esyqt6gQRgw`2oaJX)AR>{1-0@ z9F(%9#j=yVf(=&7u+6mP-WJk@)O5T(zB1KejPYARV{Hr5hjOEtSq{^^$Q5owz2Zp89_4Hlz)F%7!OyLz~k>++Jjqw#4%; zFAcvIQ^i!%<0n@6C&pp&hScAaj|uI_2pLLwp2|ZiQP0{1tD_WIQ1Bi5g%5|SnJSXE zi4%fm-ISY{^2@vx9fV!xqPAFXc?#?cM6(&o6Za9M@PF_~?y8uVS3au#D1j_1-b^1M zDuiF8_W^zN>G(d~;kXpYbPQx{VSC1$^c15+TI0aC6gV``eo>v|;EWKP9~!U3)s{O8 z&2{+KTTBV@peH!Qw8yI>1HeUbU-EeO6e9W>V?kHMCM19GrqG{!CN2hgN0oyAD4aue zms!bx*3dPu1yo;opY)gbimyOb^R$qM1j^6QPsg;^7KeBfb3Zkr^7s)^9 zXa_E~#a|OE;pgBAc3i!U){Lfs77PFJ?Sx?<18XE6McL@A*y0;cROMGm9ieUJhWG3 zTB=k~7OJQC4xm1Qp|z<*z9jw~tz9sh$HZO4EqQ{k!0OAJh~~kkZDk-<$i>bmz4d&k z1lOKw47CFjM7MlSDn*~h`&vI(#wZ7%T2eV|r#o%mebMB7k-lIvLtYm0^ZE_vu6wtnx@6jgLla(-^dkY2^;!{fOPdE9?%4fzL&Amv8%yfd}H9 zJq%ZlsmIT#H@9dTHXqt8jlz6z7Sup`h)o6`$!GW`;<)Vsl5DGr&Gdhu+E~0?w)>2K zU(_d1r|?Bmzq}dv2BaJ2RN6rQ5)QH+C>B_$#5T$T$s#|&&-C6S~ zn<0hnoia-O%YT?SjLypa!@S<=6;I0;(^xv6+QXNFZL!oMva#|-rIpiyB0o)OqdZf) z39g#|RC%RbEFL3|HAdP$g;3dROd#0c3^p5^2?hC4!JBE2BTr~xtd20@7P8%_5TVj4 zTY3=QUw|7CHqor|ROZeG&9NBzt0cAg@j(4WB};Ah}6f6%#>ZDv2t4$he1?TU5G zuOyw)Ix$~`5dJnO1XcbzrXaKFCg1^>occ9thyrRMwgx2A<5O0_`S?UF0eXQpq*2@U zm;rQ)sIG2ElYRJUMgwllkI%&}LBHuc$%5E&Mkf2OdR=Rym(w}lAy8BJ;iBZW0%dMM zo|j{-uPh?ILs4-v+#u?(T7tcfW`;H6&zeSZLzum|$+z3H5MSnU$$wBgZJnEp6Az(x z$%<5t9Ge`1)`mw9-6Nn8TgH6R6r7sf= ze;~SDhz1>MpN*klFSQ@?+qwjLnF)R($|u&QtEh|Q%J4=`Bg?Co5CWY}nHPyMcnreaqqygIs zg-ysBE(>~z5!}&)?cg%p4cY|7VcmpQzQH`S54spHPG}u=E0JgrBzRazm9>z1HRkLA3ESWqt#8={ho7o<{$JVN^@}4J5!jHHt4q zOoVn**P-O13~zD57i>E=5t=4|9Kp7d*WlQwGG{BXJlV7}L1>aVTTQiU^3?cjVo>59 zsFCfFCqnh{C&|6lcB!qv1#zXRJCTHiiCF%pP~H3+z1`CyN>9GO^_ILxm?cgttfP?V zYR@sxm1r7B3;02Do_gQ6pPO8iKn%7yV|pP~;_Aw|ThsQG2awnrj^h_o*Agvwbv#9T zf-L13^_Z!DG@o`mvelI<-JojF(LJzGm2!a{yDQ*tz(T6BbS?IbIM=>epGS_hPXb+) z+)ayxZTL)$P_IP&C0D42+*!37kpz_^nAnx-Zh8}0i%8;z8)xLvQlT15t_!fJ+V&2p zkw2Gs$stN7_E$NxiOTHelR-d9)H;MFfB#|Y+S$=}*jZSiYv%lN+0-Fx1 zkS*n{0+AvSc4oP zrRe|2RH2tIWdwu@*l*#A!I{Q0X~S-U!iQk zu8)w)%k#x1(mr_?vjyMBUO`^s)%5~$u{0p56?%zXLp5`L!kf~&qMtb7&~uemL*~xF z6mAMu8*?0M@W%}{4j+p-8>F93eWX82I~QrF4`FT8+pLNQkJ-_vLhyamEdocK zXom3zXq>f5YfVp}-$&=c5$=I7kBayf8`tFTbfT>tm6@KUT+5h4{YdT<8LyrKhqRw* zkJ-A?eNqh{Nb4J{WIwKZipSDseS_nZLnJ?FYnr|!_;Zl0-_{o~ADq9~Dt?a222-Lw zo}>IQ=kXkqJt49Q90>QO4b~D|rP}dEHqCC87=3YcDtbD2v``Ps4&r=6HCeyFw}h9_ci9NDx3C-9o^Fj@wR@bSg>0d$T$7HI z&yZb_A)5>V5o=lEuvz#R{*L!IEfYTqhOs5}4Q z<=N)0Mj~{=+KbK7s)s+jYdMmG_NYU)3>Qnk2c*v&m}m3|okdM)r&8*r-?Fa)7x?4C z@6NvXo`8#-8l3BVrA?0PK!Z}1;urQNY+2gJx+p5bIfmNvsNb9|!!c|cn;FSq`x@`U zw#Wo{3+q9jMbFBsAX8!-o0&*39BdeMnQaPK=-j9#@`ms~p(PYct@2~*hWocb8C*ZP zEq{ur>nT7|u@b^mzt{9mx`Q!Z1*#cU&E1k`VO@!p{v>LwdL}RiD)8;25^~n_Blr8Y z`=C||u7WiIqd18C(?1u>i1nA}If1NW$1dPC+vd1Tblbe!Eo?-KTgTT2e^ zM&eK5CTTOD5L!ri;u;W?N> zeI)wZGNSovJz>Suo9&A%q)=@-(?I)w6T=_m1hTAoj{x*rc|2PUDyKIi@^muk&@a(ev5R2nE`HK&5A}w zsrNR;#Q72-busB6SBRU5s+JY>17WWe;vXm^euUWy5bYUs+ILq*CP>QHrFq{gw z@`-N(EB$BNZ+W8ki*C&wbQAF}SFvO!x=Rbmx#VJQ2XaC#qi+YbRMb-%bz&!kbw)L6 zCUe7Yg6}F)49Sm*#CBANfo6oNlzq$$MZ&*zxt0P z+>a(j)#DyuZKJvi|OF!8gXg zM)~F);wtC0g(_L=Nj-@sSj!}|>Uf_q+tqsc(hY`(Ghh9Ms zvJA&>Cr?stxDUa{u`b|-keqyvOH5ge#CicL;@9A%=rVLSG#%dzC5yjD$xv1Ri}B=W z_erT$v@YASza#lBvJGA-ZpN~yUhEFKeee%+7xHCjv~(c=hiawQk60s(?GuBq?L8a~ zs0%C=o(_KZzg26b9VIKKeKYPDW$YmXp+a&4HId1ycO|kg;5QdY$Dc^$e*XHj>SLOU}OJ6;Z9e-I^&5lRm`%>36^* z%|qlwXvn@Q*!4Z8XazXTw^%wRJtiL)@iBi``tmhXQ1nKG(VhglHFaylgFjfs7$)_N zDwqa^hO=Gqn+_pBr01m1`aeZy8P>MifZ@5j!WayzCux! z#-&ZMGIw`(IdeI4`||y!zi6)JeIL0W>y02m%_&+^6l04NY3!s0EgfE5EHA_cB+bFQ zB`0Zn3;$aFhSm^Oiq)JAnj5l*_L=`J+EA2U_OC8WT$|i3?LC(-HpX5R&Pf8L6ybM~ z(omC^S+00bdsS2f+Vddb_!3$edL_5_ZHNCU$toFKLX@DET4*I}UEu)My>PiOh|r`Q z3@)uIW2c*MU_<34fzzU7iAMw2Me`RXR;*+B!S9r^zzLS@&>zbyM11tC}@nMP@I1jQW*5SDzF+nBE7s@s|kSxBA^XwOy5Ep1XOB zHi$+kn|_@lb*@$HDt)L~FEY#Lhs(qho;cR04S^amgPZ{S#BpFvPYb5DyE%2$&_eeK z${_D>`#kf7T*C%rD0|a=9T8n^xOGY!C4rsgMv(X3omiv%w}yH2V#gZhhUt8>HE;m3 zsshpvbde?+fsoy75V^1TE_H03#AzssKcoTa0dhc!x90G@pdZ*VbE*+JJrI&ne#IZp#9`0A4C7f0cNi56!jYQOzu;!Nnd=us{7F zd$4y2+?aZ(UeEO5aKi{{G&qK8z*fke#uQNlNzCI-oZtt*8F60i0%vpI2;`=6`LO6_ z?f26#ttR|aKYshHgD!R=d zLDqrQCMCZk%vUn70@}?Ly)84on%-pGEOzC`Q*+fT;yUm)X*UW&-NkZJbxzN!gNMC^ z=rm(pX^vDXPN6rkO)Oggm#!*#(7y*~l<#OfJ&oEetfYozod8$rF>Y(%82lLfN7Y>L z8Jtae8*|)!bW8j`bvbkwLZcg?GGqtq1Fr*%)sR1I2)MHKPO5{_nXC!0o)J6V*TJ=SWx#wsTO`iwD-_-Sh6`J~_CnQt<1S*VWe3DU$6v?rd9 zrX!ECWvRu1uIdv)L$b2(l&LP#Mm$7Mk*08Vq|DcxzQe9#wzQZ#&u*MxaA{tfn{k6K&}2$Em6D?sG2Z&VPYjR7=fv(E-f&#v5u` zu5qRYN#y_z)G+B~$$^1MW_+|-tmHNmlngH>ku&@RxN z?H!E2mEOyJ8ZrHa4e%TAv;MK8yEw>K1zzLXfoZh+$uepxUD+GK?r@Gmx91-)2A%(b zIZT-jUL zKc;Kg8K|0w!6WHkuJ&+<-!dJMb#f(jv8=-9i>g20TtX5s=b8=8c*3Dyzcn+6xY!-twPeKWlaJq<*^PK z5$m5l#Wx|i5zL5vtg^tMIgb66yBCX*x`bK+Ve&hA+w&k)5KI=b<-vKd(gaAO8sTbF z3YVzJJ*R^|tjoA|fn|uvR9$=mTCq0bPKU4#TBxPyAn+-6Q|qVdW5v2Fg2#{#jdvw^cKdcP zC%LXzFS0e69;nYehY79+;U&8x<8*U%?mWd%3&vw6?6uf zC*2^9NS9${av-mYK;_h658y34UDfMIjJ+4|Ch+1~Hqkr^yTmj!3+Mx>M4YG)V78-q zMLxM2zmzsX^YrJWpIjAsv!gQm+>|R93Q1-dIzZl(CsSmfHI2CLadw3v)d(MMjD@NuLC}2LcHFP%Ag8fa zra1plyfHLXn1mK+P9bg4vB)4pEZkS~kun*N@ZYtGbPPgf8Yy?+jC(X$NWMty1%|!X z(S3?4_8Ofiet;-{k+NO=L)*}_*uMgN=ZjNk>W}#6lT*=#hF47G=H*ETdkT53G+zOC zBU<*i!yZFbP2Y`=iSnY+Q2~*V=~S+=R=SuTt@z>xvF}O?)ouDleaLH0t|8a6y~wa9 z-S`f#3Y^6s3xBz=Cki2vmBKqk0_Bb1Q-b0OzHShrn zL$iu|ma#lw#5?O;yPF&0u-=dj=r48=r~3h^vEGYxalQpIxQl*R$OP1;-N<)(iRZL0 z3QxdFnWf5J*+HPS^EY7&EZ@{gIdkeU5>eO&M0q?}g!$g0Uh5XiTk1<@3=9P2tNbgm zw-A@qT<8Fgfog>~`7zmxZDu~r&LF#DAs)m*nzo)|pIWl)XeqT+2A?i6f{fOJQ@}d1fOUXxe`WvXLv9MTP=Ut7+khrAK5(DNjhDrzE?A)V+dVKvc!kFzMvCG!pu((f^`q8ykg?U0O!jvoWGHa$ko{_9E# z03bYMp{XiSo2t*v!6!RbznZw3p>q^ zQeyBPp@g_>x&=h~=YUhdMzPhwf8(|~oBJ2)R*11uwB{q)n@8ET<|c5dB~pBBS`yU+ zswE&&8g&hREq;?Y#i3TPxyJT#8?tiX7qLXSJ#+Pa!3#0_TU^^MtYhb!3JS7x`eOHz0<$gw-K%gI*<#99vT$0UH&IAO)O5z zmr{$AUQy8vs3bJna$mksP%~u2u8LcemnbK~cqz!u5sxWE?!HLnl=!0c)^#a1%Qo*D zBwJojycDYyLj2g0beluTOl^Q`mOd(;#=KEcSRHF&>S^axtw*@vpNWj(*U?1~nelQx?SdMqhN88dT z#2(TX)hOK_vMqa7Jn7=SP5CC=Ap}Kt!cW9F!(?tYKFxQ5&b)3{v8nl-``;ERD>w9?Tejp!BRw6M(E*tkvk z>=&4&)}jPe+1wbY%6oyE!bZzYWD2H1X9eqEYw)97G;xNQM(vln8CS|@ST+0_x$S*u z6oQqV!^n943%1K#A+R#HF)|)G96TYlVP=W4@QXSt&Is658T#yC>%anOeDDD`Kk+pF zn`w&Sd@)v23YaFCR(ijoOO?LBU~&=k$HF74jVtj_@I1?0t~S;L`0IU^X$fpF_cOo6 zAbG4%Rh-A~CN1)0%g5N4agp@PtO`JqTnHyf*YN7zI${uV(wqD?o?v&G3ZXoA1?sx) z3(}oS#dD0Kk=|5e$(UFRiw0kco%y%=wdfm86|4rd$4aGKV3GrKG}c8NSdZm-6R7?C zdL)I~EmkmBR!%2fr0MKFf21@N|AQP6>+xFI$<>vLL=g{#u1E_~4vH6qM|c})I#!K7 z1(#UW{5iH|uox3?oakXYj(rQ=FdO7@bfTJq?2kFXC+i(dL`*W~dIfnvY0| z(;eCKn%$GK1E^)HU9>oKz`e`%FnP6Ud@*jTY-yI{2?oR&#F*lG#$205&M7V=8reSS zZ_sIl*}fgYmi`u@^8T;77-)_3ps-qT1L88i(p(+f7OH?Xf@c|rE9q>d>jr*+laZaK zrfiHjlAI$w;~8=q(p0Prb>njP?KG|C%CYNIH#IBR-OeLOK6^vAATJ<>(AG3Z)~8#F zD!4jX8NT50?vsVKqE{wOUrY_s))?wW5xf0D~A@gcwMob_#BR;YZ? z2ILh_u%0B&!?O(&eJ=F@|3d9RFxHsNP{yV7P2C%WN5AM3k@SKHqlvLOno}Na1LlIi zzj2(ZE?wR^1Z*$*^y%&Zr?gI!?oGPv=|)5< z$*&aaBg_7h{`+LzN~H;ysEQ2G!X?yU-k5wKd3<0RdOPcZEE{MOK!Ke;GnjnhiSxW{TgdYy{<#~b>%Dgy1{;cP?40p~(pG~buqpL?2_ zORn>8*8iaHcoygnI@CEcomQwhc|e;BZUx_RqJ9LNg6%Qx5IewwO+)pQ4OdMIVVujR z)*7#vZ($LpXNZF;fh!pf<45toP%YzC#V-v(1b;vNiF6R$kNVhCfyUe?>8j%g=EQsF zj&aQgbn0|^1y$&KEt!~+)lW)it{z0&hzhlo4k+dY6wUhc(-9ZyIgllCW&oR9PBB>d| zAS&n$(6#&&Ht3z)g7197Pt`AHcat89T2@()sS|-&%wLUzYe_eBj;(GQ#*n=7nZ-zu=gS%bVvH3ilp+ube**G))_XWDFC4KCL-BWF9mfRB;) zuBDC^p0UPx1s^kK(HYuSWgCznf+dC4pS@E4;#Bz{|vA+-c1umr9!MJ%Z0; z#=y8&N#WrV_7G{vIg*AxG7EgntOiAXsO3Pw|6cI?&o+s7FaJs&~Tp=C$Cpm`dowQKy zjKn7i*d#xW_jO-PnwXj*fpWVbf)ABV$}jXId==cAZl@%p?m4a8UIb>mK^V$KM{wzI zFT978snH&gT%e>9+QN5zn-Dd(OV?5l^Nr&&kP~3C^g-9k(Aks( z>*x`VuJB2uqj;UP@=f6SVk`EJYbtU#Yduo4MGN7iFUr@-Tb_lKO?Uvxr=~go!5SJ? z5Vw#MxhvrJ#`gRN^-sDP!bx2{%b2~sQd2{rlo`_O4cHt~BT%4%ngN>tY5qsrDcB{T zC0AV=D;zhr#`d8%*m9~9@B<-aItzbRyp?9`Ibpr@%^2@*3H(*k?Ix_5e=0W4%!#Gu z71CQ_bo4>f@cn(c5?0yz;Lg!M{k%9$c;^7c4F;ePhR$PyV^acA>U5?66GSJs1Drs1 zGYt_BC$5v~s{7-GQU_=!z9{tEb)R3PJH#@~Y8FRt13h+ajjS)mV_NhhJ;l+MTf*mH zbI_mYip+D!=(tMaXx#>~4S5?K%{-2uL#BBmxti=}29Q!>JCLIR!Z;tPOWq+Ta(?jk zCf*whUxL?(Y@ivDY`Sb-vVE9ozTZHZxx6I1^q-%H*5-AU9O5sFhCRrv#oI~%d${nE zsYEr;IuWbOuJQH6M_{@5am+ycXBvZj6D~@vX@UGKj>LvgEPPw{!`lWM3C-IvP(PY| z4j2(R`ykkq84YT*E@-#*Be}8QAyp5aqFRbKO4#b&%vSSN7ABhaxigeDhFW@sv}HCR zEzrrJG699Y(qF~H`uqfdxx|X#ZzUyHl?)L$*Iv^I?7_WQRvza2O)A1pe64>tW%cn_Eg)G~@mr=$x~JLzAphI>P(MW_+n1Aoew(W{b= ziORdkvWxACt+3Kbt1R8O8?6N;FX3zCqTDFpnSP5_=e|b&KqhK`qV=60b@ig6$&Nsx z_Yv5{Z{+SY83TMK*8w`6E+2N3iy3cdNmn9Q`#X?3n4i8CbQn171t1MsgWA0-HK#sk z%rsLg$dbeaHci|U?2N0xd)PrP7wJpx54J`G_gP_$zM7mtPbFfojcTmGnu{c+OPKveJp2N{RJ z2k?qw*HC5b96tcjsJ`L*sA!?4JT@7?Z#YI;4u z;`cK3&C#a$P;1~U*B9>A^cD62Y>8i%9{E@3Q1Jvv0Yg+ZvAPfj?FhU=-I^(e58Pry zRlFm0gN^qNF)h@O0vD;;z@4dqNN>CVFc;r;zGHH1eTk0dh2#rtZLqOyPnluc7Teoa zfo*8L7HnU5A*o4_#AhRiSWD4wu`{89iluI71^B3`1+KT%3$8T8@w{y{+sybM@O0N* zq@nJ$m>oz}l;EzqWSD|iGB-^>k&XH!`Uo``ndRvRe;^=*0xS5`& z(Nr`UG+f3gQx^Ld=K0XH`9&zBc=(|Qf>lwB;e-y^0Crq)lSp%{P(5^Z8t>$ z8|^d6&&7$9jKGRUw#;0f!*DrypRd(2>%>utYm8c3Cz?M!FznSk-gep(Kon@5W!v&q7_&=@{KSaY-Q6I?qxkuq(OHGMx7%^)sau&k^nuzmr$u z*X2tRR2sta)YIe!AQ0@&j;Fht>RIQC<%EA^BhQjGjBjjqxgL1S+@$cc^=je$!bwG_ z9BWG}z8th;(+ZEt36?WV)uL=53eVo_mEKv(<5SENB&+-eT??#J8gRd)*Tf1}f72y; zoUMtYiSR?L2uoxYsU3pnc3~UHQ-(oK`TiHFqt@3nu8gmagL|iD07xySEMZr8({)4Qi z5-eEKVP*lECEYYaz8A*!#65gOU@X(b_zgH4xGX=Dd`1p#jD-@vKx6Q+7zftyc9b~B ze<3v{gO-ppXjQtKtmjp+ooK+&cx>?i$H(GV;#YHD{1MeRccemox=JkN9~Ad8Ru-FO zzQ#hunY>M2Dahod5Fy>cj#%D%2&tFtOX!8AhRsVolD{Vf%|?h4t`J^icCbG!k3E9SilR z^3kST1<~Qp^0LSY5s)dSnFucY6JTs8Cqk_x7HDxqNWGrs$JzveR8P+OOOplcPydLNlTAMrkS_myr75Ez-Rt3Y)8}2z$Ye)*cpfkm2zLQOq?h!Q(7k7 zfC_A7%nRd6cV-*jO>ANwAfLv|tlI*+j2%s)v>2Z2e*-g`dR{E?nj;%%0aoDhwY}L* z(t7wXdzHvR^5G0x417^CK%KPpw|w>yDxSDQe{n%A{}a?}Ba?y(nRPF3M=K_l>b3oNjt84OO1jEFBok z0=}y1p*85s;vwWE*A1F3%o4ur*~+Q0wMvw-H-0f43%02BrzSn zD5Xd;UkA*<4q_=rjX2xw;A`tl?D^2fdX238&{DYrR?#T1qnRFj?cY-W$rtK(v^<9HoQIetT+MsNey)jA&tD5stJNl*56 zOKN3qja&_lOdxzlh4*@u*{;t;)^R_*#RVgv_skd?B<2!|6Gz;F5TcH7Rro{Mvpbj# z`stnu(nn+n1|rKGYrQkj3dXOd@pvUMhTqCY>wB^5SqUs3$V6Mm4p%j$^Zgh2%ZVpl zUp#{&e;ruw-kx|5T&8aT#A@&O^62@I6-+;YD9T~%svk^G+u0u4lsL$F()%7R^XiOG zND^s>>_e(4dtzZ`Csmzl8NbHApZWpJgXXdDoOPRX0z@ zN20gz)v(2EPre)s6f)+;#({xBO4gazLr+hA&=xg|`78Ltp3{ zSS9mK?kJG1n_zV3)&^U#Zo@8L6>KjsQh!CYA@fn@C*2zQi7Ht)gFIpU05o*ynYJ|Q z^CC&WQ8l2RqCU%4_KgN{8jc+0HhR0bS}CcGB64B;`}{29$;=ql4BskM9TS(Zls_n) z^WF#Nnqxu)79ywdoed*_p#QU}v;S67W8h|}E8PlOC!Y6=3GDUEL1qPw@?`OXZau5B zRl$xBA9XA7pT>>oE9-nVU5^`=i)#Y|;mZ^m7#Z*gMbcF4B3u)Di*I3PNX_x9)*R_S z?odeUOQwy`oxn)GkJ-qUldoIUzOA56JYzk;wp7y6O{|b*6aUe;8R%q5!#GTfn`wg6 z@~J`}(+MdHVA2-j;3lp{Pl#ZrLH-K+nn=^AcmEfl_G#u=O&5noU9v!NtEsEV4$? zs~;jymG(s)@ut!n8-cf!1f5>#Tzy4P8Z8c9SKxn|Ih<|5JmRd%llz~eqFzPzB-6nT z$SirMc?}&IECL^^{QeqjEA#?b03?YO3^NF03kYcrm-6)z#b9TeAW$;H46%Pvh}a0c z3MM8s<$gBvDTKbTZgm85zsiO<_HXwShsTp=B?TjJZ(< zTvgbe_?4?=y6vA7cOGkp4=()2AjdiJ-%?*SV%d&2z+W*3!1B84n!o&irq$-I*h_dE zVBqceZDz5OM*4>N*gvA5s|xkkUl*zerYcJCgxHP(wMGJL?rlT=cFi*Gp?)O}W=OCUoJK!# zjN!(4yigom#CBzFDjO#6NI=?f_IOAO_L@EvDR3Hy4WAhzdS+~=K z?1DL8MgPLsaabN)zi~SRRVTXz#Z$j;yh;wFmm`cno<5k<7+Zu?rJD*1y+`>tSFHcA z7XfZ7x$TYUDCZhuBmYl{F{Wv%V0ZMc-4L~k+Y9s+`gu>mwW&V*aES6^OtGp9H-*_( z5QT^uxZrD^+WRiGa#@?y>&gD4L8X(dRA`L_OS=PxfDE~t{f?!&dxt`2IHlT&WCbr; z?gzjUK50%$d-IXvdEu@_oK09%rC68=)hqmCD@v~9PlrMAUM@C2XQ>^Bae!}Nuqglc_ubxvV7TQ7RIbq$G| zURh^beA+|Cw}lImIy&AY9V`OL4jmWiN<-1FdLV^0_Z+A>c)f9b#4 zE%0Rxgmg9!0gfqLaTF?MHbR})Bdi^^CobZkg zktYnXV60A61mcwi&~*t}%MXBU!6axR|5Cjlsh(H{k2ltJL# zF*C|#U5V>Zgs2_r5x5p;kJZGp6o$iF-h`3xWwEkxt)+pqNRxuj)pm@Jw(Oc%!(M)C zV)*#@$t8{0;braWUI_2a?HGQ%X@vbyx+mOn?VfPy^Wk9?ak^~m&h4c$d-=n&8{JIN zV4uT7>$I@9%=(xXHTFjdw6Ip$f>B+={&{V~*>zWzjjR)6C-3KlCtodB_Uyl7r7*!Ql_j&fWy5ux?Vncg`%aPV@@Zz-EfWdCejkX{DxhVx~T`{%cp5X4s!^)^s@2xB0JDJ8(cx9imiRI>$ zE&K0V_`i$_C6QOkr=c;&%d)Qz34=SkhlkBxT{gaF9sA``U&CU(N2#yJh0?x=EmA*j zeUOSg?NmmD?v(AjGvEH|>!`3ldU<&Au423Rtyg&b^D*JK)hpOrYX+CNralQ*oI1JX zAe}R9VvEh;%3Z3ZOoWc4VP};8`TcNMsNOm3>Yri1^F@``Z-qJxjUAKrY5l73I&;;u z&Wj4euTNb~8y_eO&pO_)Y{+^l?M6`6~fbBp9{B)SYLLViV4Teg6+}Z z!?YilnQ&?Sb!DHHtP79dTeoaM?}+fA!#QakX0U0GkY;5=s?;wVncFJO{IOqn+2wL& z@{8J~oo{%;-;}zw6m4-!mmnF%82e4Q%6r)Rl1?kk2E2t zzx~++AUqdR*{}Mk(kH2UyLs5^a+`hEt zLi?5FPf}C+Eh??CcuU!!HOIn_r|b_;Nr3FTReAR2I}5@yJH887n>w{@z3*`NJ=&mb zNcE($+~0r0+V{W0{bxsnL$V<}>Fw>bnVE*N)S2mNwZx$cS-Lf|Kt7K}6c&bRNLS(# zk)9D>nlB{Y$IWyUdDQV$nVI|u;V}D$Y0rIwo5j_@!Q`~Sj+Qa@)d!2(pz5T9+gnb} z+Z10Jj?#Zr7~q%i<Avc^5ndP5%zI6Ho;nQg2^|SFO0m;}*-qeDkB+zu_QN|eJJA$#H?h4bFK&k= zoqbQ$Pa28cMuem;Ol4PdG~d_MchxFH_0wHJ$2C8faLHKM1Ts7|8TCVT9(1|x3pA1) z0(|o7n)jpA#NDhz$s*&5cNsw*w`|nR);p4q&Kd;)MO8n`?!#9dM zmYo$Zc`_7^PKU%a-x~Scx zReDTnjiocv!B-D6nI!KIajwcA*E)3EQ8~YaOWSfcnec5&vUA;)-20>W4b;B?k!R*? zss-9AlFIX(e8YT;Pmi;jeBugvchZ}ETCN^`-+MNH5C74%eJd+`&yQ-+mfs(fCk&Gt z5trpkp)J;v$_BzBE8$Eeqn!1G(RLa&b=Jw{%7ZiY632!gmFB7tNOH$m^K3mJy=cayb;T zbqOuC43pLnvfq!qFjUnc7@s%TuvppS+7V5VGb1nLRu>nkhwT2NU66YdT~)X+9!c?) z73!|oN{bg5ZsJc(LoJKs`$%!1hk228w&@|Qk%fR=q1!dkXA+03Hj5V?4c%5qmR-@| z1Z;{xhvDU^X!?fV$9K#n(6h4CftbZg*4RK-y$) zvV0+VBAHHmJa6f@imElw^c;(T?`B`lYQcJ$HQZ(2D`sL&k-rO^=&MCh%IwS<_76IU zxzS=MQbQQ5oejTZJ9|{2;Y*zl=Op3pn(<1jc^SJrI;&GvU@)WwVqFbA6Rc-xK2f(%RSGvB#jR9l9X=A`p z6@-JA71HPvd58R$?ir{QIwE4ZEn_P;JQ0?wp&uP_kw*0wNo z*4rokr_$A1o~i+Y}IQd(b?Sb-UO3mVg<)q!%d(WH{gYr07CXMA7ZcWyV3x7rB0z%X{9(HU*Dphhuf1DsY)XGVPCbL)HMr zSml7lH&5H3sDeD6s4le@Q}9IzgV?q1vFZ!l=EUimnL<^?I+N6up`%&rycMvjiDMEE zQI_a;j-613P~z(-J~Q?vqS5?8b8Z-$l@R3tI;Fv~`ZavKWHES}>=&`-uZeY&BFp%Oh2YvHO&f6cx| z(ShTN_T3tdYB>rnkT?yhri8HtH+LWBscXKIj&*EV_ftlg!8CA zqA+O;+cdce^$G7vO(tu>Hh+8kBw@*GfS2Os{d04t1uLp|2ghrNa+hZX0vtytSmwr7+!+O5P}R2yaaKWjd)}pPCqK z5dSl2DDW7Mf;Mj2mT;aY#U4&sa<_RbWY0Fhz6Facpr2Ye!zUn_T7xyK8Wp{7o-O~ zfFDINS}$x>d~kCsz;;9!Zf_QU%d){-8JL;4!+NW5o(1A-uq}3elA7`>0#l^~VY+u| zupQHwd+Blpih)h;4#I`}nt{X8AL%M}mWWMQ52(#;jFIp<;|gsXuCA^^j@@TMkNcWt z^|4IXXXa|q@_QSozhG&Or=E1L&#^4&4%{gzgM9^G(N|*+!ow3;{YY$0!M|!AE!(|V zf56y++UV{=X`oDSuGkhZ!9)3<+LsF7^{Me@e>8gm7^(H3tv^#+1 z1^1G>i8wUCR?!fXKRcxk=7Z)*_23v?+sIy)h=K!r8!OF@O?tpO10lXVTp;Cv>querb3+5K~9nXGCV<1+Op zGKVH=!MNN&;$*bFwqo=(<7(AcxQSn7ERUY!FCv>YYkHqruSLS8{|say~`A4>GTlb!{-67}WH>=^^5?co?K?eI^+2AWGu zypZavqV=Lf;NS4~qNhyV;%|oC=zXAv_9kX%s4rNo?&7J;Zwc&-GX$^kdGJ1^*JaeM z^*!Ck2b)HfBD>`!S*xF@YxfynRf%Sr8kufRNEAstU{at@@9G4t7u+#}{V(%fH5NJTPa2X&oTBev6e zRPpu&Vu|>#eQ#)U@tw58;&KbD&dFPC>n2Xg>}4|;o=|ttLGWO#D=zA16XUTX@B++b zs;(8Vzx<=9JN_}0h8^o}CmQ8Mq?-6x4jZd+R&Kiguh`Rjq;NvNvSH4rBXEEOK-91yG17k* ztcZP9)i9qzO`L>A#GC|HIPSaStRerogk^ZKV|_}uq|Irs(S_QDF$Ks1sjd7+VG*~5 zBC!%%2W_&yZxX&UhQ*~-mLc+?((!@X_;PEXq|c1Z3tRy_PJSp?XEViZ{OU{sxUZd- zP}jOXuNG+spEhqH{_7dwcqBKYb$(0XS-9QqJnRg694iov+$wPi^+yZt%7qW*W=l6E=eGVcQg5k z(&w#hX-%h#<+QW#lS*^Xgl|->1?myqz(0->z7x6DTQT$^ZiWAo@WsQmKhPVGH($ahTXt>Rhn&hh?bt!|a;#Ff_S{jQ4($oPEDRAH zsbhp4S>#EFr&+I9+F5CNxim!{7@QWkBtP}0^NUlhx|ZUdl;nLK#XUgCH&<~XCmHJq z%9)6p&9~y4$y4#W(lfc6rL!cv_u?d%0X(MfYj^SkiVy9ognyGX-p%5z($0cb;bHo$ zf29YME`S?+fd;uh*bVev9mLfYR`^FCt%L@F*~zH;PT`jUYDSwUA-39-ZO&CP3m`scvXQ@RZ0Fb{0YSHzAnw85S#IKd`CD zW)JNsWw&{MahozP)Aua@5iyqTdnXt5Cw(Pf%!kC2g=@q_*2?&~0YY8wf$OWokNRaL z?rHQ;LtricgtjV)%xb{|u5O`SY^nPYeB(~ZY$YbM&piRta(EKHJn)pS9N5a9q6Tq; z>`m;^EuV%Wg}L$&^U%N>V-v?p?@L#2a$Ed9>Y1|N+XIaTMR&L`<@4GPQzLkp+?!~pdd!~2w)(naT3aGE%yccemuV7|X~#?}!Ny<_77yxG6;cMmbv2u;_2BdQ`@QK3?JV0L#{P3x)&Ir{V;$5@^dUNu ztfRUIH`K&>BJoF_R8249IZ}cv>1O7mZ@a#>Q#N$o?pDngNW0v2-U)C z;s0akECZuf)-a6w!KFZ3D%TeRrq;O_43 z95}#%o8Rq^N;Wg|&HFyjzyrbh?fK%CeQZOzAvgf~aQ79LmV$>6bxj}aa^C=4*T>q% zXd8li!As+2uqWGZ=g2k1O`=(}?E+NX^P*c=NRw^XD{x=l=C9;!ubuP~aqY}Kx0bOj z|Eka9Q*c<+>zI+YvJOi`d`b>!C%iBmR&q^rRn$`JR%B|qM%24ZM>`9Jn$%x{{!`}E zD;~d1YhHIBG--wk5E9wb=q@=R;c0GB;jcVbTv;YpEfZ5!@9X(Syz_|qaj;IhxxXju zAeoP%tsEI5+UnM9zqOuG+czbRlWIrQ-;Mb3IX}H65)!j*Hj6UPF`C;THUo+asDE~D zcYmTfg@+&CTTu3RtA-^SCN{`uoZ6UaJTqq_U7qS{%S({Ng^IgDiL~4J8*6p7p~r&* z9*0^aZlcHR{O`FE+*+SY_)&L-aRqL+f0jbtb@5$19X;0)N*IHKX|*3a`GzLPOf?#8 zi|#57ZV-!0X7r>#X6~)?3^a&XyVY=fNVy7%#eR)U;yU{Ni5)lsAIJrVsn}{DOP=S!`#J%8r`ZmxN+eb1&!V<@*m>;y$`OZ zcQ=msP`BaXt4|Icnz5nL=C)r8j^wN;SP+-p=vmwE`zDU`6vY0nR4|~?wE`y}-MHJ~ zltx|a&Mw$LG^^lJ`l*6d{hBvCIWFg5nGc5x?xpbsm2Or#bbR-;Mrr%XHJ%x9?@*rB zxN+PzP*79DlwZF4a4My+*|iS=N_1hBeNfX4+-9=H$vaQVbU}c zP*WUD*(JwJ8+#9KRjodl+lc?%*~DaM~_$CxU$f~Rx_fGe1eHl>c;mvK5xeEQ*#x!LX!THXr3Ab}Pb4jinxyQ9=rXDp%ADTTn z^at}2A37i$CAYy#U_Cs*6zEHgi*TaofDiRl^bKZ4DR>im(_Ro^)tuTE%M8~?{ibIk zF-w_ke%D6STlr?1qYpP5LP5TAvz^2j)ApcZ4g(xfq=Zg^}ckvUEd zCr9}H=3@^JlE!U2TM9Wjwhym#a|7Py%^PnI^%uBRkz$R-M$MX!o4jx91SLhU49P+R+KwH$L;SWoWe3xuhZ6l{W~^Ubtp(O>XLW){39 zUImD*kUZ0*X{>=m-JRGE!c&D&tMOADbLfMP6T}whAmW#vqc7$ri@$3x7E@p{S3iDR zyeKu+MxcPmqKTNlJI)4O>C) z<7IGV{|xzWX%J@olks%AuW18w7*C}8xlOznAi4OMpxmbHeCVsmevHkPkb-DvSfJ<%KemlEdg)FH8Y1wl(fy*uI5Rm z>qe*b9CD7+ZclQpbaj!3iH55eRi3>>&0&Vp69uy|Am-pmN3wd%bn6tiFW*7)D>Q)a zbH6cd^boq`?d03&7ZnTf5;U+Tnj|kf5pJF5YbH)s%nl{Lj{3ofO1F$s(p4pb-^mn1 z)A1|+UQka}gHcp5(@9q}v<-!;$E0NaU0jl0Ray!bt3#-UASb#4sEabmTbk&-m)+Uq z`B0z%zm5OH3d~x42dYQex$n|MW{JASN}zqg##C=x9K+kj;0wwIm)+!%^h1kmUzi`M z!=0PqNLW-EX&RY7z^azo>bp>(z&!1qN8szARiUN%-Rw_cYThhoS)og4ymZ$;LcPR~ zM}I4YhIGCtz6+UX+;!mPjP)Fd1So)QD&O4xn%kRB& z;s0#Ea<^tRW`Np!0oRyP?cDg<>2SAeRtvoiTqdK3jZ;vzG_V(Zh_Kup#l=NG`EAopt z1!Z7|dXA0pf7G@UXZ2=YFVT_Xm`-v}eV8^_-Y(JPF>(<5z^SP1QCBXUFDBn)EDPxqM;PF`0avO2pz{?S+CqF#Ib~X{7pgziqDD_Vh8piU@4C)CfhWYl+H%uR*DuLH z<3!e-s+co#ZS#cP)L#%^9_+NkT9fJ-wHgC5IjroZWoUdEZu9CsV>EpZDE2~L5c z#VK5kAUv-CnGoqLjfW}_Tk)3MqMr3ivCz1zUrGeJ&oUaJKQecq1DKMIHMo*uC<$TNv*oRHL4sYXH-O4Me{&-lS$;5u<50ahA1do1FEym+E-Bw#nX$ zgjux?$WvX9;W+k{(#%!FkcekmvayQ>?2V|O_J92UGXS^;$J@pjGwsRJ$ly)yM7pNz zOwiZT9ZMW@VqMz$n-wbjRjkzYz=V zt-QlP9o5a%cjW}i=H$REbuHb-l>;BKk7~6M6#r0VsQhxrVU(bs3#V+VQLu*LPu6gDb5&AT>ci>A?&w^9 zPAG`$95pTnXXLbjzobx$Oe=i)G==Va<5=~BcZYJ>U?7G(C5yC|37IHn@Ys?D&h z7v~#@1L#!=9h9X+oKe=D<^L7i4n;DzP2T$1_~Xzbj^#I+l*Cx(Jb#)gDcT9E(Li10 zT_gRaHxmowrTQ51D7nbJ1l9@`Xt`1+c#*rN)btdF8`$a6BHIj;0MUZ#C!Hj#XZ{^y z=gTP-3Y+Nqx4LjxDJ! zXZ#_XGFY5%cK%lAy?M9unzlwC1zji?{Ht`@(>W^$o~yUiUZ}2`nSBB@w{K*N%U6jA z=5kJI?enmtz(wP9|^88W(hlESEbfO zy7L(`0=an`cnMFS??e)5!0+C*#1$2Xud;}_^F;~WwRF0t>WDe4|#0QrlZM*5;!`oGKb#8P^1T2hDTtsGsW z?bhS?wNeS!7N^qNh|X#YZ>}YVk+!$u7Au{U5nNvOH%qSd37e$Va(cvJc$?M~U6P7# zI3cGfUzpE1Zholw8y?L(>n)>^#!~!8KS)Q!*Ni?QM~EHmQec5{Ap1W-A!FcTy*M{D zXSVaRw$_|<4bOY$T^c;c=F#Wm*<7}>Kbs^>ReIqy8NE2lQH8A|JwX3)JmYay2$a#c z;>AoBqHN`GduFq@9Nobg@?O*Gn{BpcOadN~8!M*hhnxqP0q&t`;atef05&N+jrY<= z@KZ2H@hCZ_M@%7)<_}=|XsR*6H3u(6k-q&>Du2#-NgPk`flffR4I^6;J5@qC%3SuB z#4n7m^m4(%J@HuOjR<2PElr;&jQ}{kwY(Q+nUvG6I6(;4{qZphC8U}={JCmHvbWe( z2v=;zM==hxVHd!SuvdI{xsW3nz5@YpkrVjqwjKO8Ah`bG!UMIbg=kWp^sJ?DDZ#nM z@VYeF^u2jdvgzsx7Y-9k?djxZOKE8)WsBdb)*{bv<14g;2l=YOr%sDnT`0p9qbdnq zaIYpCZ@9Y{^!xO7aLnMs*MqOlEH`$>uS3cHt(=R?HU9^j$WGdFM{5`lDAt1F!5e)PeGZJ>L*X&# zqB4Z5!+nrrxfQl+To}{O*G{g4-+^^%rhhdR$<4DQ`)U!gNrmZNr?sn5jzz9Zk5&HS zZtr=;HfK6>?`(j1Pk5CE*pT~sJs?VFgk}_{l+znTwgxTC&+#@~JZ^a!MHK_9GusON z%qh)(fpcISx!qR~^B*wqG`1%4#oi6P@^2s}^OuweleN%OOVj(Cd5Xb8I9x-|Agajk zZKFVc;|%dxwWZ!u{wDff6Y-SHT5NBDBlCF>Dv| zQt$-RFR&NBW_+eesD?7hb5kntHj)y-SGQZWN>ju<;j$|pEth`6)+qvBE1u6@Nnb)% zUroH&mO!t<71=3pNzNI@;IJ}9}B`cE9{v$fDiYNb4AKE-T7!i zX1z*V!=q3lcoF@`exCT`9Aa6d-yr&FYn8%^8bDUJ^d6AuON2T?h%NyqYeRtPhS{CS zal&C{KQ6}fM#aE(UWzzHCUOp*=NCoxQyOY6=Nh3gLwRe71Lb>rDxuFfAMR9NU=G{j zig||zF2Qo3j=MjiiDq~KAY8-POznyPggs1+Rm$T-Im@XhC{|f7BzXz=);#|%VDfR* zs2Ac|AmA$|iBKm`h~bWkTH#PycoBDNGutp)x+hOby%gFHD(c%+o7xyQuusv8kbj#` z>;UxzI}?-*>46%4S&Ti%Q6UL970lE|*p);HB=%y_uGv4O3f)^c)uUx$g$3 zP$T_^l0t=#YmyH-Wl7KWyBo%eqfcl*=XRX#<$apqsmNoUlKb= zyAaPtEb*L`W@mHG=5RCvkF!mGk8IOE|zl1HAL?`LD*++M~=^#F%TFsUj$6EDR>8&|O zKNj4^?{|K}?=403`erA$Tkdi2n0-c;G!~-CumYzlUD4m}?%EErBsa`-`sHXZ_(r~` z3SjSQU4$IGm1Nb<_Q$ZE8H_osOjWB=SiJb25?u|h8eG8o(14&%xI(BU-%p$jNV^M}CiV^T%h0kNMqpikNZPQE|;07_&Xb{p53SxKE{y{HmVmprc8=?gg} zIhh}BpW&;=ztx8X>(Jf27sbxXvFucoZF(yYnCES$qpMP%e2cS%&CD>Dn^EY6xIgYJ z=hiIjDaP2u<0w)*=8b~ajo0cCWsjJe+g85_T6=~$+S=!dW8nzp8t7ePy0r!q=8N=8igU$i?=*q-lLNHbLTkQ`a6fCMKE^o$HDDHJXQ4X^75HwOm`4OB1pDyW zN?rQ7xvm3kDj$x9vbX5=nE`L1+?R%&mA%yMZf26-Rq=i<9k91`}=D zrcHIaaEvfmXlgs8mH`!+I=HX!MDcSE^?G~u;@WXr=!uvm?dob=AN6&Alz*f0T=U## z#sy}yOXV&K!}uEbxEexgFhW_Yb<`(kUtk7p%VQ9T#^25V(q=0owo+fZyYZXhVE7KE zQ0-jTZCBzV6xKG|w??YrUnUjA&4pu`Oun8=M02zS{tm9*2160{a$H5UgIR7q-D9#p zfcd5m@SEcaEMTfh#XbE)961lpaXS5rnbT6dI!5&mDcd7AbkjEcia7 zB^~@<%v0Ecs89SfN)bc3kLFM2Q?{cqMh)eq^3-)c|5%``r5GHLm#QuhCo!kYzvZI5 z#=ibqU;9L71J7fo9DU2Z-JXh0@^kpnxH@%NYpo1dHGiz6x&aSUK@-#4-JJoeaK| z(wm*lPWGI~|Fd^>HHO`_Rjz5u-zhP<=SX?iE$8X%7iNp8u0J>ChdkPMjOpng6!;%H z!qy?{8@td%dr#t6n&d(xty6lNe9tGyMU>O~xm$~nm`gRu4=}qNE7&)54bPMGtG+~}QX&#e=3yzN zJ>NIs9eIZ??wiA;OZgkix=M4arL*`f7lGCj>my3>d7&353TyQSqrLD>s3h!d`{Z3I z@bV<{{{9%(Th9uXK~+3cU^2DK!-HZVUG2?{FlYnd5k?JmpKy%4$kl*5^;-J5kTuxS zx>Gd#_41;FE!|fPhq4^qR?Mf}eueI@j>8tcjPkD*ju+N?p(bn5$_D-Q-T^uhc%m+I ztFT6JU8X-24zIuz{4Y@=?URs?F}uX$qe9Ry|}$Pn2qKL#~bYhoRyX0JxNAO_0VE-f^&i|s%7Ga^l!&7+iAHF zUlcrrKXU%&o-oDrOd=t7wMo%D!_?+yQ?tzJ=q&XgO*Wle4avPI(f-AWj89SuQv%jv zp39SMhlrZ?Rm>b_zmy_f1>MZ9VVZIYRglN%=cH2@IA($y>U?v;dL5ns?Qs(I3N8_9 zk{!)?6@zZUaBz}48fdwDj;^40rc)%~xXm3)8yTd~KFmbthEnAWVHZdJzf8*|6^vD zjE|?T=J2woK3! zrufci(|8X9d=9mZ^@gr7Gu8VJosT4*p}k5_@6L6H^OYXgn4hh`N4TB$vd3?O~%N7%4rS)h&XvtP2QuJw@?3x&V zC~6|tlsqX5usEWfUC5ORRLg`$(!-<#B>}Y-hR`pBGpQ-KiZX_qD0BhkQzMw+ah=U~ zi8=G<+B)}<4VZA-zxE_qcW}F&lCKp%@KZU~VV*LUUd)WVNTCpaRDFvtpnX0!QQwi` z_mHnmPFv8X1YEXJ=CSE;b`z$h)S7JSY)aI?t@-WDeAG<1qHnQJ$!Q$7UpSbWg7!;G zeW%nON*$uI5`#yJ6TKynJ-ZBT2RBo+?DCFvc!*D=){5142;pf`5iW}M8BI)S0(rg=}PVR*D3#G*4Dbkjv_9wuchI(ZE`tj3(Dut<;e=U zd!|i8D`+cqi|XjC7?n;!JX_ZFepGd3voZr5$=b?nCc@Kui64m-+@YG0;jol1 z0|wC*?;w!2={6WDR1AEjCi(B!7WoWYieZge4RWo6$lc-~`+KHX^jL734T8V5agmp; zSLNQWN@72Y zotR8qfK$O5A_l0$R(6`>irg4QO5I?xGK|3NQ>MP{xpc($)nA^CX1bsu>I45wsT5Zg z4$F|}jWE&78I`r|;9k%(n4RoZ`K^2%T(`gE+QT-wHG{|RH$~wb_Fs0Zx0O*vJ}VEV zN9l+7i^we(cYUJ%A_M3$H%jy<0qSIC-M}02BJL!=5?_!d=tx=29jy`YkY$KJACx36 zaf_ty^i`ADRgFB7yM~N}U$y64LrW(#0d!P~x=JZw%4J^jcJ+UZtA|~_=GMPSkMnHa z2X+|97XATk+4{t6sSDp+`N>|Ru9K_a4Q8l~=l+J9Y?r_?%R+KBUVu;6y2@Qt7Q_!A z9oE8XH!(r{OkZXe#JbUIxI`T$3=!|3&fZg?j#w}EYQV7FLbYkaId1npu^By`eu>Xg zjm4i#IrgM)5Z=z*Gx=W6P^a{f=KF39GacSiPOID1(a_}>?J=EThUtITy;f?-8GfyMmov) zf%j}*tpuAd3_`5Y%ora#Ox?WqIdO{I$aXT*a9^09Fxm>tRrw`%o|#0o0>`*}QfZ~M z_HV7qN@4yCZmj;acNPiQ!%ZqJxw+mdTB@`WbX6FjaN$Y;jMksK8{=!a^W$rqw_PV4 zwfwPiHD@6xn%&?MeBpSF7SqD1soY*;3Ejun60Lz_(Ng&dT7&}JBc2KL1tb`l9VYiz zzqv-L<>_?oE9?vY_I+}_<7cCC!XqM9GQ=;^37nq2QUB?luf?FEd@m_0tD<}b?_x*i zy{)zx{-o=Jm)dd1KH<8$Q16THKwfwR9qMjrH9V5l2#K~zaFwg5QpTOg^!GL6Q#8t6 zM69P*(T8f~i5-C*#9lqmIIwSE!hL;&`kT(@2BPcwlklvh8b+l0Ol_%+BeSH3M&+Cq zpb<4(N#ZWS(~6H-ulA3>i~IO~gLs%hN9l#-{iudgIzEy4f>Mbk1~v$8t2xObmGXRh z_f7em;(~f8TJ5NRLRWDoe=V)C>omE^z62MyJ#~*3HkuQ*CHM0x)W+ovzD-+W?D|b>Sle^e+bYHnR3w^iP2L2zY z9NZ0^s>WqVv0OG^(+*u#ttT9t;zzK(xpDEI^v;f`;Dy{c@&|eV_vMx%udyT5%c0uZ zP`xEM4aUkYrB{Z{^1tA5@PSRvZ)&Y!{OfpYoJ1F4L*^ll@%Di`YgI%=z$v+~dO;Zz zeb&|}I6+NAv-m;qnyWotAbeo=%e4)gdMdvSP_--CT(W)ed8kk5w|^u`L9?X8(wNwW zio^d!>H=4*`M#B4dfunF(t0)7FIVFfaFQ*^erE}?=3a|8A`Qi#Ka2hYC($vUZ}edE ze#5W5wOy0@gJg1&IL|v(tm-)jo8l9}w^|>x!ki_pLQ}*p@V7c4cr@@=$gTJ#Kl29t z67{Te;smZ?!h6+ZrMWk;ht%n@xAaJJ*HyR9d}@8QeCc1`yOhCfxKz^rooFk&`B>WS zVVEoNlbPyS z?kO&Rh`9$Qpt8c%xCgi#bs(%dF@-v>4x}!iD&}3~Y2l5~gY2)001(5hAI<5{8MYqT z9JJ>)xF*8*m3SCRhh+;Gtm6y$FsnvYO`} zL+%ahXr)0f@wj6sh!rd9#kH}v2Bv8+gI;6K8!F*)%!=Hh*Rnu;n zq9AP<(>rWGec5+d(yVhV-H9LCH1V*S!v@H&sDNpi8x*%>zRXy!RAkq{MsSC_L*}g7 z1r8eKixlyRs~$4*0^DqFBoIYSTTNNy?wWaH<0qsNn>Qi>|Du-i9-QcIt`)Iw2Md@7 z)&t&--fvNMF5UWF^9xl_6ZVw29}fYgVXsh}_YMQ_BW<0xGH2#PYPNP+NTUtjz+`FEuN}3Qeqn&mVolv?8 zL-2bq@9bxqv<}Pt^_x2~0}sFqmcR1J~ufuQB8L@;a|Bt)Q-4Bk@6Q|0_-BTv}b}a zm!0Q-POnlsdh^kv?5!k$f!xca#CHc(s7O#=a)ctlXvrNCl*QC^cPH0u*FI&E?hFp& zM!^KfbfX+R8Q+f%nkWB5nq6;}^9;Yt6{WM8YsdzAZ+ACon(!}UMN8ONwF9>ftm2Z{ zWT^&MCUq7!3(nCyXO_bt=cV;FZ8bVdIsB8HNBU`D1iD2u@iUhDd&c0mwj$tz{E3?p zdQJ_ru<-|T%YdcmsoW%Wkk0B41I^_x3Ab@VZ~!vEGT$fqA0}F@ucRBL;2Gf^7=)Ul zhw2ySzv2b554Y1O>dQl6RuA6}-(X(neA3UOy11z1GC9h-)sWY3cOk!-_)~=H?z66j z`~X|JIqB)A=Gd%pVZv${gL?>t{O=`4tXGb6_1|-XtQFkrZORvNeGz%jcI6h|gjZ+_ z2f==`&ELo5-4E3-hI{mBWPxv#ubb&St%T0(nu~uL=hz3rQ@#v44P7HDaN*!ZuENY^ zqhUVm%ly!+_ysr_wNig-1}~py@~WbQ+z8RJOlKbRUW~*-oLOoSj$Isi36urJ33tZZLW|W|ctl^XZ?|q1>^si}}_I zit}@3!xtGBkZ;xX)b;ERrWAiuDFsd{GZg3_NF~A%OkBp@TE5q2wY^xfc>7Qq$aN+_lH=Hhek2DF!v!-nIu>dJXD^|Q@& z5FSon38wn5c=MQnzCFQ!ld;YSA1+>FkIFyMDd`|#8i(XJ>}926(Bo`iEkcxG+Qo0v zM|ntfgg%XWqz=@VqHF#;^t0^J!H%-UaZ2@wtMww>N6y7Kv^VPh0GBtHZ*8l>-tjFE z7X|C^te9l4#~$VKz;d>aFvatoDhxU+Il0f7_8fq3QERj=Z;P5NiK;U+NExPHQ|?HW znML@vtA%Zs$*>BEt-&z!{kH`@uuX=$l}XYvC5^3w?5Gs@>G~l)mv|TiDZnrMz|*B% z`I;J}-+4ESbzyBHQvJoeb{}CTF&tAKchNQ}^ZX6{O;Ch3iEM&Lv&E>B5yed>z*F_T zV+<@w^F|8kA@>B&8|(GE%p!fHGAQ8G?QBbat#p#E=3g0z#YK2YxDMJ$gUnpY0J*Mt zw;5)<78?tm06{14RnSRcGB-V}t2)&6Gqy|s)2F#wYG2D9&Kt;*+ptxcqe5q=A1#Y2 z!ybgQ%zJ|I;xdZzLMGWsc-G4EEom8L*iqaSezxALUJ>@If0SpS|Ds2 zh}~=ll%LF7zL$`IW1Sv&R4C^eW@eH^wvxk}Q{G&`KQb?}Pl7y?D`+P1U?J_fGu%BZ zaEw_Yt`@&4Ve$$Dl@e60uO;YdzbzC-(ea&(z0w!Bjjls9*S6(EoAl4^QR9dc_$rdY zduOt%25!T5ATCno$_1hwr+tR?gnU?u5oECrD5Z2@$5|XigWQ$MVSiOyQeFi%MhHsv z0$+qy_?Wk=ddT_JwG2E^1ifa21ltmi1J!lLHW*D}7c!NkZlPZp_cOl44fnS;+5DHS z^{8n2N7O*QJRh6?Q9A+)SGxt)kWXd5axd82zR=zbAB4q&dqcP3GZ62p%0@cZ#tg{z zF%?n0_*-g*bF4zhhv?yOk2YCYkNYqW9WXZalKB*I%Qo2+7KoPp+G2a0 z+})GpY#M6Eov|&@$Ltw``$X3h7OM*oqDreJvs>AdLk&S;FirbWtA*|X3Cc1g(yRHi zY)dIAFj>sA*W5W2ey8tByHJPVD>Re;Ta%O-+_YeQ_%dz}orR0?SNO){9AB8Lzipy& zi0-Nt^6%rWvE@zJ?-XV)RmGKR8KlnO_c4+FJ)UbQ;B6r;#+0%U9g#AWc)l9nC+PM0 zwerDMMh}x@b6&e{@^)&&?D(*(nVDARf7IcKyV?%wTv+a5*vZVM z@AcG4IEoI?|pCHtq07#)NZ0Ee(Wq27R4`Pc7>`dW3hlaA%?I-X#vz_=) z@VJ;Hl};E6c~TXIg6n=s5AHR&fp!|?4n2fFwU_Dh-I-OrevB9;0YJBrOoqajvRq((c zoij%HO;&(ql^@nfE>#{tTGVEIEADDk5ptRkDU~u)_=YCsHpJP^PRtxR9rbpcC4b_* z^k;b)c?^_d27=~FJ+_=SG+2l0#MMx3{8a8Wzb8IfH1x(~BHjgi0>Aee8>yR2HhhA1 zX@n(Kcw*nh{4i-sx10?7!arXct=07|(%#dXDG3$^r6LY1eR2z2gUn>>Dfdl&g_)6f zBa}CH+QZRX_ARxUS)8`gyM?|(l)-1+x8%cw!md=yE_yOEj3 z84jAGYV0Q^nVn4+u?-XZ`gWTuVP{+dI^x=T{-+dyNQZ4*3C{VB zXE~#NcJ$g@Z7*dCXbSG=u1z+@W#AuX0dBF|M|h*(svcb1zmHnsJ)yP*pMv$#9Ah!4 z&PB@ANrC?wM{${K59d=ox_%+}EWWxre)mgyKQRmMbT=gXfsv>uS)OW*_9>Y;>!3R; znTv3zu${SA>sDIQ|EW_l52zc9@|Dt(JC7MdZ%f3eT8rZ*JLBdU4rrWYoRtc zNxP|RmP%^Jm5D~5_|Nc(^B?#E9aVP=(45i^)(i8!Qd)5nG7IVUyjC0|-ZtNR5lWVG zOWbPVw|YN^GL0a1Im^sIj}iV>*84Z2|I+u7YorQzhu#BKLqjM(dIXC)%9?pBr`E~% z6&E3#QJU|kP*eT8V@mLw{ZjlXa)-K)e5st*K4o=O?u$))5$ZJX#kGMXq9&jzbW<`6 z&4O;{D($&xY~2j+3Du}RW}K))4&y_B}%21i@EeIJVqU(Y}`jH zizHQ=sXSJTk|n@DuDakMp2y7cMSG9yLAruK=yincvW*Rv3fB{S9;TDyw9;Cbn>_}1 z1rIFc%^bE9HU?2r3u2X?gu|q5>N0RDa1}Jr_epP|1*l!CO9TV)j&mr z+Ac|WR(c=31ODsxXV2E^WlW+EFkKZ(TqS9X(jhAkw&5r0Yh2mN4wnaYCN^*rq*0y;WoZYgBN`&^a`G0OFrxMSYR5G^B18db$qEeYxqTv%Zk}pHfKdAHEzV*^<#m z{vCgae1OK1OtvDKg%3DeM_*P&wx0<+NQW<~p>x$fQQO=Rxj_en;w3;Qw3Janak}@@r zzY$eI65w`UDr&}GU}LrFp~+(3KndsbJ^O|5xSRM5(@4f9>v|!2PKU84iJrO5)%lh= zoQzh>EATKPz}Izm<}W##f^0qAd`7y9UL`5cJHGRq#m=~_d#JMlbC@d0uZ4uCqLBd{%oV-3e3+R?y-=E%{fNQX=JoLh z;d(rZ7z4)}KTVd-N~NlEp*uyH&V6!U;YTrBneuXm(v-~#6M^n7A85!Oia+T{RL17d zwv~a$$>-pLqlAAD=nsyp$yUaJPcGeEh1=>s?Cs+p7_{rZyfRoVtL#AgSjFl>^eJ`( z=ob1*EoM7MR|z!XzQc*;+lF9TfLHn_dL?(rU!Q%8Z^Z5-%fN`hpPU1s9^M6Uk=!$C zAAi9y1)JtTUvvGqT$|r$FQyaBXLKsaMlsY6?Iiae^uk++YD%uv;cA~%!1%(scRZ(b1Z*}ytpGE(|eOwo= ziP^WfrG9c2QqN+$0v#F0!K6CpCbt9honCk^FBYIQkHT|mm}Z?+9v03~=`+MhMdyd&A}_@p+Vsz?c* zHRKok-u4B43BGgr7~WV}V}*9TSY72HPw}ev0)JOLhXxr|a&qZ&9YwDr8@ zYVynVq3}{%6QR32B)EV`H%+Fq?XBqMdHZ-S?yW0E(x43F+-uZsRFhe6nMKB%?h=w8 zgmbvn?o;Vw^PT(>vI1L|DWN2TjA*Y2~Co z0^h`{5R~Ve?QULq?Mw8VrkohNmc^fyTBfEthTu=QNZu>DiuOud$TpD5gHCFh?15T3 zldRQnbQ9a4DPRGf!tI0mb4$5)f~7DyqMbQc9BrOaWcd-?ZCy!pQNEI=_-l@{cs&{{ zR^^;G(KC?x6n%eI1=6ei2D&?UmUPkvz#AR`rxEY;m|$9l~r^hqKlEjk3Ox zv7U9Yr^u6dc3dCyxNt)_Rw<$#gL{|;LRYd}E~(c=gL0zPO>&ajn`l#Oul^l-qmG+% z*Rg0ZW6diiXx?TSyN$abh@4zGlLzz3RTFGrGNjk?U1mSCRj}vuw%nEW_&cjMXH)El zJJERD&iWCSdgDxr(q zHauBx&9NqbTB3HymzjUKrF$;nkIX~iiZ&V)q1KZb`g*TP!N#UX9xUX~@i9TeH8*(4 z=Y;|J7#AmQ&pGTi`ysU&YZZt#fiHBkyo|a_7)JR-HI!=EHZnKZ3BKteeouKH?-pSb z;P6h*4VPoBx*8UG&9o7=Q@ijfRNXa+|DzT#6~*4fYrH_|9Y0?iBm9?HD0+)6hLlh% zPE`Mq{xMdh|E(QX-k8R&VwT0Zm+Z&TL0qX;@7SugtqBOtqsGDyve{k{&ap@@wCA0@ zIk7-*no&xAp}X0Oh|vy^Hl_>no&IFB@ca*P8WhuZ#rPk=;h>qNmghmecEUe?U>XU2 z`R^LH^ubIz94AbJF^;^Po?10&zrDQsDbvf|o%>JTAtvzi@O9S)_n*wNt}b|i@3W^X zZfhS7mJxsK8@%my2c@?37ur1fbAG1-1&p}#>)I-mmtO~dQ#vS9m@Ksrb&$J7KXpgw z-L#r)fi2yfUKdinSpJ|P1{<-_lcq*ez_5|W+%LH*s^s~BzjAGu1$1e3n)p*Ji7w*4 znkWB{t(uHYYdR-C=#xk}u@kksffS9fEV z{1~rH?-AceJw#rRP4IF0)9b^er(zBLq4JU`g#z+ma2p>sKNX+ZjJ1iTLt?aeiibo# zv%C*E~Gk88s9bls(S&~ihq}Pl8g=)V?PGu<$NsB!<%$`WT#mqD_sZ3(vJ{m*2 z@DF-EZpf_BbMmID<-r3G1vf|?iSx>R+dF$_)Wfohw!2bYF~&psp;Pk>0oNs`zDQ`S zcMML3Bm_(gdw*~jTtJig26O^h-u9lIr6eYF9&XR92H*gWoUH6qd5$#e-zyc)~OvcM<&J;(~9WCb>74!kTJ0=utnzIg+ z*TP-9aaXl-aC+`b#%^X3dg=d&Ub-=Q<7(+SVSBHZGQD~KuydH7AkOxq&Qz`yG8S=j)4=@l~-BIa+#T_Xzbt9mW$=gE>c5v-U!tax$DpY8+JwCfbz|Hx`~nmw|_w zAa2&jk?%8d0IO;6IsB!R4m6=FY4f3CyC)4)A0P%YQW;x;qQR+R1K7+u3V2Ls(OEI1 z781^bE6l>+YF}ZawG>9*AiD!L(5EucPudT&MK>B`tiIpOscmv3Fjv$yHX63Kg&Fmz zrpkS-Hk=5$X+cpp`(mm(fUN=+YZ|)*&cvTk8n>A^K-R@q_Ix#t!ta^owKe2A)Gy&Z ztRXpeJQh-v4%}q3ThvmCV~fef1IMjhgBI}tc!liTz3d`fF6dbK{*-Ci9K?pi#bMRd+msi)(h}kSj%MHME7M*pWys^s@?o1`Q!v0hFOp#u^!koW z1gv?1bLQ!x0qd5A*@tQUP2E@uu}$1Vbh|0g>HYzCNZK?VJcX4qRbDMX9BX}r? zQ0IY4wI(yXp6WLnZ|<|^-aawiBX#FH!28s9dZJ#*?9+a#xrgrUe(T*soG@pUo0)#% zVc&pUL&=UF;IHZYLgvxi18w>1bQN+iz0FoJc!XKuZVu|(3Zu3BOT2?C1{}tFWrkD+ zzj5|*RlqM{gzY4kfp@|xE~zFACN2C*D^23{(LNH5|{Xe%-^|JsE6_t5Q1NF%ShY*PtjQh zrm?kKxbBil+6W=RZ3aexR5GcjZl~_<-cxsXmngw?21bF@Qg?Sx?W?<=>Ye*HzX(If z-tW8C^O*MmlbL+eN$)p)961iU4ENBiMoZzk$PHwUZaXlO*Z3;2XOLqM9sH}`WVF+* z;F_XqMttAUl46HviO39ec+6t@=u#wZ0JKB&FTo*HGCvmJ&$#fTE(Zk|y zL1WGWm+{wuHTce0U&}GzGm<6dqjtd`$qD2@D9SBT7l+!}U$FBrA@Y`bY90d;pB#zrmY=wzFV5xggy>Y|9r`893w_Ov zAy}FyzsjA(Mrl2&Z!#GWe`Df}6seP|fIgB1Oe#Qy7*vZYhpdP0via0YR=j7K?uIUljfzI5 zTZZwfc?vCauCO8&$CSco(L%y96U44xMWHSKFf;<%!${&1SCM zOqcOV=5_oxF&C0;l=&Y3C3uZUaavDz1&)R&v>3H$Pa%7uwRlH^*y^TkFqhHt)OBVg zZ`EGF&qZU}=KM$P2cfpuaf}+IP;2p)a1JM+G;zD#-}oMztB99G@qxap;D-=kJ|@~w zj#!$*ZQM1RY@)H)A;UqeUT~dA>UC2Onbl+^tC=5}EHl+GKhX~}EQdR>U-k<%tSF~In7_quMbXPds9JUj}kTh z9$U|p=GGO|W>kD~$K{J-Jq3BVKK>TV!H-5S;SceG@J0QC_z&Z6<2ZP#{wr4A+(9_K zM}aTnbD(W*Ot%)B0P)tJ=m+cb{N*S@FOPobjnRex>*-Bvtx@|)Q$75ILG);2MhyXP zH}8pF3NLnz*S1Gzf)7a*2f!=&p}}SNdj4pBx_Kq!cK?K2y7wv@#q849hD%xbpv^d$>W=% zucL8tGQ1KQg>=<+49%lY88)#SOOlM^=#%<7+=TFO_>un_`vn_8P7usM4LS|*z~fjk z2NA7KjTCouE12>?2a^_Rhc~xuWgD8+u}R@0;p^7b+T*cK=z?NN{2brh+!pZaCu8CO zD00cEW$349?B#75I~Y&-EAtX|j% zZ1)Y(&(s>gsgdf*rL>pNMPA4j@apIS4(>=$h?QXAdPSxKNvyz|8JT0 zLlT4rnlv8b%ed7Y*@oJvsaIIrF)B1 zRTbG=m@S+{q-dcLPrFFS_H7Aw z6-H72l9AvVLpVGrHVaGOD#R;7E1_GiF>DzFFodzkq;7Uq9tI-o7& zY;KL|F5jAgRL>(*C5;0LaXzq;IsrEZ2U8B@FP#}L!uw?Aa8L0N_76J=?3Ou`O{a0J z3a*cCA^Sro_^*@-I}{v>Emhtnz2V;BWD}C948A5$1ZHBr@uN_Gqz+Xn(hhBHtZh2R zymBALeeONfB>Y6iLEZ9b|L8zV)mS^;fYvDi0TP*g$qL__eWHp)$ z_-m0<(;J$JrQkb-{n#Y)R3ruMXa%tru`oT(c!W=mc4Jot28LfV4<$Fyees7VjP1m2 zxstW0%>X zvDr;Nfee2dZ|^gc{n>xw@jzE=$gmjI87FWZEQx`M)`QGYLvumJueCn1Y{;Ak$r)B` zHKs()h%Kwb*e}Ep^%_?LDg2o9mi$Or0x!YOW?rDVU}rANEoT=;=F3Mj@4QX;#LP#g zZ&ac88+~6Fa%{JJgb=zdxr1*GO~6jTY3M3wtgaL~!90zQ%gjesKpr5SsUP1fblG&R z;2FITf_&z}LAad?;jHvS{Ex1hb{p6_x+;7fYb^GBL&_@QGK-FFm8pi$0|P+< zZzwhL>Ud?*6Z_0K6Us>+r`o|z1SWC61RUbUz5fGx0lN_z#?}y7BtJ8!^CL`~mEG88 z=IKmVk&Sl_X%DHfgC#O(7_}$!bKqC>HB}ioBerJq7;F3(xYf`~*Z}>3w;4%nx_1e4 zp6r6PHC9BniqCEr7zR7zW$-RjJG=so3q$z9>LZv%?yvw;gVTE>afJhl&`R;0xIY7EeIR zEkSzFb?6FssYwskVjD&Z;FXX^Hvmq@=7dr_4tFLu2ur1I8q?tYc`c)p`DxfHGzXvW zC(A7kqZ6B zSYNPMe0_h}-j@0JO>=Xou}D=m*5vGNvMj{2$Nz-+_v z6oo)IMuw}S<r4`0@`4DAO#!j-Vw>;~QG2qZi=mO0#TH~tS23aT`1 zAanJt+za!2JRGeR4jbC}GKFNMKmQ=(I5vp;>;7d}B<80!hPS97IF(-kXK+#WwD`?( zTl*KQZ(mLg2)$!AI{NWl=_s4%cNizhIOz}Lu1GPu)Y1=fLTh!p=yYIS!wS(tn>B)a zGr-pI5S^pU1wV^@g^}Q%{QV+Ny`t$33VHi6L2!ZAg^%{0+Bjdc0c&BX!=6A&#EjB1 z(^6xq>^w7>FUqjf-$c&GRAD%jVBQkxWqc?+hI3+bpn>!hxEpZGyf8A}uYg60ec@eO zptvU_XKM&=*xEoAMCxXUC*xsz+FPDdZR4%YU)aA$r@$&!g{?80_6+1a%rU4|R2N;z z(vh|3Gg~K5GRf1!uw3}CJ`_3Nn-=NAjEy?brw*5)yKn#5Qn!#(ZTyO)vm(g>*zyskLig}J1 z#%bJ1^I|TAPUmN$L!gdm+jtYC7yJau0sbYAp;>+d_Jl4(DjK)5p-^8si4y2^@%*6x z$|LnG+kx`ov5vZ`X8Kg!253J&zE}k%LQRnF{IFCDbAY-9KEj>=D6Qwpljqq%Meofu z>0QcqG6tCePhj?9?XUp9GUPHYfcgZj$~x=;{+zzK;+HU%z5~pGiXsC2ot!7M_P>bs z18;JhnT4X%XcU$Rwgf7J%h+A;acl-VkekNz50J*Rk|+E;`f+@&rJ^EHvJgEJJs^6n z72por9OFRk6LJfXtgo$C)1$$b;Bj~&+#1?}ErPy4>)6TSgesXS5y~Me=<)m*1P7c@ zJ$AgX&(sxJnqk12p>vpLNME#;q!-#is2G^JT^(M9J_js#VRSjOO*~!O&^v6ldF-~o zbZxAb&9ZHv2#Ki!$CJK?a}7KyO0Tb66}n-;uG0(h{0sy zN1#2Thae-}IAqZd05%&|K!ZbdQmc!d>MzFI93Yz9k~ML9)*c1y$CvFXf#l9d$VcFc zz7uj>oWs2`gMx<{1@z=MqBXdvI-hCCJms&8Ck>msojCz?Sh~Y$>_FHE?}u4SpxdTD z7N?3u_zzPBb|`*^Rd`Q@7sX~FpR`{icZ5lNHSmM!y!r*!K*@3s(f^@8aZH;N)FKt3 zz1UssA3n!C5b94AMy5rpM`WJ9R08HP{1MZjWl(eIbEpXEi2tIK$gP|fUk}gVYKsJu zf1qogOME5h0a$@+LwBW*Sf}aA=;wacIA!-&%u2tP)rhU+y6{^?CaTtQAKpZ_bL~a| zXbo-C5hBBFtk8}n_^tr73M8`Ali&_bDKkdf|>GKF+-zy;sihvNn)pA=foB$07cVPco*YL zUj%FcP6XzobIkz%FRw+Hl4@-}G7icC_i*o`Q|tk}uPG)r1PI_?Toqhpo$Vj3?B`x) z&hniTiL-^!6sEIz4LKvao$4tj5AM=Gf)AnU+KbT`Iua?52jD8e3~iUlVz@jwoQ;!? zNI&e7_MSaLvm}u7#8kllVqbA@4M{-dbc4&GUB;gZ^23rggZIfZr*% zu?lD(9K-~AFNq3`xdrS%MuQav%20pof|$nL>b_-9#Vo=*VIup9Ou4{}lU8Kk|Hv zz6g({Cc>hBl&x$*{7w0#VJE+w8WwB|&k`AqgM9LEcXKzdKVPrk%~mw_*G^+|`0lmQ z2mp+Zyuum!Y2C8WcQNO+78#0d5O>m##7W8}-FZ`0pp)}d?T+ZQU^7!WezIaRHH@7G ze8heuZhT&NDNuy8HB#c|>47Rt$VX z52MxftsSFL(70ZGp03Wz9PL8IMk%*0^a4MDL<3#WbT$F#hwi7>V)bzqu!^3nC?D92 zwG+G0Qk4UF3nJPsp54L5CY}qasxrMy1uQ|QL@Z1TU|duMY%%^at_M}71sP-6+r}dX z4Vka5tFnft1$fOPHkXEZmGztBk`Q_UyninP1C8i~vVCt(cHa-e!P!&)vvehev_{bdfJ&{B* zNf-;m=8=(CU=x_-`T`Jioj$IkkpllI&s@5PR)Lg85T>tjtnR323b;THFRDyQtUZ(r zb=fpt+&oTS3VEBD+CF6BwneQu*}GnW<{3$fOxHaW6k zM-TR;F3BC`2kXXhhoW6WtJN@c*7}G}qb~^0b=Bf85Itgv*e%yANn}YV73~(erw=mS zn8(}>VPI4x>^0eakNJNK$k2AtZ4nf2**8KX(F54&2nPnSA+W%9<#wnSrXHb|YOTO7 z>JI3r-weh08m0pVuWkt1#QKrBX+|=pA~Qk_seMpIb~S8LT{M*;0(*uW8JP%G3s<7E z{Wh`XGd#Q#`x4VRCK#WX)-oJ51+$2WfV)UT`3dGOUd}Wlwh?9~&W`q`qX~%~N9{)0TdY~8FK;+Y+ z%vZ*NGziq?Mu#=HxG&*4;VKLfCoNfs8GFebhqf6{8oROIcKb|osLyn6pf*h>IQLvKe)9 z=YeFfDpx(Ehbyw<)SKda%uA6eWQt>i*v`5SZPisZT_A7j*Yi=_t^FGwhpvb<1RI&A z*^i)BV-4{fbVO5tZz|};R)#toe{&z$H0&Kalxv9PLJPqb&@wY_Zk5>p@@W$x0GmX0 zz;c`bJso_kOH_?wMnwlg|MI!9N}xTqKALP!(B)u9an`mO%moZ^+YoHXF^!dtG~kFD z*@CYDwYXHIP(QHdL(};dDh??DRAS4kYe`e-D_R2GwI^tv7;ZVI#t!MD_+Y(--j6lG z6437Q`EX(UpmHwz932QgDO%v~7^$DWE-W=wL58HPghiaRtA6k;PecEqZM^rx55d>` zS*Stdnsy1?6;L4=+LK_O42(We*EU`DvC}FAO?^dGVXy#&_2;js<&kgHWW02Ry=E{i)W%ZM2>;sPgV^j-a{bmg?28ENDWFabMa{0Yy2NT3>dL?^(a&q2ub zE*4Mf;+$NhgnowZ1$UOk z$}}9eXs~aj5ZYvF5j@M3M{479kXtAihru8?13HEc61K7Zq2Fv8Wu``iYN%UbLq%Fz zl4)9~875PI)<@to@K_V0ZDX3pRf!)&9|k8!rdm{t6N4jfz`MdZtR2}^hY)K8z7eIKp&(M(GlEpz8zJY9S~?{ z;)AD<*+@_HKDt4>*esGSux8LScyq9}SnqBd9n0X{MP-Geak82h20RT54A7> z)JSf(r8&PQ{I2ALe!X-z@+>TA=Dao6w!nzN5HI!bDX#&Y$cAUHjGpM!&& znPkmUPt%CidOF-dJZ}wwE3j`hRrryXYgk1rNjoFb*&M?-{pf&5%ZpdFOxE;knxUFZ z%|VdxqWDZf6Z^w1GQW(q1AcPrIv4oGXq$7TF2Ti7(^ZaD5|h zBPKYAPv*i}kH&-cGJ{@K^kMY3uw<`SV*_?`?Wr(^=AD)$G5=vhq2b!vxXu#smdA6k zZ_qNdzA%beCUh$4rYSXi!~R5~WSKCLd5)|%Q%0Y*F%x5pqHfz8x+FZ;JlSN$D zt?9%kyT1M}nt>&rpJ-9?5^-UOEY&-AJ?$bhA zlr4!IcCYt##!eeBWET7`VoD>?Rw2)~$A08F$0Syy5(7&+0`a@_BvCYxV=o0+!9b}7VJQ-%p za1z=YA0T=(I~zm(Nzf$gD0WiPPO#Y2h;Crc(L0db;Me#aQ%+<@IE``TGlm8<=Svyz1Or0p;XjT0& zWJzR}zb5vV7fA%H8>wJwOFy6+Xlb+~!}#_1zDyn6-`f%x zhIRnQLdVe+nq;IUd?a)zJUBeUPl-G15&UQs$ft0FMdE6}{EKUk){|Kgo48x@@FFuv z*$bR+wON*nN&TT2r;XFYb>O*-H)$3%1nwU`3U8r)VG`VGM8OPn7`!U{8g0w3;D#Do z=~(`Bik-hxirqG@ zV=iI4pzT11&=~YI1P5<%EaBZ?imQ4@0E3hu9@lretv`)I&EU%@akxoFLFHMhh`25j|%)R+zY#6j+e~A(vefK zMg~pEJoh>5Ry1AT%W_J6i?507aff8H_rIZel+yeswqoju6#vKgH@Zu#2iF7MghcTVR=Qu- zFTzdGf_+EoMi@xL*AI^m{ZS9VY(5N6g+Fmc;E(#A)|2{w@m#S_daG9bBNUEfS2E>AC?M#QA0a z*gpEFNi9UVB_)q=6n2Iwo$^SR6-B;Kw z3@5*c3Dh!NnkNMo(%JiKQ|TNH?~5#<#)!<%-J#|>iy1QfrWDr6lsT~0*ph98Ir&u7 zWR+l}qEf3oN=RmK?PFi`E3oFmaDN9bl{(3eXLg6asK<#u-F>E)#!5n4;R_}Y{SVtH zpwMuBN0hb{3Xo`y;F(a=gd=3iFO$_h(#b3S;%B*+u)K|gMt@9 zn{baEkIKYE{b%8qIy0?e%wt@^{?A*$+;O+D945U?1$}LxZqyq9t?y$T#am0rwdfB^ zj;ETiLG+r|MHfkrc~ZH@Y%=g0ZW*m-UEpN|KUNE`g6+dkSYN;|)!kq*&I`Zes|r$W zPi(4mG`!bwlOHZ7IzMAosZ{im>ACs3*c5Dn)+D#^!-G=O7JUPO=9KiTgs#+Tu#~UK zXGP{ZdGiA@=AFU3=&UKS=!l@Y=jC_nEzSJB3J8*5n$~JE0~_zxZUX9Y%x`1NE7q!U}qT*o5jA zDz=~GzvS-;zd=IC4sv9u5n9F{Lt2V`q+}-9TrH~Nb;@@9cmMqi4OxXH%)N~Vp&9Py zkwtVFH$1eHmC~oh_XjTg1fMnS@YL2{p)ZH$lILlPn@Jn#**ZN|$@dh>!5*`#v53f= z84RD|S_Plb4%Ibwvs(fUqz96ZO@-_i_+KoQS%Q6_`vrRJ2vMiVeh5Pw{U&HDm_xm1 z`{C1QAHIpf)%CQin4onpYs23AzhQ}}EAZGbDZGY!WWu~5yB3})NW%p8nGRz%@#gv( zttaj}c+vaX39D{=z&`^ofR*TQXtwwSUhx9?i(9E}Za+iqBbN)iWNCqo*c9q`bVA8y z{RjRv8;biV9tPzqcz`j3KBJMC@K7!G5H-uxK^VZa7D>rHgvQS4v0WmSql`*pHMB;5 zoZd(O#TsxDww>so+hY3a7>@nN^nhK+9po98h%dt@akrw&A{^!q!r`g-AU*}IQhZwf zox7~N!7O#F=KsmZ<#ht;JlvAan(LVGGx;&DsJBl^nKNv5A65$xSJlu}F8($;Tj6TIv zVqZk&{jXq-xYM!)eE#(KA$LEfZ6qfnAl2h_jXXaEuFqKcOZs};A9YpwNXAh%%6>P& zrhkyH?!^Z8?&?^!`BX5524epq-%MY~fk-(hSG;d*5YxmObbWLv{1Kk6zpHBw@0GVP zCy344lhFk9WBdrD@~sG3`8B{&d(+;ZwUla zNs-;)IJ!PhiF`u2`7S$3>|kwyQBBBSg%J(3Mo3!%7xM;|vTUPuOcCw$triKDg8GyA z^uw|Kw9EUG9-wxH2&RmAU=&HhR2?v%k1}f^jhGpDqwE-ut0?^ik38(-*UF956Vd zic{v*$Q^!}zYY6L`T*S-83Bq`N^@H-3+jY$`hFtMOQ5DA!$=2um}-PBraEf+m~+K_ zL?3f2?at65(Moe4IxjXXKj2e9o_&U`BWH_eyW9RL$}9um&1McU6L1^Y03wjeCK+^) z^+qm-|1%y0hta(IJDY@0v?mCcpbV%7ofY|C_%U6|RADw^iP#aZ)q^r=|O-IdF zeO2j-{C+Wf9u`F=UV$>Ta@eE#o`+3 zl=gFYE82=XtXjY%vL&doxJ|S>ly7{&x3_`OJLZ7lFE=C%13srDjzi0JEdh~l3ZM4Z zg$7_P!471O)fw2}|E}K551^Nu)8b#SV+dxBfdw8YeH!R+-Q_A_+q@O{7T^G0Zn_PZ zsB6GAO(A9sGM0Y^jrFY%oOD67kLS9f&_o;h#7{wJVF%zhb_JOX)#AtSGk`F?_~Sm} zz|rGGxbYyON2O~-&QnFU;l^y@pUH;_%kBEvwU^Z+GVl0^cMg1@28VQ>B7Oplh?*>Q!+KF-?dOBUoC*&J*;sk@g8^%ZTd%ie&2g6y^2jA(_uEy()j1D{(M`V) ztv}r%pz$qB$6Osutja2F9s(6vPqiPj7FNwA4xLhG*JFLeeX>*Y&n-U_-IA)7CS0$Z z9g?mh8VW0jDT${E6}6g}azalunt6p-mtLJ1d2bp4U3^RQ!6e!9N}NwB_eyonGM8^r$pA# zis91M1O5^t8r90)TSyRV7q!nmvuYVJ{`geFH!+WR`+ha?s;@sQuhkBs`R!EV@cn+o z;(5b~hqH<1*WJ~!rQtEe@()Sb;YtsQQ`PSL6X?0r?BV)gap|A4} zOqf|M`!CwQbkow^#FgGWadzywtfz&;iRX9!M^tTngV^wtFKyDJTz2J0XNiFWG(>mw z9I3tLE+lZ5Vq_Ar^V%L_%ICWWx_#V1L@oEShGZvX8)|eV zW_LAaANx9*P*k{+)$rVYq7<#0eW%a=h=0c4CQP$m5H)3632%HeVIfx#d78z<%dD>l zE_9ztTub^Np$Rt2T9;gr==+z;YISTK(dgH|M5mdV*?XJ^3Gw?otE1{G@#{oV>Fkz< z=8H$KBZ$Q$(Qphy?3}ZS$U1nM7}{+)QK|hTqEqQgBKSm0jBC3~TqVsX2BbaSUmL8Q zeP<0xob5X(t7n0>w0c^9qJIQEaHkVQ)H~rJrr-NR=;mxZ(CGdbLf!Q%v2n;+V)*-Y z%|4d2B+k`cN_J+dF4FvE&LxX}MCOevBpl zer})jH++!b(~c0O7jF_bJ8?u}{b2`Q-!FfV+WLp+bnX~IZL6Pck=Thxi&hZZs;xiJ zf7G_DMtjB)CtrLeg75I`9xGcD4gYfyy(4P~S;ADp$(hC=HqsDByPNO5Di{* zCS-F*5e=3dJYd?j+TOc(>MpbAi;Y!Y*i~QRPCV5>*|5~!GD&cBa2ETTYcCk|d1s1# zRvGW6_44!vTBrS_?7efO7jj6vH=lG_oy(#xz(L&zMC|7yGC>{V?!H=+BO1gClK^={gw*CPEiX9e~@~8^HwK zq`)~{Q}HK>p|#FxWNKiYXgl}<)boEY+8i2dOLGyGgO%L5ueK-EYQ6Tt&Z~-0?%TpC z#m!}-<(Z0x8dg?AzO?X*JHv&#(q%!n$LCfY^)*i~E9za)xv;T&l0uz)H;>#sKVybu zQc`gZ!3E3yqz{+*3j_B0vYPH*MalMD=Z9)_ZGSwD{NwWVl8v%6^$w;FmDSC=oOsM( zs2^69%CVHkq>P2ueYtOYT8u$J;{3KGhdNvM#hKK zp3V!YBNX-W9X0x8bP z%3IF4UQ?i)XOM04lndT`O?dEQ+c%_-KqFJD9>23(FhUUnQnhMm{Jni*VSO>HS4Ah@d_%?1;4oqtariN_( zf7D|PpVh<20h+L8aNwg;2;DPuRd-7(4IbAU{6Bz2o(0+}`h(#^%F3#L-OKF%=(S*n zK-Zm9)CHj(%Id05F4Q@@;EuAbY?6PZa%gaf!4kL=8tXhqj^5Ee|Jc$M~hbtS%)!4CeniBzDmv`E`pGb{MS{>OD* zGT6UFU6oR)UXhu=FU@N`q_&1^ihIgt8706GbwgifgFax1b4H}U0`MNx{qn}#d;J#- z^$h1jT1j31?$mAJx~>E69D9nireUczE8tQ+4Lnq&re~PG`QOP8+v`ZTr<&EU<7Kfw zeU1CjlA+4?>E&g;3%8_iEqs)e)reC~^p@pqN&8upoWZ#rwp-38k}Daky<5G`PMdUg z>QL$NhIYlFjG6hByr~MO3@eOF z4R`z}WtVbm6nv3LRGo{hq3g2YabdvGtvwW&RyBfdPBvhw6Dn9bw;Kd<&nIvIEBuh?38 zUOH4D6a;LS$k&CpH(pZcJUcZmSKp+LfgyS_&`;YnLuxoA@28b%F3Z9iM$t^xMqQ;~ zqq|P1rQ&HZpuU^3tFK|HrP}`lH z9`5OXt$O5L6ln)mlH4=t!o5|OGiC(O_+NPD`AG992u83_7zhibZ3tIp=O@&SP&U)9|EA&u&=`*^|$Z-V=YwqK*h z{zm$r`bCZg&eDL*dsp5?nv6fn{`DgQ|uC-PBJvmTzjQB z+17ELG38xoIN4!K#CzWLJavR~%sMJBr}(f#=E#sV%p06Hv(ce~OZE(9!|mM*yVi9S zowUtK-;_q>!?`~SyQEB6C)quyxJBbn1^R;S1<5I3^2pTAi9Ky29AbD%o?mQ{m!=sr z64O>nZ|wS1E@Z2{xx}`9d&<59q*%Ma=BYM z!#h3j*mXYmtInkcI_-nJKN)>p)x8nX^Fu421$L&SgfcaD_3+J2GkR&aJ72k-MKkh8 z?NSy_v(>2g#{RXzhm4Jqih=3w8J=6NuG$<0?r$HsTjPJ{hj!6VNX_cPrv*X(q?f-h)XnB~r~%~4EJ?Qir;dLyY!_OEBO_Ow#9^;Dyv zyIlq@D^R%|newf2ZvG1OZ1rc`Xji4wME#_6lkbME zW5&V2Q)e$#J-^hEU$k2zQ-AOjnxfzzS8H*O_Nt&eSdlb%>bbLwSDl5yTKG0?ci@Hp zZlr|XzIz{8y?7ONhx|vikHW#$;bkIw*vE7>bupA_R+|#T4Gdn2;g)#ua3y%4F$Nf2 zy+tBfFkPwq+Ni3mAYEuU+ind?6n$mMx+K?2Z<4IamMd~+@e;?L44o&%BPi09BW;6= zVBNm-a_$V}+l)G{L((gSX17r@)X_GvvPN0RX@kWBBvTxj&N0QfvzanW`yx424XLiH z+A2x+9Gy4?_Z!4dUn#iS%NI`39#78QTBqxRdXfu3}EA76XT(Cv14o^+#TQSOX1 zr{CsoqR_h*=P8{Z3-)*ub=%WhxZ)nj`9t|q4cmVwTh)ux9v8m$lm{VK2eR&DaOFT#?hYyJrsWqXqDsWvzKA+r4UTwnT5a>+hh zQqZt?d!>QT7A zD>L+GoHr#Y(zD+2?#bRnPo}SjoA6xmO!k+P3Z7Q-^|r6dRKEnA6i(2*RIczh$?pKR z@m`fqc29Hn@XS(<2)1+|a@Vv|{+zUW;a^*CN-OEF8~dn2!QC3A*RsM{{5dene=M+9 z^Ve5f*TMbKH$JFvxV?kJ{{VX>n+ki#<|~_Ns(UYZV)Dn!2C3W0o@)Lo=X)R9ntSHk z*5og7uD0)xEt3q8{m8iNz8=VtV~$UGoGj|OrJHU4Cq*V7=-98Sma#IWR?#V4C3|CE zEos9YgOn|01xkbLu4keot*|it60jWT2tHR1G}bamjK8%Gu)D8jMkw_SSyeXDeOmt_ zZJ@Jm@oLvTO^I^9hXNipeCOV-?O+@dy5j$?G-gCwsVV?+Te`P6?w6S}r9!Wr3}d z{cN#Pxzg4|X;A$mnOfMOUccgQ$_DNpg$I&1HBMbStbRl3`jkOM1M~W4Oe?Ca>}PK% zsck!9yP5l>h?Y!}7v^nKy)0huGD|)=vO$u9HCIiaGTOMG>#J(AkgJd+FpO$T-7vm| zjfDl&2C`LX64lS`3hj3vaJ&aC{;l*}W&!ymtoKvs9B@}cLp;0Nx~TV3Ah=SY#~xag6WEz5i@$4o@bg5#e+?OGAdy5 z02jmmdK;@WR2n@8Tn?Cu+cLHNZ%lnnYjD!LU|aiO3;!5+slG^8-d#EUTXBNT0c_pT zI0!gA-dnB-p&wz7o$}1}t_%EBU^`c+D;8EQyx?8}v;%LamopsjSJsD%Z-&l_7Ve2^ zz2c1NbELNMUr&+xgwAdB6f6MikluVrK}{HRb!C5LRHM%UJ;2JSHK61FDktUkWAv%z zQ@q8u5tku7(lD4FS!p<;uE}ausfCnghp|w$SGPy)AkW}1*w*x)|DvgJ=oS5+=QQ=r z-BTqbZ3kVBRB92G0X@`|3$If6<>|m<;J(48YvX(C{iF`7c2kLp8=kR1d2Ms)de1vW z8Yx63q#4~w`W)(xeueg;`>d~-&ur{ZqT2HQEZ;}R4%=T{T|m+Bj`z2MS5I_rG*!`U z1Rv-4lx~}7I1aV1FV&yDjUXVlG4<9J9{Nbs>@P`$qOmB zXB%k2f4gdjjX@BF)zXyRC;V}1eD zz&{Phw$1l;#Zm(R)ND9ieL`g7F^UDAlhjeF7orM`j3J;QX6OPP!O=t>ocv zsHgdNqr1ba9G!y6hSthz&K#KSFHcU;KV*+arixV!?TmXYBh7cBR`^BAL7YQsjZDX1IfRaEahu+d1=~glebtSd{aZ8)=hk6!u%X%H@LeNYckw|Wnsv>6lE^`U=a3o*7?fby?*Aeod z+(&?vGU^$-OJmFpgC?Si=?J&lye+8>e*v(@3Ib=D!@B+Ru+~{5QXiHqQ&2N_81!ch z68;u3v>bPIaeBdi+kN(i^*)+qev#G1_X4_s4bAlkDduYE_CO>i;GQUfGIWu5=i z2axsSi^cCLZs;a4PMzVnL5BH54v(-m_&53r8mo3uM))hZ7RZT68K;^%-TGQeQVK(X z@C?2MrBiodV3jj2)cKYDR^wfB71A+muA(W?!OLwy~G0#=$<)2R+raUtaxnzLN^ z3iRfl`iB%=A@@qFiE`{?!A33)J8Jx)4uki^GtL`&*Wv}9Ws!ctEXAwubiAXd>cc^J zYokDU=Z^4k??COnnn(w*NN1LVBaxYiEDk2R z;t!D)$sfUgpt{O=)@FHIsXiQw#$tPtZ!6od7)vksEKm3rm_|WyRr{cw$a;~7!cM0l zVhh`}3t>n2ma_)EGu+b)I;N@>RnGBTKdcqdU0rYFzl5EIdtJjFYeTA-qcPD9)nfXp zt(5T7u>q`z3|Ef@=LNe*_wZJdYw7#;+L3g}D58;S36g4oJ>LJsF;#pnj17L5eu*9Q zvBY__K}6Hbhc@xF?h+u_kFRh%@{gl;DASm2${`Z=*R%FcXK0)AFXmeG4*XSlfV0d# z=&QOB*hG1u96F6|@BHqnt*v60CoQMy(rbkhu5wf#^_hg|gHugvhCU=Rv~VWcjY~lbmyv7qM}?IO+R-h86Y*E6BdAJr zid4gLk7>#8W$Tg?(UM#e^aePl{Gz@~?U9L~X!W2UQ)-bfnBmwyc#>F7*{Lf_FrtPs z5Szj8hRZS6fOv8sUea5__gUT~X7iunQmkT`AqQ-~mmXu+@QCX6aFZ<27&^~JNe@H;Ni^Bo6MchINoTZ415ZB|WuYGos{fQXu)JRm0#8Ignf z&D5((zSLB$7?=$`V+Z8dMZ1PyX@czyDoK0MS$L{(m)pb+VC`J0^}CPb53pSJP4N!W zA^D595i*9WC7)P#cEyJLTjpWNP;!iSt-mWWJgJ4Xq54~CWcV98B~HpdD!1^}=2|gz zkptWs{Ju1)`V8o+qp7l0k8y0#kA`f@-@)>!bGep9$)sPYBWdA)nMfDO%aOFghg$8z z3fS0SmC#c-R$NMKs$RKxaFJY43CUraM0RtxQ(F*covy;(bRtovs9t0QccxGYXF@pI z5V4syWVIG9!&ZJS9|Iea4%k%ru-9kouA~R11eS8e=u+WuU=z?pe5ep?26{>=$Ex6T ze29r~zcy76YGD zS?0^^80aK1mLCosA@IN&&}IH%DQmn3%!_{p{LedFaG+Pr?@a&M?t29rtvJA6BE@z> zS|TsNWr3l%$c zE<6GDRQUjuH(grMHgkpTN&&<4Md&~Q0K0=$TYQ?xA-TqRao3pMhF`umV0xf;rCVG+ z@?7fAP_cLT0dRk1wa|fkuJHUeVIbE>isfwxs+ZH+2sPw;NJ9t~A4or{mbT0g^K)Xv z8{W=Z3G|Lgi>2d2=Dp0FfV29&>_2L6hG34YwNa&&Nyr%$01pDkkQLB7rfcfN@L)2G zMeG&xyKa6Y)hj9$+Qg2MT<}tDdZZWq71Ilk1Ph`Z^JXxHT}(gM`wHWd6YHsdA;wIsY?wJW0Q&$jQYVvDycQ>d0SPuysK0AV|vfxKWTb;NfNGgq3A-9$JxCml^QOx zMVY>Ae7F5|l=My_egKcQG^?p>O<^9{Ch-C2nr)*MXRVnBk|$Ri9~coo&eEjdO#Vhg zwUlW#k~<9EmyF_}>UA^rZ){QdRMQ(p~xm5pp*A2=s%kz=h^_?lsJA|7mXn&v~$V!8?bpOFZ+$%1ZukWoqWX=okiHUzGg}POvuN zW@D3uQldlX58QjIa2&HcEF<_Ag@;+t4gm;fvfv_e9P zQJd~w&h92Fhb5&SY)7~1Wsx7sd!Z(H0Wpy_^p&(hD=VzS0T$8Yk;6g>=mOInt0OHz z_EHs)v(i0*&@lSFZ9DRed&Jc9_EGL~J+wa6=kkAB@(ort1Cdyj$s@9ul1Utv8+E1# zax}XDaEhtoD1HsI2Y4iO6XS49s+TmGtH#~KzH&Qqi=`V$V+a&YkQm`xz)5zlQU)2r zOh$T%y zFLfzl)_>v6>1|wJbIs5Jp@#phR26?0oJ}1kb#xw4o2WrX{dZux@=%R(-S%#J1$sU^ zC8d+rfnH@OtNd0P3LB+LS~vKWJ*n`bbTFJ2enj=-XVV7rP5PQY#1qI;>niH3@YGTf z5JSt6pes*mr8lcKyr50NWnw?}(H}iV^kzuC!msw+0?}}SMocN-#K40}y|Ga~!Tgul z6&bi@IkAYD5g5mRwv<l&Qtky{>pAZmJEJJ-; zwTE~+_9hlDc7u0_v1mf{v|VM25vLJ{!TBN#ljU|$s|mbtwd+Y_hihq31AbsAMp=zL z5bJBJsionbL<=A@GTu5Pn%bp`-&l|87GBmQy_Xc_z=)AVjB9m~g#2&IBK2Nq5!#Ks zhu#c-(yuY|BtKnRU+UPajaAM~KSqdD=k27|JImD>^Who=2`DYSW!j zl&uGM61vz(#jWJ8nM4H*H&9U;Y>y+m@E!4sY8U(uJz3kNTv5If$rM0$dR=9`@>nm7Y^VQ~MZp#J&~@~EWJSjr(**Klu)r|c5wK)Z>Cg!65%~t( zKwS-;c9cd(rGB9Yr+!D*k}V=&zQ+;Z&iJ4DQGPai!&^{LGbJ{#!tmMW&mXfY$@G93 zmvqRYTVmPQ9?q{-jE88BVZWBQ*B7aS~?)zZBk#~XaNZyM!q_z(>O6`fhL03bopt)!r z)sS?LqC*W+yZ9#t`-Bt;qNhR@ejQ>tC#^ryPy7#99aR;DdlHM>A2>a^jV zSjnD`_Hs@_e~9M;h$8Y7HSy>H&4bD{w=FZ(cJSke&){@p1*M zfFMr@hS-Sjw>*h|X_y(kRBkZ<^GEt9e6hqyA)pgF+~2o)eY3lI3$T=6OG@G+RXmn|QaTG;nZfSmLSv*9 zQY+z~#4g;~4kT6};=oS!P0$NzySj}&Vp+&0XpfXH{zThwA`NMw#)xy-gW77f z70RKtl{HE&-hhlo|0NAXfHYILF_n^{eW#eP2S5N$si|fv|3jJ*{Zp1W4#5XQ_S8># z2c!vVqS70At!!44qVD)f*y~R459L&$Y@oivUOy^9(g}z zr`Mv%{4%W(+umC;y3!eswZY1g6M#)XroS}10s9ucalE7!!Bv%h$PqX$)$Ow43j(yh zlCBih5j1>ewKVb^e;=}iT*4o!JGsYxv#>U+F=< zf-Nk>GOwwTME@Wgv%1P@IFX-gJ;MG>DJ`|(HkeoWcKObt6NNHF9x^~eg(<*ozJ1;_ z%Rvz~j1=FP^P`EQ81X){h^s7YGlFaxTU?+S5-1peu7q|mX=qQ+6=tNZhq1m-v&|G- z{=0k|jB^G;Rd` z0zM@tGWX(JBDus*zPndR`N&@ZYT+ws-%5K<^d;!htf!Xv{J?%-O~kl_xd&dN z13;42g$@?KDm>u$r{+RxJ~)5N&Z5o8fa3EJ*VbU_!=j6kera3MenmINyc&!L>_eH| zVsCg|WS{f1RO~*B&ZPIN&CtH!Ao;jfC=JkN$`L&is?C=Y`rEGCD@ctUP4S_Q)p}2| zG_+Z6P8sQC%2IeS)?I5yEOb=FnyLHcjt)Tmgx6a2W=gNhn*1`-U47> ziPq%*MK#4`#n4q0mpjN z;Ijqogl@@d^$*~6uR~d!pM)H=txrx9n|Mnp!xBpQ`{n=RyKc<@Kl#_hrL)QC2jG%7 z(^n$#Up|j`e2*7nxuORk%mqLm~k$#3xgtQRH7^GQky&4V1o z4zy*si&ojIK$qky^e*fR8HLY~FGxEQRjr;uq=Q))q?xMfU+R9U88`}h?tUVy!UpAx zByM3Rkow#gwItkyU8PTwWBtiw8*(&QQ7MbBS$kUE!BkR2WKPOasT1$#7K<1I zA@$%3m?YpiW^SBjDLAtsHgj&l@9GVb7;|4Qo)P%T#P6f~Wj}2#Si>k(1 zYUDgip5+^xn3i!bzl7J4bjO|Kc@NCU9ha=-jJLLh&H)<>R$G?2oA}mS%i7Mmx8`<% zQ<;8=i@l$$F{ahrb>SprKwhxBeCCRGg%-kO!Y46?HsB>l-4p>Ytn9lU2J3sSCG<6) z3ViZ!f&K$ZrS$OLuU=OuHa`k@%=dsx=0^Ds#mDe&?=zqaSHoXrL$oL+@r`+`y3g>N zUxkcu93q`+1bxVrQ5sjDO#BWu$$Uz5VV2N2`ILN2$(7o%eW{SGW%M-d$1HF*bI{;< zb%a=&F3&HJA26W3yO6J-WGC^cu!KD!e-Rs*4&&Fzp2#RihF+d{CEZh#OsQ%_Y3uw! zF2sFAwp7NsQNO16=~=Q*)%gflR+%D+^g<4yZx{j1m;Zr39Cd+e$xqdm^enc0a0glk z{nx$OI~uCrcy_&aOl5Lux7%yeOLa8f)21ztIlZ6NR~s`Dz<= zdbp<8Mfn>aN+f89%njCfweHFLrb!@d|9YUokt%lpP|mwP@|Ey z<`>#huUyyoVC*jDC{T87AlqocUmT!mpS5YxoV-XwicbRYPk_=u>ZI)n9? zY3OI6Slt|)6sU^SCH^db4L$&D_uDqK;U%+NvO&h^HGno zCHg&`$2jD6fK!9gG_T!j^f!dQLoE!+%n-07)?w2b8!rrlcx0tT<8nOh#9hJ@LJ`;a zw~!2|A${E{qlz);OaFFcwEHnI7y1PKhh{S0p^`u*@)(;2y)a{F=k?{6op$#CyTRYk zn{Y>>M(j(d5Awrn5Nk*E@;+>F~-ws=9{ojVi!m^a+7%)zG`%aYEMBF7mvh^58|bD4OtE2KoxoqKa~h8b8QedYaOLuIwoE zceGz}^uUU!Vpl(LczB53N2;HCp{Q?YvatnmMmm`mLw8Jlpk~!ru4g%Z1y}1em8Rr| z=&o>jq3qfmZXC%)SEO&DZwgxBPi<#-i&B#^;hP1@-X=JN`KE12KcF1M7pOG|G-N2O z9sV5h)T45#f zc1hbcnVY8OaX&VeB+{iBP${H5Qx6zWFg}@pU*+^r_Tfzebz}*sh@Tebz#V-}<C( z>_^n2XDem&Mg9R)Ct?mElDDwK48dO2E@WOnJ|cI?wdi;5q1*`_FL%>hsMn=*@`408 z7Zl8-V4#*Xm#4#bsaDx7fgM0@^73l#kV&cM+i&{Mbf?sW(rrvNE$`i{Er*Opz_P$* zSmj7_PD*EXg5ikmsTndb89TC10+oT0wq);}yka)!wl3T6Y3$3g?y}6XJoFtgju9Np zHS>G#Roh~7-=wvsDc)V~X;7)WLr8!8n9yF@!fcgD>QK@QG=Wd@)dV`?L;W-m)mwlo z8MEMb1|p%@eZkTd2=Z;1U``LF8shK}@K@Ur=((|nc@f(eIm!$ZT8mIje`P%EM82Bt zc$(R-BW3xuY%JWFHt?mimg)!PkZX_J3;BcT*cWK4x?7(|*3t*7)%o;1O@}nUgNV+f zthF}=3eZN`7rbx9rJe?&OwAR({g#~T;OJN{G9Iu3 z8iX)Ag`I)T%(nbAxIgIiZ(-ITr@-2YmAMnlA$BSA3t?=7l0HEDfSOP(_M&0Dah4xbk&;foAzYn}BP_2LQ6QGR1E?<{G%fgvy|G!0Xl=_>9ip$|R{xvl#wGus% zdT0*@kC9=tSEOXPx!R?uqaIHeD|TSClL-yceD?a}52P)-A-^Gf(Qz{TF8imtRzD;4 zP@CA=lAD8UXp(T99qVx#KXdC->O-G6iOa3_qTqsYSV5(@<_Nd`zr4ly$NaU-F~~G% zDEug8FVhc!;5_JPR8vY2*Rr2Xe_7@iOcavYpKM#++-&( z>`QJ956Y*2lzbxZs_}f_me&Q^xU1Pteh)wZUxA`hGp!HV=|~JPU!4nd7uOnwW>!V2 zOJQg_z6-po-ih84R^Thy!(=<}32Tg!LUr80%e#@U!X2WNG#fe(H?b^G))52YF$PxN z5uejIzrf#JWnF`pbfe)3G_;UJ8{zAc+x0c030xITD zA{f9!QNkFMP}$I%=om<#=!@U4XNp@6|zcs!~@uiI>7-ebM)}W>>PcXov=1 zEM>_{QCJ!vEf->hSHvLv7&1pRq%4JxdawA~(S!0?kq0J8E#>h{f5T7lyV)wNl2VB# z)I=oFQMdYesS;F~XzXb(Zzt?T9ZoVG(G+n7*9BTBtx-yFy_YBFf&kk3}wJgtWI6ss$S&{z>_+yf&8ln> zjgS=?feuyO9Etq2^iURBD=K@@#@^rSm$GxP8vgnw&V1Z5%hNo-gJOj~=mt-Bwwoz7 z-(mXW-RCw#j+~=RB#A8{mr2hr<@@L#n%~6u$lH+dX534>V(b?9Vk>F-2N(o(v^?B! z&GOEDCwY27KAhz{7I?we4IA}8zH>qawTF68ZXVjLG+*wel>dx(O)OC%~kAP;%s=vU^!R3y*$~<~nsHzT&^}RZs zqvrV*hG0l$rVDv`F0oPiflZbN5&P*zz)I|I!r^==d{+f~fB)%FMeTrCON?{Ig!@|h zYm>0^VL}-r&R{cWQL{>-(M9!gj1CFjsnJ`&|Da>kT=1Cjuyjp&rZiBd5zj1p<)vb8 z3kVL*DZ-ou?}P1>nxd6|XuJbf^G_ny$+(;ejYX%EZ-p?mC*Cy2W+xN< z7==3lCvbbfRDKm_=dRig$gfk%3vtW}pq18v?}=`Vo)pR4BIsO4!4>CM5)JFmm0=Dp7h2I*kf|r?WekIy9DJn8qqE2UCTv8389^fGNB2DT2ZZN7# zjKgN<)`WYYMaXA%2FWNkJT3T(N{pT=O)VwR{+p)CkA+f9l|T={nAJ(T1DBJeda~dYik~I1i6&ARJzVD$VHG-=zrOBr4!H; z^^(|;o2AqtOp*)iDeQz){~)nbpu8tDZ@!mjNc5F?C%0W{p1a9YKC086MNeaKN|nTn zhCa^$7={IV{_mi|x>{zeX5_8x&!PTxNqU2=Mk%C*jjPD^#;T$l|GKyW8cgkmD z&Jlmq)@lOs1HAzbMju2Sq>WfFH5nP@jW<}uMVsr0)%lvytnmhxklPU-DOW~&Z(qwa zqGjSkr3Z2c-J2WH?HFe5Yrmj(5`5ZW@dY+s<%HIJSUK*f8tt-P z#XdyaMnLHd(ZBF4k*>Ce6GP9P{_t}18Fg1wRWHGpB9~(WG&4GkN+;Jw8OJAhL#?d6 zPy|^{d1f|Y27V-$#MDKfSXV&td^;f#@FA(lU*?^G6#jAELc>>MJ5PGbF)y@wHrUc< z%&%;#1sJ{it+UuH|MKKX!2P&iV8`UY;+L^6foX=Xz-;pgPhIAR=Ra$gDr?Ar^f#@Q z=2Euno4E9}i&5vtF; ze&U9Z$0en@4NZx&MeUGn#P@J-%e=yk#tyFfj&0#Ib&tOb(wQ3tH1q-JBW9Pcl%Xx! z$hh72$<_ zZiXBFeLfG@o+JGA;f3f-L+zMGnd7MX${MJnJp?Qfw+SPZRJ8|j73;}1QjaJBvbov` z=>e>T_D1iQM}(Oh?}2N*N0E*|RUm%(}Eah%aBj%sr>D_8Ce#Q=+{M`f<2WBPSE+8ydYesz9?0w zvhtts_sxSfR;uRe8ahW>!{wa4bk_f`vQzz^P?r9kv?Qgn`+>i#rGdG3%5`^-@`+5- zGP|mG@xuNGKx4=t-0)82ANY3WRsmlCCZ-I0gFDWQX1-ZpT8Ei_WfvKp+{Ydf2fWIe@QNSWnnvGI9w&4R7|QA|5_ zhPPYF&}Hj=+pC|mRkLO<-f3RFezc__v)cF0dM?XjYYDB)-vChF63H!-Yg_-0{bFqC zKbe`F7jjoM+XCf*Bm>NkhyO9uf+p~PmOK?pK|k}d0G@9lYz(A$>X*;jG$^s1twMY@ z(?U4Hoo0SOPgCjzQoV}kg$QIkoNtPl+k<6trap;g(EaQ$z(P+8WE}j06>M#dhcTMYBDUpx10Sh< zA(06P{gLbZUf;j|EwTy=208le#0NjCYn0bSGvP$#RGUL>A}&|&Tc;pvkUz$wavusN zJIR-k|0yo(2x4RqCTzx~^cf}Aw?TNJln`IAC$#m@Gq@*|4tc9Lw#CqGl&|PAvMjMjI4k@i-=o7U+sMaQDP*DTGBMi|gS5dS=9Y=m zty$U!dA-`7crRjN7vXO91WSS3O5EYYa_&lVgr(*S$~9p>b4<{2*%M&5Tdr_d0EjHa z?tsIkzTgjK51t^*@mvu{lksAIW-qZ`kL5-{#Y|o7l>Y_QJgSo}#!E{Lqny`ZWeUbh zl^jm8Oyr=wIa#RBCN75$l5!YMnJUbs8qqiWnf5@CPoD-{0kR4mRIctQuA&u2)(h9U z)xkZ~ctI`o!w41kle*Dt_>|JP>Ugzp?kj1J=Et*LGo1Sa_wZNrKcUO1_o>&!H&?8( zQ>#QyWh)lGRL|)Bm3Lb6qQAI1;xBb$+O%zsU_1AYLX68!1@utin8KDtCtNcewdu8d zS9G>(Vc~wLDqS;JnaZ*Mn_6ybzGt}gf$m4fN)^NmE;~D2J*iaUCnHzLlj=p!Cgcfg zQkEjWEKk^*0Z`i{^mW_S`ozJ4nmL2Lwa6V}A$yxl*!%{Y7+)G$EVN($o=-wMC@8Xm zd#J1vFyUWwz1-Du9Yy3_LIvmrwvpI_mOu{k1wbcjiNJiOlh=h+gk~yRQkuveb6S{Z z;xnTTTn|Ku=?nKn2Bug8Rfy#_Gq)$9m0+~5<=a}$SzA&&nHtdy`9$j==D#8%*_7O? zbc$<6-*r6V-aGpU*^b-xF4S%H1kI9v6;`09)$D<`i4ymPN-(W?+I#20!ngR`!X=S^ z#1@6PHcs9j{Hdv-wb7<_7ONiq;CSXdnbsri1-;6#DbgTe72k;csSk`MUTvtAiqk^2 zThefb#j-`6vJn3$$7Kw!kMx$yiZ%4Vf#GZ=?HGLyPtw*IW@@jb<^}&lOXz7r2)n=x z_SfZ>pv^5F@p?2J=D$pWZ}@k*2}| zq9Jf3Esbsf+>ca(?^`}=36|f%n#h@usx`M~A*|-rYo+y2KY@AL>^zFX!O3zgF(!Ok z8sV=`n^X)>NBZJ9#E4UgAK=sNI++(8^7$Jkp@^)_>4hWeX~vreI2St=I!P}{t+e(RCe#0^qlj&KV#=v%a86UM3h;}INd_zv!19)V0FgDCQ%HI<9S<5As1Iw8Y`Twof0(@-)d~b|l zD}jEsHgFdRXU)&S+Tfi;1k3caj&?S9!5;~&@y5a&Y_PPK*#XT7Y)9*Z8~E9IS(ZW7 z-?5+b=P{rBkJxSbhYR9?x58e&n(wGTS$#~h88@uFwI9;WTbq5A^4od}OrVazFY^WL zjJDcZ8IiD7S~G2+i`Nz2DRm@k6Jt~ZqOqm*ee%Z8%EH%zM4;3))r7SV5&-GAOuurJ zdBw{1jSHW0c~^312h)5n5kr zD3*~gC@DpsgU|h0_7(Iwv_Fl8$J%YmK5S5Ua3LT12^BfVGE?R6^lxo;(NBFiFtV@; zbu0QBXa&s?mBLTX_|$Lks))rK-Lr7aEF73NR-5Pjq@PgZ_4&@Pj&Xcn2Y_Z7euFNe z4%t_ReY=Efm_gn}Ov7R2mD&jYN2^EN1W$s+KyL?bd4v8Szet~=;K-_VE0Y_EdZ3Ch z3Y=l-rxH=`e?F2+4-^)Y^@wN0@097{KjcdFvNnzcZA01nrajPebhB_2`=Ml5M+*l6 zFMtxZ_W;XIgRcn&y#?x#C243ug1T1ms5kM8)(`>XO>J+{da-A$_3+7LJ8Y=Xkv5wS z;Emj4rRO3jzmDd!?%36whdgZ^mU1^~r*)0xmixBBnjL9tn@ttkTA-SC5&er_O4J0B!2?8wc`4aU{7!6e!%^;#$+@1GWw>j+ zC7h0`5cyUzXFY!j9OWCrd@bn2x3ffmY<5c-ANQNx5~yma4A=u6e_J<`oR{$&7~xI? zw;FBPd%4VLH^x&zNt|Uea&4nu`E{l@BXO>p(6TBFmjV5Um%vqMRpVQ ztnVoSt|hzxXs|i5*gO;2k?)0G@DXYP8R0M}0bc4`0bPj7QI~RVVs)@$MFuA@+nJZ- z2)~JajQ-8zJ)CP*APm0LS2k<=gYouHT2m3l zVb|YMi_%G^2KN8JU1(owNf2?=%X&{QbhdMhu}{<&sAbq#y$fFtZ6vgG5PDmrYjp7` ziS|2&M-!;M3Wujfc}VmLUnJ#6*~_s^%@h`!t~DfLLrixtwHs`+2dE*OA(vd$A$p_@FzB+iRM=YB)`B ziHkyC9SbQhy)X2g$b<`veo;%6Wn6iEfO^x^J7`y$rVXVRYn{{jg`3j6wO24by_XP6 zEO&XSVtQ?HZg{$0Qr%NHf?-m-7$G{{Q;jqtPyHw1Po8A%4DTQ^13kqjmapPpzB&8} zKo;uDx8QkT6xG99!!^a50UO!@aSPFkBkoJV(#-OBRkVRyGOPl>`CIr0iYxpe`~#rC z9*IX#lDJ&ms4g%>56BPZ_I z>BxvSv*XI&TVlC*<))gNy5F@8zhJLbxF)nObj>*-n5Ol|*KTbdUZmgF(zKK`RC`Ak zQBPC97qxJ#=B*fv_gq~kxRn}KxG}XtWIR1wD+m|se2PUY4ETi=gv5VRu8>~2De68!8T$~pq@lo}ywKh-v}fNCiWEB+QLqbGz0>J9Y6p(|wN)K&Coou~i8X3A*rALy?3 zHQc~)is`4V*K?8K!8eOfDY0vO`oDS)^_RT?@Q?j6Rkbjej7OTnTm7D(Lj&!xava%~ z8X08pZ^Q!o6yX=`O_^amM}Eiu!WTQP5tMzfI7O&p?MQqPUO>0N@%a~REZ#l%Qy7_8 zNv-V$7nC!^u} zsbdQVDrL1bUYvRpxukrq(U3Ga=LCygmx{#DC6O)4*KZW|h^!1B57%RNniIUm?3c}W zGGuGX8`x$*s&88IOQc86J=?*G6`(>+$?eWevx#6m=qMcR*y6esM8LSsH~iOmlkq>k zv@s=_^wIfK;U9<+)5{&;+Mqr-KUp#P{UrjwH~y7>#@O0WCvzt-Be33jzF@uWnK#w4 zFYBbYL*CuZJ+hOm83E8+wd4q2XWy5^JabuZQ@7vODXCX>hk}}q`A|UprdNx(^OZ{T8B_H!eJ>WTa-wz5a&yxa~rjyFvHcrG(#U9sv4Oc zCUfU)nNuj+Pr7<4GIuL{9k1o!io4*7Gg-}WZ3|^-lf=o+1(8>DA3BLn;MO3^wO3I4 zsM5b5?P_mA&Id~^k@=9;u0R?^-Jd!zO7EJ6=5pVa{z ztM4JU1*7k5B{$Rp`~jUIGKfXgVEkXPPWV3bg7Y9ZonOPHlfgj)Lb>o@{h94(AHi}+lB04hs#5?@4!ceK{CL+3XPC_!~w>$KM@aj z)Njnsq@%46d@JTWu?xC~%%wL6`{GNyKS+VD8>L8583)!#6@n+N3-JjWP4D7%Tjyz8 zsOBDLOd3V@>xS*BYPv- z5p&yI37BZQ7`w_{O@PphJeT{T#STuy7jAxqR0%vXZV}Aw2TB{`Ob-?6(0*!+|fbsXv|WU&an!Q2j#=$`7)Xn)l; zIaz2+rE}+~>Qa*% z{}(B_rg-BoytHsgJnPwAu!@|ZO_#=^-{qOwH(#={i!ch0y)68KnPRY_v(O*tMkO0O zj+Th_XH`^J@OzP7-eS*Qe7Rytc%ZdJS}+HZ<=$N1dunp>qiCY{mzc|Mc8?>=>w|>y z*a&n1dK9lDj2Bq%_=4fecwcE{18NZCiTcVlsyp)9I}ofzeDR{|2XSMe7Rfu65w`l4 zk%ScZ8|ffTBuwVo+({E-xFdW)8LcN`*LvmbL)z-Uq$v^sCprF;hS+0(HzKcW@Lb~d z#qLb`19l}Q=rS=uI0pV3S_sc!7Rq-yF#2ogi<~!2(;`48uAK4}F0ah=--P$u%45~& zY-O5QD(?n7iYUS#;$Owb#20-(*dn;y^if}}j0sjQS^}1HR#%&XLxZQW6y}3oUaUv9 zMbq&omJ8H#@k$h=xJ2}nq0r3W9JDc+Aa{iKknNPOdMv)0mylgTp$fX9dlcf&#F8kW zJyJ8Q0kpDqiP#cKW=f$cLK)<&Ty4pE=@eXD|1Is*7H9v2YiDnfZF&Ol2{&Xi=?zFr zxJ!5$J(fiBIIYm$4Bf$GGMBhY`WXEzbq(6!_iKx7g?0c*)b`_N#cnPtl1lFMO_%0D zqp4ti2WJ&@nYabEkoAE|)LhS5EuMH4idEy)opf!a4rJ1-5ast*f_4fCLf<6zVO^ z-M2H}Idk8c|8}Nt-cSDlK9Dp|zR!JKH!cz_DJrlevOIR2Y@c*5ZH?P+V_WzV-q{vz zP*^H0eSmm)7!fHL3f+}P><=$36Ss5OI;}Wh6K4{ytd@KgbhE^aM@vayi`8H!Bx7vh zrE6__!Mk8JUupSnZB5i2c+52cYdFKoHd5Toy|{5o`C0%A%hFQ;>gW#T?CLV)<9xD7IYTn_Tl zmIE#&_+@g?QyPV*5&@)3<8NW6&O zfv-6x3jg3u6eaWXc{2(-f%nW|t zt(Iz$a7B#08ow!;XPHVIHm$N;6l{=0l-!k|j!*W#!Ix!El1$tzNSv_5HWBpkNA6o; z$rk+Jb&@ns+zafpJcz%BUl%B()A_~{)}kf+x3IxH!u}@^QSwFbJdPAxiJmElz`uyv z>^}@XF1;pLeolzg>p;Wgvm7FAgdB3oH$>1QEUyxvl4!%e+P{wVkY?2k<-IVS#K zeSRp{KBnXY*IF{#@?82{GOpls=^SgwJi+wD@<576l8E1&w?G2%O?|^=KyG;EGNW8e z(1or&V3>V9-9=pIqOe1yYK04~SH5$!BX=hLt^7{TRQzKRQ=4drNa>3X#e# z3%;RjcPa_ERniS!ogfDO=n|k`1zTJe>b13pV{gp6Js*MTWpRSK#58V>>|op=b4h79 zexU5R#35=B{UvpqugcmMEV9K(p`=v&t@DL65Zgnt*5;O!6KjRe;`cHCCLIT67Y*d! z6m^oQOdE>ff(N#@CErAvs9qM_)Xf%*Hkn9#LgH|+V)q1alDVDvLCN`pTB4`bFHW!% zR>*uF_)N^$Q%f{j@`2~&y3;^O}-)&cbHG4-ovMpPHW<~ED^2AidHSTKc z4fU4xBB$tF`3L734@7Kpb_RNwVrZS4vPuh%N(dGI9+iN|8)aTeYtIID zUVjbDRn93`goRNd{680MMt~>w-O>RG>V&D%q2NAyE4&LgrQ|#BhkaW44(aH;eS+rp z@3wKoelEB9(DDwPF#Zfcg!d#0fk#P;50yaMfS1B~EK{*unwY?WQUOH?uWUzI5A$qd zw5cuG2J?7QL0HwB^V8}>mzvIDYSjW_F|&zTrqHXr(p0aDDkcWg*^VY)A)#}QRB{sT z`+EC!6Bxsn%8UQ!y2P)FZjg)xP9OqQX8&Y$^A0=h$Z}J(B8ePD(vBMkPl-kn3IMjP zfr=!<%@7zNY+AyXZ!YY|JtfMML>3u^L9SQ2+R8YZDKZ^A)0y%yraKl*$uQ?pzFTl8 zAqcc;(ofP}QYG$dNdsbKq+q*jYsoQdHu0Gh@VoARYU{?E&o!BwmmZMjIsp7m!eiMZ z{(Jsl{A1yLU@pI-2^TLhjRB^Z+~SCG%<@?FALkA}%jU5hvJa755vGC7Bv(u;IJ*nS z0O@XIz|bhFfIE->$M^pbqFvKG0*NAmIIV-@X*wN{yVbXkL|vwag( z?EFPoZL0t&b;#VsHOV$uae>$XjB|CBJ}>QNZzr8z#%F1!bSsbCCtEJvVvmJ~av@nP z6)yQ#)Wh*Vyo$?ng7P86TG>3)96>sp5f2HEDD;BqK$nCgiXUKIah4kq^tm*%3nHf{ta|Pgn(4E*`mUC;;+e-vb~bi@+deH zni)H>@Hb>s{F06*)6K`6uT3*#3SMiZk9BILCG zoG!9%guCEFZ4(7Wro-L>m%}amT*jx+yJDd#SzG=_GzMr9BNBf` zR>CD{H29}I%)3P@q^?l2>}f1*dXsuw6ifG}cB+Ot=YvOFO`!)dm*@o^S@bR2eS{_& z6mJDnh~I#-10o$@4+oD4y=ALxIb;qzNSr3i0WG-H!bOHrMF-dLrt*WrPqHwU6p>mh zS(bV$(8r32^ks_xp)eP3!y1$C3WgIyIQ#H?AX`+$?^49y+mw4tvR6La^+tHg_KcI< zCeK#S{>G8q)t~~tDRjXjOt)-$3y71Fe&Q^mm1UgpH_yWX*u6N(Jxuh+`-l(XyJX|c zJqyj1@IC197+TbDVSLQckhtie-1D_hLev>{Vn+-b=+0ib<1wYcUhmJ z)sk*(olCs=f9CPgO|b|?8}&YViDw$t(Q~BLBXX{H81amy${rFU85MJmSS_3c&G(p9 zAKd>#`XUQmiC}N%J9rqvx1GZGkk_dhOqz6(>6K-;^|3^2ekl4cwpsZ`P;lG)C?-l+QErw@$m_c|mA|=!eUjO#tW&v3I4IN>%w?qm zo%xqbuM`Ym?VvNwJMDvbR&(>66ZZw#8Nel)W_A#frH4DQHg@rp{C3<8(R&W=6V5C8 zXq}GRdDpf*i#QV91Wb}%IB*KMEXXRmpYXjTxp0T=%t2@D084c7Gt(qTj^K8JlA~fp zguLCaMf;@6KXafdO+_y~h5O$w8jmK+!|*m$H{yJ3P)O@JT!mz3s*S zx=K6*dRL(bQ;0DCX8&3Fe~MW724=K*iCTkp!(y1-iUDw4g_NOfzKVSBcu5vAlAaGA z$D=8G=~k$>eXu+O-zx9xo;mNRw)6egW=S*5R|GA97X{{`g1vvUc0qZ8 zw|pAw`nyuzQ~2J#%%&_&HDBJflDiYQ1b&vR15?Y}$hPD5(nF%@_z`54t%cx`Xvf~; zoEXV_VLTM?l+a$;ZCrti*pxiU7norgVLtAVaN^On21TzVkBfCJ0WalEkdd}nXKU9&wvDVm$QrZhOn=XNw^+1x8^msFBRK4FA)Yf~v zHPVp@hxXhNT@#KXLeet(OTq-DNz=-1I+Wy5i`@FNj8mL0UkY877Ft)@jrdid6S)A` z2Sr(m07{latE|s}h_coN2ds(m37j17N|suDqu%NJ?Heq7VeaqDq#k>+DI+qP_?uR; zCu(`nQC&Y5j7G=X~=&a)1+O(1pPZp9S&X2(VO0{d`jKAR%rS}%is>$bucEE85GZNx@}rwGUks>Xe^B?{k!BAr$>a? z)RsTn;$X9njA@T4R+X8#qb3>1Zyl)MNdovcedl76x^3SWQq1MrRiXppxy+=G}(vQ_7#fEJ0#UYy{!wr z44+EOk>8a?+h@7FX=l)c)lbr$KI=G0#9z~l^4Rt?d4kTl#cF1mX;snlw~BvHL=8u1*J^|;)2n_l?Qv^@#aZ_g$b(!gV{>w z=Y{VN6j{b3_9)vZI3_wTzE@VCI1Epe3`>|^f?H0OtmdT^{@UMGlqdk=P6-S7XA;ly zc}2#9SIeS>WGM@u11UUtLb`3NOpV`5T5sLWJIh*3KMB-jp9RAJeZ*Lc&~$BEZelAu zs>u%fSdN3giw{XhOJ3*mII9E~3lH(91LJtUBsRTeTE{v-BKiJ;Yjy}4iYsK>ooij4 z>5rwY%`wmx#b>#UgdrFHLRu(mBt5n_FvccU=0kU!4zykn2~LoGA!x5{x^39?I z>j3gObjajzU4-Tf(?kj|3UGr@XgxyO$4Gi4#1dn;R30g6(y{|LT=2p+i9c90y5vZC zrX!)GKQO;Q1LO<7@oCdE-q@0Klbdrw_}q>qJwEu{Qd{zX`;#}lQhk|l>Pb&@jjLV-2bc<@h-s?puBiH zv|g&h&$32_d6s?rJT3_EWl!yA9r@9h`J?d1j_!$1MFHVppuxTy_#hgCFC@R3I&l6m zt#%A_O%v5&34Tx!CYeYbB9G8fa@;Z*OF{NH(}l04-=z2XEtvo0n7NtfYhf?xRfSqU zQ8~f$)1${4kdeqYm~io^Q^-1IgCxqigW*voDu=q?QA9*Jy>xzDGq8I;Nc=_Aa{n?d zDYn`7mR%Ig=NV*YfNA2Mmg*u6B(r}gFvwN2xx@7-m#6lhZYoUlk#0t6WTcENVb9TV0-5lTdIUisx@5`2uy#J+~>^W*-S&t zRf)M}X}q#_RV9xS{>z_S9uR-q+qZPOuy2QpMa{}O7i%}69XuUp3Hk~Sl+-n|w^wrd zl>FpfIQTI?vV5IrP<&mfD?d_@Z=F<_YI;`~w1-M*;I)87*@`!d8qDYM8xE~=3b;d5 z&kZjFs00XB*h@YGO~}#kY1?zV&wSJhNdAh&|Wbe+G)!}j}zTYcIsDg z6W)G%I{qiY`NnoX5Oirc&a34*(bzlfMz}mr>oW@ zMexs(PD(4@0I4flD{e~LQX|nXj+2%z&MTg7c7Fi}SwkEK7f=ztN6ouie8gsNAz0nP zgDtjasXCc1N(Oqq$d=~WoKxvFj(Cq4Op={I=1`OEWu{lmv9NMeenGZ%Cbh|RLO$D) zjI(E_&?-s;Se&m|56UKMGqgwSelP({rvEDL4ZjCgIihigWU6B~7~{Ucy1y#Soe;I6 z1Nk4--8_~xR-GmK;zt}uI6%>6T1u_de0M*hD>RkVDQ|1v5ZZ$Zp<4e=pxnJ&BKEhp zO;5RHZdrZDw~d||0I<#8^uT=g6xC0>Uj4?S_fJwi$JRiH>1^bNR_KXH>Ek_y&8IN` zp^9hTr~bT(D(p2#fTt9j#4F&SWI6bUbOYXRXPRh?a6UDM_|vkn^>TnEI1uM8ONC@Y8SJ8X?{HNqRyv$`YnyFZOLP|+Awv-idGR^G1oWgk zz$Rrcn`XJnoJLP;D$Ks*;AD4xX`MU5vrcHw`#_C_GnvD(EodH?Aaf>ufN~XY35$3o z+#MKfjd$}M+a0t-?G6+TbrH;eadMjoZsU0Yqt5vPgCcR@l3d^1eg9Wb_?+yC(SPty%AUSDgBcc&VUv`jD#8j)_4I(8XG&X^cF^zD7t<(e_Z zj1O&su~YxbNA*UmISAet32naAITdI(K;Efib;| zE%y`zrPT+GUFOFgp6}?)Rzw92LwEEw6r{Y$O8g@x=t}7y%zGiunssuGp-r?i>y+t+ zq1Ups!;hd(4b_h$Lch@e44cx-MxcL`@ji`hi^BN8TTecbs)+3p3 z6E_*3R*8-C|2NJ!P(d4QwY!5w-op)luU^yGJt8$U`T=H4e)%@Et(nmf%!%!s*sQg& zdGsiQW7|mMt9hdjuU)JU?W(vCYFhtisBfaskUA|rNSa!>K!i zgO{}Jjqdt3!81)8vXbNn4A$JOLE)sXp`r&j8^8G48M+VYYkV-%Vo0-(F>LGJHQ24J zeTW)A-FQRuGqe6wV(7)^rN$LG?F~ET?h5XFYc~knENKkyn-Yv<`}G#Y>y49{w?E_Uk6e{yxzmZv6j!z8c z%<&n!u6UZ+Zp+nBztR>VY1q`zfQdbVKl*(Q4tsjUCW>^L18fW=vU(pAf|JVr5xjz_aj2&s0qfQySf>|GW&#r$FCUbIzKn^JN}(@`gd>R zDO}k(l(-^zSiL5+@+=x!d|wo7RHquU=ln4A)lLt|J-36a+O0Hr$Ekup*Zyv7eOp%x z93NwhnmWN)UjDmkP{wL!OyF*M95#`A7wWbc}!d3rh3f*nJ)BHmJ8Fj1{)gB~)>YHE!b)|ZdER0j&AFYu4vs@jBw}B^K zF0HRhq5J#ZdDAO&jESmu-*T-Ww9Wza1poKS6#0kx1NBthlZ=S0sp$=cJ5|;4GwQ1J z4zh)sRq1-S&=OM~oq9>}(t9{HWbaL-YF}0|dQNhSy87gYDG#dz*i&qnJ|tgNd0D@n z*d2(~!9($DT}I$`%{guzJ}Ucky-bu zq`oG*`cleQ=6c#pa&NuH_>Feg#TrKX{z^a2o~gxVtgJBj`lOCi?5^6VELP>{=J>m) zTPl}VU1p-a9_^;IBZ@IqX_W&i23DVFXyQBgX{aGr5&rj`HsZ;+C%1fzM za%m>(9kUadkujM$ryPe(r5@_~rjB*L!{B7Lqt3elU7Oj>uhR}O?uYwnU+E5PpRF-y zDWO)LtLfnzsTLrk*gp9XGmU9QSCVPea8DOeZ^cmOM;E0S>6%t?0)0ibs~F(CM)##3 zAT6nxzPE}s*8T3Mwiw-K=U}3{_PS}FyQ8Y9V~^v7s0%1pzgJT_0+^C;$!>tJsE?|y z;`!KGb-gVQF2+_l!YeD_cg{a**W&{UBWi?&72RJ{tV>-%HkdgmO_zCQ?PBSFKcC5L0x)XTCLfAB~XNItJvy)v_mc>Xud(IF-~nWeLsbholu$rD=Mk3{mEwP5nAwlxC&( zctEXtARS*>OL54wx?xZ!sIl&aYAy6k(LC*8phRm;4mlxBingnFC6rsy8y%Y2uQs2! zk#(o?u*_NB${wqCP*<4B6ff^Q^Gkb;`ocChl)A?R-rLSoy8}_~e>1q5pk`P`U4~x& z-m5|0rcrX9s=DGC=F&B<`A5x7-CR=-Wx-?7X;3ycR2E6KXL>S9ueS1ldr!rT%FltO zUXPER5E(?>jvxBr;e-M z>yNI?MHaGc*cvnv)u^(OF6u+j&J4 zL5`_g>K3XmVjGgbsx~3*)H$l#Xi{Ze;J5Y|qPG+$Mf!I$a|8LLM9B%zOgpW<`h}B= zEjGOYdjbx{GVqr80@@YrFPp2mk-S*C-u7{YnaIV*>t7v$gaY&Y#i3i<|}sm{|SsM4#V@F|$gf6l)Tzy$Kj16W%qSyzj2Rc}*Q!J9EO?FYIV zZVD6<)3piS2H6C3y>&T$#8;PL&s>n4o)wn*RRHOp>h6$Nlh;x^+;J(V$Zd+1Xg}9% zWQOC4)L^@$&ylh@r_>(K>WaS7&lwLY-_)$`e8)4Tq133sDyv7PEYU0p7N(KOqZ_tp zCwcE{(}`sN=#(R}^YjNTm$_pzfuecB_nwjQm!>nXL zZG74!W>{TjwJ3ySbT_{0d^hcYx?|m(j33F$x>kCj-dCMoJ%YOvLuy_@81*s$;u*!w zwHJ~(>PzZami-##*jiB&Sb{7;S?vH_WhTWNy)^UATMyoZX9?cRHYay)C6Q0p52j1u ze8dgiWbYJ%yTO`KvRK+Avx|13SeRZ*cl3Wv5m!#55~~KJ{-_#TeIl&~)2!N&azOnY zU7EaJzaCwri`V^9R%uUaE$Eum&Cs4wmGUmwSJ|k&fo5mCNwr+a=vgWt)~YP)0no-XPo~rBJw<@(6+!@qg5&z{^*`laniHc z_n&`2<@S_EX%F>BDh<9JmCuxsns&@_DkYFpd5I~h{$8~j{KaMWALUWDlP#0js|c3I zsm^Mfs2BVHs@xo?C7;pb;rUp%fIfMNTo_2yJ)`u2|7i<~r;=;o|ES6IZ7orxbJu8p zi2zO*bKKIeEU6;eVJ0`Yv(YKFXv(Pc*`a^a+NQ#8uP*s?qyVs$x{BixQW3C+RQRdV5Z5 zlDH{7vb57M$drQ|# zwHlnEedoG|O-`BO>8H-pgUsrbf>emDWnSi<$DH#w)|kA}$X!X1?u%&%3~K|X4n z!$`r9)LqJ(WVckEvX66epq1x3+@yMNMWyOtRe&0w!eL0=9(L|}Mz{9|{S(N2{&GCI znuji|GFQ)``-79<8*r=2l~wVTy#jlx4pl1s7X$xR7AfW@?~`W-R;h9DPydOOI=!lL zZcTaW5K`~grXH=?ft*N*#%yXMQv+^-cd~}H30Mk!QuSSVSzD=FrT>*2sk)u~Ks_5; z>Urha@A>54!VF}*0iOSxuDNkk)pmcl@u}!DMPWwWA;}VTE9fW|sh>uqq3J1FC{^8_ z%CC<1O!T#{Pj{k$s+vI+hQP%NS&gA4uKEx89DJ{(mFJ-PPs~lM(IsQ7fkt~1-ay&$ zhN3$6Vb5>6JIfxQq^@hN`lgv3G+hj*Gd4S3WsFEZ;6558JNtASLt$OA(~=r<(@<|} z+Q-zB;K}r`v@Fag(8{}}hI{%UE3;h6!Pw)>NbfRF4`hpaxm*jIv3J%F&IN3mXl=@9 z@Ddu9wvO4%^hx2$-ntqS|1ua*Wb5%wD{^ej~lWVNrKY zL3GbmqtN?AgEC#RL1FeCQ5T~#eA9HjFdrq=-Gvf~d9+M>z}v)gIlUWmLER-|i=@J#`h%y`AFTL|AJHETOfPh*_j}v%COS1h7Nc;pCO-8(<$AOQCS<;Q zJ@BlG-WBN}%~UZf(&C`meklE4YM-Z*c7(ebbSe3m_nK>U+EZ<7pCM(hdU-`FVoF80 z@1$C;tyT=tozw2eQlzsqZ6SlKUFu|H0r#|i5pq#7BCVnJn!9!8&W2-5c=D5)A4sEb zjlT?1SFWhuhvoV@`1_Da+PjzI2#z>L|?-v>i0k6Op>5 z;s;!;C+Txae)349L*?7brj=RMFDi?fU6n^@QQ(96A>~j3+I4{f z?LjQgT@|fyz9y$L_k1;;HL+dKOs(5uX{no>HGbdd-ZeXim zlzBL2Q`JZx(Oxg^8AI0i!hJt1?1sa6kNs@f>Q;Wa|DV84C`Wz1A_MtRdV>~YAIW7* zk$;f?oJ+0frrKK%Cud_{>f?0pyzT14Jur4#6zQu_?-XqI)xo!;n^)^ITG#Z{F38~e zGPTY1#-ggg`-DIG-I4RWOe!V)3SDE42!3?(=Y7Y9&$@kaD-E)z(l*hN48R>1~p2oXQcJe%QH`jDbR%n*{2YdJV z3IbzYpJ8XG}o3>fsQQe#z?JQYSd%VsKe$chnC!~-yk89`F zPEVerU#I(1LFigwF^Z>HI9h`(aplq5YQNVf*FR00n^BmyTrd!j(?zw3-a5gi`XoPJ z%Tx{vbYuMh^}dh(DcEc;9k?rTxb%TW|G#QnmrnmmZLOX_EJ~II3My;8>&!$WO+k{; z4WP}2U-ysjS9!iu$E-KBD-kqxdP;xyMa>Bh=B6;4?X;ItcM#``)ye-V6UqOQXCUX~ z{ZvlnWUXDx!l~%S)D{&|d`s%T?kh<QaBR}v_*dQ$pJj050 zt7X%ePw0MaChJ<*kmB~|RD=b}Qj1j^1dGy^dS=_zsad9>Or=_&o+Z}D5`q_gVV>X!MR`3Sehz0G1a z`=pi1tHfEdUNgj=K{>JGq8LSQe{1?cU|Lm)U+w-=^HCPOce&lzJ4t)9I@ENGvMj-@09@ z49cI9p}tLQO8!e*k+MM7TRTnrvuYrV)3?it z;LGH(I*sa60IZ%+)sYHgCXk%uovG&3zV)LTp?Xjstv`}{Lp8O!Z^gojmsJDmF4bJE zcGtE8Eh#;0=cK*$OPu$})7mHT>1bVLQQ8KoN5z?}xAt6xEInU~XhZI1DHl^;J4%s> zX%+@hm!y}g_f#qa?fl}Z&UGVvV~OtWJ+4Z(*u6B;kA75TYXr%iC2eH$Gg?*t7L;Z1 z{3jiv%=H;#Q@!8B~L&1d8)$s z82UgR#ojCbht~o-&^6LRC=pFkY3L%?S$0ox5$&Kpt9*lrwXc(;@M48U_yHLP=d!&< zkq#?#)VtFCiMzIXY~@Z&Qd!NYq0P+u@{b*F1l~{+WGZj1^z@VW223L|#2w8xMznW^ zI&_kS#^;?4O>WjdIJ9(D*74+y#_uz41!aH5WW7DOICwXscks5cZ)nAap~es8q4Icbp~lgdBdf<;!xtHbwSI(!G^96@!;ITi-u04R~t93Y-9L4 z{F-6%tM9=<0|$qKgVz{_n$gBTQbooqeaeGPf0`Sa-`O)J-yt7ro zjz5r4>HSuQs^epWPh*c8`ZiiaFOH{&?!6h+_+648y71c7IJ9*{1mM z4YsF$2Tu=aAL?pN48hoD!?Bn945``AgEQL>2o^?85B5ThLVK4StiJGkieU znjO_}>0?5pnnUgOyv4H1XW9E&=E0O?74)~)fy@y&3tm89k$-LDp(9Q`Jwq|m6N7CQ zPGj~&PK2{;kSAkb=*IqnN|4o6ronlB%`$rDl0Q zxdD16Jg97&(&1JyO`%wi2(F`_c<;H=%QAru~L`?5nL$-CkuuBm+wiTRiK29-oJ<0KGLb@Q? z3W5?!Ea(?^J^U8|lOK{4uCYiTHOFqI3|JwtLLP^WQN@Ap zW@@~Auy;OX1Wk(WzyzU5b(Z?eJq>W$!u^ov1acG_kCtL%pfz^9y9-v0h^R5$V(Va4aUC^!XMToFlHC^x-SwLmP8T(TYZe1H{n2mf36v27GK z7Rb|tTRwBwLKa`SX`}MB3-P-|Ht-s+r78<5BquN*kuml`#BZbvl!f(CkMnw^i%kF7 zKbyDHBV9sb4cvs-q3l9;=lqs`r+n;AxY99E62?xMUX%;o5}|BTX#%#Nkd2%9|u6%DEMT~ab zM1IrBE{7zXndBY~^2{{6)SkRQAG%hsN_mQCotUB==zdNOSBdZkCW(6@Ap+Xl{5l#$rTkx&$ReFbg zE7lF_0ySmUi5J0d^5%MiW3!0}aa$jSbE7bHXI$Ci=aa1^* zI_U}*WdcKXKXoT4Lg-McjeE8`0y&@vKslJ*83t{$FK~?$O~&W2_se+fec?fL06oWY z(bL{@N_EM(hR!25D?srTM!;@T9`O$m!?<-$6*Z9zqmIa0dB3~Hnl*3*zdth0Q6akx z&pNbHai2Ni-b`&$45#~`9M?^^nBNpx=UFDJQ#^BKA(`gcbO~~mC=#SNhYQ-3e}q)j z(ga-C$322BlO&?;;BIC+$z~w9Z)uNw4fzt>hR-Gjvq{rGcmerR;=IQsyABQUTn3$# zmXs+vDgMVrfEVI+ik{dMtcSM`nMG~4`M5L`PoP8_Xu0ZAd5tR{H@gl~HAF5r0bWTb zDEFCXKx}LTy>PI(sUICBn+fkgqM)0$ql&duns*0S1Rrp;a~)Hxf>+4gl3HOq#BJ9r zCOH=hdOE)#N95OD|ubAr0 z6)ePZ}5{1>y(S&#H(?XsXV-B~Y($?wv6 z&S7zttX+Mybgnakx5kYs+5nBn4R3GB65Ca0g`x%a#F-AZg7CawxYl`B`8oD)sSv#m z&SsvtrYZ-sW}}mWzDlX;G0`6l7e!MZ#aHqHQ7q^vi@=^hITG0M1<1fofw{zYRZ#RC ze{P;iy{EWnu0!uSOD}NUb?uPeQmnV}WE$UC@OeTtnrokDUnsvzeU&_e%j6F{BI2<8 zJ=&g*po{VL_8Z=nTyB0g-dwZ+t`+~maWf54I|MrqIG-aB)1b~PpTZ3HPIqmPRiFkJ z-x(`72;Q>aKv&7H0!uw(zrSe844)*hYvgOm$$*06p@kUPsycHQ#wg8r?6z)0YTZQ|P zQfCRBfcT}6%v1}5WD_IYP9|P1V|#9|qX${{?%(KVZ&SQTc1gvz%u&R-T2Pzm3e*j! zd&;3}%n?^NW#^J}NP=K5k}h$AO;m%FgQfq;&ahN}j^ntio9AR?Tl5mW#p!pRMEuBS zc&BRz@=X5KOS;Zc?SL%B_qXe zX`Z7QFkN!Mp{D*|xJZsXDPk$(vhfukuyCe~DuL#M;p`K3g8QC$61)ArDQOQPpqe^| z{3h4A{}O+e{mCGn`(Tvi33|RFnfHfNh)h*BH#b8@$sap1;U4fFw}V^_pC&V-i=}^3 zB-BVRRualOSs(P7N($MmQ^`!*F6S`RO(DQ8_dnuc*j8$|6SO3{rXq{!h4M=Ts)}(> zlpyw7#5Ro;IS!w69Vkn*Jy1ou_uJ1AJ+oUQN_T#@s0-Bd7w%l{15|x0y2&dYGbT8yl3CmVcGlfjTCAhD1De}H;GHe7++rG%# z+SbuM;Q{bqG|RIMSq#*6e5>Msd7OM8m$FJmNqW*3NCDWLsx9!@a#g1kh3H=5GTWf~ z5&r174lklFD{S~?dVs9(G(k`X#Zfu1(N_XakUoN^ z$`SIk^qK9W)a)E3*#}fwvf%`L2MRd3oRw>kB)r+B69JH5&UD#eN%ID$F~ zKe2M)3G`Pug`7#RhTGCR-G%C08Rfa}9!vf!E<@spjkZ~G50vD*MdU->=sDCxQCnai z??_1mO*ki5{x*53E1-kPmh3Oe1rroK;l1G6vaa9*5tnt5-9X>Snz)`y@w*I&=f1aa>mEvC+hg z(qXoCWn*P2ris!TEZaN^+lc**6qoiYU+8}6xdwl-^(@PEO;!RxqPYd-%}Ag4NFBW)ad8b2D*?Lu}Il3-%x6G3#A3TyBqB zU=X}uO>7sH`NBAIHbi;afLh63dK!`?MI|@ft&mmr|J~e)cwG5({mKyB`tS-wfCY<+5~h2(UbOzpP5^iMVjm6BwEQ%?s2M9 z;3bwlZUVm(AM);`zB`9=ZRoUuV{i`D)9%CjOWQgp9O@0fcF(70VI|HE*hz2^{U6Ny zA4O*w9<|nXVWhYfXsJ&}GSR)0Orf~z!Hc^d-2KJfT?(|+MluuIkty!(4hMHRxa+~c zeE<5Z?bT+o_p_e0?iJTe&rT$iE`C*C%x#jU%R3U6lm9BS$N_?@QJewFG+L&Y>o`7t!5?C3Jla4=1^+pu3VFrKuwlN8ulo_TWuaC-kpvx@qmg zeC3P}{A#wkFwpW%k8}<2EM(7tRNTeRV3)x{a#f;=dcbkQaa+A(OU}Pf9%fhJCC&TF z8Yx|CM;Gv1VtpvmHA)-F@Anmi1+|7`6=FM6z;?mSQj^RHo=;rfh63TPWB`{yHGen# zA1#~SX-kZqsgIR3vaxascLQZ*3u+nE!|wM_pnwSVpdE_Jp0N>zHJM(PJk2nMcPttKlviLl_|&e%?BIS5p(!Dxf9@a49ocDIH4za%c|#t`#^-)IqKV^H;R0G7zm{rXjhl^ljzK&F z)$`1)5bbHMR#tw4U!bsSxwJw4ktAsmR4%nsX9G{TZq$chSJcN|-l%RhvMnTkf__Rf z5)yZF(!nWif>qv>hSusid9}CA32+t_9?fD7WM?ETbb<{5a$d;#amML`F7 zXySM6Ave~k3a#YUo-qkFr6`XC3^k4-lxk)hVFGnf|IgA%lh;?2hsdeYSFxrxNgT|j zqNCCuYLNEMCGzc=H$i_O3m&7pBT9bgILnsh2ZHL#4o@@ryl^Lf2t_BwgSN&ySWLaG zeD~)GeT+GDG3%#rq79AKT3epuxl&ETO{Yl11LydTRGReDNc9xdzJp%NobVfF2A-2^ zV11*O661-`5!n-Ub^Eku#3=GOp9woLE#w#WpV*i3q?s2tzNH-;$P%0xYu!jUS`&|roRGAbEQA9Q5o*?E1baX3fgz` z>Ha_XRkaDIg=Ue1aYZhI?+-qM*MKuHlELLUp>Fbc!jzcI)$FDq;f`Z7)OcnML z?pV+?(8nw;FJ_ApyOz>&3*aO#P*JjG|4CnRHBr-*gHoEsuv=`Nb6qNNB3wT>Se+_H zaPNuTVhC%a)n?<9Pifaf4V1~wybalIO4%&uYcBy|HDx?!wi=mm2VGTPhqRlD#4cKzYec!NVy*58Tu?U(V+db2cy|!7rQ6XZWyQ-JgFhwpAjpzlgAnR*9BxD z6z6sacgl^Jw&+tT0rvdT(dqSC{f{=+7)FREEMYGIf|w;RZ%^BFf$SC zEiXFW5tH<8t|I(EtS-M&*pHHECywilmEXa)?qPDSu@)^>_ehOb?-u6iPI;MgD4pXU zVBaJUN|*em*2i;;7>z$te&u~pesB$up_T9qkye5FaW?-G?T04m?L5`tTF-UF8&A~} zdU?Rf^I3oP0J9fIV^uuwltxPm^Yh@6aBo@*lptRtAZ^yl@I9?GPBfmvZXhvGg=lE5 zkP|DFr>=Ngi%nEKG($-=No|VWFUaVV#XIOf`o4N!j^@g+2hdtnQCO*r_w+!&>_h1k zGR!~B?V?}egN7SSVy~ksLWI6KWt{t8i(}iAtHI5%o{V2cZBPsVvXS3_Pm~qdS)vH& zg$AIsU>hriJ`BtTBcuV^b?`*1$Vc#LM#s(f*A2m%3Juixc~{r&f{x^0N?&3;390wy z9kO$fx7t=(nZuC0*+t&w1#S+AW}@|FY^#))o{##%HSw?^d&gN<>FGO6)TXPWNUM=$ zxbJkxA5}|iqK(o{$PLjxp*K52O9m5Ds_1pxkEv4D^L0eq7@o#C(At#e;ht7rDvmE| zxk8X!URSfNaQmHnqj$rNIfOHC(w2sx$Fa|CF!V zZLSo1$v8?ip%VBlo(|?_u1;_W{nDIjG2z?kcY(0c<;^sQ=+fp~`4p_nHUKw)9qglO z34{5hP*2oe8;2fd+C6pRA6QM+QEVAUar|#<-u{~F$No-QVYHV9#GX=RIYOF`=E^?h zgeGAwAf`7_&NI*aU-Tlh45!Z~X@+tC~i6yJf^Cinx93dTx3 zu=h+%(iR8;4hWSqW-I=jTr$KPOI=Cf6WS5EjxYf~n7nemCD7TffO*!*@T>9-tL^No z)S7AHX|Z}js01P`4x}C4&aN))rT2rFyf(~U)PZiSs>#310@kxV1-7*5bUS>Umf&*% zARbhkCx(T;1Sd>%?~9M3{;{Qc$2hv6^Vv^>Co}`CC-K}YUzAvwNMW| zR*0@B=HkGtY_0+-ci)=Py9XL*hf$uHrW`|~gb%{|qQ@=$th;wD*-Ji|G(7Ca_Zunh z`RW{BF{Y+!0bfFQ=tUW8^@z|yJw^X2CYl9w3u}+DW*wsp{T1sRxX2A-SBYn&FiYm0 zu=D0@cC1iOz8c+HJH$ScH-QUg4>BT?b0t-&hMuW#wO1+%o5et7Ysx)3E?g9(g%;7D zK)&xLT?;#o{sFh>4iPcJKcXU{*igcMM;Y-ePO#=j(e#jx!UpHm}<`H)(l)8HD+dnGGeo-E5`4T zU;A5*gIxn@Y%NDVeTK&90>Vz|1y>P%70xRZ{X>h&8J%+i)QH+9EF4}fZAG9kT}$3Ub{0M)H%vN*^^KmaDewl} zkNcZ?lJ$e145zTFvXsFa5m5 zcDiHfQf}HJhM>ct5`1Z`Fxi#e=)c5#47Nb$xTc_A?)yy(lrrEBwI?`H`s98NhJ@E} zhqMw&1;rQHpR6gmozYgRsy^9p0_!8@P>sz}aG2SdzrePIwosA^H@yiZgs;e*WHlI6 ze!(DnLD;CP++C$B+uL#yUC&st>5)RCH|#5X1vglK^@5iE5w^O?r#kAVTqvK99aA1Ctza?n zAiQcspkX1}Zvb9-O84)VN&dY3wF#ljKiY^Ea6dmmtf}V!vW3^a~SR~c8=$d2W`mvt^ zOWC=K=uX8hS~SnviYHi(c~7OO+1l>pCg6naXdoKA2P4S3>>88|robB5Ky@Q{;QtM( zsYl2TR25&Oyn+cPkVGMF3O}7-+@;_b(P=gp4unbu?^_!7A$QMg**qFb;_C;S5?7@K z>LY&uUZL)SKJkC&)fIblCqn_S+5Ib_b7(Zk&U`9<(wy>DP4`5YyQyYctS37DIQyLV z9a>9jum&t8zw(x(u|P2*S0u5FKvDJ|wF3PI4q$Za=3j+>fQ|&G(1N=ya~_FswtAG? zkXeaNS5nEjFvhVe@v1AVRMmda*Yz~~CSFh-AU_5<(pS}wwU`@k&d**-Kcdt4+wu#o zzL3h;;Ft_A7A3?-E@iVA${VAOrHdMct)`Uo?wL$k;j*_ClT6%Glkp!~E;$N_luL5x zGl-{3L1iA8VHRZXMuWtu(sQn7r9E_8xQ1EcySFJ8cN+J_emSKynbzEb|E^gIeN%U7 z2X&}kbI%CBckZAcshwd@Z#`liYNegBS95)qt1z9=4t|C5L0u0c;SO#CQApfDb`Mpr z^vKpGd3nmgKxJm7bO!mgqec{Y)i#?h1Ydi*QPUX`J*WGVIWUqLCq~nx=ct~?7cf() zroPGiNm!K4RTrwW`7>%k&pPXd+TLn_tprXP95}`HPl?N)9`M@hiq|Aet;!#QfAd}S zs*d$aQizcHPdp-P^m!S^}|j=%+c z6=V@0Q)X~uf@}FH>>$Yri|4IHWwFD~MN|>A6Xf~2g?oc9)N8cX9iyLi3<`GvmP=9i zkuXmwiWlR)=ao`^(arHrfdy)PEr)upR8c>e`@-L-UwW*!Kbiy~Q{RznPIaFEFE*61 z6#35hX6i5eg_#R)!fDh-r3t#EOn=elk7nAwSP+RJ#bW>>o zn!(1V#{q^prh@n{#h%Ke(O#l9j#S|zDMuv-4A*cQwe;=ZSuirbH%LHIYN zQuqgm!rA0%)J1d+toI}_{n?9H2jw-@hb%)sr7q$Nqx8%j_(pA=MoQH~b$vFr5xk;| zU`lgisVrLc?=`Ra>YLGQT4Z-xATFB297FVIwjOnf`wetZ1O{mycHu88sj$1;&Qb{G zF`Lo1;JM`PLJLfTYs?wSr>sP6IXot>HFm+lW(=U2$x>lZEsHhR@I$G5t+-J)f3kUu zpGx*3cV$LPB@+LUZDast;n8`So&rpNIY+)nWf&ifiea2K19g>VzC8H_=xKFXHcyGt znB>K#N{)AqA(8nWHs)1*Cy~L{Q*N3&Dh-4KK|RfCGuNhj>KZTdMmPto91(y_1#_Kw z(m6w^CnvDJ{Jqjd=BfA>HdHo%pzOWL0{L3z-cgw zxvAeprOX`k9E|n3sN>`^>4#W2IS*Rhc#L7RWWv}4<`#QbPtmUB-PiBR1lp+|);j4y zJ_&dDHhG)nKhz^CRg;dOS#Srw2QL|Vo}HGmUrV#zs@}l=q#9NSNK$e&RNwAL9WC{1 zD>27DSv#jz@ysaD(VEX`i)`ek(>xbQLAisD$%BZ3#vOVNtSt>9&Vj;gjCDp*;Vr?n zRIU=H7vnqGYxv*1pLEd!*|=~CHW^mM+Dkpml6nWOnP&<6IW$ndL)VpQe2RV|)Y+VA zYr}rY&EYy*Dr5(Cr>l>?101k5iuRJP$)(Qk_%?qz&mSlu#k~iQv(is5iOSrF_%ZZM zwIugkbek=S?O-UZ$c&U(u>m-cZzG9N8TScGi8xPMB)geiFUeA=8zy_WJqOZYElcjkA(S@oK0PW}`=4?J<_YpYqiw%I@3 zosRMR|AHUv3Bnz-gSJDSDB1Z@>SuMH)ZcV@9y+he1K|{11K+7QzM14ijr`}7ae6*n zYVnoV8_U?W*0Z(@QqEGhSziTbVTPrLR}OY%YcZJe*|996ud%?rj!)2AIok+zw3^B?>>4_z=3u#~nRQ#8 zZ!FMGF|ElL5c=9{h19ZctG<+^VaPtgdip(ZW~d~U$?x&45ehKFjb<*4Kjs_lTY&wv zez%fxk&rc6YBiW=M0N5BRom7?S`MEu3VAe=Vw-t8xslM?j4(^5bn^`awegI=JQ(6G%cGflMqgozwF5$- z5NBgLRJtz5fs(T=yh=0*D zrUmnc5x`<;rC0In^N)dhq#AS&S37wf_ELG30r(i9Mz95b3FWdIu!sJ$!LqUk{w2Q8 zE@~eVoxz@^|7fG>ad8)=J+N0!f>KE4QAyn7G|&uOMlV6VkuhF?D}jm=TbVs>3k%@{-E={`pjSI^9T&Mn|xum|5KqzVUV)>Rxi{2i%f(Jjlh1`(i1&a%X2 z@^Q8=_r!TyjYO;18D<%y9sN=KrB*Nsu=V(c?j$MCa;@BxReuZ^W$wWlx|Y~ADI%$i zvOMRIWOHILmoY&K@gH>(Jkus2pE|+$9^3~t<=U0kxeBnQ&{gS|yE0aW+)oyOHfde* zTWXMWhg&AEHjXd{?Oox1;VEF~s~{B~4{gIn(z7{&$F&%C4b_@l4f~g=94ZzVA~Xhd zm~4o%d9G-XPIxo^Cx53|`jrIvqgFd^4fZspnk0&{TB7Ikr$B*?&8{?72u@6&)`^#IEoWSAyK;fY~nl2MK4sQ9np)b@Q-z_@?*H@MiY4S~}%uv|T zUh5wiu7BWb=Kc?_54Z7QivYIG9$R_5uJI43=4fSbd1M}-J+qV|<^%BA|CCIH>HK`| zp7z|aK>d|4HThBKmvxhSj-mp$&5C>;pPF)7`X&~KseD&yd@#>9LMfwdRgUK@<+JHY z0UusWTSLE2-tU`%R%$!Qvt}u2C*Bc{Mn8nqz-R56)J5wi9VI7t$6HO|jNFDU&fNy* zjVO8)`Y$YS_mo(%t=6pk7xy4f({O9?TC`ie!KUdg70vNEY|1-?H{uy)Hu=IV7T%$* zuRv)Hq!H#H$0E?dOacAT78Di)?zGj@-3hcrRj^rlSI|*k7(62vHUEU0@-FmWUK>;d zpRRRTTRr}HZr;a1m%YI#$x`(6g4usLxHU^3oN`lz0r{T?i1FY*=q zjcF3n!CO>>YO_^g?t;GV_F^qqRq|l-^|$Iu)+W_(tO&W~Mj6MYG|CgYsGYESKNZyr zniE!4ym6RaNf+W)dh(Pss+nU4QJ!yy^*}G>!r~O!W>HA%k~JVC=aQEtpd~maJNs%A zsK1P8Z%y|^Vk%sSikWrD4D{UHTBwe?%Zcu$`Xgq1`kl<;`mxmww99Nsc8=Nv?Eu5H za9|BK+IXH3-;{|C{fBl-f zm)KlgNwpQOGo!^*`e@Z5Ujr^Vs%O0AbV>6*(Q2Nc}|t8$&b^P5L>7-R5n~3 zS4aISG)tZr%+w-aG42`mHmRghC$E}5S3ZHI<4?80Xn~`FynwD741)2>YgF7_3@r;) z4eIn$d3N9ywbiy73t76GJ)*J7Tq!;#siinVS#9>sKcIau z)`?kEHzEmEXPNN7*hFWMpsp9iTYw?_clIRzTs@t&F5HNou9v4@Ynk8#KOHt=Lu?M3 zLDpff*{10a&?dzTp1}KJ64n@f50%LN01ESzzRf7@dCsT88u%P-R?bK5+S*3hMVVXX zBhT$Xd-Rknsr}K)!@h7%&Ud%zKjQwZCnI;rp&V0pSPsgXR6e&2eg~u+XO7icNnMHh z(sXe>SWk+ft0@aHLQB_kt@lzC)g8R?C(_lyFFwWTRUp`crjsyVX_qoM=n# z1o^I#9;;&wJe9uiSJ|?9Q&7-ys&!)fA_EB%O+<|JFa7JKXb~Q2n3T8IDm;TiLk`Vn#RwJ6Ug} zRa~*`IIV&fRnGbvb%0%^w!}{+$DF12ka4!E{(@TNE9@c2Ys5hsciT`Txh1^c@;Hi8 zBt!6dC~0F4?vK?HRLyi-4d}CAajvn_o-YkYnd{|~Tm|1fM{BAd9&c_$kCX{)ggFaL zO#dhTKkxk2}hj6S_2aEcf!5A2DfeeOdkp_l|B&LVoF;X#ku@H zRx9u_V*tvfT5C6wntD^{RQMq#LeeZxbw!#nM};lII+-q=1vXIJ;)3>Z>;{eT%+Nsc zxaCB2IL5*ZpBIO??g^N+iDt|bzk8$Q$x+XU--Y|wocKwxIovueCb=V9B3!{aNv%qa zKnE+{C(q?90-ZxA(1c(SRit?(=udL&H0?@9eF}d-+Oj5(ekztV@AJnJ8wd~0%V4Ut z&uFS=Bp1}mfn8iBYttJ?wxIUeb4b%Hn_pNSXu9)_+^?`x(tUA>cLS2?3G51Ty;c}F zP!;mL`wP8|*zXP-`&{$IBuxfa>}i2be3H^a3c4&6O;2&n;9G({UH+Rc1xBUynJ zc+OWMsj&Vr8Yi~FX5w-8D0Bt1B<~1W;2S-ZZV;Hj+!bq3Lxd(qvbNu1x<0n{l~cG> z>V|tE_e}H4pI~;*2X>VDGkBMOsS!CCJg8g)gYlVK6%B$))JJz|Jtoi&KJzBP5V=LY zo>v1wi-i}@8pI@VHgnZI%-z>j8qHm1}!prZ|qx2X2I{m z4X==MlYgS5xVxw!eis#|Lah_p>X@Iqk2RPQd{sIk9MGPr4fIO(WqPB?MaDhU$649d z(s|a|8cw&~nqQR@sFUd<9+)KlKc@{1%=VEhg6G6Z(gWC1Nwpk)ue7VdH&A0fg=>2r zVQFwOv?ZTJ!}v7yh&`7%Nmb=YaK$w!&Lnwdi?D&Nl;fZ?V&;lV%S>jktCCbqd z$oJq_zb1%qF*!^ahK@PnrG3iFoGG5`W&r*P2J&Qngwh$UTiS&A3=h-Aj5lzBW>fBx zl^hb-7W~VIA?owjnS$DG#%B37%PP~Y6JU*Ugp^_nxXZ&?F$eJ;++KeY{{S?=?*JaP z<<2hKLq3q}`6iktVC2RO`}3&spsDWyHxz#4p9WBHS<=rHmn{t|sErApV|-4*-AF#H z3@zQ$d`CW@+MxQO4IsL}Xi&GKHg zhMGq7=gQ~2FbDd&2WJQC1;3c8SulqZ=HgS?krn0<8QQzxaBVZV4i6+Zv^Mnh^h7k! zecITMfOavtF8*4%;X}Sev?Q;dwIRBt+>~mWXF)-HnDp6u5xtTNx$-$5bHcuf+wa@W z<`FNMd7uax!H+T~&_i7$SPe>e7Dy$DW!9!8jitCHTsmm1CI&|{-}on}t9lF!RJQYH z!G7r}agQ!Ul#)Jz{NOgbg39tWq}?FKIZ(?@WQhWCU!`&CWY&wisFlq~tPB_xI01?h zc6qS-A$d0Tmh8s26IF?i&+k@xE5-O+G?{!0 zY6z3H=-Bbri@mgz$KQ5b<3{LLb51(i(_gfO+AI*3Mb@?aCOuiXYn!Aka6XlWS;YGH z+(G&~HqE}6jE8ep@bFGlTeF?IHTkqZfe0yOq%U$^unmmSI_QP$D?yV$y!g@M2)Suo+oMoD0eYM>!+4{Zu;m2B(FZf%DW_t{?gul9(TK zeAYVY588wTWoyh&n1)8!gK{-vKX@dv*h8x^-IQY!kBQTeZ}nRv&KL>GDUn#cgs$QN zymIVEx7T?krBBiv>8?_pEk-S;XA809J7uks<=vJ(n%!zC)`euu)n1!LrD!K(HTy!c zfa`?28S+n2n+TcdSwhuW8($gFg!@<*c}`8BQq zJ2asuQzz;mm>Rmr)Id`)D=}lG%Pxb4fpoHj)?1yd{POp;DUz;-rHN`a^c%}U3E-xF z&-b_0@^xDI2eqd9Z~Bz+fIkAJ2dcQ|YOAz+Qb)6GXv*pb@?v3;|GM1D$j};^8x}9k zVRFlA-`zc=#mqFp>DdSR_|N4v4d?3r(}!z=gkjPnskh$*H9;rm1-b!E8FiFAqn~yq zdX3yHd!KuO+JKHTv&8+rT(CvjMeNcG3c--Y6rk=XeHDTJcYS>f6Jng{L_wCLo#-m1 zfC!C&@BGgLPno}EK{t$d)Gs2_JBF;G{Ht~2)?tt2xzawVVd#=FQ}sw41e=tt^JqVa z1r6!H=$%qOcBDJto1d{Zp|Z?r5a_#=IHkADdse|K{4V9L^N+nH)76?n zaxJHPoBV4bWaR;`QFplsQfEhRYK+xmIVoH?@g7@J-tFp)WYi8H#+FMDxT}h#bdNmA zyzK$>1N@}CCwxR&O8U9Vt2&{^?%|gD)&(`QQV%{VirE8_)f(u&w|Z`C(HS1B1mxmk zX;6TxC}q$C-5r$n@*T2$=9JB4_3Hkc;U{sFH%`C776)etHHY<&_SIsi3OmA=O#lYM zc~(NGYQj5DD!o7Hhue&QUTLGfaQHB61#eLGBoRi_9bgWLqXxVIH`45%B>R)zJ=%sd~67fr99845H!)p3-urIF|nUpe0-R>Q) zw*ZTk3vvyRSOv3kK{jnQXf5oe?Vf!pvphQk99diZPg~%hkiV(?r47AuDg-#QDL*)A zp0^|zEwA8g#&vTVOEJy$DP%deq$geX%V8>miM2ZU&Z|Mb6Ol`oV&ck*0wXue1c!oP!&U=ul0v73uB{!(tKCyhC| zs%tK_2Hy_q`D^CKa!Ov@=t*C$_Kz%#`g$k}Dz z3xC;j)9vP`+>g|0c{Oz@rc7Dm6s!p-!Q?qZzjM!InZv?Wov+PFB0D z+KK%^2ejc>0$&cBCvJC~v;N+%*e~A8+CTIVw8$J1DnU=B{v|)^iAoE0E_v6rjw}`G z2o|EVfyc^QDnd`zx_WO~XTec2Njyzg)?;nkm9khzjV23QT%Ba*9yqIXx3(_JTsdY# z?V=DDIwkc0kBK8v47OR@75pvTTq@4qUH(<&R9ko2WbI2p|e0rTQKD3!W&oF2<>_```5z$akqS85#bfiCvb}78Ep=UK-=&JD6jn*uW6yZa7j&mHzQv@L$lsx*BU< zV>jf*Gum&9<#0yp5PF5Gs8QgsK-H=Jp%lDD2YR(3NI#&r?O z3H1no@&vsud5oMEJ%fFq-hlt}j}i}>L+}l*EIyTL7ETI3R=4M71s9Wv)K%wiFaax# zXnKh7)%;_)n4@9)rdh%uilWzu3&K^@gBJBX+fpEk1)cs8dTVkhJP3QEZ^}v5r8YEP zC}+?D{VzwRafbMzCwh1LC!1T8$C{kI6}}BY?2x+9c~|zD4N+5l05vG{BAxF^k#~WK zL?LNg*=);ixJfJK$;3VCDLU32rL?Dti|t{%5EGfP{C?xjA#Q_DGm868yVQ+ypf-FTt3%PFK? zrFW5qL!UzhrM~Lgz%|NXWMaIw({hKE+4LW*1*gUz3U=A7nP;g@VjoQxivcX85Mx22 zG(Pke|3jBa?4{h+ZkeY;xc{7E3(2#~#ecMCt`~R+Ri*6mJ=-9DfU5~P10RHs_a9Ry z>IK6~fR&5(9#s~H=6S~G3S9!^O9|m}p_Qm5j8l7C?%uz&d}=GckW2`@NArC-l%*W8 zCGZ5bcT7{WBr5E0O1G0ADfRdoP!5bCV8vSDmM}uEZ#iC8Svm9bT0;~E46`&}ovmig zCldB}5t{d6Jnse6o(%0{8v!D9M; zDBvT(cH05Du>W^>t~ggH=XvScf_mYpRO2u*8VL)@nqD!~-Z~q1Z)z;BAUEdD0V}8l zaS!y$mP)%*YtJ@PtH?8@Mvmq)!}tk zXW3!$l-ZLmORR>wT@BpH=Bu2l_Pff(z<8+{8%qwhd+3r|egqPR;!_6cgm z7m<5exs52TK*~WiL#gWN3r7<@-K(NfPGTo8OSb(QI==`SPbS_UFY^8|4nTu61uawk9>ZX*Cx(1`e>DKlNakbDJZ6a;h7sC1MP@1X3%e;h zS;%C3@yEGsf%-nxQ=Oad&L<0mf7vss>(qJM{Lmd=O^-{t!S)cHRRG@-F5@QgrBt8Z zSSrpYqiDLU?l&J-J}b5s=V=2oGr(_iIr=KS548mI@X9g|U#r4;A8H^}mA=Wxf{j3Z zV*;ACc|Er-`wJMK5kYdnBg&{?15inikcww#(j7n$6u~O#<;7RhJ?gM~DcOgg#DpbE zvwM)et=KcG58oav4nqALOrw%Bv%_bk;=yj5?6KK@A)0e2%iT-q{u`&d{&-TX#B5)4 zhPoI{w9blBn{n!~qqVd}_c%*iD&c;my+t>+?xvRZwS-=mpGRMlyD$%>HI~Evl(d)J zE6)Uw=6rLsPK%+#@XtwIx!`Yi5*Mo-mUWY$v>L zLxshc#5s3qqCQzcv@0lbW;C^PxZeG zK1*yus=~m)25{0+S9{0RGH059g_l}|eB0!Sask1UmkqCT_k{LJb7wp0TcCwmE7(?P z=yP1vRZiRlK)$%s5(P~L5$SyKVyEdYZ#wztyXxC;+Zwp_b zG>`+XGNat(g*)!6bOFAVR%G+vp~uK?=d9!~CwbldolH!`n&k#1!Nv8{L{94mI^)~GA+-)euW$sEGz&J1TQC_&lvO@qN5?DoYfbfJE zyRod4Nei$Ysux&hlTobuRL)BN4in`Z?OdSZVo^QDT<6~d&v^^cAJw<|g5W>wm=u@U z2CX)Y$};uST@%g#=fdYWo3e+$PKrbab{YPK=Mg}qd7Yfg;VHBbv?kX`yg>XnuwTE7 zRtH;B_c2`_DHjz=I}6rI3Ef|H8f9NaC!X(*p`*9k*K15I=9?C+Z>2SC_AY!@JL2ud4b+f zy-VC;<{*aZmnnjtxRw4OXYl=a4vt|Y`&M^vr8JvrXlgNy^=`Gz(5_P&D9%4tDmVd$ z%L3ZyNHa3r_edG72pVJ_e26z!y3z-e%ZZcBcI@bkiYS2}3HDoeqBBZOm#Nmp2cj#o zKj~ApHzQZB5*#e;7M_8w_P2bRXyvq(7om#6Ad3Jpfr?kK@O-LfWRFet$hKhm>PVux zc8n>;FCv%IP8PQ`lO{qEox`~C=V%RGNX?-$$%BEKzHzSmeB*K$+Cg>U$Fjq+4;g1t z?&lYFOyU7u5G!sLu)cc%cTVoAoGq&<`84;_Sf^A&ov0G*eQ~PiwD44DgxqXXDp7N0 zEsNh`PtX>65`w#>ZXkr(7!Xwk)j?z^4h&ZFLiM;C;51n=e}1Te*;&h^MX6rkq&897 z6`bMeCW{uA{^O;i_^89BEb%gYsb8EUZgG( z<-i?rjv8YAh)De*&cJOf1)hxbB zNBV{*mTAnePG0G(RhC!4lI%9VtfPI#R%*4@BuQZ^Y}i2VWmnUM^kO2XJ)^EscIQ15 zEAF=5u2+Jc&~BxUvxU0|zrkDF(i#`wUN|wbm;1iWEzKukaJYAc%;1s1D_$NM+nr*#Jz04&37nte2sP884vv1)N;y9T`d?PZIM~)G} z;r2*=eDWjfld{A%gct>KLk@O=QWNVYh(ru{>YmC>qkpreaz(3CY8)H}T4{}i&uAoH zc5`JHq8}1={*oLAn&ahwH*1#Gkn2ljaOp;bvuEDWl&eOElppCXjZ7$1-g(2y5n!m1>2`pRJn>JcmW#5Q;#gK!F6z2)NQku9IQzIq$mto2=a^|Ohf^2d;dwzZ z_AfG@`W;wHwG!gXY`4e@Pnb7CEZCV}Pex=0{)(CdUK4kPNpe@-K!-xb(FW>2M~vdI zCkN)jYzEU-QaMC)io_gZ{|FUyTv=*$Uj0H1$#L{{Z=zh-6M^cZmR!0zQOuDZhD#bh zs1fGX&7D=vaTA>8ALCy9Yp^^T6gVj_SNEYnxSd?4(hj1a)ZaM;`sA?Q-Q)8fbUafV z#wGK|LIKOY^;;BN89>t{#uU5Qi{!q+ZiQZ&KAlVMtNo!%s*mxv;#4Hi#j!L^0;~B} z!Rq1^GY(Dx&6%w!eS^4qD4dnpU%!PPSM#)Q%zoq*svG^e3uZy%pgfYA86Lz3%vWwZ z_($rSUz0s#YZZt#rb*4fTif!4B=H&$h3@Dx69nHL~QUsZ5^tk2aTqo-JsfTE)M`{!lzj9St9&E)!YIX!TQYpWL6GL`Q`B z$!C@3;e0;_qS3`r4;DY~J_03{YiJ^h07PC80g=sHL*0c4g)FRt>3aa1G4w5(Q{B4wnmC*8}wM-ACiug=( zd)lkk#tVJZt8uKhhpEhb*4`6aSTg#lZXr#Rng zbEwC31kp*;jl%8%XonO+4fG;Z6x>R^C5D;@F)dCK%X28VrD-Q7sQyv$}XoC>F~?e-u|5t0(6W-^dY( z6RQBC?0`p$$M)#%4?U(!cyl`=e7gPd$> z4;`E(*G2|6B-G`$^CkFe%niQ0x!JqN%C^+j&n&3`_hFiGfcx%xjHQ7a;I3H5-a`}A zrgQ=KdhL6lc1jPD_jL!4P&c+Jd!1?ttwCM-pS(_O?{l+@=xIt#aUdM4R+GPFt+y}L zyI5(q$!7iF6?Ou=k^P4o0ym*rp3}gv_L>C9L(}Oto@CD&{ULWoC=c66sn*>lb?rA< z44w>@_A}v+)F0M=bLfWD33rSnh1Pio5^>Zp zYCE^}y7?a{RhDKp&mu`ag!wDlm#%!fhMA77rU7W4RFmB#j}xxjt*Ha?2NP=xo!Guv0Z?5Z zgw($#YI9IXP2S#0?jRQC>Tr+X?chz=)v+JDqkacdlmk?Pc1NuXn)0RnW!S{1=brl1 z1-vNjRi5&d=?TUTV!Ha2t1rmeS6~FKn~VK_yTkcqtzuk(_0v_&nx&m)o3VtlU)@7B zPn@C700)zsfiJOpxff(RX^wjdF^+tL{m>?H!tNIHn2H)^Qy?QH$}-)!W`QW>{tvHP184Q@_+q<9S#xBhds=oc3Uo zl)(n3&Q5Md_Qf9(?v#$6TY=~5M*V}@QaFT&o1KFxT1&At%Tgo2A#@uj)vz(kV{Yh4 zt`U23t?_pCAc&F2>(b6y^Fj+9C5rtxL9kD8{_fz4rnJ@@OTbQm!^*U9@2H`I_7POl(vhT$&IeH_9vkpc0_4RWGTCphH+;y zefnTyG~HF$FV4iCsH?WrwlKah`v`voD|1e_moMzSPBmuNZY~+uiD8Lu*lgj6Im+0r zuHu};_ME@O1Zj0}dPE60SWARFHkk7&TfFxr#`qyMR(_`5v^RQ}0@-8fbIHllC)m@t z9Ws?G;ot2W@BqG&kU8Do*7y?)rdp#U&@f+gK0%eY5E#ywrCcCax))ey+-;ut%j$H~aIL@&fKu_vc1wYflvzKzhf4NSePWCA3a41$Ojr&-Ut-&-z`@b~f z7kffvGvl$sgP-0Wm^Nn?xwx<4`ZUakIY>rRV}K< za-j^2SdO@T>%*V7`iCROvJur~Sw!*rrtFq(w@Zs7I-X<=t@))LP^q^@>(o?FA|_!-7qzr`l&)z$&pb zu{z*8W3-7-pW&4p^p zbtzuB0bf(a=t*)T>kPL={7JO|MTo1Wrmy!Er7Q6vt{kYs{mV98(bfI$)K*|SxrDxs9i$aHRb6jnMgQj57v5rC&=TMUcVa*d=$`VfC9oy%_){1wZtU#F<;I(jc-L14Yk~S52;5-gg==2%3`Y&7iE3(`QaHQ z1Fx4@m90S@k}rUM!Di?=j)fQS6i-%I7Owe}^md>c80DE3^<9~1Ak7c<2C>#&lCqs< zG5eUfUT==yCV^5>8fYx4^pS|9J8EqTjk3!*lX4IHr#LS$t^IDbplWSxx_T?_m$94K zMMB3ruq|OS@(-i$y4=M$g-$>m_!BxR3w_IpQ|Na)38nil=oMjJ=z;2^mU!9+eg>}! zoAf=N7DPF1LCi3Aw%X5ZsWer}=_%TJ$}Pr#61JOPB0VEiCEFcuuO}UPq!uP-hCXwI ze9T{oA_EWJj1Mdbec7h ztg4Y(<;otWfm()-QDV6L_*X66)x{oetU+DAsjvzu=@E&Wh$@v9gA9C_nPk)_d4Ff{ z1xtVrZ~F#Hdsl4!l=F>UV1l z+GZr!SF=ZY$HrjC6au3tle$B9hwn1_;E&;0*8!{yN}jyd zy9P6ZA%8^ncczt6IQM|U(L3=5a`~j6W)qO4wQvQg>D**HLv2T{Rwl9CjVIb+Uz}bI z)FFox7r3fI2k&bByZff#3vZ+IJ;%xG_-M(d=P7`qoww*8RR-hDi9Cwc-iK=8xpKTY zWV=e2(yPORbYc2^tQzW#x|HryLH~igMV0e~W1$yFH+_gGW-)Y%>cc0f&5B-Vn-UhFgSNa7#!K&>z zW_F4x%kZ(2qVIxA+AjJld`B(CpcS@vC~wV*{P9GRB4%lAnyzWb@ESr7`!()5rbM&?7(Jg z1KElAN=c;Nm|i(SUGFx)9(}>fE8@kKH4+$-135Jo}NBm{P#{)Xph~=@XGNz${GrtqwP)|LYQSd-*~OiO5H?IvGoJbjpYruBFC%Ae+{o!wRQf_p|ud^upL zr=gd60v_SMXQq>*5^!m)ywvcqlL(hG zHMH3KugeRkV0Q!>Z%k*hpZIe|r1YNAJW+hjoE1j4u#@(4ek>k*@hrrSiKp^j6J6wL z>Nk`xuPxQjpbL6ZtvU2v)w-#}?C-=Fz7`@~^r1%UZ@CBXw!Vq0O&{gE7VAUiDyY+k z#Y6L{qWFG~Pu~wGgB|K7eIPm143SOf|F(9fo`b>6TC)KAPbx3Dj5ACmU)gRVC}e#t zk%=?Au`h^(xSn;mONQ@f9rv~1i;$0*QO^C@?`VQ=p2!Nvr7=m-wzzF1$S+qC zw8s_eKFwmAgEQEwR%v^MdKNw9^rViUFXh)yGbh?MoP^#VTo`n#>A) zt-bn8e%`10y-BA+EMaFt>P5zGcO587}u zUFYOV=14Ojbk;7Z%dMTkh2#yYw^h-|SN5~Jl__ZFeN8OBX)9icswpMmbFqO~I(Cms z;YRY%rLo7Q%SK6Vk8w>uiFQMCsW0mAn62P|*}-hBb~XNSwl(AIG~ z$Y-^HpOzO7n)_}uZBb6 z-zBeb7ADVB$O4!+&g#S|)BxXM-&UitTwpzP49|Fzox%L10iDF=QqMWJVHg{jPiBfd zGl>uHHS>)JVjBKm#w_|^##7|se(fo#jRFe-Wuywi602)^I-M3@SKJotstsX-?970w zp4Fc7lhA~c?y0H$X;$URg>T39G%CT3?)%(-R5y$Qn_ZF4BI*UCZX`O2^H;@7!6G2b ztc}Hyi$PjWVP=C+)ja7xq@*Q%vPaOWXHLx0U~bMX?k(zHTVfWa*9F^nkUS{fVeYAO zLofNwv`=a+bU@zz!sH=rp6ygK?M2)(xJtau&5-Nhes~-!&K?RT7%KNri~`3Dnje{< z$DP+}7P#}<;Jmty3)#Z1=ZXtIO|#*vP<|+t`-AR**Giq2n`%e-FA5pHD?G*552s^2!hdbuCEd`W z_fL@FnUm0WdmST-+emH^_LASYo57M?Y*dMGywDjwGtP@U)I|=_EbJeEGNOfz+X=-% zhEz?=HP%J9R=TLwwRky}YmP6lig2gRt42}ZhG1{yb0CraCSUbk3hoFch3f`l6cr9+ zFPlZ6Kr~G`mB<+@0%MiQmRtXB_X2uoCh*v|se@n__meLHih|;Jy}r+@Qf=`nnkQuk6QDsnpS{chWH*wat>uTrDWDir6o`ItRx5E zCu8nmuRwKW4HYk^>Lu-ZMlbDyQrY#JRg!B0kBJTrqX)nn+!M}9;^{g3Q+O^JN=YgG zwJz*rs|wPhe&}lf31{$o$$Au-cPO-g>CO^dt~Q9fD@_qcvn|OOY{u5M{5-0R(#3EZ zBe^r0Wb{tB%;o#`DWenrUiTd@?D>m@>`Vp_JJ|BZNos*jxi^7*6wucRw~54ZZq`~(@{oJ%~jgGqOj3YVCDLAgcO#$(O*uAPqK zfqhJ4bM^Lj9+vvY3GumEr550Q=yLR4u+fuF zHxxLmF;$i9z#j>vDkIb>*>62%Kn29FIi~h8Y6$6S7+V(W(NbY$8XAk?Z~2SK!B~`c zQ1F&~3C?8Q(Ce9@>~(4f3R56uE~=Q$oSA zdP2l47UepW!c<9p5jZ4IL!AxfjeefS_z3B$v@h!!yFq!hhHTy{LF*>r2D2W1S%Iv54z4VT%XU zb529#yyr;|*$m!I|I%|3D>2E6%UqyGIv=1dU_;kZX|g*i>8@N4zqM_;d68tyx-U+3In?j~b{X=%)n_jg zk#H9k59V-c-c7r?%R^?~?~eIT;&#b7dXM_g)u zbhH-wYHZ4Kl&&c5?P-6{H<)b1{F$5bhE?#_9b`p_H~prL+nBNsStzS z#2<`CsFOgD6O+fVPngHp5`Gw(^QrP%Xk0zU-*lWZfENK8jgnv&HCx;W)-zC%W7Y>M zssoHv-)qkv<3{2z`(;YNH%3{`ypij<*6SoZYz>f(8x-*+ObFMJoB9BIo_m6BopHiq zp!yx*!UCo|w3cJdq<@unfxoC}#21BNI*G5iq2j804#;7lqiI zO|*dT8SXcmhgt5Xxr6G<_jgWLJBNCwHy7F{M`OzhWj&qADOjE~RjJQEgSUiU!BgZs z>wsC+N(JM@UPcBVCR0>g0C*wnh2EH1tc>>DcP9%=t^UGX7_jM1Tf z-SX4CD{VEr+-bdI;$QL&{m?dsye7x8HghW!%zC8F(BEm(xIbO*gd>46*lI`#0e!mb zZ#KxyVjpwegUP};tS9qcuFlY)30R&m6_ls?$TiGon}*Y4?Y`3g;H-GR>CGFIaaQW) z6$u?QV6JDbgr{JD$hFS$CzU`_2E8seQ-}u_9mi1HP_{HEVJxU`9!u`Rpxhf@FZP_h zOk8hw7YwBa*BsJjJ+32GUV5cp4L48z8O*_wXbevX^&-zeUtE&WCFh)LrL@dmh}?rI z#%k26w^uP;ezvq#gt-(-ke?bez%_S=Dg<_p8feXuQ7ex8yV030Ye!@jH-1(5gUOS3 ztD~6&B^+4F*?KvtJvR|!t)u#N@^HA1x~B4bX`*=qYfE%Ql$9Ew$@*o)C`wUugNpF}o1JS#!9_|9_e`yfYq)+eIUZX;e9Wtenw8VGZBiB?B2QOr zh(cROv*sm_gWuRpv#^-V9cBc)Hu(+~6Ivy2R|-e%)uxZ zQYyKla+mwpoNZLas=!@XabuoT#y^#tWbCH;dgd|3Ii9$TvikYH5e3-U4speSOObX5#0;#UbkYXVv{D7_w?<`6q5qfAgspEiCjTQ71V!nwDIeZhf;_*$ z{dO(varAI*9#0r^a*q&^ZgJy5gkHPr`^fB*mM)}Zsy(~tmm2z?skGPLIR9om=tMNo#=_A1?Ov)KJ zee=PTtJ+cWY3Od!Cuhslt*|B02&r&o{b%ey`2S#gR(FpI(OS2>Or;SW*xm>c;Oc^1 z!jJ?9{ERQgIHD@bE_=Cy%u0}{_Tz@Zqvp4~(PW~z&9lj@z%|kbQrCnU_GhDN@;J6N z_gFb4yu|`yCS5`s411ZK%qhlp%H<9OzWQpzAiX*?);l`76`EL1z|mMWNgiJ~SHB>*M&O=+S zwdy=E;Mj)x4FA>hVFxgl@4<5xbnE11>WTUU-{JmbHLxeJCqXW{sWkQ6cBYekxbgZ7 zK-#D=FZ@bF?Q}v^_-~Xcc*->4G47`1P*RO{SRUJ0X@l|&2}Dihr*IQ4hkMjBNF8#z z2g2$pC8)8~Lwl7mhvdma%6|6~GhMG8ss?u9BiTRM*7OrF2@#bSVhP6b&{-*!{!eej zS&kXrM~FMHQ|xN(=FZ!b@uwW(Rmt-#OFQL?7`L~#l8W7a0C+@g);!U@uBdTHd(Ec6f|4$#ff>aA5$E;QX9epfzsL!;UBpK*(tjh ztc+D(cZ+=J?g@EUr484$(?St%U0B{~sWc)U8aK5nXmd8hZbxp_Yf4+_W!l>WuSpmg zW~Nxlx{c1YkJMDPJevXv{5PqJ?0?EvaCEejN~rCm+!1OHT4IIdEgy;-)XeG9)P zI!W6j_U0w1Vf@c+Q>A=*qtGU)Do8huvv-pY8}4vT<3o}Q#;WXn$K##Hw^;|)EQXl8GKLZGDYtNO$Ji%Jk4tNC14We)KdTtwd`Gt3w6cyDv$ zE2>Y-ljch`%>C*lcr*H1@>lm4@|I_O{9!O7tSS@WC2~k;XXaCKpxM-D6Ccap3ausE zi|6E>Fhcf&V$AxuVR@PGwG#%D`RXyV#E@;VG0Zwo8@dP8i);$|VZ*qc(pCi#ft>dl5JY zW@#_Q`!ERKvHdHR*Gr)eofKs=C~loq#^O```@#jRL=9*Ak{yU>V>=nC?9vA-y+C(z zP-qYvW6w738|U%OmINZx8wMsS&y7*)0K2Nu#Hb@~$f<9Bq$aD^z$bGd+1hLYH^PO? zt_+X+jZr()kE#PIDP_WC!3;+SeQjWby93qGTtFFS2)6aih9=zu5%@XbnMXmIV$GZ; zc3qc1z0qD6FFd+l&*~h023~n`*m{Ia&NmJyE93THSL{sLiEi8DtQYD#eaJS7neXXf zj0_i%lbCV{kh+oT!dHoZlwXO;Vyh8(fmM|!fqz5o@#*11*(E@C^C=m_ZD$)IercRm z5{@Q+anlfoZ4`G=tP!IU+e9;SddgL<8?_vKMr@h!aGKdlndkj2`8L+mRY^=h99gg3 zI#9{m+wdzYce?5qt&Fh-6m!%kM>`jIdSY`F3VATwNcWiLf~4Hk`*S?##xLZDrVM}u zu2`WP`vyDft_g||+6Am@jI2M`& z6VTSf4QeU!9oG?_vwniBIYrY3V*gWj5am#Md9r;k)EFjlQS336w=Qt0_)FoPwG+gn zw9K@uG{S>lkZ&1tgo@g7I*m<}SIB@n04~7kmHsxGS{U+S{7*zPgy&=q#eBn|T`nVq z9bn7~`h-gCe)}ol8 zgX5TfW_`MH%2e#7#;d)BqZ-fG0;c7U#^?xNz_G@cB-Wxo_{aD!fIGnx+6}?4ZXg?b zFe^XzhjhqjuVHyrxmwQOjMPMs^8~c>EK^If^AT%#i06#+nVhdr6Dyl(nwI+A39wRn zF1HL^CaQ3>Ti}((k zU{{%4@(!>k_Yb+US<$?X^nAgyBDFGTifky!Eaf6RxFQH`)negqNbBOk zNydXuW?Bicbg6J3c7yi@s4wk9go?2?pu6$YjJd=oF5q7swLv=1?V}WM)a(J~DipCn zJwxtRAMs_GxB9>Ce(XkSfSKSur7~Q%><(s8?2oNSO^cmw2;K<$ZkFk-1ezHSwPt!N z@V7pm8DN&zHmj)#=cH`jhp0=Jtz89?NcEqe^ogB8riSY99`Xct1aAe;p!DiKW264w z7$=l~6~T0!<#f;soqLTk_raw?dZ=vtccZeW5naN6GdYqZU4>oE%zL3tcrK`o_%-wL zYs4UqkY}oNmo)<=utDLXKgk?rKAdD7?^i@=McTuB~@KgMYKajl^`~%lz z?_x^ff5xn{TEvNNUe`z;cZ97EHWCI*g-`hiu+*+imI@X%(zwyCO6)~QI)Bd}Ddm$5 zq-NH{xE*3MZW9r-ic((tb6CIbItt6k%Tlzqmgj58^NQXYzcu zj~Ec<+eN_{N|f(2&nYf=)>v&`ln!&;j-tFq_tu9Aixl9P4Nim2@e`QEq5Gsy+3%^x zz9;u6w-LSSR>EMzMf{c6&-mSt%@W*X>a`VC`oRzv2Dh+^V%en6Tvv92Sk=bKHmKtc zs&(LUr89dcbRYRP8TSxRdu_`0jne#>CFB~bwl=|cF|L5GK#|6)Ei9aF*Gy`_o%3ue z_R(x77(_mrF`Y^?I+~3#Q^7p$7nkH~A{7-%Yc13TdL#0fk*)volvhV;E$J<43*#R+ zRiohpz054cx#&>Lr+=zA|2_$N|#^#qp2Eas{Ll3lMop)&E# zd`6&`aVIbxmb3=rCFn6=yzeL83B!aqcwb(tY=q^s&)ZFSlt0e?jT+f`uy6ZWt}^Er zkF&_FqpTB#AucX@H?Ob352Et~hgz6CZ5)6f^cVQ>a2nUm?!sMl{m1QBm%~<|L`E+i z6jb3~z$(0l{wWVe{Eu$@ZF-4rNPm+qyo%M&nPe|E`upZ-G7}-THVphBV$>wVwajB~ zf_Vuw3_L)I(`%09R2k9Fbb^bMTLo&EU0|B^bAt@+Xh}?0r^ByY6_CBhG5u-MuYiaA~!f+u!6EBtmLb+gqc+p%# zEw*z67gd}Pd^7FT0?3xZaKTWUQwd9J__Ywr^#Fq7GhEQ*HP zNU=6%3+b-kupE5?NHnj3HqIvaTh#p2OT7pq;Zf_HE7Gh0(QIrVQ9n|V>P)f)7{crm zRjsTr+W(%7Q$A@F-a*@}pEVs~Hr^F%rbN4~Z;}0oj}j&ukzT(NOWsm=?0315%LCtU z!%!Re<50HH&(|^j3R}wGnyn3**r%)o48@$63D*{QnXHv?KRg&Oa7}ZKj)5m`5Blox%{X7rcyrh!QBvt^F{H&VW}!w(no|QjvY^WKWuYOz^Os(~82z*plSG zwT7IeeOERa9x7x{K>BEBbBxr=_&^*G=B;{SR?t%c9ki$u#6s@2wi9KjMf)<_A3pMb zbu(Pi4b_8<*=Ej2^9o|zr>p;g>xe4z7Z^ZsLP(vQP+I@S-&FIIm5x8w9-+!vCEdML zj4tM3LJzDK7D|$#{`maR17EJ(TA7}k;Zd~vhz&O0T?ou$i;}Y~Is4zhAx~pCC3h6n zpE)5h>@Qm3H&Ib>h>e4KIIin;q!M%%lff?58-zoi`rAJn@5tv=M~QP@(YuGPIEFB5 zrRvl=Ab53rGW$d7hg!$>InK}z62f!|_g0W1{_f$pE3U8jboC4AW6Sz>i>;AbG!8y> z>T**#?3%y~bNhpD)S_f6*e~>=3x^&i-XIq^j?y8y8+X;?MNV29Y$oTT?O{vTXtT7i zf_AWIXQH*m#_0$MZhw~;SehK|Xh?SoepbqAE7S(=O>8d8G^7e$*aI~0iIqO9$#M;_ z5>(mhhRwk=1egY{i-B8_0+M5~0u*np?wY!D-O5aVdg|`q9qXt=%>k-k# z>|5@pewTa;`{MDgcA-plXz>@cv~`a-I;k=l5rLlnbp#^Kc>` zZ?tgkk;cVekR*B_`8xC)Jj$wGbcSfF)j2|JYU}W(bX&QuRp#Eh5~=d$67mW*I{&A2 z-D|n~Yj>z7>?BQL&XehIi=pJmW@pgINBeiX>k&2WALdBA6V)8?%&N&r#`o;6bU9pz z>B%m_XXtL@mfB-S6EV_shPH|IAlm=wmGe(3h2ly%?@(^zs67*uA=l?LS6y@qxHSC1 zJY$X(Di9^%6}ucTxUBrah{;;Z=%L%{A;kDtDD<)p<13iq;xh{Mf|^qHyqI!m?{X`! z(3!kQ#Q2W(8mtf8Z8f8+Sq(h9*c@;?R89U&m!!IE zX--@tqNRsab-D|C!Fo}xxsnLG!CLIY*x^!lJghzg547<_yhP`=R~J(S*m^KO5wmK# z%O)3qQN|A4DU<+S>iW+0sL^K&`J0flU6Y0y zZa4rgW2T2sfLTl^rD^OT&+_Wc1P@Vw) zEI&bRW?f6Y9CBe%>KaF*{0eNMyhMBEY`vpIV0z9EF@~s3O%qDO)Xf=KdderdobiYF z1uG?Il5vP>zn_6*3u*~!RN81>2$A4z_8R&LY)>YGcJve1Cw({Ezx{t&*MtGy;$j2( zs1#>#8*d8ncn7H-C?<3#=3u~J(Zc} zPd*B+K&hu%e3EUc3HjAQRhx%J!8uP8h=bNvrf(9ibJdMf<`gv-`#|f=Lp2yEAnS3@ za{j`el<5vqO)rC4AN?=gL2-xIOBL1HGM|lMuq^i{;ni2>@MaOMF<;eFQn{xVsEw7& z+X-Q^yp*o$O_s{S@$wBcg={dR`BFkxwzj-VxoK=;>smMHGW1X_3h$~+30w^|L2e4v z!e^9Imsn-Y_v9vJsnJMo$t>WS>jNPL=2=JR(_~ksD3PgjQuo17(husQm|(sjYQaL@@rZ}$X7%>S5LeO zF}j@gDt%kEr(IlR3kAERbQ;n#TrkrcbUg@TIo(Zv0Blqv_aNI zXL(7af0MUx<;j+TFYt9{3$!?YUJNh~!$Tj|d%mn3&)EC6NziW-O*?|?H zW?na`XYvyIkah^$EBDhYsmFqsnS(M zQ@#A;BC)4l6_kKGDcdX;tN&UiN3@4cDPU@v;%PpN;|t}8n8OCRgZPW1pgNiL%fl;;75?F=}54x zRUIj3Z?OJ&O!j;-?5d>X%ir{JLOOs>sQ;S*MUvm!&9Q&`z*#{VPK z4pZUVR)P7!6N~z6U)VRzBcLm}hwN*`sa4SLs-7nRXvZ&mC#ey;)ZOX>^8qd*9hB8- zWlv)}!P@eya2dyN=idbCcaZ)?P3isdlc+y^Q+`*WGMmb+OiuFt&eb&b+U42bIV$Nh zydb7lJ`{+i&ZE6?8D^+CLFgk7ls^ibA#uk+KDsk~hVwk_K$Y;T;0yQJoYw~F@jI;W zWBNVykfgos&?$9_nFBZE56;d7G`EkPCtR`*z<*#Wby-D4MqGhWS9_{_CVxgO$QUS2 zV2=ya5^7k5wEM1=Tmt>TnD2OlQovi1r}N3!MYd;bMf;;Yo?oW=QqFT1loDV(^5%9) zN^zVwtNG3vfBFVet9*&f7*n>JZC?g-V(i+dh@Vo2_ z@B*~}K8iI#Z=tJDNV}5q%Rhv@XHT_AX)8)FMDp!J;c$0H9=X?^y6v{w7qXs%=0UUz zZ7V&~HkiLDhJo5|g}yN$`K0}wxuEi(7jZ-R5tb5W*^i7mzBt4x$yUF!7ufw&W?~mt zHOO=OwBz`htdYz(6uVa?dMM(<&jQd9`wWbS` z14ZT>Gv4UPVZu~IW$4QkVmPM0ep>9u2fTa9UhG7@lW-5LW)#na7(8x8{#E5QIgofQ z{7nxu^3|FCiF{6a9kQQ(!&iwf!?uyTGLpgz>p6t|)04I9iHY=gFY6P7S3-|N)0M;g z1+tAHkUnE6Qo#n$-I;jDb>kqs8@esk*CRp)m6viHQ`1or{q{Ye2hfxHqnzk|Gw8d{ z4w9$qlG#!DS6qVs5!@I$DKw<{)Vruxs4Bh^BCi45RK7JwD@T-#>`bl*J5*jz);0{@ z0S|EV$-4Liz6X0%CxR=@mFT%;=UxihxF)`*&d`pOQm8BM1)>pL${w06hE%R>g{dGq zIL2|<-YBM%$wp86MS5xDc=)NdPT55+BVvdv`d0BxXqYh^OqGj^zeqn-7WjqpvdyNG zlbEU47-l1=WDE^`*C$n-$gGi?aJaGo6b{}M?zviM>7k3E)b)4Y^JAD)ao8=?nhmz~VfP$<97X79xZQ)?RaA<{-8PY6$vG zZE5dJ`b)a3tmC=`&vQBM(YaOGN5m?mFf0SlwOn9Zx2PF8qsG)bNUYk(5lhnlbGLEDex+u4=WFh40B;HHu06SZsW#Sq zw1?;!%BsA>*T5&o7Grnno28lVQxYf4Q>%mb$?McE%6a%EWgycmG#CcNGTIx|x?bA; zmfTOSkLJ!fo-*o8rMTP;_`xY(6xKUuKkDGB3rez`xffz7FAHa?=YekhBi5!SF{B=_ zyLf+vBGB}I#Dngy=J&u*wG7eGL$g!eWnfRHgrm2!ksT+VXQ$a4Ln@fS|xd=gtR~P-?`zG5xvw?ITY5;%R@nnf-NZ}V9})H zyj=OYO!Ed<0ah6!edGD};8A=pU!>G9a>0PtH6faT zE>tg9fe})Ip~-47jp0UC3%pt>6iCx1T!Rd?*y)I@Q?f?5<7wz*)U;3q{4i+7eKkvyU-LT=YbDBEBs`m&%GM&?X6w)eu+VIT&ehshCkdCfDO0tv z?)z+nb~yBseNHqNzrjg99F&kYne!=N4fo7Z{`F^~K84lHT`*nwor!{fpt*ECC@+kp zn!!1~GfI#<$#-YoXjQ!fgc4Del_6jqQbP~Iqe`~|T$rowvp>eif{M6@iPxgZhpGYV zsTk`8Z}6JF=lFf+QKKE7rca@^6`SOZu_qER^fhiA-9V6pCnRsg>7O>2VkfFQtYQfZ z()uf$bb+}UIKTdw@DQ=fp1`n{414;_#0Vx8Z_5R-#l{fS2s}AV37gy%0*yi0P=SUz zCn#B(s`B#6#29Zir6zkFDuGw-iE17G6lKcw$)Sk7vCl4RMIg?@Oq_Hj(tpA7PTAF% zFUu$4wS@-Ocf?a0qczfJ!5)!)Lg;=2cCfFF&PF>f6}!cLgCD3@T9(=l^yR8^Q_OKx zJGKJ8PM?T~SIzA)nP;Fw?_xmwFHbq)1J(l!w=N2$DfWPQM=2Pkl|F115YstC9CAwuwSEHn_z0x&*5j6xxa^Sucj|SoX~+O_rMfpkoV6Gj)--zJyLVJntI^oC{�fBe;Cg} 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