// Copyright (c) 2012 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 "base/utf_string_conversions.h" #include "chrome/test/base/chrome_render_view_test.h" #include "components/autofill/common/autofill_messages.h" #include "components/autofill/common/form_data.h" #include "components/autofill/common/form_field_data.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/WebKit/Source/Platform/chromium/public/WebString.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebInputElement.h" using WebKit::WebDocument; using WebKit::WebFrame; using WebKit::WebInputElement; using WebKit::WebString; namespace autofill { TEST_F(ChromeRenderViewTest, SendForms) { // Don't want any delay for form state sync changes. This will still post a // message so updates will get coalesced, but as soon as we spin the message // loop, it will generate an update. SendContentStateImmediately(); LoadHTML("
" " " " " " " " " " " "
"); // Verify that "FormsSeen" sends the expected number of fields. const IPC::Message* message = render_thread_->sink().GetFirstMessageMatching( AutofillHostMsg_FormsSeen::ID); ASSERT_NE(static_cast(NULL), message); AutofillHostMsg_FormsSeen::Param params; AutofillHostMsg_FormsSeen::Read(message, ¶ms); const std::vector& forms = params.a; ASSERT_EQ(1UL, forms.size()); ASSERT_EQ(4UL, forms[0].fields.size()); FormFieldData expected; expected.name = ASCIIToUTF16("firstname"); expected.value = string16(); expected.form_control_type = "text"; expected.max_length = WebInputElement::defaultMaxLength(); EXPECT_FORM_FIELD_DATA_EQUALS(expected, forms[0].fields[0]); expected.name = ASCIIToUTF16("middlename"); expected.value = string16(); expected.form_control_type = "text"; expected.max_length = WebInputElement::defaultMaxLength(); EXPECT_FORM_FIELD_DATA_EQUALS(expected, forms[0].fields[1]); expected.name = ASCIIToUTF16("lastname"); expected.value = string16(); expected.form_control_type = "text"; expected.autocomplete_attribute = "off"; expected.max_length = WebInputElement::defaultMaxLength(); EXPECT_FORM_FIELD_DATA_EQUALS(expected, forms[0].fields[2]); expected.autocomplete_attribute = std::string(); // reset expected.name = ASCIIToUTF16("state"); expected.value = ASCIIToUTF16("?"); expected.form_control_type = "select-one"; expected.max_length = 0; EXPECT_FORM_FIELD_DATA_EQUALS(expected, forms[0].fields[3]); // Verify that |didAcceptAutofillSuggestion()| sends the expected number of // fields. WebFrame* web_frame = GetMainFrame(); WebDocument document = web_frame->document(); WebInputElement firstname = document.getElementById("firstname").to(); // Make sure to query for Autofill suggestions before selecting one. autofill_agent_->element_ = firstname; autofill_agent_->QueryAutofillSuggestions(firstname, false); // Accept suggestion that contains a label. Labeled items indicate Autofill // as opposed to Autocomplete. We're testing this distinction below with // the |AutofillHostMsg_FillAutofillFormData::ID| message. autofill_agent_->didAcceptAutofillSuggestion( firstname, WebKit::WebString::fromUTF8("Johnny"), WebKit::WebString::fromUTF8("Home"), 1, -1); ProcessPendingMessages(); const IPC::Message* message2 = render_thread_->sink().GetUniqueMessageMatching( AutofillHostMsg_FillAutofillFormData::ID); ASSERT_NE(static_cast(NULL), message2); AutofillHostMsg_FillAutofillFormData::Param params2; AutofillHostMsg_FillAutofillFormData::Read(message2, ¶ms2); const FormData& form2 = params2.b; ASSERT_EQ(3UL, form2.fields.size()); expected.name = ASCIIToUTF16("firstname"); expected.value = string16(); expected.form_control_type = "text"; expected.max_length = WebInputElement::defaultMaxLength(); EXPECT_FORM_FIELD_DATA_EQUALS(expected, form2.fields[0]); expected.name = ASCIIToUTF16("middlename"); expected.value = string16(); expected.form_control_type = "text"; expected.max_length = WebInputElement::defaultMaxLength(); EXPECT_FORM_FIELD_DATA_EQUALS(expected, form2.fields[1]); expected.name = ASCIIToUTF16("state"); expected.value = ASCIIToUTF16("?"); expected.form_control_type = "select-one"; expected.max_length = 0; EXPECT_FORM_FIELD_DATA_EQUALS(expected, form2.fields[2]); } TEST_F(ChromeRenderViewTest, SendDynamicForms) { // Don't want any delay for form state sync changes. This will still post a // message so updates will get coalesced, but as soon as we spin the message // loop, it will generate an update. SendContentStateImmediately(); LoadHTML("
" " " " " " " " " " " "
"); // Verify that "FormsSeen" sends the expected number of fields. const IPC::Message* message = render_thread_->sink().GetFirstMessageMatching( AutofillHostMsg_FormsSeen::ID); ASSERT_NE(static_cast(NULL), message); AutofillHostMsg_FormsSeen::Param params; AutofillHostMsg_FormsSeen::Read(message, ¶ms); const std::vector& forms = params.a; ASSERT_EQ(1UL, forms.size()); ASSERT_EQ(4UL, forms[0].fields.size()); autofill_agent_->OnAutocheckoutSupported(); render_thread_->sink().ClearMessages(); ExecuteJavaScript("var newInput=document.createElement(\"input\");" "newInput.setAttribute(\"type\",\"text\");" "newInput.setAttribute(\"id\", \"telephone\");" "document.getElementById(\"testform\")" ".appendChild(newInput);"); msg_loop_.RunUntilIdle(); // Verify that FormsSeen is present with the new field. const IPC::Message* message2 = render_thread_->sink().GetFirstMessageMatching( AutofillHostMsg_FormsSeen::ID); ASSERT_NE(static_cast(NULL), message2); AutofillHostMsg_FormsSeen::Read(message2, ¶ms); const std::vector& new_forms = params.a; ASSERT_EQ(1UL, new_forms.size()); ASSERT_EQ(5UL, new_forms[0].fields.size()); FormFieldData expected; expected.name = ASCIIToUTF16("telephone"); expected.value = string16(); expected.form_control_type = "text"; expected.max_length = WebInputElement::defaultMaxLength(); EXPECT_FORM_FIELD_DATA_EQUALS(expected, forms[0].fields[4]); } TEST_F(ChromeRenderViewTest, FillFormElement) { // Don't want any delay for form state sync changes. This will still post a // message so updates will get coalesced, but as soon as we spin the message // loop, it will generate an update. SendContentStateImmediately(); LoadHTML("
" " " " " "
"); // Verify that "FormsSeen" isn't sent, as there are too few fields. const IPC::Message* message = render_thread_->sink().GetFirstMessageMatching( AutofillHostMsg_FormsSeen::ID); ASSERT_NE(static_cast(NULL), message); AutofillHostMsg_FormsSeen::Param params; AutofillHostMsg_FormsSeen::Read(message, ¶ms); const std::vector& forms = params.a; ASSERT_EQ(0UL, forms.size()); // Verify that |didAcceptAutofillSuggestion()| sets the value of the expected // field. WebFrame* web_frame = GetMainFrame(); WebDocument document = web_frame->document(); WebInputElement firstname = document.getElementById("firstname").to(); WebInputElement middlename = document.getElementById("middlename").to(); middlename.setAutofilled(true); // Make sure to query for Autofill suggestions before selecting one. autofill_agent_->element_ = firstname; autofill_agent_->QueryAutofillSuggestions(firstname, false); // Accept a suggestion in a form that has been auto-filled. This triggers // the direct filling of the firstname element with value parameter. autofill_agent_->didAcceptAutofillSuggestion(firstname, WebString::fromUTF8("David"), WebString(), 0, 0); ProcessPendingMessages(); const IPC::Message* message2 = render_thread_->sink().GetUniqueMessageMatching( AutofillHostMsg_FillAutofillFormData::ID); // No message should be sent in this case. |firstname| is filled directly. ASSERT_EQ(static_cast(NULL), message2); EXPECT_EQ(firstname.value(), WebKit::WebString::fromUTF8("David")); } TEST_F(ChromeRenderViewTest, ShowAutofillWarning) { // Don't want any delay for form state sync changes. This will still post a // message so updates will get coalesced, but as soon as we spin the message // loop, it will generate an update. SendContentStateImmediately(); LoadHTML("
" " " " " " " "
"); // Verify that "QueryFormFieldAutofill" isn't sent prior to a user // interaction. const IPC::Message* message0 = render_thread_->sink().GetFirstMessageMatching( AutofillHostMsg_QueryFormFieldAutofill::ID); EXPECT_EQ(static_cast(NULL), message0); WebFrame* web_frame = GetMainFrame(); WebDocument document = web_frame->document(); WebInputElement firstname = document.getElementById("firstname").to(); WebInputElement middlename = document.getElementById("middlename").to(); // Simulate attempting to Autofill the form from the first element, which // specifies autocomplete="off". This should still not trigger an IPC, as we // don't show warnings for elements that have autocomplete="off". autofill_agent_->InputElementClicked(firstname, true, true); const IPC::Message* message1 = render_thread_->sink().GetFirstMessageMatching( AutofillHostMsg_QueryFormFieldAutofill::ID); EXPECT_EQ(static_cast(NULL), message1); // Simulate attempting to Autofill the form from the second element, which // does not specify autocomplete="off". This *should* trigger an IPC, as we // *do* show warnings for elements that don't themselves set // autocomplete="off", but for which the form does. autofill_agent_->InputElementClicked(middlename, true, true); const IPC::Message* message2 = render_thread_->sink().GetFirstMessageMatching( AutofillHostMsg_QueryFormFieldAutofill::ID); ASSERT_NE(static_cast(NULL), message2); // TODO(isherman): It would be nice to verify here that the message includes // the correct data. I'm not sure how to extract that information from an // IPC::Message though. } } // namespace autofill