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 #ifndef CHROME_BROWSER_UI_LOGIN_LOGIN_PROMPT_H_
6 #define CHROME_BROWSER_UI_LOGIN_LOGIN_PROMPT_H_
10 #include "base/basictypes.h"
11 #include "base/synchronization/lock.h"
12 #include "chrome/browser/password_manager/password_manager.h"
13 #include "content/public/browser/notification_observer.h"
14 #include "content/public/browser/resource_dispatcher_host_login_delegate.h"
19 class RenderViewHostDelegate
;
20 class NotificationRegistrar
;
21 } // namespace content
24 class AuthChallengeInfo
;
25 class HttpNetworkSession
;
29 // This is the base implementation for the OS-specific classes that route
30 // authentication info to the net::URLRequest that needs it. These functions
31 // must be implemented in a thread safe manner.
32 class LoginHandler
: public content::ResourceDispatcherHostLoginDelegate
,
33 public LoginModelObserver
,
34 public content::NotificationObserver
{
36 LoginHandler(net::AuthChallengeInfo
* auth_info
, net::URLRequest
* request
);
38 // Builds the platform specific LoginHandler. Used from within
39 // CreateLoginPrompt() which creates tasks.
40 static LoginHandler
* Create(net::AuthChallengeInfo
* auth_info
,
41 net::URLRequest
* request
);
43 // ResourceDispatcherHostLoginDelegate implementation:
44 virtual void OnRequestCancelled() OVERRIDE
;
46 // Initializes the underlying platform specific view.
47 virtual void BuildViewForPasswordManager(
48 PasswordManager
* manager
,
49 const base::string16
& explanation
) = 0;
51 // Sets information about the authentication type (|form|) and the
52 // |password_manager| for this profile.
53 void SetPasswordForm(const autofill::PasswordForm
& form
);
54 void SetPasswordManager(PasswordManager
* password_manager
);
56 // Returns the WebContents that needs authentication.
57 content::WebContents
* GetWebContentsForLogin() const;
59 // Resend the request with authentication credentials.
60 // This function can be called from either thread.
61 void SetAuth(const base::string16
& username
, const base::string16
& password
);
63 // Display the error page without asking for credentials again.
64 // This function can be called from either thread.
67 // Implements the content::NotificationObserver interface.
68 // Listens for AUTH_SUPPLIED and AUTH_CANCELLED notifications from other
69 // LoginHandlers so that this LoginHandler has the chance to dismiss itself
70 // if it was waiting for the same authentication.
71 virtual void Observe(int type
,
72 const content::NotificationSource
& source
,
73 const content::NotificationDetails
& details
) OVERRIDE
;
75 // Who/where/what asked for the authentication.
76 const net::AuthChallengeInfo
* auth_info() const { return auth_info_
.get(); }
78 // Returns whether authentication had been handled (SetAuth or CancelAuth).
79 bool WasAuthHandled() const;
82 virtual ~LoginHandler();
84 void SetModel(LoginModel
* model
);
86 // Notify observers that authentication is needed.
87 void NotifyAuthNeeded();
89 // Performs necessary cleanup before deletion.
92 // Closes the native dialog.
93 virtual void CloseDialog() = 0;
96 // Starts observing notifications from other LoginHandlers.
99 // Stops observing notifications from other LoginHandlers.
100 void RemoveObservers();
102 // Notify observers that authentication is supplied.
103 void NotifyAuthSupplied(const base::string16
& username
,
104 const base::string16
& password
);
106 // Notify observers that authentication is cancelled.
107 void NotifyAuthCancelled();
109 // Marks authentication as handled and returns the previous handled
111 bool TestAndSetAuthHandled();
113 // Calls SetAuth from the IO loop.
114 void SetAuthDeferred(const base::string16
& username
,
115 const base::string16
& password
);
117 // Calls CancelAuth from the IO loop.
118 void CancelAuthDeferred();
120 // Closes the view_contents from the UI loop.
121 void CloseContentsDeferred();
123 // True if we've handled auth (SetAuth or CancelAuth has been called).
125 mutable base::Lock handled_auth_lock_
;
127 // Who/where/what asked for the authentication.
128 scoped_refptr
<net::AuthChallengeInfo
> auth_info_
;
130 // The request that wants login data.
131 // This should only be accessed on the IO loop.
132 net::URLRequest
* request_
;
134 // The HttpNetworkSession |request_| is associated with.
135 const net::HttpNetworkSession
* http_network_session_
;
137 // The PasswordForm sent to the PasswordManager. This is so we can refer to it
138 // when later notifying the password manager if the credentials were accepted
140 // This should only be accessed on the UI loop.
141 autofill::PasswordForm password_form_
;
143 // Points to the password manager owned by the WebContents requesting auth.
144 // This should only be accessed on the UI loop.
145 PasswordManager
* password_manager_
;
147 // Cached from the net::URLRequest, in case it goes NULL on us.
148 int render_process_host_id_
;
149 int render_frame_id_
;
151 // If not null, points to a model we need to notify of our own destruction
152 // so it doesn't try and access this when its too late.
153 LoginModel
* login_model_
;
155 // Observes other login handlers so this login handler can respond.
156 // This is only accessed on the UI thread.
157 scoped_ptr
<content::NotificationRegistrar
> registrar_
;
160 // Details to provide the content::NotificationObserver. Used by the automation
161 // proxy for testing.
162 class LoginNotificationDetails
{
164 explicit LoginNotificationDetails(LoginHandler
* handler
)
165 : handler_(handler
) {}
166 LoginHandler
* handler() const { return handler_
; }
169 LoginNotificationDetails() {}
171 LoginHandler
* handler_
; // Where to send the response.
173 DISALLOW_COPY_AND_ASSIGN(LoginNotificationDetails
);
176 // Details to provide the NotificationObserver. Used by the automation proxy
177 // for testing and by other LoginHandlers to dismiss themselves when an
178 // identical auth is supplied.
179 class AuthSuppliedLoginNotificationDetails
: public LoginNotificationDetails
{
181 AuthSuppliedLoginNotificationDetails(LoginHandler
* handler
,
182 const base::string16
& username
,
183 const base::string16
& password
)
184 : LoginNotificationDetails(handler
),
186 password_(password
) {}
187 const base::string16
& username() const { return username_
; }
188 const base::string16
& password() const { return password_
; }
191 // The username that was used for the authentication.
192 const base::string16 username_
;
194 // The password that was used for the authentication.
195 const base::string16 password_
;
197 DISALLOW_COPY_AND_ASSIGN(AuthSuppliedLoginNotificationDetails
);
200 // Prompts the user for their username and password. This is designed to
201 // be called on the background (I/O) thread, in response to
202 // net::URLRequest::Delegate::OnAuthRequired. The prompt will be created
203 // on the main UI thread via a call to UI loop's InvokeLater, and will send the
204 // credentials back to the net::URLRequest on the calling thread.
205 // A LoginHandler object (which lives on the calling thread) is returned,
206 // which can be used to set or cancel authentication programmatically. The
207 // caller must invoke OnRequestCancelled() on this LoginHandler before
208 // destroying the net::URLRequest.
209 LoginHandler
* CreateLoginPrompt(net::AuthChallengeInfo
* auth_info
,
210 net::URLRequest
* request
);
212 // Helper to remove the ref from an net::URLRequest to the LoginHandler.
213 // Should only be called from the IO thread, since it accesses an
215 void ResetLoginHandlerForRequest(net::URLRequest
* request
);
217 // Get the signon_realm under which the identity should be saved.
218 std::string
GetSignonRealm(const GURL
& url
,
219 const net::AuthChallengeInfo
& auth_info
);
221 #endif // CHROME_BROWSER_UI_LOGIN_LOGIN_PROMPT_H_