summaryrefslogtreecommitdiffstats
path: root/chrome/browser/android/intercept_download_resource_throttle.cc
blob: 4f5662fa13f0afc3063a4d1087bd63476739a4bd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// 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 "chrome/browser/android/intercept_download_resource_throttle.h"

#include "base/feature_list.h"
#include "base/metrics/histogram_macros.h"
#include "chrome/browser/android/chrome_feature_list.h"
#include "components/data_reduction_proxy/core/common/data_reduction_proxy_headers.h"
#include "content/public/browser/android/download_controller_android.h"
#include "content/public/browser/resource_controller.h"
#include "net/http/http_request_headers.h"
#include "net/http/http_response_headers.h"
#include "net/url_request/url_request.h"

namespace {

// UMA histogram for tracking reasons that chrome fails to intercept the
// download. Keep this in sync with MobileDownloadInterceptFailureReasons in
// histograms.xml.
enum MobileDownloadInterceptFailureReason {
  NO_FAILURE = 0,
  EMPTY_URL,
  NON_HTTP_OR_HTTPS,
  NON_GET_METHODS,
  NO_REQUEST_HEADERS,
  USE_HTTP_AUTH,
  USE_CHANNEL_BOUND_COOKIES,
  // FAILURE_REASON_SIZE should always be last - this is a count of the number
  // of items in this enum.
  FAILURE_REASON_SIZE,
};

void RecordInterceptFailureReasons(
    MobileDownloadInterceptFailureReason reason) {
  UMA_HISTOGRAM_ENUMERATION("MobileDownload.InterceptFailureReason",
                            reason,
                            FAILURE_REASON_SIZE);
}

}  // namespace

namespace chrome {

// static
bool InterceptDownloadResourceThrottle::IsDownloadInterceptionEnabled() {
  return base::FeatureList::IsEnabled(chrome::android::kSystemDownloadManager);
}

InterceptDownloadResourceThrottle::InterceptDownloadResourceThrottle(
    net::URLRequest* request,
    int render_process_id,
    int render_view_id,
    int request_id)
    : request_(request),
      render_process_id_(render_process_id),
      render_view_id_(render_view_id),
      request_id_(request_id) {
}

InterceptDownloadResourceThrottle::~InterceptDownloadResourceThrottle() {
}

void InterceptDownloadResourceThrottle::WillProcessResponse(bool* defer) {
  ProcessDownloadRequest();
}

const char* InterceptDownloadResourceThrottle::GetNameForLogging() const {
  return "InterceptDownloadResourceThrottle";
}

void InterceptDownloadResourceThrottle::ProcessDownloadRequest() {
  if (!IsDownloadInterceptionEnabled())
    return;

  if (request_->url_chain().empty()) {
    RecordInterceptFailureReasons(EMPTY_URL);
    return;
  }

  GURL url = request_->url_chain().back();
  if (!url.SchemeIsHTTPOrHTTPS()) {
    RecordInterceptFailureReasons(NON_HTTP_OR_HTTPS);
    return;
  }

  if (request_->method() != net::HttpRequestHeaders::kGetMethod) {
    RecordInterceptFailureReasons(NON_GET_METHODS);
    return;
  }

  net::HttpRequestHeaders headers;
  if (!request_->GetFullRequestHeaders(&headers)) {
    RecordInterceptFailureReasons(NO_REQUEST_HEADERS);
    return;
  }

  // In general, if the request uses HTTP authorization, either with the origin
  // or a proxy, then the network stack should handle the download. The one
  // exception is a request that is fetched via the Chrome Proxy and does not
  // authenticate with the origin.
  if (request_->response_info().did_use_http_auth) {
    if (headers.HasHeader(net::HttpRequestHeaders::kAuthorization) ||
        !(request_->response_info().headers.get() &&
          data_reduction_proxy::HasDataReductionProxyViaHeader(
              request_->response_info().headers.get(), NULL))) {
      RecordInterceptFailureReasons(USE_HTTP_AUTH);
      return;
    }
  }

  // If the cookie is possibly channel-bound, don't pass it to android download
  // manager.
  // TODO(qinmin): add a test for this. http://crbug.com/430541.
  if (request_->ssl_info().channel_id_sent) {
    RecordInterceptFailureReasons(USE_CHANNEL_BOUND_COOKIES);
    return;
  }

  content::DownloadControllerAndroid::Get()->CreateGETDownload(
      render_process_id_, render_view_id_, request_id_);
  controller()->Cancel();
  RecordInterceptFailureReasons(NO_FAILURE);
}

}  // namespace chrome