chromium/chrome/browser/autofill/android/java/src/org/chromium/chrome/browser/autofill/PersonalDataManager.java

// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

package org.chromium.chrome.browser.autofill;

import android.content.Context;
import android.graphics.Bitmap;
import android.text.TextUtils;

import androidx.annotation.Nullable;

import org.jni_zero.CalledByNative;
import org.jni_zero.JNINamespace;
import org.jni_zero.JniType;
import org.jni_zero.NativeMethods;

import org.chromium.base.ResettersForTesting;
import org.chromium.base.ThreadUtils;
import org.chromium.base.lifetime.Destroyable;
import org.chromium.chrome.browser.preferences.Pref;
import org.chromium.chrome.browser.profiles.Profile;
import org.chromium.components.autofill.AutofillProfile;
import org.chromium.components.autofill.IbanRecordType;
import org.chromium.components.autofill.VirtualCardEnrollmentState;
import org.chromium.components.autofill.payments.BankAccount;
import org.chromium.components.image_fetcher.ImageFetcher;
import org.chromium.components.prefs.PrefService;
import org.chromium.components.user_prefs.UserPrefs;
import org.chromium.url.GURL;

import java.util.ArrayList;
import java.util.Calendar;
import java.util.List;
import java.util.Objects;
import java.util.Optional;

/**
 * Android wrapper of the PersonalDataManager which provides access from the Java layer.
 *
 * <p>Only usable from the UI thread as it's primary purpose is for supporting the Android
 * preferences UI.
 *
 * <p>See chrome/browser/autofill/personal_data_manager.h for more details.
 */
@JNINamespace("autofill")
public class PersonalDataManager implements Destroyable {
    private static final String TAG = "PersonalDataManager";

    /** Observer of PersonalDataManager events. */
    public interface PersonalDataManagerObserver {
        /** Called when the data is changed. */
        void onPersonalDataChanged();
    }

    /** Autofill credit card information. */
    public static class CreditCard {
        // Note that while some of these fields are numbers, they're predominantly read,
        // marshaled and compared as strings. To save conversions, we sometimes use strings.
        private String mGUID;
        private String mOrigin;
        private boolean mIsLocal;
        private boolean mIsVirtual;
        private String mName;
        private String mNumber;
        private String mNetworkAndLastFourDigits;
        private String mMonth;
        private String mYear;
        private String mBasicCardIssuerNetwork;
        private int mIssuerIconDrawableId;
        private String mBillingAddressId;
        private final String mServerId;
        private final long mInstrumentId;
        // The label set for the card. This could be one of Card Network + LastFour, Nickname +
        // LastFour or a Google specific string for Google-issued cards. This is used for displaying
        // the card in PaymentMethods in Settings.
        private String mCardLabel;
        private String mNickname;
        private GURL mCardArtUrl;
        private String mCvc;
        private final @VirtualCardEnrollmentState int mVirtualCardEnrollmentState;
        private final String mProductDescription;
        private final String mCardNameForAutofillDisplay;
        private final String mObfuscatedLastFourDigits;

        @CalledByNative("CreditCard")
        public static CreditCard create(
                String guid,
                String origin,
                boolean isLocal,
                boolean isVirtual,
                String name,
                String number,
                String networkAndLastFourDigits,
                String month,
                String year,
                String basicCardIssuerNetwork,
                int iconId,
                String billingAddressId,
                String serverId,
                long instrumentId,
                String cardLabel,
                String nickname,
                GURL cardArtUrl,
                @VirtualCardEnrollmentState int virtualCardEnrollmentState,
                String productDescription,
                String cardNameForAutofillDisplay,
                String obfuscatedLastFourDigits,
                String cvc) {
            return new CreditCard(
                    guid,
                    origin,
                    isLocal,
                    isVirtual,
                    name,
                    number,
                    networkAndLastFourDigits,
                    month,
                    year,
                    basicCardIssuerNetwork,
                    iconId,
                    billingAddressId,
                    serverId,
                    instrumentId,
                    cardLabel,
                    nickname,
                    cardArtUrl,
                    virtualCardEnrollmentState,
                    productDescription,
                    cardNameForAutofillDisplay,
                    obfuscatedLastFourDigits,
                    cvc);
        }

        public CreditCard(
                String guid,
                String origin,
                boolean isLocal,
                String name,
                String number,
                String networkAndLastFourDigits,
                String month,
                String year,
                String basicCardIssuerNetwork,
                int issuerIconDrawableId,
                String billingAddressId,
                String serverId) {
            this(
                    guid,
                    origin,
                    isLocal,
                    /* isVirtual= */ false,
                    name,
                    number,
                    networkAndLastFourDigits,
                    month,
                    year,
                    basicCardIssuerNetwork,
                    issuerIconDrawableId,
                    billingAddressId,
                    serverId,
                    /* instrumentId= */ 0,
                    /* cardLabel= */ networkAndLastFourDigits,
                    /* nickname= */ "",
                    /* cardArtUrl= */ null,
                    /* virtualCardEnrollmentState= */ VirtualCardEnrollmentState.UNSPECIFIED,
                    /* productDescription= */ "",
                    /* cardNameForAutofillDisplay= */ "",
                    /* obfuscatedLastFourDigits= */ "",
                    /* cvc= */ "");
        }

        public CreditCard(
                String guid,
                String origin,
                boolean isLocal,
                boolean isVirtual,
                String name,
                String number,
                String networkAndLastFourDigits,
                String month,
                String year,
                String basicCardIssuerNetwork,
                int issuerIconDrawableId,
                String billingAddressId,
                String serverId,
                long instrumentId,
                String cardLabel,
                String nickname,
                GURL cardArtUrl,
                @VirtualCardEnrollmentState int virtualCardEnrollmentState,
                String productDescription,
                String cardNameForAutofillDisplay,
                String obfuscatedLastFourDigits,
                String cvc) {
            mGUID = guid;
            mOrigin = origin;
            mIsLocal = isLocal;
            mIsVirtual = isVirtual;
            mName = name;
            mNumber = number;
            mNetworkAndLastFourDigits = networkAndLastFourDigits;
            mMonth = month;
            mYear = year;
            mBasicCardIssuerNetwork = basicCardIssuerNetwork;
            mIssuerIconDrawableId = issuerIconDrawableId;
            mBillingAddressId = billingAddressId;
            mServerId = serverId;
            mInstrumentId = instrumentId;
            mCardLabel = cardLabel;
            mNickname = nickname;
            mCardArtUrl = cardArtUrl;
            mVirtualCardEnrollmentState = virtualCardEnrollmentState;
            mProductDescription = productDescription;
            mCardNameForAutofillDisplay = cardNameForAutofillDisplay;
            mObfuscatedLastFourDigits = obfuscatedLastFourDigits;
            mCvc = cvc;
        }

        public CreditCard() {
            this(
                    /* guid= */ "",
                    /* origin= */ AutofillEditorBase.SETTINGS_ORIGIN,
                    /* isLocal= */ true,
                    /* name= */ "",
                    /* number= */ "",
                    /* networkAndLastFourDigits= */ "",
                    /* month= */ "",
                    /* year= */ "",
                    /* basicCardIssuerNetwork= */ "",
                    /* issuerIconDrawableId= */ 0,
                    /* billingAddressId= */ "",
                    /* serverId= */ "");
        }

        @CalledByNative("CreditCard")
        public String getGUID() {
            return mGUID;
        }

        @CalledByNative("CreditCard")
        public String getOrigin() {
            return mOrigin;
        }

        @CalledByNative("CreditCard")
        public String getName() {
            return mName;
        }

        @CalledByNative("CreditCard")
        public String getNumber() {
            return mNumber;
        }

        public String getNetworkAndLastFourDigits() {
            return mNetworkAndLastFourDigits;
        }

        @CalledByNative("CreditCard")
        public String getMonth() {
            return mMonth;
        }

        @CalledByNative("CreditCard")
        public String getYear() {
            return mYear;
        }

        public String getFormattedExpirationDate(Context context) {
            String twoDigityear = getYear().substring(2);
            return getMonth()
                    + context.getResources().getString(R.string.autofill_expiration_date_separator)
                    + twoDigityear;
        }

        public String getFormattedExpirationDateWithCvcSavedMessage(Context context) {
            return context.getResources()
                    .getString(
                            R.string.autofill_settings_page_summary_separated_by_pipe,
                            getFormattedExpirationDate(context),
                            context.getResources()
                                    .getString(R.string.autofill_settings_page_cvc_saved_label));
        }

        @CalledByNative("CreditCard")
        public boolean getIsLocal() {
            return mIsLocal;
        }

        @CalledByNative("CreditCard")
        public boolean getIsVirtual() {
            return mIsVirtual;
        }

        @CalledByNative("CreditCard")
        public String getBasicCardIssuerNetwork() {
            return mBasicCardIssuerNetwork;
        }

        public int getIssuerIconDrawableId() {
            return mIssuerIconDrawableId;
        }

        @CalledByNative("CreditCard")
        public String getBillingAddressId() {
            return mBillingAddressId;
        }

        @CalledByNative("CreditCard")
        public String getServerId() {
            return mServerId;
        }

        @CalledByNative("CreditCard")
        public long getInstrumentId() {
            return mInstrumentId;
        }

        public String getCardLabel() {
            return mCardLabel;
        }

        @CalledByNative("CreditCard")
        public String getNickname() {
            return mNickname;
        }

        @CalledByNative("CreditCard")
        public GURL getCardArtUrl() {
            return mCardArtUrl;
        }

        @CalledByNative("CreditCard")
        public @VirtualCardEnrollmentState int getVirtualCardEnrollmentState() {
            return mVirtualCardEnrollmentState;
        }

        @CalledByNative("CreditCard")
        public String getProductDescription() {
            return mProductDescription;
        }

        @CalledByNative("CreditCard")
        public String getCvc() {
            return mCvc;
        }

        public String getCardNameForAutofillDisplay() {
            return mCardNameForAutofillDisplay;
        }

        public String getObfuscatedLastFourDigits() {
            return mObfuscatedLastFourDigits;
        }

        public void setGUID(String guid) {
            mGUID = guid;
        }

        public void setOrigin(String origin) {
            mOrigin = origin;
        }

        public void setName(String name) {
            mName = name;
        }

        public void setNumber(String number) {
            mNumber = number;
        }

        public void setNetworkAndLastFourDigits(String networkAndLastFourDigits) {
            mNetworkAndLastFourDigits = networkAndLastFourDigits;
        }

        public void setMonth(String month) {
            mMonth = month;
        }

        public void setYear(String year) {
            mYear = year;
        }

        public void setCvc(String cvc) {
            mCvc = cvc;
        }

        public void setBasicCardIssuerNetwork(String network) {
            mBasicCardIssuerNetwork = network;
        }

        public void setIssuerIconDrawableId(int id) {
            mIssuerIconDrawableId = id;
        }

        public void setBillingAddressId(String id) {
            mBillingAddressId = id;
        }

        public void setCardLabel(String cardLabel) {
            mCardLabel = cardLabel;
        }

        public void setNickname(String nickname) {
            mNickname = nickname;
        }

        public void setCardArtUrl(GURL cardArtUrl) {
            mCardArtUrl = cardArtUrl;
        }

        public boolean hasValidCreditCardExpirationDate() {
            if (mMonth.isEmpty() || mYear.isEmpty()) return false;

            Calendar expiryDate = Calendar.getInstance();
            // The mMonth value is 1 based but the month in calendar is 0 based.
            expiryDate.set(Calendar.MONTH, Integer.parseInt(mMonth) - 1);
            expiryDate.set(Calendar.YEAR, Integer.parseInt(mYear));
            // Add 1 minute to the expiry instance to ensure that the card is still valid on its
            // exact expiration date.
            expiryDate.add(Calendar.MINUTE, 1);
            return Calendar.getInstance().before(expiryDate);
        }
    }

    /** Autofill IBAN information. */
    public static class Iban {
        @Nullable private String mGuid;
        @Nullable private Long mInstrumentId;

        // Obfuscated IBAN value. This is used for displaying the IBAN in the Payment methods page.
        private String mLabel;

        private String mNickname;
        private @IbanRecordType int mRecordType;
        private String mValue;

        private Iban(
                String guid,
                Long instrumentId,
                String label,
                String nickname,
                @IbanRecordType int recordType,
                String value) {
            mGuid = guid;
            mInstrumentId = instrumentId;
            mLabel = Objects.requireNonNull(label, "Label can't be null");
            mNickname = Objects.requireNonNull(nickname, "Nickname can't be null");
            mRecordType = recordType;
            mValue = Objects.requireNonNull(value, "Iban value can't be null");
        }

        // Creates an Iban instance that is not stored on a server nor locally,
        // yet. This Iban has type IbanRecordType.UNKNOWN and has neither a
        // Guid nor an instrumentId.
        @CalledByNative("Iban")
        public static Iban createEphemeral(String label, String nickname, String value) {
            return new Iban.Builder()
                    .setLabel(label)
                    .setNickname(nickname)
                    .setRecordType(IbanRecordType.UNKNOWN)
                    .setValue(value)
                    .build();
        }

        @CalledByNative("Iban")
        public static Iban createLocal(String guid, String label, String nickname, String value) {
            return new Iban.Builder()
                    .setGuid(guid)
                    .setLabel(label)
                    .setNickname(nickname)
                    .setRecordType(IbanRecordType.LOCAL_IBAN)
                    .setValue(value)
                    .build();
        }

        @CalledByNative("Iban")
        public static Iban createServer(
                long instrumentId, String label, String nickname, String value) {
            return new Iban.Builder()
                    .setInstrumentId(Long.valueOf(instrumentId))
                    .setLabel(label)
                    .setNickname(nickname)
                    .setRecordType(IbanRecordType.SERVER_IBAN)
                    .setValue(value)
                    .build();
        }

        @CalledByNative("Iban")
        public String getGuid() {
            assert mRecordType != IbanRecordType.SERVER_IBAN;
            return mGuid;
        }

        public Long getInstrumentId() {
            assert mInstrumentId != null;
            assert mRecordType == IbanRecordType.SERVER_IBAN;
            return mInstrumentId;
        }

        public String getLabel() {
            return mLabel;
        }

        @CalledByNative("Iban")
        public String getNickname() {
            return mNickname;
        }

        @CalledByNative("Iban")
        public @IbanRecordType int getRecordType() {
            return mRecordType;
        }

        @CalledByNative("Iban")
        public String getValue() {
            return mValue;
        }

        public void updateNickname(String nickname) {
            mNickname = nickname;
        }

        public void updateValue(String value) {
            mValue = value;
        }

        @Override
        public boolean equals(Object obj) {
            if (obj == null) return false;
            if (this == obj) return true;
            if (getClass() != obj.getClass()) return false;

            Iban otherIban = (Iban) obj;

            return Objects.equals(mLabel, otherIban.getLabel())
                    && Objects.equals(mNickname, otherIban.getNickname())
                    && mRecordType == otherIban.getRecordType()
                    && (mRecordType != IbanRecordType.SERVER_IBAN
                            || Objects.equals(mInstrumentId, otherIban.getInstrumentId()))
                    && (mRecordType != IbanRecordType.LOCAL_IBAN
                            || Objects.equals(mGuid, otherIban.getGuid()))
                    && Objects.equals(mValue, otherIban.getValue());
        }

        @Override
        public int hashCode() {
            return Objects.hash(mGuid, mLabel, mNickname, mRecordType, mValue);
        }

        /** Builder for {@link Iban}. */
        public static final class Builder {
            private String mGuid;
            private Long mInstrumentId;
            private String mLabel;
            private String mNickname;
            private @IbanRecordType int mRecordType;
            private String mValue;

            public Builder setGuid(String guid) {
                mGuid = guid;
                return this;
            }

            public Builder setInstrumentId(Long instrumentId) {
                mInstrumentId = instrumentId;
                return this;
            }

            public Builder setLabel(String label) {
                mLabel = label;
                return this;
            }

            public Builder setNickname(String nickname) {
                mNickname = nickname;
                return this;
            }

            public Builder setRecordType(@IbanRecordType int recordType) {
                mRecordType = recordType;
                return this;
            }

            public Builder setValue(String value) {
                mValue = value;
                return this;
            }

            public Iban build() {
                switch (mRecordType) {
                    case IbanRecordType.UNKNOWN:
                        assert mGuid == null && mInstrumentId == null
                                : "IBANs with 'UNKNOWN' record type must have an empty GUID and"
                                        + " InstrumentId.";
                        break;
                    case IbanRecordType.LOCAL_IBAN:
                        assert !TextUtils.isEmpty(mGuid) && mInstrumentId == null
                                : "Local IBANs must have a non-empty GUID and null InstrumentID.";
                        break;
                    case IbanRecordType.SERVER_IBAN:
                        assert mInstrumentId != null
                                        && mInstrumentId != 0L
                                        && TextUtils.isEmpty(mGuid)
                                : "Server IBANs must have a non-zero instrumentId and empty GUID.";
                        break;
                }
                return new Iban(mGuid, mInstrumentId, mLabel, mNickname, mRecordType, mValue);
            }
        }
    }

    private final PrefService mPrefService;
    private final List<PersonalDataManagerObserver> mDataObservers =
            new ArrayList<PersonalDataManagerObserver>();

    private long mPersonalDataManagerAndroid;
    private AutofillImageFetcher mImageFetcher;

    PersonalDataManager(Profile profile) {
        mPersonalDataManagerAndroid = PersonalDataManagerJni.get().init(this, profile);
        mPrefService = UserPrefs.get(profile);
        // Get the AutofillImageFetcher instance that was created during browser startup.
        mImageFetcher =
                PersonalDataManagerJni.get()
                        .getOrCreateJavaImageFetcher(mPersonalDataManagerAndroid);
    }

    @Override
    public void destroy() {
        PersonalDataManagerJni.get().destroy(mPersonalDataManagerAndroid);
        mPersonalDataManagerAndroid = 0;
    }

    /** Called from native when template URL service is done loading. */
    @CalledByNative
    private void personalDataChanged() {
        ThreadUtils.assertOnUiThread();
        for (PersonalDataManagerObserver observer : mDataObservers) {
            observer.onPersonalDataChanged();
        }
        fetchCreditCardArtImages();
    }

    /** Registers a PersonalDataManagerObserver on the native side. */
    public boolean registerDataObserver(PersonalDataManagerObserver observer) {
        ThreadUtils.assertOnUiThread();
        assert !mDataObservers.contains(observer);
        mDataObservers.add(observer);
        return PersonalDataManagerJni.get().isDataLoaded(mPersonalDataManagerAndroid);
    }

    /** Unregisters the provided observer. */
    public void unregisterDataObserver(PersonalDataManagerObserver observer) {
        ThreadUtils.assertOnUiThread();
        assert (mDataObservers.size() > 0);
        assert (mDataObservers.contains(observer));
        mDataObservers.remove(observer);
    }

    /**
     * TODO(crbug.com/41256488): Reduce the number of Java to Native calls when getting profiles.
     *
     * <p>Gets the profiles to show in the settings page. Returns all the profiles without any
     * processing.
     *
     * @return The list of profiles to show in the settings.
     */
    public List<AutofillProfile> getProfilesForSettings() {
        ThreadUtils.assertOnUiThread();
        return getProfilesWithLabels(
                PersonalDataManagerJni.get()
                        .getProfileLabelsForSettings(mPersonalDataManagerAndroid),
                PersonalDataManagerJni.get()
                        .getProfileGUIDsForSettings(mPersonalDataManagerAndroid));
    }

    /**
     * TODO(crbug.com/41256488): Reduce the number of Java to Native calls when getting profiles
     *
     * <p>Gets the profiles to suggest when filling a form or completing a transaction. The profiles
     * will have been processed to be more relevant to the user.
     *
     * @param includeNameInLabel Whether to include the name in the profile's label.
     * @return The list of profiles to suggest to the user.
     */
    public ArrayList<AutofillProfile> getProfilesToSuggest(boolean includeNameInLabel) {
        ThreadUtils.assertOnUiThread();
        return getProfilesWithLabels(
                PersonalDataManagerJni.get()
                        .getProfileLabelsToSuggest(
                                mPersonalDataManagerAndroid,
                                includeNameInLabel,
                                /* includeOrganizationInLabel= */ true,
                                /* includeCountryInLabel= */ true),
                PersonalDataManagerJni.get().getProfileGUIDsToSuggest(mPersonalDataManagerAndroid));
    }

    /**
     * TODO(crbug.com/41256488): Reduce the number of Java to Native calls when getting profiles.
     *
     * <p>Gets the profiles to suggest when associating a billing address to a credit card. The
     * profiles will have been processed to be more relevant to the user.
     *
     * @param includeOrganizationInLabel Whether the organization name should be included in the
     *     label.
     * @return The list of billing addresses to suggest to the user.
     */
    public ArrayList<AutofillProfile> getBillingAddressesToSuggest(
            boolean includeOrganizationInLabel) {
        ThreadUtils.assertOnUiThread();
        return getProfilesWithLabels(
                PersonalDataManagerJni.get()
                        .getProfileLabelsToSuggest(
                                mPersonalDataManagerAndroid,
                                /* includeNameInLabel= */ true,
                                includeOrganizationInLabel,
                                /* includeCountryInLabel= */ false),
                PersonalDataManagerJni.get().getProfileGUIDsToSuggest(mPersonalDataManagerAndroid));
    }

    private ArrayList<AutofillProfile> getProfilesWithLabels(
            String[] profileLabels, String[] profileGUIDs) {
        ArrayList<AutofillProfile> profiles = new ArrayList<AutofillProfile>(profileGUIDs.length);
        for (int i = 0; i < profileGUIDs.length; i++) {
            AutofillProfile profile =
                    new AutofillProfile(
                            PersonalDataManagerJni.get()
                                    .getProfileByGUID(
                                            mPersonalDataManagerAndroid, profileGUIDs[i]));
            profile.setLabel(profileLabels[i]);
            profiles.add(profile);
        }

        return profiles;
    }

    public AutofillProfile getProfile(String guid) {
        ThreadUtils.assertOnUiThread();
        return new AutofillProfile(
                PersonalDataManagerJni.get().getProfileByGUID(mPersonalDataManagerAndroid, guid));
    }

    public void deleteProfile(String guid) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get().removeByGUID(mPersonalDataManagerAndroid, guid);
    }

    public String setProfile(AutofillProfile profile) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .setProfile(mPersonalDataManagerAndroid, profile, profile.getGUID());
    }

    public String setProfileToLocal(AutofillProfile profile) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .setProfileToLocal(mPersonalDataManagerAndroid, profile, profile.getGUID());
    }

    /** Gets the number of credit cards for the settings page. */
    public int getCreditCardCountForSettings() {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getCreditCardGUIDsForSettings(mPersonalDataManagerAndroid)
                .length;
    }

    /**
     * Gets the credit cards to show in the settings page. Returns all the cards without any
     * processing.
     */
    public List<CreditCard> getCreditCardsForSettings() {
        ThreadUtils.assertOnUiThread();
        return getCreditCards(
                PersonalDataManagerJni.get()
                        .getCreditCardGUIDsForSettings(mPersonalDataManagerAndroid));
    }

    /**
     * Gets the credit cards to suggest when filling a form or completing a transaction. The cards
     * will have been processed to be more relevant to the user.
     */
    public ArrayList<CreditCard> getCreditCardsToSuggest() {
        ThreadUtils.assertOnUiThread();
        return getCreditCards(
                PersonalDataManagerJni.get()
                        .getCreditCardGUIDsToSuggest(mPersonalDataManagerAndroid));
    }

    private ArrayList<CreditCard> getCreditCards(String[] creditCardGUIDs) {
        ArrayList<CreditCard> cards = new ArrayList<CreditCard>(creditCardGUIDs.length);
        for (int i = 0; i < creditCardGUIDs.length; i++) {
            cards.add(
                    PersonalDataManagerJni.get()
                            .getCreditCardByGUID(mPersonalDataManagerAndroid, creditCardGUIDs[i]));
        }
        return cards;
    }

    public CreditCard getCreditCard(String guid) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get().getCreditCardByGUID(mPersonalDataManagerAndroid, guid);
    }

    public CreditCard getCreditCardForNumber(String cardNumber) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getCreditCardForNumber(mPersonalDataManagerAndroid, cardNumber);
    }

    public String setCreditCard(CreditCard card) {
        ThreadUtils.assertOnUiThread();
        assert card.getIsLocal();
        return PersonalDataManagerJni.get().setCreditCard(mPersonalDataManagerAndroid, card);
    }

    public void updateServerCardBillingAddress(CreditCard card) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get()
                .updateServerCardBillingAddress(mPersonalDataManagerAndroid, card);
    }

    public static String getBasicCardIssuerNetwork(String cardNumber, boolean emptyIfInvalid) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getBasicCardIssuerNetwork(
                        cardNumber,
                        emptyIfInvalid);
    }

    public void addServerCreditCardForTest(CreditCard card) {
        ThreadUtils.assertOnUiThread();
        assert !card.getIsLocal();
        PersonalDataManagerJni.get()
                .addServerCreditCardForTest(mPersonalDataManagerAndroid, card); // IN-TEST
    }

    public void addServerCreditCardForTestWithAdditionalFields(
            CreditCard card, String nickname, int cardIssuer) {
        ThreadUtils.assertOnUiThread();
        assert !card.getIsLocal();
        PersonalDataManagerJni.get()
                .addServerCreditCardForTestWithAdditionalFields(
                        mPersonalDataManagerAndroid, card, nickname, cardIssuer);
    }

    public void deleteCreditCard(String guid) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get().removeByGUID(mPersonalDataManagerAndroid, guid);
    }

    /** Deletes all local credit cards. */
    public void deleteAllLocalCreditCards() {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get().deleteAllLocalCreditCards(mPersonalDataManagerAndroid);
    }

    public String getShippingAddressLabelWithCountryForPaymentRequest(AutofillProfile profile) {
        return PersonalDataManagerJni.get()
                .getShippingAddressLabelForPaymentRequest(
                        mPersonalDataManagerAndroid,
                        profile,
                        profile.getGUID(),
                        /* includeCountry= */ true);
    }

    public String getShippingAddressLabelWithoutCountryForPaymentRequest(AutofillProfile profile) {
        return PersonalDataManagerJni.get()
                .getShippingAddressLabelForPaymentRequest(
                        mPersonalDataManagerAndroid,
                        profile,
                        profile.getGUID(),
                        /* includeCountry= */ false);
    }

    public Iban getIban(String guid) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get().getIbanByGuid(mPersonalDataManagerAndroid, guid);
    }

    public Iban[] getLocalIbansForSettings() {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get().getLocalIbansForSettings(mPersonalDataManagerAndroid);
    }

    public String addOrUpdateLocalIban(Iban iban) {
        ThreadUtils.assertOnUiThread();
        assert iban.getRecordType() == IbanRecordType.UNKNOWN
                        || iban.getRecordType() == IbanRecordType.LOCAL_IBAN
                : "Add or update local IBANs only.";
        return PersonalDataManagerJni.get().addOrUpdateLocalIban(mPersonalDataManagerAndroid, iban);
    }

    public void deleteIban(String guid) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get().removeByGUID(mPersonalDataManagerAndroid, guid);
    }

    public boolean isValidIban(String ibanValue) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get().isValidIban(mPersonalDataManagerAndroid, ibanValue);
    }

    public BankAccount[] getMaskedBankAccounts() {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get().getMaskedBankAccounts(mPersonalDataManagerAndroid);
    }

    public void addMaskedBankAccountForTest(BankAccount bankAccount) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get()
                .addMaskedBankAccountForTest(mPersonalDataManagerAndroid, bankAccount);
    }

    /**
     * Records the use of the profile associated with the specified {@code guid}. Effectively
     * increments the use count of the profile and sets its use date to the current time. Also logs
     * usage metrics.
     *
     * @param guid The GUID of the profile.
     */
    public void recordAndLogProfileUse(String guid) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get().recordAndLogProfileUse(mPersonalDataManagerAndroid, guid);
    }

    protected void setProfileUseStatsForTesting(String guid, int count, int daysSinceLastUsed) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get()
                .setProfileUseStatsForTesting(
                        mPersonalDataManagerAndroid, guid, count, daysSinceLastUsed);
    }

    int getProfileUseCountForTesting(String guid) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getProfileUseCountForTesting(mPersonalDataManagerAndroid, guid); // IN-TEST
    }

    long getProfileUseDateForTesting(String guid) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getProfileUseDateForTesting(mPersonalDataManagerAndroid, guid); // IN-TEST
    }

    /**
     * Records the use of the credit card associated with the specified {@code guid}. Effectively
     * increments the use count of the credit card and set its use date to the current time. Also
     * logs usage metrics.
     *
     * @param guid The GUID of the credit card.
     */
    public void recordAndLogCreditCardUse(String guid) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get().recordAndLogCreditCardUse(mPersonalDataManagerAndroid, guid);
    }

    protected void setCreditCardUseStatsForTesting(String guid, int count, int daysSinceLastUsed) {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get()
                .setCreditCardUseStatsForTesting(
                        mPersonalDataManagerAndroid, guid, count, daysSinceLastUsed);
    }

    int getCreditCardUseCountForTesting(String guid) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getCreditCardUseCountForTesting(mPersonalDataManagerAndroid, guid); // IN-TEST
    }

    long getCreditCardUseDateForTesting(String guid) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getCreditCardUseDateForTesting(mPersonalDataManagerAndroid, guid); // IN-TEST
    }

    long getCurrentDateForTesting() {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getCurrentDateForTesting(mPersonalDataManagerAndroid); // IN-TEST
    }

    long getDateNDaysAgoForTesting(int days) {
        ThreadUtils.assertOnUiThread();
        return PersonalDataManagerJni.get()
                .getDateNDaysAgoForTesting( // IN-TEST
                        mPersonalDataManagerAndroid, days);
    }

    protected void clearServerDataForTesting() {
        ThreadUtils.assertOnUiThread();
        PersonalDataManagerJni.get()
                .clearServerDataForTesting(mPersonalDataManagerAndroid); // IN-TEST
    }

    protected void clearImageDataForTesting() {
        if (mImageFetcher == null) {
            return;
        }

        ThreadUtils.assertOnUiThread();
        mImageFetcher.clearCachedImagesForTesting();
    }

    /**
     * Determines whether the logged in user (if any) is eligible to store Autofill address profiles
     * to their account.
     */
    public boolean isEligibleForAddressAccountStorage() {
        return PersonalDataManagerJni.get()
                .isEligibleForAddressAccountStorage(mPersonalDataManagerAndroid);
    }

    /** Determines the country code for a newly created address profile. */
    public String getDefaultCountryCodeForNewAddress() {
        return PersonalDataManagerJni.get()
                .getDefaultCountryCodeForNewAddress(mPersonalDataManagerAndroid);
    }

    /**
     * Users based in unsupported countries and profiles with a country value set to an unsupported
     * country are not eligible for account storage. This function determines if the `country_code`
     * is eligible.
     */
    public boolean isCountryEligibleForAccountStorage(String countryCode) {
        return PersonalDataManagerJni.get()
                .isCountryEligibleForAccountStorage(mPersonalDataManagerAndroid, countryCode);
    }

    /**
     * Checks whether the Autofill PersonalDataManager has profiles.
     *
     * @return True If there are profiles.
     */
    public boolean hasProfiles() {
        return PersonalDataManagerJni.get().hasProfiles(mPersonalDataManagerAndroid);
    }

    /**
     * Checks whether the Autofill PersonalDataManager has credit cards.
     *
     * @return True If there are credit cards.
     */
    public boolean hasCreditCards() {
        return PersonalDataManagerJni.get().hasCreditCards(mPersonalDataManagerAndroid);
    }

    /**
     * @return Whether FIDO authentication is available.
     */
    public boolean isFidoAuthenticationAvailable() {
        return isAutofillCreditCardEnabled()
                && PersonalDataManagerJni.get()
                        .isFidoAuthenticationAvailable(mPersonalDataManagerAndroid);
    }

    /**
     * @return Whether the Autofill feature for Profiles (addresses) is enabled.
     */
    public boolean isAutofillProfileEnabled() {
        return mPrefService.getBoolean(Pref.AUTOFILL_PROFILE_ENABLED);
    }

    /**
     * @return Whether the Autofill feature for Credit Cards is enabled.
     */
    public boolean isAutofillCreditCardEnabled() {
        return mPrefService.getBoolean(Pref.AUTOFILL_CREDIT_CARD_ENABLED);
    }

    /**
     * Enables or disables the Autofill feature for Profiles.
     *
     * @param enable True to disable profile Autofill, false otherwise.
     */
    public void setAutofillProfileEnabled(boolean enable) {
        mPrefService.setBoolean(Pref.AUTOFILL_PROFILE_ENABLED, enable);
    }

    /**
     * Enables or disables the Autofill feature for Credit Cards.
     *
     * @param enable True to disable credit card Autofill, false otherwise.
     */
    public void setAutofillCreditCardEnabled(boolean enable) {
        mPrefService.setBoolean(Pref.AUTOFILL_CREDIT_CARD_ENABLED, enable);
    }

    /**
     * @return Whether the Autofill feature for FIDO authentication is enabled.
     */
    public boolean isAutofillCreditCardFidoAuthEnabled() {
        return mPrefService.getBoolean(Pref.AUTOFILL_CREDIT_CARD_FIDO_AUTH_ENABLED);
    }

    /**
     * Enables or disables the Autofill feature for FIDO authentication. We are trying to align this
     * pref with the server's source of truth, but any mismatches between this pref and the server
     * should imply the user's intention to opt in/out.
     *
     * @param enable True to enable credit card FIDO authentication, false otherwise.
     */
    public void setAutofillCreditCardFidoAuthEnabled(boolean enable) {
        mPrefService.setBoolean(Pref.AUTOFILL_CREDIT_CARD_FIDO_AUTH_ENABLED, enable);
    }

    /**
     * @return Whether the Autofill feature for payment methods mandatory reauth is enabled.
     */
    public boolean isPaymentMethodsMandatoryReauthEnabled() {
        return mPrefService.getBoolean(Pref.AUTOFILL_PAYMENT_METHODS_MANDATORY_REAUTH);
    }

    /**
     * Enables or disables the Autofill feature for payment methods mandatory reauth.
     *
     * @param enable True to enable payment methods mandatory reauth, false otherwise.
     */
    public void setAutofillPaymentMethodsMandatoryReauth(boolean enable) {
        mPrefService.setBoolean(Pref.AUTOFILL_PAYMENT_METHODS_MANDATORY_REAUTH, enable);
    }

    /**
     * @return Whether the Autofill feature for payment cvc storage is enabled.
     */
    public boolean isPaymentCvcStorageEnabled() {
        return mPrefService.getBoolean(Pref.AUTOFILL_PAYMENT_CVC_STORAGE);
    }

    /**
     * Enables or disables the Autofill feature for payment cvc storage.
     *
     * @param enable True to enable payment cvc storage, false otherwise.
     */
    public void setAutofillPaymentCvcStorage(boolean enable) {
        mPrefService.setBoolean(Pref.AUTOFILL_PAYMENT_CVC_STORAGE, enable);
    }

    /**
     * @return Whether the Autofill feature is managed.
     */
    public boolean isAutofillManaged() {
        return PersonalDataManagerJni.get().isAutofillManaged(mPersonalDataManagerAndroid);
    }

    /**
     * @return Whether the Autofill feature for Profiles (addresses) is managed.
     */
    public boolean isAutofillProfileManaged() {
        return PersonalDataManagerJni.get().isAutofillProfileManaged(mPersonalDataManagerAndroid);
    }

    /**
     * @return Whether the Autofill feature for Credit Cards is managed.
     */
    public boolean isAutofillCreditCardManaged() {
        return PersonalDataManagerJni.get()
                .isAutofillCreditCardManaged(mPersonalDataManagerAndroid);
    }

    public void setSyncServiceForTesting() {
        PersonalDataManagerJni.get().setSyncServiceForTesting(mPersonalDataManagerAndroid);
    }

    private void fetchCreditCardArtImages() {
        mImageFetcher.prefetchImages(
                getCreditCardsToSuggest().stream()
                        .map(card -> card.getCardArtUrl())
                        .toArray(GURL[]::new));
    }

    /**
     * Return the card art image for the given `customImageUrl`.
     * @param customImageUrl  URL of the image. If the image is available, it is returned, otherwise
     *         it is fetched from this URL.
     * @param cardIconSpecs {@code CardIconSpecs} instance containing the specs for the card icon.
     * @return Bitmap image if found in the local cache, else return an empty object.
     */
    public Optional<Bitmap> getCustomImageForAutofillSuggestionIfAvailable(
            GURL customImageUrl, AutofillUiUtils.CardIconSpecs cardIconSpecs) {
        return mImageFetcher.getImageIfAvailable(customImageUrl, cardIconSpecs);
    }

    /**
     * Returns the {@link AutofillImageFetcher} that is used to download and cache icons for payment
     * methods.
     */
    public AutofillImageFetcher getImageFetcherForTesting() {
        return mImageFetcher;
    }

    public void setImageFetcherForTesting(ImageFetcher imageFetcher) {
        var oldValue = this.mImageFetcher;
        this.mImageFetcher = new AutofillImageFetcher(imageFetcher);
        ResettersForTesting.register(() -> this.mImageFetcher = oldValue);
    }

    /** Sets the preference value for supporting payments using Pix. */
    public void setFacilitatedPaymentsPixPref(boolean value) {
        mPrefService.setBoolean(Pref.FACILITATED_PAYMENTS_PIX, value);
    }

    /** Returns the preference value for supporting payments using Pix. */
    public boolean getFacilitatedPaymentsPixPref() {
        return mPrefService.getBoolean(Pref.FACILITATED_PAYMENTS_PIX);
    }

    @NativeMethods
    interface Natives {
        long init(PersonalDataManager caller, @JniType("Profile*") Profile profile);

        void destroy(long nativePersonalDataManagerAndroid);

        boolean isDataLoaded(long nativePersonalDataManagerAndroid);

        String[] getProfileGUIDsForSettings(long nativePersonalDataManagerAndroid);

        String[] getProfileGUIDsToSuggest(long nativePersonalDataManagerAndroid);

        String[] getProfileLabelsForSettings(long nativePersonalDataManagerAndroid);

        String[] getProfileLabelsToSuggest(
                long nativePersonalDataManagerAndroid,
                boolean includeNameInLabel,
                boolean includeOrganizationInLabel,
                boolean includeCountryInLabel);

        AutofillProfile getProfileByGUID(long nativePersonalDataManagerAndroid, String guid);

        boolean isEligibleForAddressAccountStorage(long nativePersonalDataManagerAndroid);

        String getDefaultCountryCodeForNewAddress(long nativePersonalDataManagerAndroid);

        boolean isCountryEligibleForAccountStorage(
                long nativePersonalDataManagerAndroid, String countryCode);

        String setProfile(
                long nativePersonalDataManagerAndroid, AutofillProfile profile, String guid);

        String setProfileToLocal(
                long nativePersonalDataManagerAndroid, AutofillProfile profile, String guid);

        String getShippingAddressLabelForPaymentRequest(
                long nativePersonalDataManagerAndroid,
                AutofillProfile profile,
                String guid,
                boolean includeCountry);

        String[] getCreditCardGUIDsForSettings(long nativePersonalDataManagerAndroid);

        String[] getCreditCardGUIDsToSuggest(long nativePersonalDataManagerAndroid);

        CreditCard getCreditCardByGUID(long nativePersonalDataManagerAndroid, String guid);

        CreditCard getCreditCardForNumber(long nativePersonalDataManagerAndroid, String cardNumber);

        void deleteAllLocalCreditCards(long nativePersonalDataManagerAndroid);

        String setCreditCard(long nativePersonalDataManagerAndroid, CreditCard card);

        long getDateNDaysAgoForTesting(long nativePersonalDataManagerAndroid, int days); // IN-TEST

        void updateServerCardBillingAddress(long nativePersonalDataManagerAndroid, CreditCard card);

        String getBasicCardIssuerNetwork(String cardNumber, boolean emptyIfInvalid);

        void addServerCreditCardForTest(
                long nativePersonalDataManagerAndroid, CreditCard card); // IN-TEST

        void addServerCreditCardForTestWithAdditionalFields(
                long nativePersonalDataManagerAndroid,
                CreditCard card,
                String nickname,
                int cardIssuer);

        void removeByGUID(long nativePersonalDataManagerAndroid, String guid);

        void recordAndLogProfileUse(long nativePersonalDataManagerAndroid, String guid);

        void setProfileUseStatsForTesting(
                long nativePersonalDataManagerAndroid,
                String guid,
                int count,
                int daysSinceLastUsed);

        int getProfileUseCountForTesting(
                long nativePersonalDataManagerAndroid, String guid); // IN-TEST

        long getProfileUseDateForTesting(
                long nativePersonalDataManagerAndroid, String guid); // IN-TEST

        void recordAndLogCreditCardUse(long nativePersonalDataManagerAndroid, String guid);

        void setCreditCardUseStatsForTesting(
                long nativePersonalDataManagerAndroid,
                String guid,
                int count,
                int daysSinceLastUsed);

        int getCreditCardUseCountForTesting(
                long nativePersonalDataManagerAndroid, String guid); // IN-TEST

        long getCreditCardUseDateForTesting(
                long nativePersonalDataManagerAndroid, String guid); // IN-TEST

        long getCurrentDateForTesting(long nativePersonalDataManagerAndroid); // IN-TEST

        void clearServerDataForTesting(long nativePersonalDataManagerAndroid); // IN-TEST

        boolean hasProfiles(long nativePersonalDataManagerAndroid);

        boolean hasCreditCards(long nativePersonalDataManagerAndroid);

        boolean isFidoAuthenticationAvailable(long nativePersonalDataManagerAndroid);

        boolean isAutofillManaged(long nativePersonalDataManagerAndroid);

        boolean isAutofillProfileManaged(long nativePersonalDataManagerAndroid);

        boolean isAutofillCreditCardManaged(long nativePersonalDataManagerAndroid);

        String toCountryCode(String countryName);

        void setSyncServiceForTesting(long nativePersonalDataManagerAndroid);

        AutofillImageFetcher getOrCreateJavaImageFetcher(long nativePersonalDataManagerAndroid);

        Iban getIbanByGuid(long nativePersonalDataManagerAndroid, String guid);

        Iban[] getLocalIbansForSettings(long nativePersonalDataManagerAndroid);

        String addOrUpdateLocalIban(long nativePersonalDataManagerAndroid, Iban iban);

        boolean isValidIban(long nativePersonalDataManagerAndroid, String ibanValue);

        BankAccount[] getMaskedBankAccounts(long nativePersonalDataManagerAndroid);

        void addMaskedBankAccountForTest(
                long nativePersonalDataManagerAndroid, BankAccount bankAccount); // IN-TEST
    }
}