From 52707be8d62c9577b5e513b4dc636e8a7b05f872 Mon Sep 17 00:00:00 2001 From: Michael Shafrir Date: Mon, 25 Nov 2019 12:09:29 -0500 Subject: [PATCH 1/2] Update configuration of example app Summary - Add logic to read configuration settings from local `gradle.properties` or environment variables and write out to `AndroidManifest.xml` at build time. - Update `Settings` to be a class that takes a `Context` to read metadata from `AndroidManifest.xml`. Retain constants in `Settings` to preserve existing functionality. - Update `RetrofitFactory` accordingly. Motivation Enable Espresso testing of the example app without hardcoding config values by setting them in Travis environment variables [0]. [0] https://docs.travis-ci.com/user/environment-variables/ --- example/AndroidManifest.xml | 10 +++ example/build.gradle | 25 ++++++ .../main/java/com/stripe/example/Settings.kt | 78 ++++++++++++++----- .../activity/CustomerSessionActivity.kt | 2 +- .../example/activity/FpxPaymentActivity.java | 2 +- .../activity/FragmentExamplesActivity.kt | 12 ++- .../example/activity/LauncherActivity.kt | 2 +- .../example/activity/PaymentAuthActivity.kt | 8 +- .../activity/PaymentSessionActivity.kt | 2 +- ...etrofitFactory.kt => BackendApiFactory.kt} | 14 ++-- .../service/ExampleEphemeralKeyProvider.kt | 13 +++- 11 files changed, 128 insertions(+), 40 deletions(-) rename example/src/main/java/com/stripe/example/module/{RetrofitFactory.kt => BackendApiFactory.kt} (76%) diff --git a/example/AndroidManifest.xml b/example/AndroidManifest.xml index b55e70319f1..f8067422145 100644 --- a/example/AndroidManifest.xml +++ b/example/AndroidManifest.xml @@ -15,6 +15,16 @@ android:supportsRtl="true" tools:ignore="AllowBackup"> + + + + launchAddPaymentMethod()); diff --git a/example/src/main/java/com/stripe/example/activity/FragmentExamplesActivity.kt b/example/src/main/java/com/stripe/example/activity/FragmentExamplesActivity.kt index 43358155087..07ec7c62ea9 100644 --- a/example/src/main/java/com/stripe/example/activity/FragmentExamplesActivity.kt +++ b/example/src/main/java/com/stripe/example/activity/FragmentExamplesActivity.kt @@ -10,6 +10,7 @@ import android.widget.ProgressBar import android.widget.TextView import android.widget.Toast import androidx.appcompat.app.AppCompatActivity +import androidx.fragment.app.Fragment import com.stripe.android.ApiResultCallback import com.stripe.android.CustomerSession import com.stripe.android.PaymentConfiguration @@ -25,7 +26,7 @@ import com.stripe.android.model.ConfirmSetupIntentParams import com.stripe.android.model.Customer import com.stripe.android.view.ShippingInfoWidget import com.stripe.example.R -import com.stripe.example.module.RetrofitFactory +import com.stripe.example.module.BackendApiFactory import com.stripe.example.service.BackendApi import com.stripe.example.service.ExampleEphemeralKeyProvider import io.reactivex.android.schedulers.AndroidSchedulers @@ -54,7 +55,7 @@ class FragmentExamplesActivity : AppCompatActivity() { .commit() } - class LauncherFragment : androidx.fragment.app.Fragment() { + class LauncherFragment : Fragment() { private val compositeDisposable = CompositeDisposable() @@ -71,7 +72,7 @@ class FragmentExamplesActivity : AppCompatActivity() { override fun onActivityCreated(savedInstanceState: Bundle?) { super.onActivityCreated(savedInstanceState) - backendApi = RetrofitFactory.instance.create(BackendApi::class.java) + backendApi = BackendApiFactory(requireContext()).create() stripe = Stripe( requireContext(), PaymentConfiguration.getInstance(requireContext()).publishableKey @@ -274,7 +275,10 @@ class FragmentExamplesActivity : AppCompatActivity() { } private fun createCustomerSession(): CustomerSession { - CustomerSession.initCustomerSession(requireContext(), ExampleEphemeralKeyProvider()) + CustomerSession.initCustomerSession( + requireContext(), + ExampleEphemeralKeyProvider(requireContext()) + ) val customerSession = CustomerSession.getInstance() customerSession.retrieveCurrentCustomer( object : CustomerSession.CustomerRetrievalListener { diff --git a/example/src/main/java/com/stripe/example/activity/LauncherActivity.kt b/example/src/main/java/com/stripe/example/activity/LauncherActivity.kt index 7e294654d3f..42ee05d92e5 100644 --- a/example/src/main/java/com/stripe/example/activity/LauncherActivity.kt +++ b/example/src/main/java/com/stripe/example/activity/LauncherActivity.kt @@ -20,7 +20,7 @@ class LauncherActivity : AppCompatActivity() { super.onCreate(savedInstanceState) setContentView(R.layout.activity_launcher) - PaymentConfiguration.init(this, Settings.PUBLISHABLE_KEY) + PaymentConfiguration.init(this, Settings(this).publishableKey) val linearLayoutManager = LinearLayoutManager(this) .apply { diff --git a/example/src/main/java/com/stripe/example/activity/PaymentAuthActivity.kt b/example/src/main/java/com/stripe/example/activity/PaymentAuthActivity.kt index 4de5301e48a..a9efb772eef 100644 --- a/example/src/main/java/com/stripe/example/activity/PaymentAuthActivity.kt +++ b/example/src/main/java/com/stripe/example/activity/PaymentAuthActivity.kt @@ -16,7 +16,7 @@ import com.stripe.android.model.ConfirmSetupIntentParams import com.stripe.android.model.PaymentMethodCreateParams import com.stripe.example.R import com.stripe.example.Settings -import com.stripe.example.module.RetrofitFactory +import com.stripe.example.module.BackendApiFactory import com.stripe.example.service.BackendApi import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.disposables.CompositeDisposable @@ -39,12 +39,12 @@ class PaymentAuthActivity : AppCompatActivity() { private lateinit var backendApi: BackendApi private lateinit var statusTextView: TextView - private val stripeAccountId: String? = Settings.STRIPE_ACCOUNT_ID - override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_payment_auth) + val stripeAccountId = Settings(this).stripeAccountId + val uiCustomization = PaymentAuthConfig.Stripe3ds2UiCustomization.Builder().build() PaymentAuthConfig.init(PaymentAuthConfig.Builder() @@ -59,7 +59,7 @@ class PaymentAuthActivity : AppCompatActivity() { statusTextView.text = savedInstanceState.getString(STATE_STATUS) } - backendApi = RetrofitFactory.instance.create(BackendApi::class.java) + backendApi = BackendApiFactory(this).create() val publishableKey = PaymentConfiguration.getInstance(this).publishableKey stripe = Stripe(this, publishableKey, stripeAccountId = stripeAccountId, diff --git a/example/src/main/java/com/stripe/example/activity/PaymentSessionActivity.kt b/example/src/main/java/com/stripe/example/activity/PaymentSessionActivity.kt index 71eb2c18c10..5b28e3e853c 100644 --- a/example/src/main/java/com/stripe/example/activity/PaymentSessionActivity.kt +++ b/example/src/main/java/com/stripe/example/activity/PaymentSessionActivity.kt @@ -94,7 +94,7 @@ class PaymentSessionActivity : AppCompatActivity() { private fun createCustomerSession(): CustomerSession { CustomerSession.initCustomerSession( this, - ExampleEphemeralKeyProvider(), + ExampleEphemeralKeyProvider(this), false ) return CustomerSession.getInstance() diff --git a/example/src/main/java/com/stripe/example/module/RetrofitFactory.kt b/example/src/main/java/com/stripe/example/module/BackendApiFactory.kt similarity index 76% rename from example/src/main/java/com/stripe/example/module/RetrofitFactory.kt rename to example/src/main/java/com/stripe/example/module/BackendApiFactory.kt index 6a2c62311b7..275f7cbdcdc 100644 --- a/example/src/main/java/com/stripe/example/module/RetrofitFactory.kt +++ b/example/src/main/java/com/stripe/example/module/BackendApiFactory.kt @@ -1,8 +1,10 @@ package com.stripe.example.module +import android.content.Context import com.facebook.stetho.okhttp3.StethoInterceptor import com.google.gson.GsonBuilder import com.stripe.example.Settings +import com.stripe.example.service.BackendApi import okhttp3.OkHttpClient import okhttp3.logging.HttpLoggingInterceptor import retrofit2.Retrofit @@ -12,10 +14,11 @@ import retrofit2.converter.gson.GsonConverterFactory /** * Factory to generate our Retrofit instance. */ -object RetrofitFactory { - val instance: Retrofit +class BackendApiFactory internal constructor(private val backendUrl: String) { - init { + constructor(context: Context) : this(Settings(context).backendUrl) + + fun create(): BackendApi { // Set your desired log level. Use Level.BODY for debugging errors. // Adding Rx so the calls can be Observable, and adding a Gson converter with // leniency to make parsing the results simple. @@ -31,11 +34,12 @@ object RetrofitFactory { .setLenient() .create() - instance = Retrofit.Builder() + return Retrofit.Builder() .addConverterFactory(GsonConverterFactory.create(gson)) .addCallAdapterFactory(RxJava2CallAdapterFactory.create()) - .baseUrl(Settings.BASE_URL) + .baseUrl(backendUrl) .client(httpClient) .build() + .create(BackendApi::class.java) } } diff --git a/example/src/main/java/com/stripe/example/service/ExampleEphemeralKeyProvider.kt b/example/src/main/java/com/stripe/example/service/ExampleEphemeralKeyProvider.kt index 88b91efceb8..55e91a8753c 100644 --- a/example/src/main/java/com/stripe/example/service/ExampleEphemeralKeyProvider.kt +++ b/example/src/main/java/com/stripe/example/service/ExampleEphemeralKeyProvider.kt @@ -1,10 +1,12 @@ package com.stripe.example.service +import android.content.Context import android.util.Log import androidx.annotation.Size import com.stripe.android.EphemeralKeyProvider import com.stripe.android.EphemeralKeyUpdateListener -import com.stripe.example.module.RetrofitFactory +import com.stripe.example.Settings +import com.stripe.example.module.BackendApiFactory import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.disposables.CompositeDisposable import io.reactivex.schedulers.Schedulers @@ -14,11 +16,14 @@ import java.io.IOException * An implementation of [EphemeralKeyProvider] that can be used to generate * ephemeral keys on the backend. */ -class ExampleEphemeralKeyProvider : EphemeralKeyProvider { +internal class ExampleEphemeralKeyProvider constructor( + backendUrl: String +) : EphemeralKeyProvider { + + constructor(context: Context) : this(Settings(context).backendUrl) private val compositeDisposable: CompositeDisposable = CompositeDisposable() - private val backendApi: BackendApi = - RetrofitFactory.instance.create(BackendApi::class.java) + private val backendApi: BackendApi = BackendApiFactory(backendUrl).create() override fun createEphemeralKey( @Size(min = 4) apiVersion: String, From 907b5abb031e92a0d45d314b5a9d21c34778a0a2 Mon Sep 17 00:00:00 2001 From: Michael Shafrir Date: Mon, 25 Nov 2019 13:15:32 -0500 Subject: [PATCH 2/2] Fix test failure --- example/build.gradle | 23 ++++++++++++++--------- 1 file changed, 14 insertions(+), 9 deletions(-) diff --git a/example/build.gradle b/example/build.gradle index d7e863984e4..3d1156184ad 100644 --- a/example/build.gradle +++ b/example/build.gradle @@ -11,21 +11,26 @@ configurations { // Read values from gradle.properties or system environment variable def getBackendUrl() { - return hasProperty('STRIPE_EXAMPLE_BACKEND_URL') ? - STRIPE_EXAMPLE_BACKEND_URL : - System.getenv('STRIPE_EXAMPLE_BACKEND_URL') + return readProperty('STRIPE_EXAMPLE_BACKEND_URL') } def getPublishableKey() { - return hasProperty('STRIPE_EXAMPLE_PUBLISHABLE_KEY') ? - STRIPE_EXAMPLE_PUBLISHABLE_KEY : - System.getenv('STRIPE_EXAMPLE_PUBLISHABLE_KEY') + return readProperty('STRIPE_EXAMPLE_PUBLISHABLE_KEY') } def getAccountId() { - return hasProperty('STRIPE_ACCOUNT_ID') ? - STRIPE_ACCOUNT_ID : - System.getenv('STRIPE_ACCOUNT_ID') + return readProperty('STRIPE_ACCOUNT_ID') +} + +private def readProperty(name) { + final String propValue + if (hasProperty(name)) { + propValue = property(name) + } else { + propValue = System.getenv(name) + } + + return propValue?.trim() ? propValue : "" } dependencies {