1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/strings/string_util.h"
6 #include "base/strings/utf_string_conversions.h"
7 #include "chrome/renderer/autofill/password_generation_test_utils.h"
8 #include "chrome/test/base/chrome_render_view_test.h"
9 #include "components/autofill/content/common/autofill_messages.h"
10 #include "components/autofill/content/renderer/autofill_agent.h"
11 #include "components/autofill/content/renderer/form_autofill_util.h"
12 #include "components/autofill/content/renderer/password_autofill_agent.h"
13 #include "components/autofill/content/renderer/test_password_autofill_agent.h"
14 #include "components/autofill/content/renderer/test_password_generation_agent.h"
15 #include "components/autofill/core/common/autofill_constants.h"
16 #include "components/autofill/core/common/autofill_switches.h"
17 #include "components/autofill/core/common/form_data.h"
18 #include "components/autofill/core/common/form_field_data.h"
19 #include "components/autofill/core/common/password_form_field_prediction_map.h"
20 #include "content/public/renderer/render_frame.h"
21 #include "testing/gtest/include/gtest/gtest.h"
22 #include "third_party/WebKit/public/platform/WebString.h"
23 #include "third_party/WebKit/public/platform/WebVector.h"
24 #include "third_party/WebKit/public/web/WebDocument.h"
25 #include "third_party/WebKit/public/web/WebElement.h"
26 #include "third_party/WebKit/public/web/WebFormControlElement.h"
27 #include "third_party/WebKit/public/web/WebFormElement.h"
28 #include "third_party/WebKit/public/web/WebInputElement.h"
29 #include "third_party/WebKit/public/web/WebLocalFrame.h"
30 #include "third_party/WebKit/public/web/WebNode.h"
31 #include "third_party/WebKit/public/web/WebView.h"
32 #include "ui/events/keycodes/keyboard_codes.h"
34 using autofill::PasswordForm
;
35 using base::ASCIIToUTF16
;
36 using base::UTF16ToUTF8
;
37 using blink::WebDocument
;
38 using blink::WebElement
;
39 using blink::WebFrame
;
40 using blink::WebInputElement
;
41 using blink::WebString
;
46 const int kPasswordFillFormDataId
= 1234;
48 // The name of the username/password element in the form.
49 const char kUsernameName
[] = "username";
50 const char kPasswordName
[] = "password";
51 const char kEmailName
[] = "email";
52 const char kCreditCardOwnerName
[] = "creditcardowner";
53 const char kCreditCardNumberName
[] = "creditcardnumber";
54 const char kCreditCardVerificationName
[] = "cvc";
56 const char kAliceUsername
[] = "alice";
57 const char kAlicePassword
[] = "password";
58 const char kBobUsername
[] = "bob";
59 const char kBobPassword
[] = "secret";
60 const char kCarolUsername
[] = "Carol";
61 const char kCarolPassword
[] = "test";
62 const char kCarolAlternateUsername
[] = "RealCarolUsername";
64 const char kFormHTML
[] =
65 "<FORM name='LoginTestForm' action='http://www.bidule.com'>"
66 " <INPUT type='text' id='random_field'/>"
67 " <INPUT type='text' id='username'/>"
68 " <INPUT type='password' id='password'/>"
69 " <INPUT type='submit' value='Login'/>"
72 const char kVisibleFormWithNoUsernameHTML
[] =
73 "<head> <style> form {display: inline;} </style> </head>"
75 " <form name='LoginTestForm' action='http://www.bidule.com'>"
77 " <input type='password' id='password'/>"
82 const char kEmptyFormHTML
[] =
83 "<head> <style> form {display: inline;} </style> </head>"
84 "<body> <form> </form> </body>";
86 const char kNonVisibleFormHTML
[] =
87 "<head> <style> form {display: none;} </style> </head>"
91 " <input type='password' id='password'/>"
96 const char kSignupFormHTML
[] =
97 "<FORM name='LoginTestForm' action='http://www.bidule.com'>"
98 " <INPUT type='text' id='random_info'/>"
99 " <INPUT type='password' id='new_password'/>"
100 " <INPUT type='password' id='confirm_password'/>"
101 " <INPUT type='submit' value='Login'/>"
104 const char kEmptyWebpage
[] =
112 const char kRedirectionWebpage
[] =
115 " <meta http-equiv='Content-Type' content='text/html'>"
116 " <title>Redirection page</title>"
120 " <script type='text/javascript'>"
126 const char kSimpleWebpage
[] =
129 " <meta charset='utf-8' />"
130 " <title>Title</title>"
133 " <form name='LoginTestForm'>"
134 " <input type='text' id='username'/>"
135 " <input type='password' id='password'/>"
136 " <input type='submit' value='Login'/>"
141 const char kWebpageWithDynamicContent
[] =
144 " <meta charset='utf-8' />"
145 " <title>Title</title>"
148 " <script type='text/javascript'>"
149 " function addParagraph() {"
150 " var p = document.createElement('p');"
151 " document.body.appendChild(p);"
153 " window.onload = addParagraph;"
158 const char kJavaScriptClick
[] =
159 "var event = new MouseEvent('click', {"
162 " 'cancelable': true"
164 "var form = document.getElementById('myform1');"
165 "form.dispatchEvent(event);"
166 "console.log('clicked!');";
168 const char kOnChangeDetectionScript
[] =
170 " usernameOnchangeCalled = false;"
171 " passwordOnchangeCalled = false;"
172 " document.getElementById('username').onchange = function() {"
173 " usernameOnchangeCalled = true;"
175 " document.getElementById('password').onchange = function() {"
176 " passwordOnchangeCalled = true;"
180 const char kFormHTMLWithTwoTextFields
[] =
181 "<FORM name='LoginTestForm' id='LoginTestForm' "
182 "action='http://www.bidule.com'>"
183 " <INPUT type='text' id='username'/>"
184 " <INPUT type='text' id='email'/>"
185 " <INPUT type='password' id='password'/>"
186 " <INPUT type='submit' value='Login'/>"
189 const char kPasswordChangeFormHTML
[] =
190 "<FORM name='ChangeWithUsernameForm' action='http://www.bidule.com'>"
191 " <INPUT type='text' id='username'/>"
192 " <INPUT type='password' id='password'/>"
193 " <INPUT type='password' id='newpassword'/>"
194 " <INPUT type='password' id='confirmpassword'/>"
195 " <INPUT type='submit' value='Login'/>"
198 const char kCreditCardFormHTML
[] =
199 "<FORM name='ChangeWithUsernameForm' action='http://www.bidule.com'>"
200 " <INPUT type='text' id='creditcardowner'/>"
201 " <INPUT type='text' id='creditcardnumber'/>"
202 " <INPUT type='password' id='cvc'/>"
203 " <INPUT type='submit' value='Submit'/>"
206 const char kNoFormHTML
[] =
207 " <INPUT type='text' id='username'/>"
208 " <INPUT type='password' id='password'/>";
210 // Sets the "readonly" attribute of |element| to the value given by |read_only|.
211 void SetElementReadOnly(WebInputElement
& element
, bool read_only
) {
212 element
.setAttribute(WebString::fromUTF8("readonly"),
213 read_only
? WebString::fromUTF8("true") : WebString());
220 class PasswordAutofillAgentTest
: public ChromeRenderViewTest
{
222 PasswordAutofillAgentTest() {
225 // Simulates the fill password form message being sent to the renderer.
226 // We use that so we don't have to make RenderView::OnFillPasswordForm()
228 void SimulateOnFillPasswordForm(
229 const PasswordFormFillData
& fill_data
) {
230 AutofillMsg_FillPasswordForm
msg(0, kPasswordFillFormDataId
, fill_data
);
231 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
232 ->OnMessageReceived(msg
);
235 // As above, but fills for an iframe.
236 void SimulateOnFillPasswordFormForFrame(
238 const PasswordFormFillData
& fill_data
) {
239 AutofillMsg_FillPasswordForm
msg(0, kPasswordFillFormDataId
, fill_data
);
240 content::RenderFrame::FromWebFrame(frame
)->OnMessageReceived(msg
);
243 void SendVisiblePasswordForms() {
244 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
248 void SetUp() override
{
249 ChromeRenderViewTest::SetUp();
251 // Add a preferred login and an additional login to the FillData.
252 username1_
= ASCIIToUTF16(kAliceUsername
);
253 password1_
= ASCIIToUTF16(kAlicePassword
);
254 username2_
= ASCIIToUTF16(kBobUsername
);
255 password2_
= ASCIIToUTF16(kBobPassword
);
256 username3_
= ASCIIToUTF16(kCarolUsername
);
257 password3_
= ASCIIToUTF16(kCarolPassword
);
258 alternate_username3_
= ASCIIToUTF16(kCarolAlternateUsername
);
260 FormFieldData username_field
;
261 username_field
.name
= ASCIIToUTF16(kUsernameName
);
262 username_field
.value
= username1_
;
263 fill_data_
.username_field
= username_field
;
265 FormFieldData password_field
;
266 password_field
.name
= ASCIIToUTF16(kPasswordName
);
267 password_field
.value
= password1_
;
268 password_field
.form_control_type
= "password";
269 fill_data_
.password_field
= password_field
;
271 PasswordAndRealm password2
;
272 password2
.password
= password2_
;
273 fill_data_
.additional_logins
[username2_
] = password2
;
274 PasswordAndRealm password3
;
275 password3
.password
= password3_
;
276 fill_data_
.additional_logins
[username3_
] = password3
;
278 UsernamesCollectionKey key
;
279 key
.username
= username3_
;
280 key
.password
= password3_
;
281 key
.realm
= "google.com";
282 fill_data_
.other_possible_usernames
[key
].push_back(alternate_username3_
);
284 // We need to set the origin so it matches the frame URL and the action so
285 // it matches the form action, otherwise we won't autocomplete.
286 UpdateOriginForHTML(kFormHTML
);
287 fill_data_
.action
= GURL("http://www.bidule.com");
291 // Necessary for SimulateElementClick() to work correctly.
292 GetWebWidget()->resize(blink::WebSize(500, 500));
293 GetWebWidget()->setFocus(true);
295 // Now retrieve the input elements so the test can access them.
296 UpdateUsernameAndPasswordElements();
299 void TearDown() override
{
300 username_element_
.reset();
301 password_element_
.reset();
302 ChromeRenderViewTest::TearDown();
305 void UpdateOriginForHTML(const std::string
& html
) {
306 std::string origin
= "data:text/html;charset=utf-8," + html
;
307 fill_data_
.origin
= GURL(origin
);
310 void UpdateUsernameAndPasswordElements() {
311 WebDocument document
= GetMainFrame()->document();
313 document
.getElementById(WebString::fromUTF8(kUsernameName
));
314 ASSERT_FALSE(element
.isNull());
315 username_element_
= element
.to
<blink::WebInputElement
>();
316 element
= document
.getElementById(WebString::fromUTF8(kPasswordName
));
317 ASSERT_FALSE(element
.isNull());
318 password_element_
= element
.to
<blink::WebInputElement
>();
321 blink::WebInputElement
GetInputElementByID(const std::string
& id
) {
322 WebDocument document
= GetMainFrame()->document();
324 document
.getElementById(WebString::fromUTF8(id
.c_str()));
325 return element
.to
<blink::WebInputElement
>();
328 void ClearUsernameAndPasswordFields() {
329 username_element_
.setValue("");
330 username_element_
.setAutofilled(false);
331 password_element_
.setValue("");
332 password_element_
.setAutofilled(false);
335 void SimulateDidEndEditing(WebFrame
* input_frame
, WebInputElement
& input
) {
336 static_cast<blink::WebAutofillClient
*>(autofill_agent_
)
337 ->textFieldDidEndEditing(input
);
340 void SimulateSuggestionChoice(WebInputElement
& username_input
) {
341 base::string16
username(base::ASCIIToUTF16(kAliceUsername
));
342 base::string16
password(base::ASCIIToUTF16(kAlicePassword
));
343 SimulateSuggestionChoiceOfUsernameAndPassword(username_input
, username
,
347 void SimulateSuggestionChoiceOfUsernameAndPassword(
348 WebInputElement
& input
,
349 const base::string16
& username
,
350 const base::string16
& password
) {
351 // This call is necessary to setup the autofill agent appropriate for the
352 // user selection; simulates the menu actually popping up.
353 render_thread_
->sink().ClearMessages();
354 static_cast<autofill::PageClickListener
*>(autofill_agent_
)
355 ->FormControlElementClicked(input
, false);
357 AutofillMsg_FillPasswordSuggestion
msg(0, username
, password
);
358 static_cast<content::RenderFrameObserver
*>(autofill_agent_
)
359 ->OnMessageReceived(msg
);
362 void SimulateUsernameChange(const std::string
& username
) {
363 SimulateUserInputChangeForElement(&username_element_
, username
);
366 void SimulatePasswordChange(const std::string
& password
) {
367 SimulateUserInputChangeForElement(&password_element_
, password
);
370 void CheckTextFieldsStateForElements(const WebInputElement
& username_element
,
371 const std::string
& username
,
372 bool username_autofilled
,
373 const WebInputElement
& password_element
,
374 const std::string
& password
,
375 bool password_autofilled
,
376 bool checkSuggestedValue
) {
378 static_cast<std::string
>(username_element
.value().utf8()));
379 EXPECT_EQ(username_autofilled
, username_element
.isAutofilled());
381 static_cast<std::string
>(
382 checkSuggestedValue
? password_element
.suggestedValue().utf8()
383 : password_element
.value().utf8()))
384 << "checkSuggestedValue == " << checkSuggestedValue
;
385 EXPECT_EQ(password_autofilled
, password_element
.isAutofilled());
388 // Checks the DOM-accessible value of the username element and the
389 // *suggested* value of the password element.
390 void CheckTextFieldsState(const std::string
& username
,
391 bool username_autofilled
,
392 const std::string
& password
,
393 bool password_autofilled
) {
394 CheckTextFieldsStateForElements(username_element_
,
403 // Checks the DOM-accessible value of the username element and the
404 // DOM-accessible value of the password element.
405 void CheckTextFieldsDOMState(const std::string
& username
,
406 bool username_autofilled
,
407 const std::string
& password
,
408 bool password_autofilled
) {
409 CheckTextFieldsStateForElements(username_element_
,
418 void CheckUsernameSelection(int start
, int end
) {
419 EXPECT_EQ(start
, username_element_
.selectionStart());
420 EXPECT_EQ(end
, username_element_
.selectionEnd());
423 // Checks the message sent to PasswordAutofillManager to build the suggestion
424 // list. |username| is the expected username field value, and |show_all| is
425 // the expected flag for the PasswordAutofillManager, whether to show all
426 // suggestions, or only those starting with |username|.
427 void CheckSuggestions(const std::string
& username
, bool show_all
) {
428 const IPC::Message
* message
=
429 render_thread_
->sink().GetFirstMessageMatching(
430 AutofillHostMsg_ShowPasswordSuggestions::ID
);
431 ASSERT_TRUE(message
);
432 base::Tuple
<int, base::i18n::TextDirection
, base::string16
, int, gfx::RectF
>
434 AutofillHostMsg_ShowPasswordSuggestions::Read(message
, &args
);
435 EXPECT_EQ(kPasswordFillFormDataId
, base::get
<0>(args
));
436 EXPECT_EQ(ASCIIToUTF16(username
), base::get
<2>(args
));
438 static_cast<bool>(base::get
<3>(args
) & autofill::SHOW_ALL
));
440 render_thread_
->sink().ClearMessages();
443 void ExpectFormSubmittedWithUsernameAndPasswords(
444 const std::string
& username_value
,
445 const std::string
& password_value
,
446 const std::string
& new_password_value
) {
447 const IPC::Message
* message
=
448 render_thread_
->sink().GetFirstMessageMatching(
449 AutofillHostMsg_PasswordFormSubmitted::ID
);
450 ASSERT_TRUE(message
);
451 base::Tuple
<autofill::PasswordForm
> args
;
452 AutofillHostMsg_PasswordFormSubmitted::Read(message
, &args
);
453 EXPECT_EQ(ASCIIToUTF16(username_value
), base::get
<0>(args
).username_value
);
454 EXPECT_EQ(ASCIIToUTF16(password_value
), base::get
<0>(args
).password_value
);
455 EXPECT_EQ(ASCIIToUTF16(new_password_value
),
456 base::get
<0>(args
).new_password_value
);
459 void ExpectInPageNavigationWithUsernameAndPasswords(
460 const std::string
& username_value
,
461 const std::string
& password_value
,
462 const std::string
& new_password_value
) {
463 const IPC::Message
* message
=
464 render_thread_
->sink().GetFirstMessageMatching(
465 AutofillHostMsg_InPageNavigation::ID
);
466 ASSERT_TRUE(message
);
467 base::Tuple
<autofill::PasswordForm
> args
;
468 AutofillHostMsg_InPageNavigation::Read(message
, &args
);
469 EXPECT_EQ(ASCIIToUTF16(username_value
), base::get
<0>(args
).username_value
);
470 EXPECT_EQ(ASCIIToUTF16(password_value
), base::get
<0>(args
).password_value
);
471 EXPECT_EQ(ASCIIToUTF16(new_password_value
),
472 base::get
<0>(args
).new_password_value
);
475 base::string16 username1_
;
476 base::string16 username2_
;
477 base::string16 username3_
;
478 base::string16 password1_
;
479 base::string16 password2_
;
480 base::string16 password3_
;
481 base::string16 alternate_username3_
;
482 PasswordFormFillData fill_data_
;
484 WebInputElement username_element_
;
485 WebInputElement password_element_
;
488 DISALLOW_COPY_AND_ASSIGN(PasswordAutofillAgentTest
);
491 // Tests that the password login is autocompleted as expected when the browser
492 // sends back the password info.
493 TEST_F(PasswordAutofillAgentTest
, InitialAutocomplete
) {
495 * Right now we are not sending the message to the browser because we are
496 * loading a data URL and the security origin canAccessPasswordManager()
497 * returns false. May be we should mock URL loading to cirmcuvent this?
498 TODO(jcivelli): find a way to make the security origin not deny access to the
499 password manager and then reenable this code.
501 // The form has been loaded, we should have sent the browser a message about
503 const IPC::Message* msg = render_thread_.sink().GetFirstMessageMatching(
504 AutofillHostMsg_PasswordFormsParsed::ID);
505 ASSERT_TRUE(msg != NULL);
507 base::Tuple1<std::vector<PasswordForm> > forms;
508 AutofillHostMsg_PasswordFormsParsed::Read(msg, &forms);
509 ASSERT_EQ(1U, forms.a.size());
510 PasswordForm password_form = forms.a[0];
511 EXPECT_EQ(PasswordForm::SCHEME_HTML, password_form.scheme);
512 EXPECT_EQ(ASCIIToUTF16(kUsernameName), password_form.username_element);
513 EXPECT_EQ(ASCIIToUTF16(kPasswordName), password_form.password_element);
516 // Simulate the browser sending back the login info, it triggers the
518 SimulateOnFillPasswordForm(fill_data_
);
520 // The username and password should have been autocompleted.
521 CheckTextFieldsState(kAliceUsername
, true, kAlicePassword
, true);
524 // Tests that we correctly fill forms having an empty 'action' attribute.
525 TEST_F(PasswordAutofillAgentTest
, InitialAutocompleteForEmptyAction
) {
526 const char kEmptyActionFormHTML
[] =
527 "<FORM name='LoginTestForm'>"
528 " <INPUT type='text' id='username'/>"
529 " <INPUT type='password' id='password'/>"
530 " <INPUT type='submit' value='Login'/>"
532 LoadHTML(kEmptyActionFormHTML
);
534 // Retrieve the input elements so the test can access them.
535 WebDocument document
= GetMainFrame()->document();
537 document
.getElementById(WebString::fromUTF8(kUsernameName
));
538 ASSERT_FALSE(element
.isNull());
539 username_element_
= element
.to
<blink::WebInputElement
>();
540 element
= document
.getElementById(WebString::fromUTF8(kPasswordName
));
541 ASSERT_FALSE(element
.isNull());
542 password_element_
= element
.to
<blink::WebInputElement
>();
544 // Set the expected form origin and action URLs.
545 UpdateOriginForHTML(kEmptyActionFormHTML
);
546 fill_data_
.action
= fill_data_
.origin
;
548 // Simulate the browser sending back the login info, it triggers the
550 SimulateOnFillPasswordForm(fill_data_
);
552 // The username and password should have been autocompleted.
553 CheckTextFieldsState(kAliceUsername
, true, kAlicePassword
, true);
556 // Tests that if a password is marked as readonly, neither field is autofilled
558 TEST_F(PasswordAutofillAgentTest
, NoInitialAutocompleteForReadOnlyPassword
) {
559 SetElementReadOnly(password_element_
, true);
561 // Simulate the browser sending back the login info, it triggers the
563 SimulateOnFillPasswordForm(fill_data_
);
565 CheckTextFieldsState(std::string(), false, std::string(), false);
568 // Can still fill a password field if the username is set to a value that
570 TEST_F(PasswordAutofillAgentTest
,
571 AutocompletePasswordForReadonlyUsernameMatched
) {
572 username_element_
.setValue(username3_
);
573 SetElementReadOnly(username_element_
, true);
575 // Filled even though username is not the preferred match.
576 SimulateOnFillPasswordForm(fill_data_
);
577 CheckTextFieldsState(UTF16ToUTF8(username3_
), false,
578 UTF16ToUTF8(password3_
), true);
581 // If a username field is empty and readonly, don't autofill.
582 TEST_F(PasswordAutofillAgentTest
,
583 NoAutocompletePasswordForReadonlyUsernameUnmatched
) {
584 username_element_
.setValue(WebString::fromUTF8(""));
585 SetElementReadOnly(username_element_
, true);
587 SimulateOnFillPasswordForm(fill_data_
);
588 CheckTextFieldsState(std::string(), false, std::string(), false);
591 // Tests that having a non-matching username precludes the autocomplete.
592 TEST_F(PasswordAutofillAgentTest
, NoAutocompleteForFilledFieldUnmatched
) {
593 username_element_
.setValue(WebString::fromUTF8("bogus"));
595 // Simulate the browser sending back the login info, it triggers the
597 SimulateOnFillPasswordForm(fill_data_
);
599 // Neither field should be autocompleted.
600 CheckTextFieldsState("bogus", false, std::string(), false);
603 // Don't try to complete a prefilled value even if it's a partial match
605 TEST_F(PasswordAutofillAgentTest
, NoPartialMatchForPrefilledUsername
) {
606 username_element_
.setValue(WebString::fromUTF8("ali"));
608 SimulateOnFillPasswordForm(fill_data_
);
610 CheckTextFieldsState("ali", false, std::string(), false);
613 TEST_F(PasswordAutofillAgentTest
, InputWithNoForms
) {
614 const char kNoFormInputs
[] =
615 "<input type='text' id='username'/>"
616 "<input type='password' id='password'/>";
617 LoadHTML(kNoFormInputs
);
619 SimulateOnFillPasswordForm(fill_data_
);
621 // Input elements that aren't in a <form> won't autofill.
622 CheckTextFieldsState(std::string(), false, std::string(), false);
625 TEST_F(PasswordAutofillAgentTest
, NoAutocompleteForTextFieldPasswords
) {
626 const char kTextFieldPasswordFormHTML
[] =
627 "<FORM name='LoginTestForm' action='http://www.bidule.com'>"
628 " <INPUT type='text' id='username'/>"
629 " <INPUT type='text' id='password'/>"
630 " <INPUT type='submit' value='Login'/>"
632 LoadHTML(kTextFieldPasswordFormHTML
);
634 // Retrieve the input elements so the test can access them.
635 WebDocument document
= GetMainFrame()->document();
637 document
.getElementById(WebString::fromUTF8(kUsernameName
));
638 ASSERT_FALSE(element
.isNull());
639 username_element_
= element
.to
<blink::WebInputElement
>();
640 element
= document
.getElementById(WebString::fromUTF8(kPasswordName
));
641 ASSERT_FALSE(element
.isNull());
642 password_element_
= element
.to
<blink::WebInputElement
>();
644 // Set the expected form origin URL.
645 UpdateOriginForHTML(kTextFieldPasswordFormHTML
);
647 SimulateOnFillPasswordForm(fill_data_
);
649 // Fields should still be empty.
650 CheckTextFieldsState(std::string(), false, std::string(), false);
653 TEST_F(PasswordAutofillAgentTest
, NoAutocompleteForPasswordFieldUsernames
) {
654 const char kPasswordFieldUsernameFormHTML
[] =
655 "<FORM name='LoginTestForm' action='http://www.bidule.com'>"
656 " <INPUT type='password' id='username'/>"
657 " <INPUT type='password' id='password'/>"
658 " <INPUT type='submit' value='Login'/>"
660 LoadHTML(kPasswordFieldUsernameFormHTML
);
662 // Retrieve the input elements so the test can access them.
663 WebDocument document
= GetMainFrame()->document();
665 document
.getElementById(WebString::fromUTF8(kUsernameName
));
666 ASSERT_FALSE(element
.isNull());
667 username_element_
= element
.to
<blink::WebInputElement
>();
668 element
= document
.getElementById(WebString::fromUTF8(kPasswordName
));
669 ASSERT_FALSE(element
.isNull());
670 password_element_
= element
.to
<blink::WebInputElement
>();
672 // Set the expected form origin URL.
673 UpdateOriginForHTML(kPasswordFieldUsernameFormHTML
);
675 SimulateOnFillPasswordForm(fill_data_
);
677 // Fields should still be empty.
678 CheckTextFieldsState(std::string(), false, std::string(), false);
681 // Tests that having a matching username does not preclude the autocomplete.
682 TEST_F(PasswordAutofillAgentTest
, InitialAutocompleteForMatchingFilledField
) {
683 username_element_
.setValue(WebString::fromUTF8(kAliceUsername
));
685 // Simulate the browser sending back the login info, it triggers the
687 SimulateOnFillPasswordForm(fill_data_
);
689 // The username and password should have been autocompleted.
690 CheckTextFieldsState(kAliceUsername
, true, kAlicePassword
, true);
693 TEST_F(PasswordAutofillAgentTest
, PasswordNotClearedOnEdit
) {
694 // Simulate the browser sending back the login info, it triggers the
696 SimulateOnFillPasswordForm(fill_data_
);
698 // Simulate the user changing the username to some unknown username.
699 SimulateUsernameChange("alicia");
701 // The password should not have been cleared.
702 CheckTextFieldsDOMState("alicia", false, kAlicePassword
, true);
705 // Tests that we only autocomplete on focus lost and with a full username match
706 // when |wait_for_username| is true.
707 TEST_F(PasswordAutofillAgentTest
, WaitUsername
) {
708 // Simulate the browser sending back the login info.
709 fill_data_
.wait_for_username
= true;
710 SimulateOnFillPasswordForm(fill_data_
);
712 // No auto-fill should have taken place.
713 CheckTextFieldsState(std::string(), false, std::string(), false);
715 // No autocomplete should happen when text is entered in the username.
716 SimulateUsernameChange("a");
717 CheckTextFieldsState("a", false, std::string(), false);
718 SimulateUsernameChange("al");
719 CheckTextFieldsState("al", false, std::string(), false);
720 SimulateUsernameChange(kAliceUsername
);
721 CheckTextFieldsState(kAliceUsername
, false, std::string(), false);
723 // Autocomplete should happen only when the username textfield is blurred with
725 SimulateUsernameChange("a");
726 static_cast<blink::WebAutofillClient
*>(autofill_agent_
)
727 ->textFieldDidEndEditing(username_element_
);
728 CheckTextFieldsState("a", false, std::string(), false);
729 SimulateUsernameChange("al");
730 static_cast<blink::WebAutofillClient
*>(autofill_agent_
)
731 ->textFieldDidEndEditing(username_element_
);
732 CheckTextFieldsState("al", false, std::string(), false);
733 SimulateUsernameChange("alices");
734 static_cast<blink::WebAutofillClient
*>(autofill_agent_
)
735 ->textFieldDidEndEditing(username_element_
);
736 CheckTextFieldsState("alices", false, std::string(), false);
737 SimulateUsernameChange(kAliceUsername
);
738 static_cast<blink::WebAutofillClient
*>(autofill_agent_
)
739 ->textFieldDidEndEditing(username_element_
);
740 CheckTextFieldsDOMState(kAliceUsername
, true, kAlicePassword
, true);
743 TEST_F(PasswordAutofillAgentTest
, IsWebNodeVisibleTest
) {
744 blink::WebVector
<blink::WebFormElement
> forms1
, forms2
, forms3
;
745 blink::WebFrame
* frame
;
747 LoadHTML(kVisibleFormWithNoUsernameHTML
);
748 frame
= GetMainFrame();
749 frame
->document().forms(forms1
);
750 ASSERT_EQ(1u, forms1
.size());
751 EXPECT_TRUE(IsWebNodeVisible(forms1
[0]));
753 LoadHTML(kEmptyFormHTML
);
754 frame
= GetMainFrame();
755 frame
->document().forms(forms2
);
756 ASSERT_EQ(1u, forms2
.size());
757 EXPECT_FALSE(IsWebNodeVisible(forms2
[0]));
759 LoadHTML(kNonVisibleFormHTML
);
760 frame
= GetMainFrame();
761 frame
->document().forms(forms3
);
762 ASSERT_EQ(1u, forms3
.size());
763 EXPECT_FALSE(IsWebNodeVisible(forms3
[0]));
766 TEST_F(PasswordAutofillAgentTest
, SendPasswordFormsTest
) {
767 render_thread_
->sink().ClearMessages();
768 LoadHTML(kVisibleFormWithNoUsernameHTML
);
769 const IPC::Message
* message
= render_thread_
->sink()
770 .GetFirstMessageMatching(AutofillHostMsg_PasswordFormsRendered::ID
);
771 EXPECT_TRUE(message
);
772 base::Tuple
<std::vector
<autofill::PasswordForm
>, bool> param
;
773 AutofillHostMsg_PasswordFormsRendered::Read(message
, ¶m
);
774 EXPECT_TRUE(base::get
<0>(param
).size());
776 render_thread_
->sink().ClearMessages();
777 LoadHTML(kEmptyFormHTML
);
778 message
= render_thread_
->sink().GetFirstMessageMatching(
779 AutofillHostMsg_PasswordFormsRendered::ID
);
780 EXPECT_TRUE(message
);
781 AutofillHostMsg_PasswordFormsRendered::Read(message
, ¶m
);
782 EXPECT_FALSE(base::get
<0>(param
).size());
784 render_thread_
->sink().ClearMessages();
785 LoadHTML(kNonVisibleFormHTML
);
786 message
= render_thread_
->sink().GetFirstMessageMatching(
787 AutofillHostMsg_PasswordFormsRendered::ID
);
788 EXPECT_TRUE(message
);
789 AutofillHostMsg_PasswordFormsRendered::Read(message
, ¶m
);
790 EXPECT_FALSE(base::get
<0>(param
).size());
793 TEST_F(PasswordAutofillAgentTest
, SendPasswordFormsTest_Redirection
) {
794 render_thread_
->sink().ClearMessages();
795 LoadHTML(kEmptyWebpage
);
796 EXPECT_FALSE(render_thread_
->sink().GetFirstMessageMatching(
797 AutofillHostMsg_PasswordFormsRendered::ID
));
799 render_thread_
->sink().ClearMessages();
800 LoadHTML(kRedirectionWebpage
);
801 EXPECT_FALSE(render_thread_
->sink().GetFirstMessageMatching(
802 AutofillHostMsg_PasswordFormsRendered::ID
));
804 render_thread_
->sink().ClearMessages();
805 LoadHTML(kSimpleWebpage
);
806 EXPECT_TRUE(render_thread_
->sink().GetFirstMessageMatching(
807 AutofillHostMsg_PasswordFormsRendered::ID
));
809 render_thread_
->sink().ClearMessages();
810 LoadHTML(kWebpageWithDynamicContent
);
811 EXPECT_TRUE(render_thread_
->sink().GetFirstMessageMatching(
812 AutofillHostMsg_PasswordFormsRendered::ID
));
815 // Tests that a password will only be filled as a suggested and will not be
816 // accessible by the DOM until a user gesture has occurred.
817 TEST_F(PasswordAutofillAgentTest
, GestureRequiredTest
) {
818 // Trigger the initial autocomplete.
819 SimulateOnFillPasswordForm(fill_data_
);
821 // The username and password should have been autocompleted.
822 CheckTextFieldsState(kAliceUsername
, true, kAlicePassword
, true);
824 // However, it should only have completed with the suggested value, as tested
825 // above, and it should not have completed into the DOM accessible value for
826 // the password field.
827 CheckTextFieldsDOMState(kAliceUsername
, true, std::string(), true);
829 // Simulate a user click so that the password field's real value is filled.
830 SimulateElementClick(kUsernameName
);
831 CheckTextFieldsDOMState(kAliceUsername
, true, kAlicePassword
, true);
834 // Verfies that a DOM-activated UI event will not cause an autofill.
835 TEST_F(PasswordAutofillAgentTest
, NoDOMActivationTest
) {
836 // Trigger the initial autocomplete.
837 SimulateOnFillPasswordForm(fill_data_
);
839 ExecuteJavaScriptForTests(kJavaScriptClick
);
840 CheckTextFieldsDOMState(kAliceUsername
, true, "", true);
843 // Verifies that password autofill triggers onChange events in JavaScript for
844 // forms that are filled on page load.
845 TEST_F(PasswordAutofillAgentTest
,
846 PasswordAutofillTriggersOnChangeEventsOnLoad
) {
847 std::string html
= std::string(kFormHTML
) + kOnChangeDetectionScript
;
848 LoadHTML(html
.c_str());
849 UpdateOriginForHTML(html
);
850 UpdateUsernameAndPasswordElements();
852 // Simulate the browser sending back the login info, it triggers the
854 SimulateOnFillPasswordForm(fill_data_
);
856 // The username and password should have been autocompleted...
857 CheckTextFieldsState(kAliceUsername
, true, kAlicePassword
, true);
858 // ... but since there hasn't been a user gesture yet, the autocompleted
859 // password should only be visible to the user.
860 CheckTextFieldsDOMState(kAliceUsername
, true, std::string(), true);
862 // A JavaScript onChange event should have been triggered for the username,
863 // but not yet for the password.
864 int username_onchange_called
= -1;
865 int password_onchange_called
= -1;
867 ExecuteJavaScriptAndReturnIntValue(
868 ASCIIToUTF16("usernameOnchangeCalled ? 1 : 0"),
869 &username_onchange_called
));
870 EXPECT_EQ(1, username_onchange_called
);
872 ExecuteJavaScriptAndReturnIntValue(
873 ASCIIToUTF16("passwordOnchangeCalled ? 1 : 0"),
874 &password_onchange_called
));
875 // TODO(isherman): Re-enable this check once http://crbug.com/333144 is fixed.
876 // EXPECT_EQ(0, password_onchange_called);
878 // Simulate a user click so that the password field's real value is filled.
879 SimulateElementClick(kUsernameName
);
880 CheckTextFieldsDOMState(kAliceUsername
, true, kAlicePassword
, true);
882 // Now, a JavaScript onChange event should have been triggered for the
885 ExecuteJavaScriptAndReturnIntValue(
886 ASCIIToUTF16("passwordOnchangeCalled ? 1 : 0"),
887 &password_onchange_called
));
888 EXPECT_EQ(1, password_onchange_called
);
891 // Verifies that password autofill triggers onChange events in JavaScript for
892 // forms that are filled after page load.
893 TEST_F(PasswordAutofillAgentTest
,
894 PasswordAutofillTriggersOnChangeEventsWaitForUsername
) {
895 std::string html
= std::string(kFormHTML
) + kOnChangeDetectionScript
;
896 LoadHTML(html
.c_str());
897 UpdateOriginForHTML(html
);
898 UpdateUsernameAndPasswordElements();
900 // Simulate the browser sending back the login info, it triggers the
902 fill_data_
.wait_for_username
= true;
903 SimulateOnFillPasswordForm(fill_data_
);
905 // The username and password should not yet have been autocompleted.
906 CheckTextFieldsState(std::string(), false, std::string(), false);
908 // Simulate a click just to force a user gesture, since the username value is
910 SimulateElementClick(kUsernameName
);
912 // Simulate the user entering the first letter of her username and selecting
913 // the matching autofill from the dropdown.
914 SimulateUsernameChange("a");
915 SimulateSuggestionChoice(username_element_
);
917 // The username and password should now have been autocompleted.
918 CheckTextFieldsDOMState(kAliceUsername
, true, kAlicePassword
, true);
920 // JavaScript onChange events should have been triggered both for the username
921 // and for the password.
922 int username_onchange_called
= -1;
923 int password_onchange_called
= -1;
925 ExecuteJavaScriptAndReturnIntValue(
926 ASCIIToUTF16("usernameOnchangeCalled ? 1 : 0"),
927 &username_onchange_called
));
928 EXPECT_EQ(1, username_onchange_called
);
930 ExecuteJavaScriptAndReturnIntValue(
931 ASCIIToUTF16("passwordOnchangeCalled ? 1 : 0"),
932 &password_onchange_called
));
933 EXPECT_EQ(1, password_onchange_called
);
936 // Tests that |FillSuggestion| properly fills the username and password.
937 TEST_F(PasswordAutofillAgentTest
, FillSuggestion
) {
938 // Simulate the browser sending the login info, but set |wait_for_username|
939 // to prevent the form from being immediately filled.
940 fill_data_
.wait_for_username
= true;
941 SimulateOnFillPasswordForm(fill_data_
);
943 // Neither field should have been autocompleted.
944 CheckTextFieldsDOMState(std::string(), false, std::string(), false);
946 // If the password field is not autocompletable, it should not be affected.
947 SetElementReadOnly(password_element_
, true);
948 EXPECT_FALSE(password_autofill_agent_
->FillSuggestion(
949 username_element_
, kAliceUsername
, kAlicePassword
));
950 CheckTextFieldsDOMState(std::string(), false, std::string(), false);
951 SetElementReadOnly(password_element_
, false);
953 // After filling with the suggestion, both fields should be autocompleted.
954 EXPECT_TRUE(password_autofill_agent_
->FillSuggestion(
955 username_element_
, kAliceUsername
, kAlicePassword
));
956 CheckTextFieldsDOMState(kAliceUsername
, true, kAlicePassword
, true);
957 int username_length
= strlen(kAliceUsername
);
958 CheckUsernameSelection(username_length
, username_length
);
960 // Try Filling with a suggestion with password different from the one that was
961 // initially sent to the renderer.
962 EXPECT_TRUE(password_autofill_agent_
->FillSuggestion(
963 username_element_
, kBobUsername
, kCarolPassword
));
964 CheckTextFieldsDOMState(kBobUsername
, true, kCarolPassword
, true);
965 username_length
= strlen(kBobUsername
);
966 CheckUsernameSelection(username_length
, username_length
);
969 // Tests that |PreviewSuggestion| properly previews the username and password.
970 TEST_F(PasswordAutofillAgentTest
, PreviewSuggestion
) {
971 // Simulate the browser sending the login info, but set |wait_for_username|
972 // to prevent the form from being immediately filled.
973 fill_data_
.wait_for_username
= true;
974 SimulateOnFillPasswordForm(fill_data_
);
976 // Neither field should have been autocompleted.
977 CheckTextFieldsDOMState(std::string(), false, std::string(), false);
979 // If the password field is not autocompletable, it should not be affected.
980 SetElementReadOnly(password_element_
, true);
981 EXPECT_FALSE(password_autofill_agent_
->PreviewSuggestion(
982 username_element_
, kAliceUsername
, kAlicePassword
));
983 EXPECT_EQ(std::string(), username_element_
.suggestedValue().utf8());
984 EXPECT_FALSE(username_element_
.isAutofilled());
985 EXPECT_EQ(std::string(), password_element_
.suggestedValue().utf8());
986 EXPECT_FALSE(password_element_
.isAutofilled());
987 SetElementReadOnly(password_element_
, false);
989 // After selecting the suggestion, both fields should be previewed
990 // with suggested values.
991 EXPECT_TRUE(password_autofill_agent_
->PreviewSuggestion(
992 username_element_
, kAliceUsername
, kAlicePassword
));
995 static_cast<std::string
>(username_element_
.suggestedValue().utf8()));
996 EXPECT_TRUE(username_element_
.isAutofilled());
999 static_cast<std::string
>(password_element_
.suggestedValue().utf8()));
1000 EXPECT_TRUE(password_element_
.isAutofilled());
1001 int username_length
= strlen(kAliceUsername
);
1002 CheckUsernameSelection(0, username_length
);
1004 // Try previewing with a password different from the one that was initially
1005 // sent to the renderer.
1006 EXPECT_TRUE(password_autofill_agent_
->PreviewSuggestion(
1007 username_element_
, kBobUsername
, kCarolPassword
));
1010 static_cast<std::string
>(username_element_
.suggestedValue().utf8()));
1011 EXPECT_TRUE(username_element_
.isAutofilled());
1014 static_cast<std::string
>(password_element_
.suggestedValue().utf8()));
1015 EXPECT_TRUE(password_element_
.isAutofilled());
1016 username_length
= strlen(kBobUsername
);
1017 CheckUsernameSelection(0, username_length
);
1020 // Tests that |PreviewSuggestion| properly sets the username selection range.
1021 TEST_F(PasswordAutofillAgentTest
, PreviewSuggestionSelectionRange
) {
1022 username_element_
.setValue(WebString::fromUTF8("ali"));
1023 username_element_
.setSelectionRange(3, 3);
1024 username_element_
.setAutofilled(true);
1026 CheckTextFieldsDOMState("ali", true, std::string(), false);
1028 // Simulate the browser sending the login info, but set |wait_for_username|
1029 // to prevent the form from being immediately filled.
1030 fill_data_
.wait_for_username
= true;
1031 SimulateOnFillPasswordForm(fill_data_
);
1033 EXPECT_TRUE(password_autofill_agent_
->PreviewSuggestion(
1034 username_element_
, kAliceUsername
, kAlicePassword
));
1037 static_cast<std::string
>(username_element_
.suggestedValue().utf8()));
1038 EXPECT_TRUE(username_element_
.isAutofilled());
1041 static_cast<std::string
>(password_element_
.suggestedValue().utf8()));
1042 EXPECT_TRUE(password_element_
.isAutofilled());
1043 int username_length
= strlen(kAliceUsername
);
1044 CheckUsernameSelection(3, username_length
);
1047 // Tests that |ClearPreview| properly clears previewed username and password
1048 // with password being previously autofilled.
1049 TEST_F(PasswordAutofillAgentTest
, ClearPreviewWithPasswordAutofilled
) {
1050 password_element_
.setValue(WebString::fromUTF8("sec"));
1051 password_element_
.setAutofilled(true);
1053 // Simulate the browser sending the login info, but set |wait_for_username|
1054 // to prevent the form from being immediately filled.
1055 fill_data_
.wait_for_username
= true;
1056 SimulateOnFillPasswordForm(fill_data_
);
1058 CheckTextFieldsDOMState(std::string(), false, "sec", true);
1060 EXPECT_TRUE(password_autofill_agent_
->PreviewSuggestion(
1061 username_element_
, kAliceUsername
, kAlicePassword
));
1064 password_autofill_agent_
->DidClearAutofillSelection(username_element_
));
1066 EXPECT_TRUE(username_element_
.value().isEmpty());
1067 EXPECT_TRUE(username_element_
.suggestedValue().isEmpty());
1068 EXPECT_FALSE(username_element_
.isAutofilled());
1069 EXPECT_EQ(ASCIIToUTF16("sec"), password_element_
.value());
1070 EXPECT_TRUE(password_element_
.suggestedValue().isEmpty());
1071 EXPECT_TRUE(password_element_
.isAutofilled());
1072 CheckUsernameSelection(0, 0);
1075 // Tests that |ClearPreview| properly clears previewed username and password
1076 // with username being previously autofilled.
1077 TEST_F(PasswordAutofillAgentTest
, ClearPreviewWithUsernameAutofilled
) {
1078 username_element_
.setValue(WebString::fromUTF8("ali"));
1079 username_element_
.setSelectionRange(3, 3);
1080 username_element_
.setAutofilled(true);
1082 // Simulate the browser sending the login info, but set |wait_for_username|
1083 // to prevent the form from being immediately filled.
1084 fill_data_
.wait_for_username
= true;
1085 SimulateOnFillPasswordForm(fill_data_
);
1087 CheckTextFieldsDOMState("ali", true, std::string(), false);
1089 EXPECT_TRUE(password_autofill_agent_
->PreviewSuggestion(
1090 username_element_
, kAliceUsername
, kAlicePassword
));
1093 password_autofill_agent_
->DidClearAutofillSelection(username_element_
));
1095 EXPECT_EQ(ASCIIToUTF16("ali"), username_element_
.value());
1096 EXPECT_TRUE(username_element_
.suggestedValue().isEmpty());
1097 EXPECT_TRUE(username_element_
.isAutofilled());
1098 EXPECT_TRUE(password_element_
.value().isEmpty());
1099 EXPECT_TRUE(password_element_
.suggestedValue().isEmpty());
1100 EXPECT_FALSE(password_element_
.isAutofilled());
1101 CheckUsernameSelection(3, 3);
1104 // Tests that |ClearPreview| properly clears previewed username and password
1105 // with username and password being previously autofilled.
1106 TEST_F(PasswordAutofillAgentTest
,
1107 ClearPreviewWithAutofilledUsernameAndPassword
) {
1108 username_element_
.setValue(WebString::fromUTF8("ali"));
1109 username_element_
.setSelectionRange(3, 3);
1110 username_element_
.setAutofilled(true);
1111 password_element_
.setValue(WebString::fromUTF8("sec"));
1112 password_element_
.setAutofilled(true);
1114 // Simulate the browser sending the login info, but set |wait_for_username|
1115 // to prevent the form from being immediately filled.
1116 fill_data_
.wait_for_username
= true;
1117 SimulateOnFillPasswordForm(fill_data_
);
1119 CheckTextFieldsDOMState("ali", true, "sec", true);
1121 EXPECT_TRUE(password_autofill_agent_
->PreviewSuggestion(
1122 username_element_
, kAliceUsername
, kAlicePassword
));
1125 password_autofill_agent_
->DidClearAutofillSelection(username_element_
));
1127 EXPECT_EQ(ASCIIToUTF16("ali"), username_element_
.value());
1128 EXPECT_TRUE(username_element_
.suggestedValue().isEmpty());
1129 EXPECT_TRUE(username_element_
.isAutofilled());
1130 EXPECT_EQ(ASCIIToUTF16("sec"), password_element_
.value());
1131 EXPECT_TRUE(password_element_
.suggestedValue().isEmpty());
1132 EXPECT_TRUE(password_element_
.isAutofilled());
1133 CheckUsernameSelection(3, 3);
1136 // Tests that |ClearPreview| properly clears previewed username and password
1137 // with neither username nor password being previously autofilled.
1138 TEST_F(PasswordAutofillAgentTest
,
1139 ClearPreviewWithNotAutofilledUsernameAndPassword
) {
1140 // Simulate the browser sending the login info, but set |wait_for_username|
1141 // to prevent the form from being immediately filled.
1142 fill_data_
.wait_for_username
= true;
1143 SimulateOnFillPasswordForm(fill_data_
);
1145 CheckTextFieldsDOMState(std::string(), false, std::string(), false);
1147 EXPECT_TRUE(password_autofill_agent_
->PreviewSuggestion(
1148 username_element_
, kAliceUsername
, kAlicePassword
));
1151 password_autofill_agent_
->DidClearAutofillSelection(username_element_
));
1153 EXPECT_TRUE(username_element_
.value().isEmpty());
1154 EXPECT_TRUE(username_element_
.suggestedValue().isEmpty());
1155 EXPECT_FALSE(username_element_
.isAutofilled());
1156 EXPECT_TRUE(password_element_
.value().isEmpty());
1157 EXPECT_TRUE(password_element_
.suggestedValue().isEmpty());
1158 EXPECT_FALSE(password_element_
.isAutofilled());
1159 CheckUsernameSelection(0, 0);
1162 // Tests that logging is off by default.
1163 TEST_F(PasswordAutofillAgentTest
, OnChangeLoggingState_NoMessage
) {
1164 render_thread_
->sink().ClearMessages();
1165 SendVisiblePasswordForms();
1166 const IPC::Message
* message
= render_thread_
->sink().GetFirstMessageMatching(
1167 AutofillHostMsg_RecordSavePasswordProgress::ID
);
1168 EXPECT_FALSE(message
);
1171 // Test that logging can be turned on by a message.
1172 TEST_F(PasswordAutofillAgentTest
, OnChangeLoggingState_Activated
) {
1173 // Turn the logging on.
1174 AutofillMsg_SetLoggingState
msg_activate(0, true);
1175 // Up-cast to access OnMessageReceived, which is private in the agent.
1176 EXPECT_TRUE(static_cast<IPC::Listener
*>(password_autofill_agent_
)
1177 ->OnMessageReceived(msg_activate
));
1179 render_thread_
->sink().ClearMessages();
1180 SendVisiblePasswordForms();
1181 const IPC::Message
* message
= render_thread_
->sink().GetFirstMessageMatching(
1182 AutofillHostMsg_RecordSavePasswordProgress::ID
);
1183 EXPECT_TRUE(message
);
1186 // Test that logging can be turned off by a message.
1187 TEST_F(PasswordAutofillAgentTest
, OnChangeLoggingState_Deactivated
) {
1188 // Turn the logging on and then off.
1189 AutofillMsg_SetLoggingState
msg_activate(0, /*active=*/true);
1190 // Up-cast to access OnMessageReceived, which is private in the agent.
1191 EXPECT_TRUE(static_cast<IPC::Listener
*>(password_autofill_agent_
)
1192 ->OnMessageReceived(msg_activate
));
1193 AutofillMsg_SetLoggingState
msg_deactivate(0, /*active=*/false);
1194 EXPECT_TRUE(static_cast<IPC::Listener
*>(password_autofill_agent_
)
1195 ->OnMessageReceived(msg_deactivate
));
1197 render_thread_
->sink().ClearMessages();
1198 SendVisiblePasswordForms();
1199 const IPC::Message
* message
= render_thread_
->sink().GetFirstMessageMatching(
1200 AutofillHostMsg_RecordSavePasswordProgress::ID
);
1201 EXPECT_FALSE(message
);
1204 // Test that the agent sends an IPC call to get the current activity state of
1205 // password saving logging soon after construction.
1206 TEST_F(PasswordAutofillAgentTest
, SendsLoggingStateUpdatePingOnConstruction
) {
1207 const IPC::Message
* message
= render_thread_
->sink().GetFirstMessageMatching(
1208 AutofillHostMsg_PasswordAutofillAgentConstructed::ID
);
1209 EXPECT_TRUE(message
);
1212 // Tests that one user click on a username field is sufficient to bring up a
1213 // credential suggestion popup, and the user can autocomplete the password by
1214 // selecting the credential from the popup.
1215 TEST_F(PasswordAutofillAgentTest
, ClickAndSelect
) {
1216 // SimulateElementClick() is called so that a user gesture is actually made
1217 // and the password can be filled. However, SimulateElementClick() does not
1218 // actually lead to the AutofillAgent's InputElementClicked() method being
1219 // called, so SimulateSuggestionChoice has to manually call
1220 // InputElementClicked().
1221 ClearUsernameAndPasswordFields();
1222 SimulateOnFillPasswordForm(fill_data_
);
1223 SimulateElementClick(kUsernameName
);
1224 SimulateSuggestionChoice(username_element_
);
1225 CheckSuggestions(kAliceUsername
, true);
1227 CheckTextFieldsDOMState(kAliceUsername
, true, kAlicePassword
, true);
1230 // Tests the autosuggestions that are given when the element is clicked.
1231 // Specifically, tests when the user clicks on the username element after page
1232 // load and the element is autofilled, when the user clicks on an element that
1233 // has a non-matching username, and when the user clicks on an element that's
1234 // already been autofilled and they've already modified.
1235 TEST_F(PasswordAutofillAgentTest
, CredentialsOnClick
) {
1236 // Simulate the browser sending back the login info.
1237 SimulateOnFillPasswordForm(fill_data_
);
1239 // Clear the text fields to start fresh.
1240 ClearUsernameAndPasswordFields();
1242 // Call SimulateElementClick() to produce a user gesture on the page so
1243 // autofill will actually fill.
1244 SimulateElementClick(kUsernameName
);
1246 // Simulate a user clicking on the username element. This should produce a
1247 // message with all the usernames.
1248 render_thread_
->sink().ClearMessages();
1249 static_cast<PageClickListener
*>(autofill_agent_
)
1250 ->FormControlElementClicked(username_element_
, false);
1251 CheckSuggestions(std::string(), false);
1253 // Now simulate a user typing in an unrecognized username and then
1254 // clicking on the username element. This should also produce a message with
1255 // all the usernames.
1256 SimulateUsernameChange("baz");
1257 render_thread_
->sink().ClearMessages();
1258 static_cast<PageClickListener
*>(autofill_agent_
)
1259 ->FormControlElementClicked(username_element_
, true);
1260 CheckSuggestions("baz", true);
1261 ClearUsernameAndPasswordFields();
1264 // Tests that there are no autosuggestions from the password manager when the
1265 // user clicks on the password field and the username field is editable when
1266 // FillOnAccountSelect is enabled.
1267 TEST_F(PasswordAutofillAgentTest
,
1268 FillOnAccountSelectOnlyNoCredentialsOnPasswordClick
) {
1269 base::CommandLine::ForCurrentProcess()->AppendSwitch(
1270 autofill::switches::kEnableFillOnAccountSelect
);
1272 // Simulate the browser sending back the login info.
1273 SimulateOnFillPasswordForm(fill_data_
);
1275 // Clear the text fields to start fresh.
1276 ClearUsernameAndPasswordFields();
1278 // Call SimulateElementClick() to produce a user gesture on the page so
1279 // autofill will actually fill.
1280 SimulateElementClick(kUsernameName
);
1282 // Simulate a user clicking on the password element. This should produce no
1284 render_thread_
->sink().ClearMessages();
1285 static_cast<PageClickListener
*>(autofill_agent_
)
1286 ->FormControlElementClicked(password_element_
, false);
1287 EXPECT_FALSE(render_thread_
->sink().GetFirstMessageMatching(
1288 AutofillHostMsg_ShowPasswordSuggestions::ID
));
1291 // Tests the autosuggestions that are given when a password element is clicked,
1292 // the username element is not editable, and FillOnAccountSelect is enabled.
1293 // Specifically, tests when the user clicks on the password element after page
1294 // load, and the corresponding username element is readonly (and thus
1295 // uneditable), that the credentials for the already-filled username are
1297 TEST_F(PasswordAutofillAgentTest
,
1298 FillOnAccountSelectOnlyCredentialsOnPasswordClick
) {
1299 base::CommandLine::ForCurrentProcess()->AppendSwitch(
1300 autofill::switches::kEnableFillOnAccountSelect
);
1302 // Simulate the browser sending back the login info.
1303 SimulateOnFillPasswordForm(fill_data_
);
1305 // Clear the text fields to start fresh.
1306 ClearUsernameAndPasswordFields();
1308 // Simulate the page loading with a prefilled username element that is
1310 username_element_
.setValue("alicia");
1311 SetElementReadOnly(username_element_
, true);
1313 // Call SimulateElementClick() to produce a user gesture on the page so
1314 // autofill will actually fill.
1315 SimulateElementClick(kUsernameName
);
1317 // Simulate a user clicking on the password element. This should produce a
1318 // message with "alicia" suggested as the credential.
1319 render_thread_
->sink().ClearMessages();
1320 static_cast<PageClickListener
*>(autofill_agent_
)
1321 ->FormControlElementClicked(password_element_
, false);
1322 CheckSuggestions("alicia", false);
1325 // Tests that there are no autosuggestions from the password manager when the
1326 // user clicks on the password field (not the username field).
1327 TEST_F(PasswordAutofillAgentTest
, NoCredentialsOnPasswordClick
) {
1328 // Simulate the browser sending back the login info.
1329 SimulateOnFillPasswordForm(fill_data_
);
1331 // Clear the text fields to start fresh.
1332 ClearUsernameAndPasswordFields();
1334 // Call SimulateElementClick() to produce a user gesture on the page so
1335 // autofill will actually fill.
1336 SimulateElementClick(kUsernameName
);
1338 // Simulate a user clicking on the password element. This should produce no
1340 render_thread_
->sink().ClearMessages();
1341 static_cast<PageClickListener
*>(autofill_agent_
)
1342 ->FormControlElementClicked(password_element_
, false);
1343 EXPECT_FALSE(render_thread_
->sink().GetFirstMessageMatching(
1344 AutofillHostMsg_ShowPasswordSuggestions::ID
));
1347 // The user types in a username and a password, but then just before sending
1348 // the form off, a script clears them. This test checks that
1349 // PasswordAutofillAgent can still remember the username and the password
1350 // typed by the user.
1351 TEST_F(PasswordAutofillAgentTest
,
1352 RememberLastNonEmptyUsernameAndPasswordOnSubmit_ScriptCleared
) {
1353 SimulateUsernameChange("temp");
1354 SimulatePasswordChange("random");
1356 // Simulate that the username and the password value was cleared by the
1357 // site's JavaScript before submit.
1358 username_element_
.setValue(WebString());
1359 password_element_
.setValue(WebString());
1360 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1361 ->WillSubmitForm(username_element_
.form());
1363 // Observe that the PasswordAutofillAgent still remembered the last non-empty
1364 // username and password and sent that to the browser.
1365 ExpectFormSubmittedWithUsernameAndPasswords("temp", "random", "");
1368 // Similar to RememberLastNonEmptyPasswordOnSubmit_ScriptCleared, but this time
1369 // it's the user who clears the username and the password. This test checks
1370 // that in that case, the last non-empty username and password are not
1372 TEST_F(PasswordAutofillAgentTest
,
1373 RememberLastNonEmptyUsernameAndPasswordOnSubmit_UserCleared
) {
1374 SimulateUsernameChange("temp");
1375 SimulatePasswordChange("random");
1377 // Simulate that the user actually cleared the username and password again.
1378 SimulateUsernameChange("");
1379 SimulatePasswordChange("");
1380 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1381 ->WillSubmitForm(username_element_
.form());
1383 // Observe that the PasswordAutofillAgent respects the user having cleared the
1385 ExpectFormSubmittedWithUsernameAndPasswords("", "", "");
1388 // Similar to RememberLastNonEmptyPasswordOnSubmit_ScriptCleared, but uses the
1389 // new password instead of the current password.
1390 TEST_F(PasswordAutofillAgentTest
,
1391 RememberLastNonEmptyUsernameAndPasswordOnSubmit_New
) {
1392 const char kNewPasswordFormHTML
[] =
1393 "<FORM name='LoginTestForm'>"
1394 " <INPUT type='text' id='username' autocomplete='username'/>"
1395 " <INPUT type='password' id='password' autocomplete='new-password'/>"
1396 " <INPUT type='submit' value='Login'/>"
1398 LoadHTML(kNewPasswordFormHTML
);
1399 UpdateUsernameAndPasswordElements();
1401 SimulateUsernameChange("temp");
1402 SimulatePasswordChange("random");
1404 // Simulate that the username and the password value was cleared by
1405 // the site's JavaScript before submit.
1406 username_element_
.setValue(WebString());
1407 password_element_
.setValue(WebString());
1408 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1409 ->WillSubmitForm(username_element_
.form());
1411 // Observe that the PasswordAutofillAgent still remembered the last non-empty
1412 // password and sent that to the browser.
1413 ExpectFormSubmittedWithUsernameAndPasswords("temp", "", "random");
1416 // The user first accepts a suggestion, but then overwrites the password. This
1417 // test checks that the overwritten password is not reverted back if the user
1418 // triggers autofill through focusing (but not changing) the username again.
1419 TEST_F(PasswordAutofillAgentTest
,
1420 NoopEditingDoesNotOverwriteManuallyEditedPassword
) {
1421 // Simulate having credentials which needed to wait until the user starts
1422 // typing the username to be filled (e.g., PSL-matched credentials). Those are
1423 // the ones which can be filled as a result of TextFieldDidEndEditing.
1424 fill_data_
.wait_for_username
= true;
1425 SimulateOnFillPasswordForm(fill_data_
);
1426 // Simulate that the user typed her name to make the autofill work.
1427 SimulateUsernameChange(kAliceUsername
);
1428 SimulateDidEndEditing(GetMainFrame(), username_element_
);
1429 const std::string
old_username(username_element_
.value().utf8());
1430 const std::string
old_password(password_element_
.value().utf8());
1431 const std::string
new_password(old_password
+ "modify");
1433 // The user changes the password.
1434 SimulatePasswordChange(new_password
);
1436 // The user switches back into the username field, but leaves that without
1438 SimulateDidEndEditing(GetMainFrame(), username_element_
);
1440 // The password should have stayed as the user changed it.
1441 CheckTextFieldsDOMState(old_username
, true, new_password
, false);
1442 // The password should not have a suggested value.
1443 CheckTextFieldsState(old_username
, true, std::string(), false);
1446 // The user types in a username and a password, but then just before sending
1447 // the form off, a script changes them. This test checks that
1448 // PasswordAutofillAgent can still remember the username and the password
1449 // typed by the user.
1450 TEST_F(PasswordAutofillAgentTest
,
1451 RememberLastTypedUsernameAndPasswordOnSubmit_ScriptChanged
) {
1452 SimulateUsernameChange("temp");
1453 SimulatePasswordChange("random");
1455 // Simulate that the username and the password value was changed by the
1456 // site's JavaScript before submit.
1457 username_element_
.setValue(WebString("new username"));
1458 password_element_
.setValue(WebString("new password"));
1459 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1460 ->WillSendSubmitEvent(username_element_
.form());
1461 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1462 ->WillSubmitForm(username_element_
.form());
1464 // Observe that the PasswordAutofillAgent still remembered the last typed
1465 // username and password and sent that to the browser.
1466 ExpectFormSubmittedWithUsernameAndPasswords("temp", "random", "");
1469 // The username/password is autofilled by password manager then just before
1470 // sending the form off, a script changes them. This test checks that
1471 // PasswordAutofillAgent can still get the username and the password autofilled.
1472 TEST_F(PasswordAutofillAgentTest
,
1473 RememberLastAutofilledUsernameAndPasswordOnSubmit_ScriptChanged
) {
1474 SimulateOnFillPasswordForm(fill_data_
);
1476 // Simulate that the username and the password value was changed by the
1477 // site's JavaScript before submit.
1478 username_element_
.setValue(WebString("new username"));
1479 password_element_
.setValue(WebString("new password"));
1480 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1481 ->WillSendSubmitEvent(username_element_
.form());
1482 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1483 ->WillSubmitForm(username_element_
.form());
1485 // Observe that the PasswordAutofillAgent still remembered the autofilled
1486 // username and password and sent that to the browser.
1487 ExpectFormSubmittedWithUsernameAndPasswords(kAliceUsername
, kAlicePassword
,
1491 // The username/password is autofilled by password manager then user types in a
1492 // username and a password. Then just before sending the form off, a script
1493 // changes them. This test checks that PasswordAutofillAgent can still remember
1494 // the username and the password typed by the user.
1496 PasswordAutofillAgentTest
,
1497 RememberLastTypedAfterAutofilledUsernameAndPasswordOnSubmit_ScriptChanged
) {
1498 SimulateOnFillPasswordForm(fill_data_
);
1500 SimulateUsernameChange("temp");
1501 SimulatePasswordChange("random");
1503 // Simulate that the username and the password value was changed by the
1504 // site's JavaScript before submit.
1505 username_element_
.setValue(WebString("new username"));
1506 password_element_
.setValue(WebString("new password"));
1507 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1508 ->WillSendSubmitEvent(username_element_
.form());
1509 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1510 ->WillSubmitForm(username_element_
.form());
1512 // Observe that the PasswordAutofillAgent still remembered the last typed
1513 // username and password and sent that to the browser.
1514 ExpectFormSubmittedWithUsernameAndPasswords("temp", "random", "");
1517 // The user starts typing username then it is autofilled.
1518 // PasswordAutofillAgent should remember the username that was autofilled,
1520 TEST_F(PasswordAutofillAgentTest
, RememberAutofilledUsername
) {
1521 SimulateUsernameChange("Te");
1522 // Simulate that the username was changed by autofilling.
1523 username_element_
.setValue(WebString("temp"));
1524 SimulatePasswordChange("random");
1526 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1527 ->WillSendSubmitEvent(username_element_
.form());
1528 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1529 ->WillSubmitForm(username_element_
.form());
1531 // Observe that the PasswordAutofillAgent still remembered the last typed
1532 // username and password and sent that to the browser.
1533 ExpectFormSubmittedWithUsernameAndPasswords("temp", "random", "");
1536 TEST_F(PasswordAutofillAgentTest
, FormFillDataMustHaveUsername
) {
1537 ClearUsernameAndPasswordFields();
1539 PasswordFormFillData no_username_fill_data
= fill_data_
;
1540 no_username_fill_data
.username_field
.name
= base::string16();
1541 SimulateOnFillPasswordForm(no_username_fill_data
);
1543 // The username and password should not have been autocompleted.
1544 CheckTextFieldsState("", false, "", false);
1547 TEST_F(PasswordAutofillAgentTest
, FillOnAccountSelectOnly
) {
1548 base::CommandLine::ForCurrentProcess()->AppendSwitch(
1549 autofill::switches::kEnableFillOnAccountSelect
);
1551 ClearUsernameAndPasswordFields();
1553 // Simulate the browser sending back the login info for an initial page load.
1554 SimulateOnFillPasswordForm(fill_data_
);
1556 CheckTextFieldsState(std::string(), true, std::string(), false);
1559 TEST_F(PasswordAutofillAgentTest
, FillOnAccountSelectOnlyReadonlyUsername
) {
1560 base::CommandLine::ForCurrentProcess()->AppendSwitch(
1561 autofill::switches::kEnableFillOnAccountSelect
);
1563 ClearUsernameAndPasswordFields();
1565 username_element_
.setValue("alice");
1566 SetElementReadOnly(username_element_
, true);
1568 // Simulate the browser sending back the login info for an initial page load.
1569 SimulateOnFillPasswordForm(fill_data_
);
1571 CheckTextFieldsState(std::string("alice"), false, std::string(), true);
1574 TEST_F(PasswordAutofillAgentTest
,
1575 FillOnAccountSelectOnlyReadonlyNotPreferredUsername
) {
1576 base::CommandLine::ForCurrentProcess()->AppendSwitch(
1577 autofill::switches::kEnableFillOnAccountSelect
);
1579 ClearUsernameAndPasswordFields();
1581 username_element_
.setValue("Carol");
1582 SetElementReadOnly(username_element_
, true);
1584 // Simulate the browser sending back the login info for an initial page load.
1585 SimulateOnFillPasswordForm(fill_data_
);
1587 CheckTextFieldsState(std::string("Carol"), false, std::string(), true);
1590 TEST_F(PasswordAutofillAgentTest
, FillOnAccountSelectOnlyNoUsername
) {
1591 base::CommandLine::ForCurrentProcess()->AppendSwitch(
1592 autofill::switches::kEnableFillOnAccountSelect
);
1594 // Load a form with no username and update test data.
1595 LoadHTML(kVisibleFormWithNoUsernameHTML
);
1596 username_element_
.reset();
1597 WebDocument document
= GetMainFrame()->document();
1598 WebElement element
=
1599 document
.getElementById(WebString::fromUTF8(kPasswordName
));
1600 ASSERT_FALSE(element
.isNull());
1601 password_element_
= element
.to
<blink::WebInputElement
>();
1602 fill_data_
.username_field
= FormFieldData();
1603 UpdateOriginForHTML(kVisibleFormWithNoUsernameHTML
);
1604 fill_data_
.additional_logins
.clear();
1605 fill_data_
.other_possible_usernames
.clear();
1607 password_element_
.setValue("");
1608 password_element_
.setAutofilled(false);
1610 // Simulate the browser sending back the login info for an initial page load.
1611 SimulateOnFillPasswordForm(fill_data_
);
1613 EXPECT_TRUE(password_element_
.suggestedValue().isEmpty());
1614 EXPECT_TRUE(password_element_
.isAutofilled());
1617 TEST_F(PasswordAutofillAgentTest
, ShowPopupOnEmptyPasswordField
) {
1618 // Load a form with no username and update test data.
1619 LoadHTML(kVisibleFormWithNoUsernameHTML
);
1620 username_element_
.reset();
1621 WebDocument document
= GetMainFrame()->document();
1622 WebElement element
=
1623 document
.getElementById(WebString::fromUTF8(kPasswordName
));
1624 ASSERT_FALSE(element
.isNull());
1625 password_element_
= element
.to
<blink::WebInputElement
>();
1626 fill_data_
.username_field
= FormFieldData();
1627 UpdateOriginForHTML(kVisibleFormWithNoUsernameHTML
);
1628 fill_data_
.additional_logins
.clear();
1629 fill_data_
.other_possible_usernames
.clear();
1631 password_element_
.setValue("");
1632 password_element_
.setAutofilled(false);
1634 // Simulate the browser sending back the login info for an initial page load.
1635 SimulateOnFillPasswordForm(fill_data_
);
1637 // Show popup suggesstion when the password field is empty.
1638 password_element_
.setValue("");
1639 password_element_
.setAutofilled(false);
1641 SimulateSuggestionChoiceOfUsernameAndPassword(
1642 password_element_
, base::string16(), ASCIIToUTF16(kAlicePassword
));
1643 CheckSuggestions(std::string(), false);
1644 EXPECT_EQ(ASCIIToUTF16(kAlicePassword
), password_element_
.value());
1645 EXPECT_TRUE(password_element_
.isAutofilled());
1648 TEST_F(PasswordAutofillAgentTest
, ShowPopupOnAutofilledPasswordField
) {
1649 // Load a form with no username and update test data.
1650 LoadHTML(kVisibleFormWithNoUsernameHTML
);
1651 username_element_
.reset();
1652 WebDocument document
= GetMainFrame()->document();
1653 WebElement element
=
1654 document
.getElementById(WebString::fromUTF8(kPasswordName
));
1655 ASSERT_FALSE(element
.isNull());
1656 password_element_
= element
.to
<blink::WebInputElement
>();
1657 fill_data_
.username_field
= FormFieldData();
1658 UpdateOriginForHTML(kVisibleFormWithNoUsernameHTML
);
1659 fill_data_
.additional_logins
.clear();
1660 fill_data_
.other_possible_usernames
.clear();
1662 password_element_
.setValue("");
1663 password_element_
.setAutofilled(false);
1665 // Simulate the browser sending back the login info for an initial page load.
1666 SimulateOnFillPasswordForm(fill_data_
);
1668 // Show popup suggesstion when the password field is autofilled.
1669 password_element_
.setValue("123");
1670 password_element_
.setAutofilled(true);
1672 SimulateSuggestionChoiceOfUsernameAndPassword(
1673 password_element_
, base::string16(), ASCIIToUTF16(kAlicePassword
));
1674 CheckSuggestions(std::string(), false);
1675 EXPECT_EQ(ASCIIToUTF16(kAlicePassword
), password_element_
.value());
1676 EXPECT_TRUE(password_element_
.isAutofilled());
1679 TEST_F(PasswordAutofillAgentTest
, NotShowPopupPasswordField
) {
1680 // Load a form with no username and update test data.
1681 LoadHTML(kVisibleFormWithNoUsernameHTML
);
1682 username_element_
.reset();
1683 WebDocument document
= GetMainFrame()->document();
1684 WebElement element
=
1685 document
.getElementById(WebString::fromUTF8(kPasswordName
));
1686 ASSERT_FALSE(element
.isNull());
1687 password_element_
= element
.to
<blink::WebInputElement
>();
1688 fill_data_
.username_field
= FormFieldData();
1689 UpdateOriginForHTML(kVisibleFormWithNoUsernameHTML
);
1690 fill_data_
.additional_logins
.clear();
1691 fill_data_
.other_possible_usernames
.clear();
1693 password_element_
.setValue("");
1694 password_element_
.setAutofilled(false);
1696 // Simulate the browser sending back the login info for an initial page load.
1697 SimulateOnFillPasswordForm(fill_data_
);
1699 // Do not show popup suggesstion when the password field is not-empty and not
1701 password_element_
.setValue("123");
1702 password_element_
.setAutofilled(false);
1704 SimulateSuggestionChoiceOfUsernameAndPassword(
1705 password_element_
, base::string16(), ASCIIToUTF16(kAlicePassword
));
1706 ASSERT_FALSE(render_thread_
->sink().GetFirstMessageMatching(
1707 AutofillHostMsg_ShowPasswordSuggestions::ID
));
1710 // Tests with fill-on-account-select enabled that if the username element is
1711 // read-only and filled with an unknown username, then the password field is not
1712 // highlighted as autofillable (regression test for https://crbug.com/442564).
1713 TEST_F(PasswordAutofillAgentTest
,
1714 FillOnAccountSelectOnlyReadonlyUnknownUsername
) {
1715 base::CommandLine::ForCurrentProcess()->AppendSwitch(
1716 autofill::switches::kEnableFillOnAccountSelect
);
1718 ClearUsernameAndPasswordFields();
1720 username_element_
.setValue("foobar");
1721 SetElementReadOnly(username_element_
, true);
1723 CheckTextFieldsState(std::string("foobar"), false, std::string(), false);
1726 // Test that the last plain text field before a password field is chosen as a
1727 // username, in a form with 2 plain text fields without username predictions.
1728 TEST_F(PasswordAutofillAgentTest
, FindingUsernameWithoutAutofillPredictions
) {
1729 LoadHTML(kFormHTMLWithTwoTextFields
);
1730 UpdateUsernameAndPasswordElements();
1731 blink::WebInputElement email_element
= GetInputElementByID(kEmailName
);
1732 SimulateUsernameChange("temp");
1733 SimulateUserInputChangeForElement(&email_element
, "temp@google.com");
1734 SimulatePasswordChange("random");
1735 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1736 ->WillSendSubmitEvent(username_element_
.form());
1737 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1738 ->WillSubmitForm(username_element_
.form());
1740 // Observe that the PasswordAutofillAgent identifies the second field (e-mail)
1742 ExpectFormSubmittedWithUsernameAndPasswords("temp@google.com", "random", "");
1745 // Tests that field predictions are followed when identifying the username
1746 // and password in a password form with two plain text fields.
1747 TEST_F(PasswordAutofillAgentTest
, FindingFieldsWithAutofillPredictions
) {
1748 LoadHTML(kFormHTMLWithTwoTextFields
);
1749 UpdateUsernameAndPasswordElements();
1750 blink::WebInputElement email_element
= GetInputElementByID(kEmailName
);
1751 SimulateUsernameChange("temp");
1752 SimulateUserInputChangeForElement(&email_element
, "temp@google.com");
1753 SimulatePasswordChange("random");
1754 // Find FormData for visible password form.
1755 blink::WebFormElement form_element
= username_element_
.form();
1757 ASSERT_TRUE(WebFormElementToFormData(form_element
,
1758 blink::WebFormControlElement(),
1759 EXTRACT_NONE
, &form_data
, nullptr));
1760 // Simulate Autofill predictions: the first field is username, the third
1762 std::map
<autofill::FormData
, PasswordFormFieldPredictionMap
> predictions
;
1763 predictions
[form_data
][form_data
.fields
[0]] = PREDICTION_USERNAME
;
1764 predictions
[form_data
][form_data
.fields
[2]] = PREDICTION_NEW_PASSWORD
;
1765 AutofillMsg_AutofillUsernameAndPasswordDataReceived
msg(0, predictions
);
1766 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1767 ->OnMessageReceived(msg
);
1769 // The predictions should still match even if the form changes, as long
1770 // as the particular elements don't change.
1771 std::string add_field_to_form
=
1772 "var form = document.getElementById('LoginTestForm');"
1773 "var new_input = document.createElement('input');"
1774 "new_input.setAttribute('type', 'text');"
1775 "new_input.setAttribute('id', 'other_field');"
1776 "form.appendChild(new_input);";
1777 ExecuteJavaScriptForTests(add_field_to_form
.c_str());
1779 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1780 ->WillSendSubmitEvent(username_element_
.form());
1781 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1782 ->WillSubmitForm(username_element_
.form());
1784 // Observe that the PasswordAutofillAgent identifies the first field as
1786 // TODO(msramek): We should also test that adding another password field
1787 // won't override the password field prediction either. However, the password
1788 // field predictions are not taken into account yet.
1789 ExpectFormSubmittedWithUsernameAndPasswords("temp", "random", "");
1792 // The user types in a username and a password. Then JavaScript changes password
1793 // field to readonly state before submit. PasswordAutofillAgent can correctly
1794 // process readonly password field. This test models behaviour of gmail.com.
1795 TEST_F(PasswordAutofillAgentTest
, ReadonlyPasswordFieldOnSubmit
) {
1796 SimulateUsernameChange("temp");
1797 SimulatePasswordChange("random");
1799 // Simulate that JavaScript makes password field readonly.
1800 SetElementReadOnly(password_element_
, true);
1801 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1802 ->WillSubmitForm(username_element_
.form());
1804 // Observe that the PasswordAutofillAgent can correctly process submitted
1806 ExpectFormSubmittedWithUsernameAndPasswords("temp", "random", "");
1809 // Verify that typed passwords are saved correctly when autofill and generation
1810 // both trigger. Regression test for https://crbug.com/493455
1811 TEST_F(PasswordAutofillAgentTest
, PasswordGenerationTriggered_TypedPassword
) {
1812 SimulateOnFillPasswordForm(fill_data_
);
1814 SetNotBlacklistedMessage(password_generation_
, kFormHTML
);
1815 SetAccountCreationFormsDetectedMessage(password_generation_
,
1816 GetMainFrame()->document(),
1819 SimulateUsernameChange("NewGuy");
1820 SimulatePasswordChange("NewPassword");
1821 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1822 ->WillSendSubmitEvent(username_element_
.form());
1823 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1824 ->WillSubmitForm(username_element_
.form());
1826 ExpectFormSubmittedWithUsernameAndPasswords("NewGuy", "NewPassword", "");
1829 // Verify that generated passwords are saved correctly when autofill and
1830 // generation both trigger. Regression test for https://crbug.com/493455.
1831 TEST_F(PasswordAutofillAgentTest
,
1832 PasswordGenerationTriggered_GeneratedPassword
) {
1833 SimulateOnFillPasswordForm(fill_data_
);
1835 SetNotBlacklistedMessage(password_generation_
, kFormHTML
);
1836 SetAccountCreationFormsDetectedMessage(password_generation_
,
1837 GetMainFrame()->document(),
1840 base::string16 password
= base::ASCIIToUTF16("NewPass22");
1841 AutofillMsg_GeneratedPasswordAccepted
msg(0, password
);
1842 password_generation_
->OnMessageReceived(msg
);
1844 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1845 ->WillSendSubmitEvent(username_element_
.form());
1846 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
1847 ->WillSubmitForm(username_element_
.form());
1849 ExpectFormSubmittedWithUsernameAndPasswords(kAliceUsername
, "NewPass22", "");
1852 // If password generation is enabled for a field, password autofill should not
1854 TEST_F(PasswordAutofillAgentTest
, PasswordGenerationSupersedesAutofill
) {
1855 LoadHTML(kSignupFormHTML
);
1857 // Update password_element_;
1858 WebDocument document
= GetMainFrame()->document();
1859 WebElement element
=
1860 document
.getElementById(WebString::fromUTF8("new_password"));
1861 ASSERT_FALSE(element
.isNull());
1862 password_element_
= element
.to
<blink::WebInputElement
>();
1864 // Update fill_data_ for the new form and simulate filling. Pretend as if
1865 // the password manager didn't detect a username field so it will try to
1866 // show UI when the password field is focused.
1867 fill_data_
.wait_for_username
= true;
1868 fill_data_
.username_field
= FormFieldData();
1869 fill_data_
.password_field
.name
= base::ASCIIToUTF16("new_password");
1870 UpdateOriginForHTML(kSignupFormHTML
);
1871 SimulateOnFillPasswordForm(fill_data_
);
1873 // Simulate generation triggering.
1874 SetNotBlacklistedMessage(password_generation_
,
1876 SetAccountCreationFormsDetectedMessage(password_generation_
,
1877 GetMainFrame()->document(),
1880 // Simulate the field being clicked to start typing. This should trigger
1881 // generation but not password autofill.
1882 SetFocused(password_element_
);
1883 SimulateElementClick("new_password");
1885 render_thread_
->sink().GetFirstMessageMatching(
1886 AutofillHostMsg_ShowPasswordSuggestions::ID
));
1887 EXPECT_TRUE(render_thread_
->sink().GetFirstMessageMatching(
1888 AutofillHostMsg_ShowPasswordGenerationPopup::ID
));
1891 // Tests that a password change form is properly filled with the username and
1893 TEST_F(PasswordAutofillAgentTest
, FillSuggestionPasswordChangeForms
) {
1894 LoadHTML(kPasswordChangeFormHTML
);
1895 UpdateOriginForHTML(kPasswordChangeFormHTML
);
1896 UpdateUsernameAndPasswordElements();
1897 // Simulate the browser sending the login info, but set |wait_for_username|
1898 // to prevent the form from being immediately filled.
1899 fill_data_
.wait_for_username
= true;
1900 fill_data_
.is_possible_change_password_form
= true;
1901 SimulateOnFillPasswordForm(fill_data_
);
1903 // Neither field should have been autocompleted.
1904 CheckTextFieldsDOMState(std::string(), false, std::string(), false);
1906 EXPECT_TRUE(password_autofill_agent_
->FillSuggestion(
1907 username_element_
, kAliceUsername
, kAlicePassword
));
1908 CheckTextFieldsDOMState(kAliceUsername
, true, kAlicePassword
, true);
1911 // Tests that a password change form is properly filled with the password when
1912 // the user click on the password field.
1913 TEST_F(PasswordAutofillAgentTest
,
1914 FillSuggestionPasswordChangeFormsOnlyPassword
) {
1915 LoadHTML(kPasswordChangeFormHTML
);
1916 UpdateOriginForHTML(kPasswordChangeFormHTML
);
1917 UpdateUsernameAndPasswordElements();
1918 // Simulate the browser sending the login info, but set |wait_for_username|
1919 // to prevent the form from being immediately filled.
1920 fill_data_
.wait_for_username
= true;
1921 fill_data_
.is_possible_change_password_form
= true;
1922 SimulateOnFillPasswordForm(fill_data_
);
1924 // Neither field should have been autocompleted.
1925 CheckTextFieldsDOMState(std::string(), false, std::string(), false);
1927 EXPECT_TRUE(password_autofill_agent_
->FillSuggestion(
1928 password_element_
, kAliceUsername
, kAlicePassword
));
1929 CheckTextFieldsDOMState("", false, kAlicePassword
, true);
1932 // Tests that one user click on a username field is sufficient to bring up a
1933 // credential suggestion popup on a change password form.
1934 TEST_F(PasswordAutofillAgentTest
,
1935 SuggestionsOnUsernameFieldOfChangePasswordForm
) {
1936 LoadHTML(kPasswordChangeFormHTML
);
1937 UpdateOriginForHTML(kPasswordChangeFormHTML
);
1938 UpdateUsernameAndPasswordElements();
1940 ClearUsernameAndPasswordFields();
1941 fill_data_
.wait_for_username
= true;
1942 fill_data_
.is_possible_change_password_form
= true;
1943 SimulateOnFillPasswordForm(fill_data_
);
1944 // Simulate a user clicking on the username element. This should produce a
1946 render_thread_
->sink().ClearMessages();
1947 static_cast<PageClickListener
*>(autofill_agent_
)
1948 ->FormControlElementClicked(username_element_
, true);
1949 CheckSuggestions("", true);
1952 // Tests that one user click on a password field is sufficient to bring up a
1953 // credential suggestion popup on a change password form.
1954 TEST_F(PasswordAutofillAgentTest
,
1955 SuggestionsOnPasswordFieldOfChangePasswordForm
) {
1956 LoadHTML(kPasswordChangeFormHTML
);
1957 UpdateOriginForHTML(kPasswordChangeFormHTML
);
1958 UpdateUsernameAndPasswordElements();
1960 ClearUsernameAndPasswordFields();
1961 fill_data_
.wait_for_username
= true;
1962 fill_data_
.is_possible_change_password_form
= true;
1963 SimulateOnFillPasswordForm(fill_data_
);
1964 // Simulate a user clicking on the password element. This should produce a
1966 render_thread_
->sink().ClearMessages();
1967 static_cast<PageClickListener
*>(autofill_agent_
)
1968 ->FormControlElementClicked(password_element_
, true);
1969 CheckSuggestions("", false);
1972 // Tests that there are no autosuggestions from the password manager when the
1973 // user clicks on the password field of change password form after the user
1974 // typed in the username field.
1975 TEST_F(PasswordAutofillAgentTest
,
1976 NoSuggestionsOnPasswordFieldOfChangePasswordFormAfterUsernameTyping
) {
1977 LoadHTML(kPasswordChangeFormHTML
);
1978 UpdateOriginForHTML(kPasswordChangeFormHTML
);
1979 UpdateUsernameAndPasswordElements();
1981 ClearUsernameAndPasswordFields();
1982 fill_data_
.wait_for_username
= true;
1983 fill_data_
.is_possible_change_password_form
= true;
1984 SimulateOnFillPasswordForm(fill_data_
);
1986 // Clear the text fields to start fresh.
1987 SimulateUsernameChange("temp");
1989 // Simulate a user clicking on the password element. This should produce no
1991 render_thread_
->sink().ClearMessages();
1992 static_cast<PageClickListener
*>(autofill_agent_
)
1993 ->FormControlElementClicked(password_element_
, false);
1994 EXPECT_FALSE(render_thread_
->sink().GetFirstMessageMatching(
1995 AutofillHostMsg_ShowPasswordSuggestions::ID
));
1998 // Tests that NOT_PASSWORD field predictions are followed so that no password
1999 // form is submitted.
2000 TEST_F(PasswordAutofillAgentTest
, IgnoreNotPasswordFields
) {
2001 LoadHTML(kCreditCardFormHTML
);
2002 blink::WebInputElement credit_card_owner_element
=
2003 GetInputElementByID(kCreditCardOwnerName
);
2004 blink::WebInputElement credit_card_number_element
=
2005 GetInputElementByID(kCreditCardNumberName
);
2006 blink::WebInputElement credit_card_verification_element
=
2007 GetInputElementByID(kCreditCardVerificationName
);
2008 SimulateUserInputChangeForElement(&credit_card_owner_element
, "JohnSmith");
2009 SimulateUserInputChangeForElement(&credit_card_number_element
,
2010 "1234123412341234");
2011 SimulateUserInputChangeForElement(&credit_card_verification_element
, "123");
2012 // Find FormData for visible form.
2013 blink::WebFormElement form_element
= credit_card_number_element
.form();
2015 ASSERT_TRUE(WebFormElementToFormData(form_element
,
2016 blink::WebFormControlElement(),
2017 EXTRACT_NONE
, &form_data
, nullptr));
2018 // Simulate Autofill predictions: the third field is not a password.
2019 std::map
<autofill::FormData
, PasswordFormFieldPredictionMap
> predictions
;
2020 predictions
[form_data
][form_data
.fields
[2]] = PREDICTION_NOT_PASSWORD
;
2021 AutofillMsg_AutofillUsernameAndPasswordDataReceived
msg(0, predictions
);
2022 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
2023 ->OnMessageReceived(msg
);
2025 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
2026 ->WillSendSubmitEvent(form_element
);
2027 static_cast<content::RenderFrameObserver
*>(password_autofill_agent_
)
2028 ->WillSubmitForm(form_element
);
2030 const IPC::Message
* message
= render_thread_
->sink().GetFirstMessageMatching(
2031 AutofillHostMsg_PasswordFormSubmitted::ID
);
2032 ASSERT_FALSE(message
);
2035 // Tests that only the password field is autocompleted when the browser sends
2036 // back data with only one credentials and empty username.
2037 TEST_F(PasswordAutofillAgentTest
, NotAutofillNoUsername
) {
2038 fill_data_
.username_field
.value
.clear();
2039 fill_data_
.additional_logins
.clear();
2040 SimulateOnFillPasswordForm(fill_data_
);
2042 CheckTextFieldsState("", false, kAlicePassword
, true);
2045 // Tests that both the username and the password fields are autocompleted
2046 // despite the empty username, when the browser sends back data more than one
2048 TEST_F(PasswordAutofillAgentTest
,
2049 AutofillNoUsernameWhenOtherCredentialsStored
) {
2050 fill_data_
.username_field
.value
.clear();
2051 ASSERT_FALSE(fill_data_
.additional_logins
.empty());
2052 SimulateOnFillPasswordForm(fill_data_
);
2054 CheckTextFieldsState("", true, kAlicePassword
, true);
2057 TEST_F(PasswordAutofillAgentTest
, NoForm_PromptForAJAXSubmitWithoutNavigation
) {
2058 LoadHTML(kNoFormHTML
);
2059 UpdateUsernameAndPasswordElements();
2061 SimulateUsernameChange("Bob");
2062 SimulatePasswordChange("mypassword");
2064 username_element_
.setAttribute("style", "display:none;");
2065 password_element_
.setAttribute("style", "display:none;");
2067 password_autofill_agent_
->AJAXSucceeded();
2069 ExpectInPageNavigationWithUsernameAndPasswords("Bob", "mypassword", "");
2072 TEST_F(PasswordAutofillAgentTest
,
2073 NoForm_NoPromptForAJAXSubmitWithoutNavigationAndElementsVisible
) {
2074 LoadHTML(kNoFormHTML
);
2075 UpdateUsernameAndPasswordElements();
2077 SimulateUsernameChange("Bob");
2078 SimulatePasswordChange("mypassword");
2080 password_autofill_agent_
->AJAXSucceeded();
2082 const IPC::Message
* message
=
2083 render_thread_
->sink().GetFirstMessageMatching(
2084 AutofillHostMsg_PasswordFormSubmitted::ID
);
2085 ASSERT_FALSE(message
);
2088 } // namespace autofill