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. 4 5 #include "chrome/browser/ui/omnibox/omnibox_navigation_observer.h" 6 7 #include "chrome/browser/history/shortcuts_backend.h" 8 #include "chrome/browser/history/shortcuts_backend_factory.h" 9 #include "chrome/browser/infobars/infobar_service.h" 10 #include "chrome/browser/intranet_redirect_detector.h" 11 #include "chrome/browser/ui/omnibox/alternate_nav_infobar_delegate.h" 12 #include "content/public/browser/browser_context.h" 13 #include "content/public/browser/navigation_controller.h" 14 #include "content/public/browser/navigation_details.h" 15 #include "content/public/browser/navigation_entry.h" 16 #include "content/public/browser/notification_service.h" 17 #include "content/public/browser/notification_types.h" 18 #include "content/public/browser/web_contents.h" 19 #include "net/base/load_flags.h" 20 #include "net/base/registry_controlled_domains/registry_controlled_domain.h" 21 #include "net/url_request/url_fetcher.h" 22 #include "net/url_request/url_request.h" 23 24 25 // Helpers -------------------------------------------------------------------- 26 27 namespace { 28 29 // HTTP 2xx, 401, and 407 all indicate that the target address exists. 30 bool ResponseCodeIndicatesSuccess(int response_code) { 31 return ((response_code / 100) == 2) || (response_code == 401) || 32 (response_code == 407); 33 } 34 35 // Returns true if |final_url| doesn't represent an ISP hijack of 36 // |original_url|, based on the IntranetRedirectDetector's RedirectOrigin(). 37 bool IsValidNavigation(const GURL& original_url, const GURL& final_url) { 38 const GURL& redirect_url(IntranetRedirectDetector::RedirectOrigin()); 39 return !redirect_url.is_valid() || 40 net::registry_controlled_domains::SameDomainOrHost( 41 original_url, final_url, 42 net::registry_controlled_domains::EXCLUDE_PRIVATE_REGISTRIES) || 43 !net::registry_controlled_domains::SameDomainOrHost( 44 final_url, redirect_url, 45 net::registry_controlled_domains::EXCLUDE_PRIVATE_REGISTRIES); 46 } 47 48 } // namespace 49 50 51 // OmniboxNavigationObserver -------------------------------------------------- 52 53 OmniboxNavigationObserver::OmniboxNavigationObserver( 54 Profile* profile, 55 const base::string16& text, 56 const AutocompleteMatch& match, 57 const AutocompleteMatch& alternate_nav_match) 58 : text_(text), 59 match_(match), 60 alternate_nav_match_(alternate_nav_match), 61 shortcuts_backend_(ShortcutsBackendFactory::GetForProfile(profile)), 62 load_state_(LOAD_NOT_SEEN), 63 fetch_state_(FETCH_NOT_COMPLETE) { 64 if (alternate_nav_match_.destination_url.is_valid()) { 65 fetcher_.reset(net::URLFetcher::Create(alternate_nav_match_.destination_url, 66 net::URLFetcher::HEAD, this)); 67 fetcher_->SetLoadFlags(net::LOAD_DO_NOT_SAVE_COOKIES); 68 fetcher_->SetStopOnRedirect(true); 69 } 70 // We need to start by listening to AllSources, since we don't know which tab 71 // the navigation might occur in. 72 registrar_.Add(this, content::NOTIFICATION_NAV_ENTRY_PENDING, 73 content::NotificationService::AllSources()); 74 } 75 76 OmniboxNavigationObserver::~OmniboxNavigationObserver() { 77 } 78 79 void OmniboxNavigationObserver::OnSuccessfulNavigation() { 80 if (shortcuts_backend_) 81 shortcuts_backend_->AddOrUpdateShortcut(text_, match_); 82 } 83 84 void OmniboxNavigationObserver::Observe( 85 int type, 86 const content::NotificationSource& source, 87 const content::NotificationDetails& details) { 88 DCHECK_EQ(content::NOTIFICATION_NAV_ENTRY_PENDING, type); 89 registrar_.Remove(this, content::NOTIFICATION_NAV_ENTRY_PENDING, 90 content::NotificationService::AllSources()); 91 content::NavigationController* controller = 92 content::Source<content::NavigationController>(source).ptr(); 93 if (fetcher_) { 94 fetcher_->SetRequestContext( 95 controller->GetBrowserContext()->GetRequestContext()); 96 } 97 WebContentsObserver::Observe(controller->GetWebContents()); 98 // DidStartNavigationToPendingEntry() will be called for this load as well. 99 } 100 101 void OmniboxNavigationObserver::NavigationEntryCommitted( 102 const content::LoadCommittedDetails& load_details) { 103 load_state_ = LOAD_COMMITTED; 104 if (ResponseCodeIndicatesSuccess(load_details.http_status_code) && 105 IsValidNavigation(match_.destination_url, load_details.entry->GetURL())) 106 OnSuccessfulNavigation(); 107 if (!fetcher_ || (fetch_state_ != FETCH_NOT_COMPLETE)) 108 OnAllLoadingFinished(); // deletes |this|! 109 } 110 111 void OmniboxNavigationObserver::WebContentsDestroyed( 112 content::WebContents* web_contents) { 113 delete this; 114 } 115 116 void OmniboxNavigationObserver::DidStartNavigationToPendingEntry( 117 const GURL& url, 118 content::NavigationController::ReloadType reload_type) { 119 if (load_state_ == LOAD_NOT_SEEN) { 120 load_state_ = LOAD_PENDING; 121 if (fetcher_) 122 fetcher_->Start(); 123 } else { 124 delete this; 125 } 126 } 127 128 void OmniboxNavigationObserver::OnURLFetchComplete( 129 const net::URLFetcher* source) { 130 DCHECK_EQ(fetcher_.get(), source); 131 const net::URLRequestStatus& status = source->GetStatus(); 132 int response_code = source->GetResponseCode(); 133 fetch_state_ = 134 (status.is_success() && ResponseCodeIndicatesSuccess(response_code)) || 135 ((status.status() == net::URLRequestStatus::CANCELED) && 136 ((response_code / 100) == 3) && 137 IsValidNavigation(alternate_nav_match_.destination_url, 138 source->GetURL())) ? 139 FETCH_SUCCEEDED : FETCH_FAILED; 140 if (load_state_ == LOAD_COMMITTED) 141 OnAllLoadingFinished(); // deletes |this|! 142 } 143 144 void OmniboxNavigationObserver::OnAllLoadingFinished() { 145 if (fetch_state_ == FETCH_SUCCEEDED) { 146 AlternateNavInfoBarDelegate::Create( 147 web_contents(), text_, alternate_nav_match_, match_.destination_url); 148 } 149 delete this; 150 } 151