// Copyright (c) 2011 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/browser/browser_url_handler.h" #include "base/string_util.h" #include "chrome/browser/browser_about_handler.h" #include "chrome/browser/extensions/extension_web_ui.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/webui/chrome_web_ui_factory.h" #include "chrome/common/url_constants.h" #include "content/browser/webui/web_ui.h" #include "googleurl/src/gurl.h" // Handles rewriting view-source URLs for what we'll actually load. static bool HandleViewSource(GURL* url, Profile* profile) { if (url->SchemeIs(chrome::kViewSourceScheme)) { // Load the inner URL instead. *url = GURL(url->path()); // Bug 26129: limit view-source to view the content and not any // other kind of 'active' url scheme like 'javascript' or 'data'. static const char* const allowed_sub_schemes[] = { chrome::kHttpScheme, chrome::kHttpsScheme, chrome::kFtpScheme, chrome::kChromeDevToolsScheme, chrome::kChromeUIScheme, chrome::kFileScheme }; bool is_sub_scheme_allowed = false; for (size_t i = 0; i < arraysize(allowed_sub_schemes); i++) { if (url->SchemeIs(allowed_sub_schemes[i])) { is_sub_scheme_allowed = true; break; } } if (!is_sub_scheme_allowed) { *url = GURL(chrome::kAboutBlankURL); return false; } return true; } return false; } // Turns a non view-source URL into the corresponding view-source URL. static bool ReverseViewSource(GURL* url, Profile* profile) { // No action necessary if the URL is already view-source: if (url->SchemeIs(chrome::kViewSourceScheme)) return false; url_canon::Replacements repl; repl.SetScheme(chrome::kViewSourceScheme, url_parse::Component(0, strlen(chrome::kViewSourceScheme))); repl.SetPath(url->spec().c_str(), url_parse::Component(0, url->spec().size())); *url = url->ReplaceComponents(repl); return true; } // Handles rewriting Web UI URLs. static bool HandleWebUI(GURL* url, Profile* profile) { if (!ChromeWebUIFactory::GetInstance()->UseWebUIForURL(profile, *url)) return false; // Special case the new tab page. In older versions of Chrome, the new tab // page was hosted at chrome-internal:. This might be in people's saved // sessions or bookmarks, so we say any URL with that scheme triggers the new // tab page. if (url->SchemeIs(chrome::kChromeInternalScheme)) { // Rewrite it with the proper new tab URL. *url = GURL(chrome::kChromeUINewTabURL); } return true; } // static BrowserURLHandler* BrowserURLHandler::GetInstance() { return Singleton::get(); } // static BrowserURLHandler::URLHandler BrowserURLHandler::null_handler() { // Visual Studio 2010 has problems converting NULL to the null pointer for // std::pair. See http://connect.microsoft.com/VisualStudio/feedback/details/520043/error-converting-from-null-to-a-pointer-type-in-std-pair // It will work if we pass nullptr. #if defined(_MSC_VER) && _MSC_VER >= 1600 return nullptr; #else return NULL; #endif } BrowserURLHandler::BrowserURLHandler() { } BrowserURLHandler::~BrowserURLHandler() { } void BrowserURLHandler::AddHandlerPair(URLHandler handler, URLHandler reverse_handler) { url_handlers_.push_back(HandlerPair(handler, reverse_handler)); } void BrowserURLHandler::InitURLHandlers() { if (!url_handlers_.empty()) return; // Add the default URL handlers. AddHandlerPair(&ExtensionWebUI::HandleChromeURLOverride, null_handler()); AddHandlerPair(null_handler(), &ExtensionWebUI::HandleChromeURLOverrideReverse); // about: AddHandlerPair(&WillHandleBrowserAboutURL, null_handler()); // chrome: & friends. AddHandlerPair(&HandleWebUI, null_handler()); // view-source: AddHandlerPair(&HandleViewSource, &ReverseViewSource); } void BrowserURLHandler::RewriteURLIfNecessary(GURL* url, Profile* profile, bool* reverse_on_redirect) { if (url_handlers_.empty()) InitURLHandlers(); for (size_t i = 0; i < url_handlers_.size(); ++i) { URLHandler handler = *url_handlers_[i].first; if (handler && handler(url, profile)) { *reverse_on_redirect = (url_handlers_[i].second != NULL); return; } } } bool BrowserURLHandler::ReverseURLRewrite( GURL* url, const GURL& original, Profile* profile) { for (size_t i = 0; i < url_handlers_.size(); ++i) { URLHandler reverse_rewriter = *url_handlers_[i].second; if (reverse_rewriter) { GURL test_url(original); URLHandler handler = *url_handlers_[i].first; if (!handler) { if (reverse_rewriter(url, profile)) return true; } else if (handler(&test_url, profile)) { return reverse_rewriter(url, profile); } } } return false; }