Skip to content

Commit

Permalink
Issue mozilla-mobile#10140 - Part 3: Implement the new Autocomplete.S…
Browse files Browse the repository at this point in the history
…torageDelegate interface in GeckoStorageDelegateWrapper

- Uses the new GV `Autocomplete.StorageDelegate` interface in GeckoStorageDelegateWrapper
- Implements `onCreditCardFetch`
  • Loading branch information
gabrielluong committed Apr 22, 2021
1 parent 62acdcb commit a236c11
Show file tree
Hide file tree
Showing 7 changed files with 120 additions and 15 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -9,30 +9,58 @@ import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.launch
import mozilla.components.browser.engine.gecko.ext.toLogin
import mozilla.components.browser.engine.gecko.ext.toLoginEntry
import mozilla.components.concept.storage.Login
import mozilla.components.concept.storage.CreditCardsAddressesStorageDelegate
import mozilla.components.concept.storage.LoginStorageDelegate
import org.mozilla.geckoview.Autocomplete
import org.mozilla.geckoview.GeckoResult

/**
* This class exists only to convert incoming [LoginEntry] arguments into [Login]s, then forward
* them to [storageDelegate]. This allows us to avoid duplicating [LoginStorageDelegate] code
* between different versions of GeckoView, by duplicating this wrapper instead.
* Gecko credit card and login storage delegate that handles runtime storage requests. This allows
* the Gecko runtime to call the underlying storage to handle requests for fetching, saving and
* updating of autocomplete items in the storage.
*/
@Suppress("Deprecation")
// This will be addressed in https://github.com/mozilla-mobile/android-components/issues/10093
class GeckoLoginDelegateWrapper(private val storageDelegate: LoginStorageDelegate) :
Autocomplete.LoginStorageDelegate {
class GeckoStorageDelegateWrapper(
private val creditCardsAddressesStorageDelegate: CreditCardsAddressesStorageDelegate,
private val loginStorageDelegate: LoginStorageDelegate
) : Autocomplete.StorageDelegate {

override fun onCreditCardFetch(): GeckoResult<Array<Autocomplete.CreditCard>>? {
val result = GeckoResult<Array<Autocomplete.CreditCard>>()

GlobalScope.launch(IO) {
val creditCards = creditCardsAddressesStorageDelegate.onCreditCardsFetch().await()
.mapNotNull {
val plaintextCardNumber =
creditCardsAddressesStorageDelegate.decrypt(it.encryptedCardNumber)?.number

if (plaintextCardNumber == null) {
null
} else {
Autocomplete.CreditCard.Builder()
.guid(it.guid)
.name(it.billingName)
.number(plaintextCardNumber)
.expirationMonth(it.expiryMonth.toString())
.expirationYear(it.expiryYear.toString())
.build()
}
}
.toTypedArray()
result.complete(creditCards)
}

return result
}

override fun onLoginSave(login: Autocomplete.LoginEntry) {
storageDelegate.onLoginSave(login.toLogin())
loginStorageDelegate.onLoginSave(login.toLogin())
}

override fun onLoginFetch(domain: String): GeckoResult<Array<Autocomplete.LoginEntry>>? {
val result = GeckoResult<Array<Autocomplete.LoginEntry>>()

GlobalScope.launch(IO) {
val storedLogins = storageDelegate.onLoginFetch(domain)
val storedLogins = loginStorageDelegate.onLoginFetch(domain)

val logins = storedLogins.await()
.map { it.toLoginEntry() }
Expand Down
2 changes: 2 additions & 0 deletions components/concept/storage/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,8 @@ dependencies {
// dependency, but it will crash at runtime.
// Included via 'api' because this module is unusable without coroutines.
api Dependencies.kotlin_coroutines

api project(':lib-dataprotect')
}

apply from: '../../../publish.gradle'
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ package mozilla.components.concept.storage
import android.os.Parcelable
import kotlinx.android.parcel.Parcelize
import kotlinx.coroutines.Deferred
import mozilla.components.lib.dataprotect.ManagedKey

/**
* An interface which defines read/write methods for credit card and address data.
Expand Down Expand Up @@ -104,6 +105,32 @@ interface CreditCardsAddressesStorage {
* @param guid Unique identifier for the desired address.
*/
suspend fun touchAddress(guid: String)

/**
* Returns an instance of [CreditCardCrypto] that knows how to encrypt and decrypt credit card
* numbers.
*
* @return [CreditCardCrypto] instance.
*/
fun getCreditCardCrypto(): CreditCardCrypto
}

/**
*
*/
interface CreditCardCrypto {

/**
* Encrypt a [CreditCardNumber.Plaintext] using provided key.
* A `null` result means a bad key was provided. In that case caller should obtain a new key and try again.
*/
fun encrypt(key: ManagedKey, plaintextCardNumber: CreditCardNumber.Plaintext): CreditCardNumber.Encrypted?

/**
* Decrypt a [CreditCardNumber.Encrypted] using provided key.
* A `null` result means a bad key was provided. In that case caller should obtain a new key and try again.
*/
fun decrypt(key: ManagedKey, encryptedCardNumber: CreditCardNumber.Encrypted): CreditCardNumber.Plaintext?
}

/**
Expand Down Expand Up @@ -273,6 +300,15 @@ data class UpdatableAddressFields(
*/
interface CreditCardsAddressesStorageDelegate {

/**
* Decrypt the provided encrypted credit card number into its plaintext equivalent or `null` if
* it fails to decrypt.
*
* @param encryptedCardNumber The encrypted credit card number.
* @return the decrypted plaintext credit card number.
*/
fun decrypt(encryptedCardNumber: CreditCardNumber.Encrypted): CreditCardNumber.Plaintext?

/**
* Returns all stored addresses. This is called when the engine believes an address field
* should be autofilled.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import mozilla.components.concept.storage.UpdatableAddressFields
import mozilla.components.concept.storage.UpdatableCreditCardFields
import mozilla.components.concept.sync.SyncableStore
import mozilla.components.lib.dataprotect.KeyGenerationReason
import mozilla.components.lib.dataprotect.KeyProvider
import mozilla.components.lib.dataprotect.KeyRecoveryHandler
import mozilla.components.lib.dataprotect.SecureAbove22Preferences
import mozilla.components.support.base.log.logger.Logger
Expand Down Expand Up @@ -157,6 +158,10 @@ class AutofillCreditCardsAddressesStorage(
conn.getStorage().touchAddress(guid)
}

override fun getCreditCardCrypto(): AutofillCrypto {
return crypto
}

override fun registerWithSyncManager() {
conn.getStorage().registerWithSyncManager()
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import mozilla.appservices.autofill.ErrorException
import mozilla.appservices.autofill.createKey
import mozilla.appservices.autofill.decryptString
import mozilla.appservices.autofill.encryptString
import mozilla.components.concept.storage.CreditCardCrypto
import mozilla.components.concept.storage.CreditCardNumber
import mozilla.components.lib.dataprotect.KeyGenerationReason
import mozilla.components.lib.dataprotect.KeyProvider
Expand All @@ -33,15 +34,15 @@ class AutofillCrypto(
private val context: Context,
private val securePrefs: SecureAbove22Preferences,
private val keyRecoveryHandler: KeyRecoveryHandler
) : KeyProvider {
) : CreditCardCrypto, KeyProvider {
private val logger = Logger("AutofillCrypto")
private val plaintextPrefs by lazy { context.getSharedPreferences(AUTOFILL_PREFS, Context.MODE_PRIVATE) }

/**
* Encrypt a [CreditCardNumber.Plaintext] using provided key.
* A `null` result means a bad key was provided. In that case caller should obtain a new key and try again.
*/
fun encrypt(key: ManagedKey, plaintextCardNumber: CreditCardNumber.Plaintext): CreditCardNumber.Encrypted? {
override fun encrypt(key: ManagedKey, plaintextCardNumber: CreditCardNumber.Plaintext): CreditCardNumber.Encrypted? {
return try {
CreditCardNumber.Encrypted(encrypt(key, plaintextCardNumber.number))
} catch (e: ErrorException.JsonError) {
Expand All @@ -57,7 +58,7 @@ class AutofillCrypto(
* Decrypt a [CreditCardNumber.Encrypted] using provided key.
* A `null` result means a bad key was provided. In that case caller should obtain a new key and try again.
*/
fun decrypt(key: ManagedKey, encryptedCardNumber: CreditCardNumber.Encrypted): CreditCardNumber.Plaintext? {
override fun decrypt(key: ManagedKey, encryptedCardNumber: CreditCardNumber.Encrypted): CreditCardNumber.Plaintext? {
return try {
CreditCardNumber.Plaintext(decrypt(key, encryptedCardNumber.number))
} catch (e: ErrorException.JsonError) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.async
import mozilla.components.concept.storage.Address
import mozilla.components.concept.storage.CreditCard
import mozilla.components.concept.storage.CreditCardNumber
import mozilla.components.concept.storage.CreditCardsAddressesStorage
import mozilla.components.concept.storage.CreditCardsAddressesStorageDelegate

Expand All @@ -21,6 +22,12 @@ class GeckoCreditCardsAddressesStorageDelegate(
private val scope: CoroutineScope = CoroutineScope(Dispatchers.IO)
) : CreditCardsAddressesStorageDelegate {

override fun decrypt(encryptedCardNumber: CreditCardNumber.Encrypted): CreditCardNumber.Plaintext? {
val crypto = storage.value.getCreditCardCrypto()
val key = crypto.key()
return crypto.decrypt(key, encryptedCardNumber)
}

override fun onAddressesFetch(): Deferred<List<Address>> {
return scope.async {
storage.value.getAllAddresses()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,11 @@ import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.launch
import kotlinx.coroutines.runBlocking
import kotlinx.coroutines.test.TestCoroutineScope
import mozilla.components.support.test.mock
import mozilla.components.concept.storage.CreditCardNumber
import mozilla.components.concept.storage.NewCreditCardFields
import mozilla.components.lib.dataprotect.SecureAbove22Preferences
import mozilla.components.support.test.robolectric.testContext
import org.junit.Assert.assertEquals
import org.junit.Before
import org.junit.Test
import org.junit.runner.RunWith
Expand All @@ -21,7 +24,8 @@ import org.mockito.Mockito.verify
@RunWith(AndroidJUnit4::class)
class GeckoCreditCardsAddressesStorageDelegateTest {

private val storage = AutofillCreditCardsAddressesStorage(testContext, mock())
private lateinit var storage: AutofillCreditCardsAddressesStorage
private lateinit var securePrefs: SecureAbove22Preferences
private lateinit var delegate: GeckoCreditCardsAddressesStorageDelegate
private lateinit var scope: TestCoroutineScope

Expand All @@ -32,9 +36,31 @@ class GeckoCreditCardsAddressesStorageDelegateTest {
@Before
fun before() = runBlocking {
scope = TestCoroutineScope()
// forceInsecure is set in the tests because a keystore wouldn't be configured in the test environment.
securePrefs = SecureAbove22Preferences(testContext, "autofill", forceInsecure = true)
storage = AutofillCreditCardsAddressesStorage(testContext, lazy { securePrefs })
delegate = GeckoCreditCardsAddressesStorageDelegate(lazy { storage }, scope)
}

@Test
fun `decrypt`() = runBlocking {
val plaintextNumber = CreditCardNumber.Plaintext("4111111111111111")
val creditCardFields = NewCreditCardFields(
billingName = "Jon Doe",
plaintextCardNumber = plaintextNumber,
cardNumberLast4 = "1111",
expiryMonth = 12,
expiryYear = 2028,
cardType = "amex"
)
val creditCard = storage.addCreditCard(creditCardFields)

assertEquals(
plaintextNumber,
delegate.decrypt(creditCard.encryptedCardNumber)
)
}

@Test
fun `onAddressFetch`() {
scope.launch {
Expand Down

0 comments on commit a236c11

Please sign in to comment.