| // Copyright 2013 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. |
| |
| #include <vector> |
| |
| #include "base/files/file_enumerator.h" |
| #include "base/files/file_path.h" |
| #include "base/mac/foundation_util.h" |
| #include "base/memory/ptr_util.h" |
| #include "base/path_service.h" |
| #include "base/strings/string_util.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "base/task_scheduler/task_scheduler.h" |
| #import "base/test/ios/wait_util.h" |
| #include "base/test/scoped_feature_list.h" |
| #include "components/autofill/core/browser/autofill_manager.h" |
| #include "components/autofill/core/browser/data_driven_test.h" |
| #include "components/autofill/core/common/autofill_features.h" |
| #import "components/autofill/ios/browser/autofill_agent.h" |
| #include "components/autofill/ios/browser/autofill_driver_ios.h" |
| #import "ios/chrome/browser/autofill/autofill_controller.h" |
| #import "ios/chrome/browser/autofill/form_suggestion_controller.h" |
| #include "ios/chrome/browser/browser_state/test_chrome_browser_state.h" |
| #include "ios/chrome/browser/chrome_paths.h" |
| #include "ios/chrome/browser/infobars/infobar_manager_impl.h" |
| #include "ios/chrome/browser/web/chrome_web_client.h" |
| #import "ios/chrome/browser/web/chrome_web_test.h" |
| #import "ios/web/public/web_state/web_state.h" |
| |
| #if !defined(__has_feature) || !__has_feature(objc_arc) |
| #error "This file requires ARC support." |
| #endif |
| |
| namespace autofill { |
| |
| namespace { |
| |
| const base::FilePath::CharType kTestName[] = FILE_PATH_LITERAL("heuristics"); |
| |
| const base::FilePath& GetTestDataDir() { |
| CR_DEFINE_STATIC_LOCAL(base::FilePath, dir, ()); |
| if (dir.empty()) |
| PathService::Get(ios::DIR_TEST_DATA, &dir); |
| return dir; |
| } |
| |
| base::FilePath GetIOSInputDirectory() { |
| base::FilePath dir; |
| CHECK(PathService::Get(base::DIR_SOURCE_ROOT, &dir)); |
| |
| return dir.AppendASCII("components") |
| .AppendASCII("test") |
| .AppendASCII("data") |
| .AppendASCII("autofill") |
| .Append(kTestName) |
| .AppendASCII("input"); |
| } |
| |
| base::FilePath GetIOSOutputDirectory() { |
| base::FilePath dir; |
| CHECK(PathService::Get(base::DIR_SOURCE_ROOT, &dir)); |
| |
| return dir.AppendASCII("components") |
| .AppendASCII("test") |
| .AppendASCII("data") |
| .AppendASCII("autofill") |
| .Append(kTestName) |
| .AppendASCII("output"); |
| } |
| |
| const std::vector<base::FilePath> GetTestFiles() { |
| base::FilePath dir(GetIOSInputDirectory()); |
| base::FileEnumerator input_files(dir, false, base::FileEnumerator::FILES); |
| std::vector<base::FilePath> files; |
| for (base::FilePath input_file = input_files.Next(); !input_file.empty(); |
| input_file = input_files.Next()) { |
| files.push_back(input_file); |
| } |
| std::sort(files.begin(), files.end()); |
| |
| base::mac::ClearAmIBundledCache(); |
| return files; |
| } |
| |
| const std::set<std::string>& GetFailingTestNames() { |
| static std::set<std::string>* failing_test_names = |
| new std::set<std::string>{}; |
| return *failing_test_names; |
| } |
| |
| } // namespace |
| |
| // Test fixture for verifying Autofill heuristics. Each input is an HTML |
| // file that contains one or more forms. The corresponding output file lists the |
| // heuristically detected type for each field. |
| // This is based on FormStructureBrowserTest from the Chromium Project. |
| // TODO(crbug.com/245246): Unify the tests. |
| class FormStructureBrowserTest |
| : public ChromeWebTest, |
| public DataDrivenTest, |
| public ::testing::WithParamInterface<base::FilePath> { |
| protected: |
| FormStructureBrowserTest(); |
| ~FormStructureBrowserTest() override {} |
| |
| void SetUp() override; |
| void TearDown() override; |
| |
| // DataDrivenTest: |
| void GenerateResults(const std::string& input, std::string* output) override; |
| |
| // Serializes the given |forms| into a string. |
| std::string FormStructuresToString( |
| const std::vector<std::unique_ptr<FormStructure>>& forms); |
| |
| FormSuggestionController* suggestionController_; |
| AutofillController* autofillController_; |
| |
| private: |
| base::test::ScopedFeatureList feature_list_; |
| DISALLOW_COPY_AND_ASSIGN(FormStructureBrowserTest); |
| }; |
| |
| FormStructureBrowserTest::FormStructureBrowserTest() |
| : ChromeWebTest(std::make_unique<ChromeWebClient>()), |
| DataDrivenTest(GetTestDataDir()) {} |
| |
| void FormStructureBrowserTest::SetUp() { |
| ChromeWebTest::SetUp(); |
| feature_list_.InitAndDisableFeature( |
| autofill::features::kAutofillEnforceMinRequiredFieldsForUpload); |
| |
| InfoBarManagerImpl::CreateForWebState(web_state()); |
| AutofillAgent* autofillAgent = [[AutofillAgent alloc] |
| initWithPrefService:chrome_browser_state_->GetPrefs() |
| webState:web_state()]; |
| suggestionController_ = |
| [[FormSuggestionController alloc] initWithWebState:web_state() |
| providers:@[ autofillAgent ]]; |
| autofillController_ = [[AutofillController alloc] |
| initWithBrowserState:chrome_browser_state_.get() |
| webState:web_state() |
| autofillAgent:autofillAgent |
| passwordGenerationManager:nullptr |
| downloadEnabled:NO]; |
| } |
| |
| void FormStructureBrowserTest::TearDown() { |
| [autofillController_ detachFromWebState]; |
| |
| // TODO(crbug.com/776330): remove this manual sync. |
| // This is a workaround to manually sync the tasks posted by |
| // |CRWCertVerificationController verifyTrust:completionHandler:|. |
| // |WaitForBackgroundTasks| currently fails to wait for completion of them. |
| base::test::ios::SpinRunLoopWithMinDelay(base::TimeDelta::FromSecondsD(0.1)); |
| ChromeWebTest::TearDown(); |
| } |
| |
| void FormStructureBrowserTest::GenerateResults(const std::string& input, |
| std::string* output) { |
| LoadHtml(input); |
| base::TaskScheduler::GetInstance()->FlushForTesting(); |
| AutofillManager* autofill_manager = |
| AutofillDriverIOS::FromWebState(web_state())->autofill_manager(); |
| ASSERT_NE(nullptr, autofill_manager); |
| const std::vector<std::unique_ptr<FormStructure>>& forms = |
| autofill_manager->form_structures_; |
| *output = FormStructureBrowserTest::FormStructuresToString(forms); |
| } |
| |
| std::string FormStructureBrowserTest::FormStructuresToString( |
| const std::vector<std::unique_ptr<FormStructure>>& forms) { |
| std::string forms_string; |
| for (const std::unique_ptr<FormStructure>& form : forms) { |
| for (const auto& field : *form) { |
| std::string name = base::UTF16ToUTF8(field->name); |
| if (base::StartsWith(name, "gChrome~field~", |
| base::CompareCase::SENSITIVE)) { |
| // The name has been generated by iOS JavaScript. Output an empty name |
| // to have a behavior similar to other platforms. |
| name = ""; |
| } |
| std::string section = field->section(); |
| if (base::StartsWith(section, "gChrome~field~", |
| base::CompareCase::SENSITIVE)) { |
| // The name has been generated by iOS JavaScript. Output an empty name |
| // to have a behavior similar to other platforms. |
| size_t first_underscore = section.find_first_of('_'); |
| section = section.substr(first_underscore); |
| } |
| forms_string += field->Type().ToString(); |
| forms_string += " | " + name; |
| forms_string += " | " + base::UTF16ToUTF8(field->label); |
| forms_string += " | " + base::UTF16ToUTF8(field->value); |
| forms_string += " | " + section; |
| forms_string += "\n"; |
| } |
| } |
| return forms_string; |
| } |
| |
| TEST_P(FormStructureBrowserTest, DataDrivenHeuristics) { |
| bool is_expected_to_pass = |
| !base::ContainsKey(GetFailingTestNames(), GetParam().BaseName().value()); |
| RunOneDataDrivenTest(GetParam(), GetIOSOutputDirectory(), |
| is_expected_to_pass); |
| } |
| |
| INSTANTIATE_TEST_CASE_P(AllForms, |
| FormStructureBrowserTest, |
| testing::ValuesIn(GetTestFiles())); |
| |
| } // namespace autofill |