Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(6357)

Unified Diff: chrome/browser/login_prompt.cc

Issue 1528012: Notify all active login prompts when one login prompt is submitted. This allo... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: Created 10 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « chrome/browser/login_prompt.h ('k') | chrome/browser/login_prompt_gtk.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: chrome/browser/login_prompt.cc
===================================================================
--- chrome/browser/login_prompt.cc (revision 43269)
+++ chrome/browser/login_prompt.cc (working copy)
@@ -67,18 +67,26 @@
// ----------------------------------------------------------------------------
// LoginHandler
-LoginHandler::LoginHandler(URLRequest* request)
+LoginHandler::LoginHandler(net::AuthChallengeInfo* auth_info,
+ URLRequest* request)
: handled_auth_(false),
dialog_(NULL),
+ auth_info_(auth_info),
request_(request),
password_manager_(NULL),
login_model_(NULL) {
// This constructor is called on the I/O thread, so we cannot load the nib
// here. BuildViewForPasswordManager() will be invoked on the UI thread
// later, so wait with loading the nib until then.
- DCHECK(request_) << "LoginHandlerMac constructed with NULL request";
+ DCHECK(request_) << "LoginHandler constructed with NULL request";
+ DCHECK(auth_info_) << "LoginHandler constructed with NULL auth info";
AddRef(); // matched by ReleaseSoon::ReleaseSoon().
+
+ ChromeThread::PostTask(
+ ChromeThread::UI, FROM_HERE,
+ NewRunnableMethod(this, &LoginHandler::AddObservers));
+
if (!ResourceDispatcherHost::RenderViewForRequest(
request_, &render_process_host_id_, &tab_contents_id_)) {
NOTREACHED();
@@ -121,11 +129,12 @@
NewRunnableMethod(this, &LoginHandler::CloseContentsDeferred));
ChromeThread::PostTask(
ChromeThread::UI, FROM_HERE,
- NewRunnableMethod(this, &LoginHandler::SendNotifications));
+ NewRunnableMethod(
+ this, &LoginHandler::NotifyAuthSupplied, username, password));
ChromeThread::PostTask(
ChromeThread::IO, FROM_HERE,
- NewRunnableMethod(this, &LoginHandler::SetAuthDeferred,
- username, password));
+ NewRunnableMethod(
+ this, &LoginHandler::SetAuthDeferred, username, password));
}
void LoginHandler::CancelAuth() {
@@ -137,7 +146,7 @@
NewRunnableMethod(this, &LoginHandler::CloseContentsDeferred));
ChromeThread::PostTask(
ChromeThread::UI, FROM_HERE,
- NewRunnableMethod(this, &LoginHandler::SendNotifications));
+ NewRunnableMethod(this, &LoginHandler::NotifyAuthCancelled));
ChromeThread::PostTask(
ChromeThread::IO, FROM_HERE,
NewRunnableMethod(this, &LoginHandler::CancelAuthDeferred));
@@ -154,6 +163,63 @@
CancelAuth();
}
+void LoginHandler::AddObservers() {
+ DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI));
+
+ registrar_.Add(this, NotificationType::AUTH_SUPPLIED,
+ NotificationService::AllSources());
+ registrar_.Add(this, NotificationType::AUTH_CANCELLED,
+ NotificationService::AllSources());
+}
+
+void LoginHandler::RemoveObservers() {
+ DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI));
+
+ registrar_.Remove(this, NotificationType::AUTH_SUPPLIED,
+ NotificationService::AllSources());
+ registrar_.Remove(this, NotificationType::AUTH_CANCELLED,
+ NotificationService::AllSources());
+
+ DCHECK(registrar_.IsEmpty());
+}
+
+void LoginHandler::Observe(NotificationType type,
+ const NotificationSource& source,
+ const NotificationDetails& details) {
+ DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI));
+ DCHECK(type == NotificationType::AUTH_SUPPLIED ||
+ type == NotificationType::AUTH_CANCELLED);
+
+ TabContents* requesting_contents = GetTabContentsForLogin();
+ if (!requesting_contents)
+ return;
+
+ NavigationController* this_controller = &requesting_contents->controller();
+ NavigationController* that_controller =
+ Source<NavigationController>(source).ptr();
+
+ // Only handle notifications from other handlers.
+ if (this_controller == that_controller)
+ return;
+
+ LoginNotificationDetails* login_details =
+ Details<LoginNotificationDetails>(details).ptr();
+
+ // Only handle notification for the identical auth info.
+ if (*login_details->handler()->auth_info() != *auth_info())
+ return;
+
+ // Set or cancel the auth in this handler.
+ if (type == NotificationType::AUTH_SUPPLIED) {
+ AuthSuppliedLoginNotificationDetails* supplied_details =
+ Details<AuthSuppliedLoginNotificationDetails>(details).ptr();
+ SetAuth(supplied_details->username(), supplied_details->password());
+ } else {
+ DCHECK(type == NotificationType::AUTH_CANCELLED);
+ CancelAuth();
+ }
+}
+
void LoginHandler::SetModel(LoginModel* model) {
if (login_model_)
login_model_->SetObserver(NULL);
@@ -166,30 +232,59 @@
dialog_ = dialog;
}
-// Notify observers that authentication is needed or received. The automation
-// proxy uses this for testing.
-void LoginHandler::SendNotifications() {
+void LoginHandler::NotifyAuthNeeded() {
DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI));
+ if (WasAuthHandled(false))
+ return;
+ TabContents* requesting_contents = GetTabContentsForLogin();
+ if (!requesting_contents)
+ return;
+
NotificationService* service = NotificationService::current();
+ NavigationController* controller = &requesting_contents->controller();
+ LoginNotificationDetails details(this);
+
+ service->Notify(NotificationType::AUTH_NEEDED,
+ Source<NavigationController>(controller),
+ Details<LoginNotificationDetails>(&details));
+}
+
+void LoginHandler::NotifyAuthCancelled() {
+ DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI));
+ DCHECK(WasAuthHandled(false));
+
TabContents* requesting_contents = GetTabContentsForLogin();
if (!requesting_contents)
return;
+ NotificationService* service = NotificationService::current();
NavigationController* controller = &requesting_contents->controller();
+ LoginNotificationDetails details(this);
- if (!WasAuthHandled(false)) {
- LoginNotificationDetails details(this);
- service->Notify(NotificationType::AUTH_NEEDED,
- Source<NavigationController>(controller),
- Details<LoginNotificationDetails>(&details));
- } else {
- service->Notify(NotificationType::AUTH_SUPPLIED,
- Source<NavigationController>(controller),
- NotificationService::NoDetails());
- }
+ service->Notify(NotificationType::AUTH_CANCELLED,
+ Source<NavigationController>(controller),
+ Details<LoginNotificationDetails>(&details));
}
+void LoginHandler::NotifyAuthSupplied(const std::wstring& username,
+ const std::wstring& password) {
+ DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI));
+ DCHECK(WasAuthHandled(false));
+
+ TabContents* requesting_contents = GetTabContentsForLogin();
+ if (!requesting_contents)
+ return;
+
+ NotificationService* service = NotificationService::current();
+ NavigationController* controller = &requesting_contents->controller();
+ AuthSuppliedLoginNotificationDetails details(this, username, password);
+
+ service->Notify(NotificationType::AUTH_SUPPLIED,
+ Source<NavigationController>(controller),
+ Details<AuthSuppliedLoginNotificationDetails>(&details));
+}
+
void LoginHandler::ReleaseSoon() {
if (!WasAuthHandled(true)) {
ChromeThread::PostTask(
@@ -197,9 +292,13 @@
NewRunnableMethod(this, &LoginHandler::CancelAuthDeferred));
ChromeThread::PostTask(
ChromeThread::UI, FROM_HERE,
- NewRunnableMethod(this, &LoginHandler::SendNotifications));
+ NewRunnableMethod(this, &LoginHandler::NotifyAuthCancelled));
}
+ ChromeThread::PostTask(
+ ChromeThread::UI, FROM_HERE,
+ NewRunnableMethod(this, &LoginHandler::RemoveObservers));
+
// Delete this object once all InvokeLaters have been called.
ChromeThread::ReleaseSoon(ChromeThread::IO, FROM_HERE, this);
}
@@ -231,7 +330,7 @@
if (request_) {
request_->CancelAuth();
- // Verify that CancelAuth does destroy the request via our delegate.
+ // Verify that CancelAuth doesn't destroy the request via our delegate.
DCHECK(request_ != NULL);
ResetLoginHandlerForRequest(request_);
}
@@ -303,7 +402,7 @@
std::string host_and_port(WideToASCII(auth_info_->host_and_port));
if (net::GetHostAndPort(request_url_) != host_and_port) {
dialog_form.origin = GURL();
- NOTREACHED();
+ NOTREACHED(); // crbug.com/32718
} else if (auth_info_->is_proxy) {
std::string origin = host_and_port;
// We don't expect this to already start with http:// or https://.
@@ -329,7 +428,7 @@
// This is owned by the ResourceDispatcherHost that invoked us.
LoginHandler* handler_;
- DISALLOW_EVIL_CONSTRUCTORS(LoginDialogTask);
+ DISALLOW_COPY_AND_ASSIGN(LoginDialogTask);
};
// ----------------------------------------------------------------------------
@@ -337,7 +436,7 @@
LoginHandler* CreateLoginPrompt(net::AuthChallengeInfo* auth_info,
URLRequest* request) {
- LoginHandler* handler = LoginHandler::Create(request);
+ LoginHandler* handler = LoginHandler::Create(auth_info, request);
ChromeThread::PostTask(
ChromeThread::UI, FROM_HERE, new LoginDialogTask(
request->url(), auth_info, handler));
« no previous file with comments | « chrome/browser/login_prompt.h ('k') | chrome/browser/login_prompt_gtk.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698