|  | // Copyright (c) 2012 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 "content/browser/browser_url_handler_impl.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  |  | 
|  | #include "base/cxx17_backports.h" | 
|  | #include "base/ranges/algorithm.h" | 
|  | #include "base/strings/string_util.h" | 
|  | #include "content/browser/renderer_host/debug_urls.h" | 
|  | #include "content/browser/webui/web_ui_impl.h" | 
|  | #include "content/public/browser/content_browser_client.h" | 
|  | #include "content/public/common/content_client.h" | 
|  | #include "content/public/common/url_constants.h" | 
|  | #include "content/public/common/url_utils.h" | 
|  | #include "third_party/blink/public/common/chrome_debug_urls.h" | 
|  | #include "url/gurl.h" | 
|  |  | 
|  | namespace content { | 
|  |  | 
|  | // Handles rewriting view-source URLs for what we'll actually load. | 
|  | static bool HandleViewSource(GURL* url, BrowserContext* browser_context) { | 
|  | if (url->SchemeIs(kViewSourceScheme)) { | 
|  | // Load the inner URL instead. | 
|  | *url = GURL(url->GetContent()); | 
|  |  | 
|  | // 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 default_allowed_sub_schemes[] = { | 
|  | url::kHttpScheme, | 
|  | url::kHttpsScheme, | 
|  | url::kFtpScheme, | 
|  | kChromeUIScheme, | 
|  | url::kFileScheme, | 
|  | url::kFileSystemScheme | 
|  | }; | 
|  |  | 
|  | // Merge all the schemes for which view-source is allowed by default, with | 
|  | // the view-source schemes defined by the ContentBrowserClient. | 
|  | std::vector<std::string> all_allowed_sub_schemes; | 
|  | for (size_t i = 0; i < base::size(default_allowed_sub_schemes); ++i) | 
|  | all_allowed_sub_schemes.push_back(default_allowed_sub_schemes[i]); | 
|  | GetContentClient()->browser()->GetAdditionalViewSourceSchemes( | 
|  | &all_allowed_sub_schemes); | 
|  |  | 
|  | bool is_sub_scheme_allowed = false; | 
|  | for (size_t i = 0; i < all_allowed_sub_schemes.size(); ++i) { | 
|  | if (url->SchemeIs(all_allowed_sub_schemes[i].c_str())) { | 
|  | is_sub_scheme_allowed = true; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!is_sub_scheme_allowed) { | 
|  | *url = GURL(url::kAboutBlankURL); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Turns a non view-source URL into the corresponding view-source URL. | 
|  | static bool ReverseViewSource(GURL* url, BrowserContext* browser_context) { | 
|  | // No action necessary if the URL is already view-source: | 
|  | if (url->SchemeIs(kViewSourceScheme)) | 
|  | return false; | 
|  | // Recreate the url with the view-source scheme. | 
|  | *url = GURL(kViewSourceScheme + std::string(":") + url->spec()); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool DebugURLHandler(GURL* url, BrowserContext* browser_context) { | 
|  | // Circumvent processing URLs that the renderer process will handle. | 
|  | return blink::IsRendererDebugURL(*url); | 
|  | } | 
|  |  | 
|  | // static | 
|  | BrowserURLHandler* BrowserURLHandler::GetInstance() { | 
|  | return BrowserURLHandlerImpl::GetInstance(); | 
|  | } | 
|  |  | 
|  | // static | 
|  | BrowserURLHandler::URLHandler BrowserURLHandler::null_handler() { | 
|  | // Required for VS2010: http://connect.microsoft.com/VisualStudio/feedback/details/520043/error-converting-from-null-to-a-pointer-type-in-std-pair | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | // static | 
|  | BrowserURLHandlerImpl* BrowserURLHandlerImpl::GetInstance() { | 
|  | return base::Singleton<BrowserURLHandlerImpl>::get(); | 
|  | } | 
|  |  | 
|  | BrowserURLHandlerImpl::BrowserURLHandlerImpl() { | 
|  | AddHandlerPair(&DebugURLHandler, BrowserURLHandlerImpl::null_handler()); | 
|  |  | 
|  | // view-source: should take precedence over other rewriters, so it's | 
|  | // important to add it before calling up to the content client. | 
|  | AddHandlerPair(&HandleViewSource, &ReverseViewSource); | 
|  |  | 
|  | GetContentClient()->browser()->BrowserURLHandlerCreated(this); | 
|  | } | 
|  |  | 
|  | BrowserURLHandlerImpl::~BrowserURLHandlerImpl() { | 
|  | } | 
|  |  | 
|  | void BrowserURLHandlerImpl::AddHandlerPair(URLHandler handler, | 
|  | URLHandler reverse_handler) { | 
|  | url_handlers_.push_back(HandlerPair(handler, reverse_handler)); | 
|  | } | 
|  |  | 
|  | void BrowserURLHandlerImpl::RewriteURLIfNecessary( | 
|  | GURL* url, | 
|  | BrowserContext* browser_context) { | 
|  | DCHECK(url); | 
|  | DCHECK(browser_context); | 
|  | bool ignored_reverse_on_redirect; | 
|  | RewriteURLIfNecessary(url, browser_context, &ignored_reverse_on_redirect); | 
|  | } | 
|  |  | 
|  | std::vector<GURL> BrowserURLHandlerImpl::GetPossibleRewrites( | 
|  | const GURL& url, | 
|  | BrowserContext* browser_context) { | 
|  | std::vector<GURL> rewrites; | 
|  | for (const auto& it : url_handlers_) { | 
|  | const URLHandler& handler = it.first; | 
|  | if (!handler) | 
|  | continue; | 
|  |  | 
|  | GURL mutable_url(url); | 
|  | if (handler(&mutable_url, browser_context)) | 
|  | rewrites.push_back(std::move(mutable_url)); | 
|  | } | 
|  |  | 
|  | return rewrites; | 
|  | } | 
|  |  | 
|  | void BrowserURLHandlerImpl::RewriteURLIfNecessary( | 
|  | GURL* url, | 
|  | BrowserContext* browser_context, | 
|  | bool* reverse_on_redirect) { | 
|  | DCHECK(url); | 
|  | DCHECK(browser_context); | 
|  | DCHECK(reverse_on_redirect); | 
|  |  | 
|  | if (!url->is_valid()) { | 
|  | *reverse_on_redirect = false; | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (const auto& it : url_handlers_) { | 
|  | const URLHandler& handler = it.first; | 
|  | bool has_reverse_rewriter = it.second; | 
|  | if (handler && handler(url, browser_context)) { | 
|  | *reverse_on_redirect = has_reverse_rewriter; | 
|  | return; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | bool BrowserURLHandlerImpl::ReverseURLRewrite( | 
|  | GURL* url, const GURL& original, BrowserContext* browser_context) { | 
|  | for (const auto& it : url_handlers_) { | 
|  | const URLHandler& handler = it.first; | 
|  | const URLHandler& reverse_rewriter = it.second; | 
|  | if (reverse_rewriter) { | 
|  | GURL test_url(original); | 
|  | if (!handler) { | 
|  | if (reverse_rewriter(url, browser_context)) | 
|  | return true; | 
|  | } else if (handler(&test_url, browser_context)) { | 
|  | return reverse_rewriter(url, browser_context); | 
|  | } | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void BrowserURLHandlerImpl::RemoveHandlerForTesting(URLHandler handler) { | 
|  | const auto it = | 
|  | base::ranges::find(url_handlers_, handler, &HandlerPair::first); | 
|  | DCHECK(url_handlers_.end() != it); | 
|  | url_handlers_.erase(it); | 
|  | } | 
|  |  | 
|  | }  // namespace content |