| // Copyright 2017 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "content/browser/renderer_host/blocked_scheme_navigation_throttle.h" |
| |
| #include "base/feature_list.h" |
| #include "base/strings/stringprintf.h" |
| #include "build/build_config.h" |
| #include "content/browser/renderer_host/frame_tree.h" |
| #include "content/browser/renderer_host/frame_tree_node.h" |
| #include "content/browser/renderer_host/navigation_request.h" |
| #include "content/common/navigation_params_utils.h" |
| #include "content/public/browser/navigation_handle.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/common/content_features.h" |
| #include "third_party/blink/public/mojom/devtools/console_message.mojom.h" |
| #include "url/url_constants.h" |
| |
| namespace content { |
| |
| namespace { |
| const char kConsoleError[] = "Not allowed to navigate top frame to %s URL: %s"; |
| const char kAnyFrameConsoleError[] = "Not allowed to navigate to %s URL: %s"; |
| } |
| |
| BlockedSchemeNavigationThrottle::BlockedSchemeNavigationThrottle( |
| NavigationHandle* navigation_handle) |
| : NavigationThrottle(navigation_handle) {} |
| |
| BlockedSchemeNavigationThrottle::~BlockedSchemeNavigationThrottle() {} |
| |
| NavigationThrottle::ThrottleCheckResult |
| BlockedSchemeNavigationThrottle::WillStartRequest() { |
| NavigationRequest* request = NavigationRequest::From(navigation_handle()); |
| if (!request->GetURL().SchemeIs(url::kFileSystemScheme)) |
| return PROCEED; |
| |
| if (base::FeatureList::IsEnabled(blink::features::kFileSystemUrlNavigation)) |
| return PROCEED; |
| |
| RenderFrameHost* top_frame = |
| request->frame_tree_node()->frame_tree()->root()->current_frame_host(); |
| top_frame->AddMessageToConsole( |
| blink::mojom::ConsoleMessageLevel::kError, |
| base::StringPrintf(kAnyFrameConsoleError, |
| request->GetURL().scheme().c_str(), |
| request->GetURL().spec().c_str())); |
| |
| return CANCEL; |
| } |
| |
| NavigationThrottle::ThrottleCheckResult |
| BlockedSchemeNavigationThrottle::WillProcessResponse() { |
| NavigationRequest* request = NavigationRequest::From(navigation_handle()); |
| if (request->IsDownload()) |
| return PROCEED; |
| |
| RenderFrameHost* top_frame = |
| request->frame_tree_node()->frame_tree()->root()->current_frame_host(); |
| top_frame->AddMessageToConsole( |
| blink::mojom::ConsoleMessageLevel::kError, |
| base::StringPrintf(kConsoleError, request->GetURL().scheme().c_str(), |
| request->GetURL().spec().c_str())); |
| return CANCEL; |
| } |
| |
| const char* BlockedSchemeNavigationThrottle::GetNameForLogging() { |
| return "BlockedSchemeNavigationThrottle"; |
| } |
| |
| // static |
| std::unique_ptr<NavigationThrottle> |
| BlockedSchemeNavigationThrottle::CreateThrottleForNavigation( |
| NavigationHandle* navigation_handle) { |
| NavigationRequest* request = NavigationRequest::From(navigation_handle); |
| // Create throttles when going to blocked schemes via renderer-initiated |
| // navigations (which are cross-document in the main frame). Note that history |
| // navigations can bypass this, because the blocked scheme must have |
| // originally committed in a permitted case (e.g., omnibox navigation). |
| if (request->IsInMainFrame() && request->IsRendererInitiated() && |
| !request->IsSameDocument() && |
| !NavigationTypeUtils::IsHistory( |
| request->common_params().navigation_type) && |
| (request->GetURL().SchemeIs(url::kDataScheme) || |
| request->GetURL().SchemeIs(url::kFileSystemScheme)) && |
| !base::FeatureList::IsEnabled( |
| features::kAllowContentInitiatedDataUrlNavigations)) { |
| return std::make_unique<BlockedSchemeNavigationThrottle>(request); |
| } |
| // Block all renderer initiated navigations to filesystem: URLs. These won't |
| // load anyway since no URL Loader exists for them, but the throttle lets us |
| // add a message to the console. |
| if (!base::FeatureList::IsEnabled( |
| blink::features::kFileSystemUrlNavigation) && |
| request->IsRendererInitiated() && |
| request->GetURL().SchemeIs(url::kFileSystemScheme)) { |
| return std::make_unique<BlockedSchemeNavigationThrottle>(request); |
| } |
| return nullptr; |
| } |
| |
| } // namespace content |