| // Copyright 2019 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #import "ios/chrome/browser/ui/autofill/autofill_app_interface.h" |
| |
| #include "base/memory/singleton.h" |
| #include "base/strings/sys_string_conversions.h" |
| #include "base/strings/utf_string_conversions.h" |
| #import "base/test/ios/wait_util.h" |
| #include "components/autofill/core/browser/autofill_test_utils.h" |
| #include "components/autofill/core/browser/form_data_importer.h" |
| #include "components/autofill/core/browser/payments/credit_card_save_manager.h" |
| #include "components/autofill/core/browser/personal_data_manager.h" |
| #include "components/autofill/core/common/autofill_prefs.h" |
| #include "components/autofill/ios/browser/autofill_driver_ios.h" |
| #import "components/autofill/ios/browser/credit_card_save_manager_test_observer_bridge.h" |
| #include "components/autofill/ios/browser/ios_test_event_waiter.h" |
| #include "components/keyed_service/core/service_access_type.h" |
| #include "components/password_manager/core/browser/password_store_consumer.h" |
| #include "components/password_manager/core/browser/password_store_interface.h" |
| #include "ios/chrome/browser/application_context.h" |
| #include "ios/chrome/browser/autofill/personal_data_manager_factory.h" |
| #include "ios/chrome/browser/browser_state/chrome_browser_state.h" |
| #include "ios/chrome/browser/passwords/ios_chrome_password_store_factory.h" |
| #import "ios/chrome/test/app/chrome_test_util.h" |
| #import "ios/chrome/test/app/tab_test_util.h" |
| #import "ios/public/provider/chrome/browser/risk_data/risk_data_api.h" |
| #import "ios/web/public/js_messaging/web_frames_manager.h" |
| #import "ios/web/public/web_state.h" |
| #include "services/network/public/cpp/weak_wrapper_shared_url_loader_factory.h" |
| #include "services/network/test/test_url_loader_factory.h" |
| |
| #if !defined(__has_feature) || !__has_feature(objc_arc) |
| #error "This file requires ARC support." |
| #endif |
| |
| namespace { |
| |
| const char16_t kExampleUsername[] = u"concrete username"; |
| const char16_t kExamplePassword[] = u"concrete password"; |
| |
| // Gets the current password store. |
| scoped_refptr<password_manager::PasswordStoreInterface> GetPasswordStore() { |
| // ServiceAccessType governs behaviour in Incognito: only modifications with |
| // EXPLICIT_ACCESS, which correspond to user's explicit gesture, succeed. |
| // This test does not deal with Incognito, and should not run in Incognito |
| // context. Therefore IMPLICIT_ACCESS is used to let the test fail if in |
| // Incognito context. |
| return IOSChromePasswordStoreFactory::GetInterfaceForBrowserState( |
| chrome_test_util::GetOriginalBrowserState(), |
| ServiceAccessType::IMPLICIT_ACCESS); |
| } |
| |
| // This class is used to obtain results from the PasswordStore and hence both |
| // check the success of store updates and ensure that store has finished |
| // processing. |
| class TestStoreConsumer : public password_manager::PasswordStoreConsumer { |
| public: |
| void OnGetPasswordStoreResults( |
| std::vector<std::unique_ptr<password_manager::PasswordForm>> obtained) |
| override { |
| obtained_ = std::move(obtained); |
| } |
| |
| const std::vector<password_manager::PasswordForm>& GetStoreResults() { |
| results_.clear(); |
| ResetObtained(); |
| GetPasswordStore()->GetAllLogins(this); |
| #pragma clang diagnostic push |
| #pragma clang diagnostic ignored "-Wunused-result" |
| base::test::ios::WaitUntilConditionOrTimeout( |
| base::test::ios::kWaitForFileOperationTimeout, ^bool { |
| return !AreObtainedReset(); |
| }); |
| #pragma clang diagnostic pop |
| AppendObtainedToResults(); |
| return results_; |
| } |
| |
| private: |
| // Puts |obtained_| in a known state not corresponding to any PasswordStore |
| // state. |
| void ResetObtained() { |
| obtained_.clear(); |
| obtained_.emplace_back(nullptr); |
| } |
| |
| // Returns true if |obtained_| are in the reset state. |
| bool AreObtainedReset() { return obtained_.size() == 1 && !obtained_[0]; } |
| |
| void AppendObtainedToResults() { |
| for (const auto& source : obtained_) { |
| results_.emplace_back(*source); |
| } |
| ResetObtained(); |
| } |
| |
| // Temporary cache of obtained store results. |
| std::vector<std::unique_ptr<password_manager::PasswordForm>> obtained_; |
| |
| // Combination of fillable and blocked credentials from the store. |
| std::vector<password_manager::PasswordForm> results_; |
| }; |
| |
| // Saves |form| to the password store and waits until the async processing is |
| // done. |
| void SaveToPasswordStore(const password_manager::PasswordForm& form) { |
| GetPasswordStore()->AddLogin(form); |
| // When we retrieve the form from the store, |in_store| should be set. |
| password_manager::PasswordForm expected_form = form; |
| expected_form.in_store = password_manager::PasswordForm::Store::kProfileStore; |
| // TODO(crbug.com/1223022): Once all places that operate changes on forms |
| // via UpdateLogin properly set |password_issues|, setting them to an empty |
| // map should be part of the default constructor. |
| expected_form.password_issues = |
| base::flat_map<password_manager::InsecureType, |
| password_manager::InsecurityMetadata>(); |
| // Check the result and ensure PasswordStore processed this. |
| TestStoreConsumer consumer; |
| for (const auto& result : consumer.GetStoreResults()) { |
| if (result == expected_form) |
| return; |
| } |
| } |
| |
| // Saves an example form in the store. |
| void SaveExamplePasswordForm() { |
| password_manager::PasswordForm example; |
| example.username_value = kExampleUsername; |
| example.password_value = kExamplePassword; |
| example.url = GURL("https://example.com/"); |
| example.signon_realm = example.url.spec(); |
| SaveToPasswordStore(example); |
| } |
| |
| // Saves an example form in the store for the passed URL. |
| void SaveLocalPasswordForm(const GURL& url) { |
| password_manager::PasswordForm localForm; |
| localForm.username_value = kExampleUsername; |
| localForm.password_value = kExamplePassword; |
| localForm.url = url; |
| localForm.signon_realm = localForm.url.spec(); |
| SaveToPasswordStore(localForm); |
| } |
| |
| // Removes all credentials stored. |
| void ClearPasswordStore() { |
| GetPasswordStore()->RemoveLoginsCreatedBetween(base::Time(), base::Time(), |
| base::DoNothing()); |
| TestStoreConsumer consumer; |
| } |
| |
| // Saves an example profile in the store. |
| void AddAutofillProfile(autofill::PersonalDataManager* personalDataManager) { |
| autofill::AutofillProfile profile = autofill::test::GetFullProfile(); |
| // If the test profile is already in the store, adding it will be a no-op. |
| // In that case, early return. |
| for (autofill::AutofillProfile* p : personalDataManager->GetProfiles()) { |
| if (p->Compare(profile) == 0) { |
| return; |
| } |
| } |
| size_t profileCount = personalDataManager->GetProfiles().size(); |
| |
| personalDataManager->AddProfile(profile); |
| |
| ConditionBlock conditionBlock = ^bool { |
| return profileCount < personalDataManager->GetProfiles().size(); |
| }; |
| base::test::ios::TimeUntilCondition( |
| nil, conditionBlock, false, |
| base::Seconds(base::test::ios::kWaitForActionTimeout)); |
| } |
| |
| } // namespace |
| |
| namespace autofill { |
| |
| // Helper class that provides access to private members of |
| // BrowserAutofillManager, FormDataImporter and CreditCardSaveManager. This |
| // class is friend with some autofill internal classes to access private fields. |
| class SaveCardInfobarEGTestHelper |
| : public CreditCardSaveManager::ObserverForTest { |
| public: |
| static SaveCardInfobarEGTestHelper* SharedInstance() { |
| return base::Singleton<SaveCardInfobarEGTestHelper>::get(); |
| } |
| |
| SaveCardInfobarEGTestHelper() {} |
| ~SaveCardInfobarEGTestHelper() override {} |
| SaveCardInfobarEGTestHelper(const SaveCardInfobarEGTestHelper&) = delete; |
| SaveCardInfobarEGTestHelper& operator=(const SaveCardInfobarEGTestHelper&) = |
| delete; |
| |
| // Access the CreditCardSaveManager. |
| static CreditCardSaveManager* GetCreditCardSaveManager() { |
| web::WebState* web_state = chrome_test_util::GetCurrentWebState(); |
| web::WebFrame* main_frame = |
| web_state->GetWebFramesManager()->GetMainWebFrame(); |
| return AutofillDriverIOS::FromWebStateAndWebFrame(web_state, main_frame) |
| ->autofill_manager() |
| ->client() |
| ->GetFormDataImporter() |
| ->credit_card_save_manager_.get(); |
| } |
| |
| // Access the PaymentsClient. |
| static payments::PaymentsClient* GetPaymentsClient() { |
| web::WebState* web_state = chrome_test_util::GetCurrentWebState(); |
| web::WebFrame* main_frame = |
| web_state->GetWebFramesManager()->GetMainWebFrame(); |
| DCHECK(web_state); |
| return AutofillDriverIOS::FromWebStateAndWebFrame(web_state, main_frame) |
| ->autofill_manager() |
| ->client() |
| ->GetPaymentsClient(); |
| } |
| |
| // Delete all failed attempds registered on every cards. |
| static void ClearCreditCardSaveStrikes() { |
| GetCreditCardSaveManager() |
| ->GetCreditCardSaveStrikeDatabase() |
| ->ClearAllStrikes(); |
| } |
| |
| // Set the number of failed attempds registered on a card. |
| static void SetFormFillMaxStrikes(NSString* card_key, int max) { |
| // The strike key is made of CreditCardSaveStrikeDatabase's project prefix |
| // and the last 4 digits of the card used in fillAndSubmitForm(), which can |
| // be found in credit_card_upload_form_address_and_cc.html. |
| GetCreditCardSaveManager() |
| ->GetCreditCardSaveStrikeDatabase() |
| ->strike_database_->SetStrikeData(base::SysNSStringToUTF8(card_key), |
| max); |
| } |
| |
| // Reset the IOSTestEventWaiter and make it watch |events|. |
| void ResetEventWaiterForEvents(NSArray* events, NSTimeInterval timeout) { |
| std::list<CreditCardSaveManagerObserverEvent> events_list; |
| for (NSNumber* e : events) { |
| events_list.push_back( |
| static_cast<CreditCardSaveManagerObserverEvent>([e intValue])); |
| } |
| event_waiter_ = std::make_unique< |
| IOSTestEventWaiter<CreditCardSaveManagerObserverEvent>>( |
| std::move(events_list), timeout); |
| } |
| |
| void OnOfferLocalSave() override { |
| OnEvent(CreditCardSaveManagerObserverEvent::kOnOfferLocalSaveCalled); |
| } |
| |
| void OnDecideToRequestUploadSave() override { |
| OnEvent( |
| CreditCardSaveManagerObserverEvent::kOnDecideToRequestUploadSaveCalled); |
| } |
| |
| void OnReceivedGetUploadDetailsResponse() override { |
| OnEvent(CreditCardSaveManagerObserverEvent:: |
| kOnReceivedGetUploadDetailsResponseCalled); |
| } |
| |
| void OnSentUploadCardRequest() override { |
| OnEvent(CreditCardSaveManagerObserverEvent::kOnSentUploadCardRequestCalled); |
| } |
| |
| void OnReceivedUploadCardResponse() override { |
| OnEvent(CreditCardSaveManagerObserverEvent:: |
| kOnReceivedUploadCardResponseCalled); |
| } |
| |
| void OnShowCardSavedFeedback() override { |
| OnEvent(CreditCardSaveManagerObserverEvent::kOnShowCardSavedFeedbackCalled); |
| } |
| |
| void OnStrikeChangeComplete() override { |
| OnEvent(CreditCardSaveManagerObserverEvent::kOnStrikeChangeCompleteCalled); |
| } |
| |
| // Triggers |event| on the IOSTestEventWaiter. |
| bool OnEvent(CreditCardSaveManagerObserverEvent event) { |
| return event_waiter_->OnEvent(event); |
| } |
| |
| // Waits until the expected events are triggered. |
| bool WaitForEvents() { return event_waiter_->Wait(); } |
| |
| // Sets the response to payment server requests. |
| void SetPaymentsResponse(NSString* request, NSString* response, int error) { |
| test_url_loader_factory_->AddResponse( |
| base::SysNSStringToUTF8(request), base::SysNSStringToUTF8(response), |
| static_cast<net::HttpStatusCode>(error)); |
| } |
| |
| void SetPaymentsRiskData(const std::string& risk_data) { |
| GetCreditCardSaveManager()->OnDidGetUploadRiskData(risk_data); |
| } |
| |
| void SetUp() { |
| test_url_loader_factory_ = |
| std::make_unique<network::TestURLLoaderFactory>(); |
| // Set up the URL loader factory for the PaymentsClient so we can intercept |
| // those network requests. |
| shared_url_loader_factory_ = |
| base::MakeRefCounted<network::WeakWrapperSharedURLLoaderFactory>( |
| test_url_loader_factory_.get()); |
| |
| payments::PaymentsClient* payments_client = |
| SaveCardInfobarEGTestHelper::GetPaymentsClient(); |
| payments_client->set_url_loader_factory_for_testing( |
| shared_url_loader_factory_); |
| |
| // Observe actions in CreditCardSaveManager. |
| CreditCardSaveManager* credit_card_save_manager = |
| SaveCardInfobarEGTestHelper::GetCreditCardSaveManager(); |
| credit_card_save_manager->SetEventObserverForTesting(this); |
| } |
| |
| void TearDown() { |
| ClearCreditCardSaveStrikes(); |
| CreditCardSaveManager* credit_card_save_manager = |
| SaveCardInfobarEGTestHelper::GetCreditCardSaveManager(); |
| credit_card_save_manager->SetEventObserverForTesting(nullptr); |
| event_waiter_.reset(); |
| shared_url_loader_factory_.reset(); |
| test_url_loader_factory_.reset(); |
| } |
| |
| private: |
| std::unique_ptr<network::TestURLLoaderFactory> test_url_loader_factory_; |
| scoped_refptr<network::SharedURLLoaderFactory> shared_url_loader_factory_; |
| std::unique_ptr<IOSTestEventWaiter<CreditCardSaveManagerObserverEvent>> |
| event_waiter_; |
| }; |
| } |
| |
| @implementation AutofillAppInterface |
| |
| + (void)clearPasswordStore { |
| ClearPasswordStore(); |
| } |
| |
| + (void)saveExamplePasswordForm { |
| SaveExamplePasswordForm(); |
| } |
| |
| + (void)savePasswordFormForURLSpec:(NSString*)URLSpec { |
| SaveLocalPasswordForm(GURL(base::SysNSStringToUTF8(URLSpec))); |
| } |
| |
| + (NSInteger)profilesCount { |
| autofill::PersonalDataManager* personalDataManager = |
| [self personalDataManager]; |
| return personalDataManager->GetProfiles().size(); |
| } |
| |
| + (void)setAutoAcceptAddressImports:(BOOL)autoAccept { |
| autofill::PersonalDataManager* personalDataManager = |
| [self personalDataManager]; |
| return personalDataManager->set_auto_accept_address_imports_for_testing( |
| autoAccept); |
| } |
| |
| + (void)clearProfilesStore { |
| ChromeBrowserState* browserState = |
| chrome_test_util::GetOriginalBrowserState(); |
| autofill::PersonalDataManager* personalDataManager = |
| autofill::PersonalDataManagerFactory::GetForBrowserState(browserState); |
| for (const auto* profile : personalDataManager->GetProfiles()) { |
| personalDataManager->RemoveByGUID(profile->guid()); |
| } |
| |
| ConditionBlock conditionBlock = ^bool { |
| return 0 == personalDataManager->GetProfiles().size(); |
| }; |
| base::test::ios::TimeUntilCondition( |
| nil, conditionBlock, false, |
| base::Seconds(base::test::ios::kWaitForActionTimeout)); |
| |
| autofill::prefs::SetAutofillProfileEnabled(browserState->GetPrefs(), YES); |
| } |
| |
| + (void)saveExampleProfile { |
| AddAutofillProfile([self personalDataManager]); |
| } |
| |
| + (NSString*)exampleProfileName { |
| autofill::AutofillProfile profile = autofill::test::GetFullProfile(); |
| std::u16string name = |
| profile.GetInfo(autofill::AutofillType(autofill::NAME_FULL), |
| GetApplicationContext()->GetApplicationLocale()); |
| return base::SysUTF16ToNSString(name); |
| } |
| |
| + (void)clearCreditCardStore { |
| autofill::PersonalDataManager* personalDataManager = |
| [self personalDataManager]; |
| for (const auto* creditCard : personalDataManager->GetCreditCards()) { |
| personalDataManager->RemoveByGUID(creditCard->guid()); |
| } |
| |
| ChromeBrowserState* browserState = |
| chrome_test_util::GetOriginalBrowserState(); |
| autofill::prefs::SetAutofillCreditCardEnabled(browserState->GetPrefs(), YES); |
| } |
| |
| + (NSString*)saveLocalCreditCard { |
| autofill::PersonalDataManager* personalDataManager = |
| [self personalDataManager]; |
| autofill::CreditCard card = autofill::test::GetCreditCard(); |
| size_t card_count = personalDataManager->GetCreditCards().size(); |
| personalDataManager->AddCreditCard(card); |
| ConditionBlock conditionBlock = ^bool { |
| return card_count < personalDataManager->GetCreditCards().size(); |
| }; |
| base::test::ios::TimeUntilCondition( |
| nil, conditionBlock, false, |
| base::Seconds(base::test::ios::kWaitForFileOperationTimeout)); |
| personalDataManager->NotifyPersonalDataObserver(); |
| return base::SysUTF16ToNSString(card.NetworkAndLastFourDigits()); |
| } |
| |
| + (NSInteger)localCreditCount { |
| return [self personalDataManager] -> GetCreditCards().size(); |
| } |
| |
| + (void)saveMaskedCreditCard { |
| autofill::PersonalDataManager* personalDataManager = |
| [self personalDataManager]; |
| autofill::CreditCard card = autofill::test::GetMaskedServerCard(); |
| DCHECK(card.record_type() != autofill::CreditCard::LOCAL_CARD); |
| |
| personalDataManager->AddServerCreditCardForTest( |
| std::make_unique<autofill::CreditCard>(card)); |
| personalDataManager->NotifyPersonalDataObserver(); |
| } |
| |
| + (void)setUpSaveCardInfobarEGTestHelper { |
| autofill::SaveCardInfobarEGTestHelper::SharedInstance()->SetUp(); |
| } |
| |
| + (void)tearDownSaveCardInfobarEGTestHelper { |
| autofill::SaveCardInfobarEGTestHelper::SharedInstance()->TearDown(); |
| } |
| |
| + (void)resetEventWaiterForEvents:(NSArray*)events |
| timeout:(NSTimeInterval)timeout { |
| autofill::SaveCardInfobarEGTestHelper::SharedInstance() |
| ->ResetEventWaiterForEvents(events, timeout); |
| } |
| |
| + (BOOL)waitForEvents { |
| return autofill::SaveCardInfobarEGTestHelper::SharedInstance() |
| ->WaitForEvents(); |
| } |
| |
| + (void)setPaymentsResponse:(NSString*)response |
| forRequest:(NSString*)request |
| withErrorCode:(int)error { |
| return autofill::SaveCardInfobarEGTestHelper::SharedInstance() |
| ->SetPaymentsResponse(request, response, error); |
| } |
| |
| + (void)setFormFillMaxStrikes:(int)max forCard:(NSString*)card { |
| return autofill::SaveCardInfobarEGTestHelper::SharedInstance() |
| ->SetFormFillMaxStrikes(card, max); |
| } |
| |
| + (void)setPaymentsRiskData:(NSString*)riskData { |
| return autofill::SaveCardInfobarEGTestHelper::SharedInstance() |
| ->SetPaymentsRiskData(base::SysNSStringToUTF8(riskData)); |
| } |
| |
| + (NSString*)paymentsRiskData { |
| return ios::provider::GetRiskData(); |
| } |
| |
| #pragma mark - Private |
| |
| // The PersonalDataManager instance for the current browser state. |
| + (autofill::PersonalDataManager*)personalDataManager { |
| ChromeBrowserState* browserState = |
| chrome_test_util::GetOriginalBrowserState(); |
| autofill::PersonalDataManager* personalDataManager = |
| autofill::PersonalDataManagerFactory::GetForBrowserState(browserState); |
| personalDataManager->SetSyncingForTest(true); |
| return personalDataManager; |
| } |
| |
| @end |