// 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 "chrome/browser/download/android/intercept_oma_download_navigation_throttle.h"
#include "base/functional/bind.h"
#include "base/memory/ptr_util.h"
#include "chrome/browser/download/android/download_controller_base.h"
#include "components/embedder_support/user_agent_utils.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/navigation_handle.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/browser/render_view_host.h"
#include "content/public/browser/web_contents.h"
#include "third_party/blink/public/mojom/loader/referrer.mojom.h"
using content::BrowserThread;
// static
std::unique_ptr<content::NavigationThrottle>
InterceptOMADownloadNavigationThrottle::Create(
content::NavigationHandle* handle) {
return base::WrapUnique(new InterceptOMADownloadNavigationThrottle(handle));
}
InterceptOMADownloadNavigationThrottle::
~InterceptOMADownloadNavigationThrottle() = default;
content::NavigationThrottle::ThrottleCheckResult
InterceptOMADownloadNavigationThrottle::WillProcessResponse() {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
if (!navigation_handle()->IsDownload())
return content::NavigationThrottle::PROCEED;
if (!navigation_handle()->GetURL().SchemeIsHTTPOrHTTPS())
return content::NavigationThrottle::PROCEED;
if (navigation_handle()->IsPost())
return content::NavigationThrottle::PROCEED;
const net::HttpResponseHeaders* headers =
navigation_handle()->GetResponseHeaders();
if (!headers)
return content::NavigationThrottle::PROCEED;
std::string mime_type;
if (!headers->GetMimeType(&mime_type))
return content::NavigationThrottle::PROCEED;
if (!base::EqualsCaseInsensitiveASCII(mime_type, kOMADrmMessageMimeType) &&
!base::EqualsCaseInsensitiveASCII(mime_type, kOMADrmContentMimeType) &&
!base::EqualsCaseInsensitiveASCII(mime_type, kOMADrmRightsMimeType1) &&
!base::EqualsCaseInsensitiveASCII(mime_type, kOMADrmRightsMimeType2)) {
return content::NavigationThrottle::PROCEED;
}
// Make the throttle prerender-aware. Prerender must not have any observable
// side effects so avoid calling InterceptDownload. The prerender should also
// be cancelled in this case but that requires a reference to the
// RenderFrameHost which isn't easily available for a download. Instead,
// proceed in this throttle; it will be cancelled from
// PrerenderNavigationThrottle eventually.
// TODO(robertlin): Find a way to port PrerenderHost status checks to Java,
// and add a test on Android to verify the cancellation of an OMA download.
content::RenderFrameHost* parent_render_frame_host =
navigation_handle()->GetParentFrame();
if ((parent_render_frame_host &&
content::RenderFrameHost::LifecycleState::kPrerendering ==
parent_render_frame_host->GetLifecycleState()) ||
navigation_handle()->IsInPrerenderedMainFrame()) {
return content::NavigationThrottle::PROCEED;
}
InterceptDownload();
return content::NavigationThrottle::CANCEL;
}
const char* InterceptOMADownloadNavigationThrottle::GetNameForLogging() {
return "InterceptDownloadNavigationThrottle";
}
InterceptOMADownloadNavigationThrottle::InterceptOMADownloadNavigationThrottle(
content::NavigationHandle* handle)
: content::NavigationThrottle(handle) {}
void InterceptOMADownloadNavigationThrottle::InterceptDownload() {
GURL original_url;
const std::vector<GURL>& url_chain = navigation_handle()->GetRedirectChain();
if (!url_chain.empty())
original_url = url_chain.front();
std::string content_disposition;
std::string mime_type;
const net::HttpResponseHeaders* headers =
navigation_handle()->GetResponseHeaders();
headers->GetMimeType(&mime_type);
headers->GetNormalizedHeader("content-disposition", &content_disposition);
content::WebContents* web_contents = navigation_handle()->GetWebContents();
int process_id =
web_contents ? web_contents->GetRenderViewHost()->GetProcess()->GetID()
: 0;
int routing_id =
web_contents ? web_contents->GetRenderViewHost()->GetRoutingID() : 0;
DownloadControllerBase::Get()->CreateAndroidDownload(
base::BindRepeating(&GetWebContents, process_id, routing_id),
DownloadInfo(navigation_handle()->GetURL(), original_url,
content_disposition, mime_type,
embedder_support::GetUserAgent(),
// TODO(qinmin): Get the cookie from cookie store.
std::string(), navigation_handle()->GetReferrer().url));
}