|  | // Copyright 2019 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/portal/portal_navigation_throttle.h" | 
|  |  | 
|  | #include "base/feature_list.h" | 
|  | #include "base/memory/ptr_util.h" | 
|  | #include "base/strings/stringprintf.h" | 
|  | #include "content/browser/frame_host/render_frame_host_impl.h" | 
|  | #include "content/browser/portal/portal.h" | 
|  | #include "content/browser/web_contents/web_contents_impl.h" | 
|  | #include "content/public/browser/navigation_handle.h" | 
|  | #include "content/public/common/url_utils.h" | 
|  | #include "third_party/blink/public/common/features.h" | 
|  | #include "url/gurl.h" | 
|  | #include "url/origin.h" | 
|  |  | 
|  | namespace content { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // A URL where developers can learn more about why this navigation throttle may | 
|  | // have cancelled their request. | 
|  | const char* GetBlockedInfoURL() { | 
|  | return "https://www.chromium.org/blink/origin-trials/portals"; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | // static | 
|  | std::unique_ptr<PortalNavigationThrottle> | 
|  | PortalNavigationThrottle::MaybeCreateThrottleFor( | 
|  | NavigationHandle* navigation_handle) { | 
|  | if (!Portal::IsEnabled() || !navigation_handle->IsInMainFrame()) | 
|  | return nullptr; | 
|  |  | 
|  | return base::WrapUnique(new PortalNavigationThrottle(navigation_handle)); | 
|  | } | 
|  |  | 
|  | PortalNavigationThrottle::PortalNavigationThrottle( | 
|  | NavigationHandle* navigation_handle) | 
|  | : NavigationThrottle(navigation_handle) {} | 
|  |  | 
|  | PortalNavigationThrottle::~PortalNavigationThrottle() = default; | 
|  |  | 
|  | const char* PortalNavigationThrottle::GetNameForLogging() { | 
|  | return "PortalNavigationThrottle"; | 
|  | } | 
|  |  | 
|  | NavigationThrottle::ThrottleCheckResult | 
|  | PortalNavigationThrottle::WillStartRequest() { | 
|  | return WillStartOrRedirectRequest(); | 
|  | } | 
|  |  | 
|  | NavigationThrottle::ThrottleCheckResult | 
|  | PortalNavigationThrottle::WillRedirectRequest() { | 
|  | return WillStartOrRedirectRequest(); | 
|  | } | 
|  |  | 
|  | NavigationThrottle::ThrottleCheckResult | 
|  | PortalNavigationThrottle::WillStartOrRedirectRequest() { | 
|  | WebContentsImpl* web_contents = | 
|  | static_cast<WebContentsImpl*>(navigation_handle()->GetWebContents()); | 
|  | Portal* portal = web_contents->portal(); | 
|  | if (!portal) | 
|  | return PROCEED; | 
|  |  | 
|  | GURL url = navigation_handle()->GetURL(); | 
|  | CHECK(!HasWebUIScheme(url)) | 
|  | << "Portals should not even be able to attempt to reach WebUI"; | 
|  | if (!url.SchemeIsHTTPOrHTTPS()) { | 
|  | base::StringPiece scheme = url.scheme_piece(); | 
|  | portal->owner_render_frame_host()->AddMessageToConsole( | 
|  | blink::mojom::ConsoleMessageLevel::kWarning, | 
|  | base::StringPrintf("Navigating a portal to scheme '%.*s' was blocked.", | 
|  | static_cast<int>(scheme.size()), scheme.data())); | 
|  | return CANCEL; | 
|  | } | 
|  |  | 
|  | if (!base::FeatureList::IsEnabled(blink::features::kPortalsCrossOrigin)) { | 
|  | url::Origin origin = url::Origin::Create(url); | 
|  | url::Origin first_party_origin = | 
|  | portal->owner_render_frame_host()->GetLastCommittedOrigin(); | 
|  |  | 
|  | if (origin != first_party_origin) { | 
|  | portal->owner_render_frame_host()->AddMessageToConsole( | 
|  | blink::mojom::ConsoleMessageLevel::kWarning, | 
|  | base::StringPrintf( | 
|  | "Navigating a portal to cross-origin content (from %s) " | 
|  | "is not currently permitted and was blocked. " | 
|  | "See %s for more information.", | 
|  | origin.Serialize().c_str(), GetBlockedInfoURL())); | 
|  | return CANCEL; | 
|  | } | 
|  | } | 
|  |  | 
|  | return PROCEED; | 
|  | } | 
|  |  | 
|  | }  // namespace content |