Home | History | Annotate | Download | only in net
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "android_webview/browser/net/aw_url_request_context_getter.h"
      6 
      7 #include <vector>
      8 
      9 #include "android_webview/browser/aw_content_browser_client.h"
     10 #include "android_webview/browser/aw_request_interceptor.h"
     11 #include "android_webview/browser/net/aw_network_delegate.h"
     12 #include "android_webview/browser/net/aw_url_request_job_factory.h"
     13 #include "android_webview/browser/net/init_native_callback.h"
     14 #include "android_webview/common/aw_switches.h"
     15 #include "base/command_line.h"
     16 #include "base/strings/string_number_conversions.h"
     17 #include "base/threading/sequenced_worker_pool.h"
     18 #include "content/public/browser/browser_thread.h"
     19 #include "content/public/browser/content_browser_client.h"
     20 #include "content/public/browser/cookie_store_factory.h"
     21 #include "content/public/common/content_client.h"
     22 #include "content/public/common/content_switches.h"
     23 #include "content/public/common/url_constants.h"
     24 #include "net/base/cache_type.h"
     25 #include "net/cookies/cookie_store.h"
     26 #include "net/dns/mapped_host_resolver.h"
     27 #include "net/http/http_cache.h"
     28 #include "net/http/http_stream_factory.h"
     29 #include "net/proxy/proxy_service.h"
     30 #include "net/url_request/data_protocol_handler.h"
     31 #include "net/url_request/file_protocol_handler.h"
     32 #include "net/url_request/protocol_intercept_job_factory.h"
     33 #include "net/url_request/url_request_context_builder.h"
     34 #include "net/url_request/url_request_context.h"
     35 
     36 using content::BrowserThread;
     37 
     38 namespace android_webview {
     39 
     40 
     41 namespace {
     42 
     43 void ApplyCmdlineOverridesToURLRequestContextBuilder(
     44     net::URLRequestContextBuilder* builder) {
     45   const CommandLine& command_line = *CommandLine::ForCurrentProcess();
     46   if (command_line.HasSwitch(switches::kHostResolverRules)) {
     47     // If hostname remappings were specified on the command-line, layer these
     48     // rules on top of the real host resolver. This allows forwarding all
     49     // requests through a designated test server.
     50     scoped_ptr<net::MappedHostResolver> host_resolver(
     51         new net::MappedHostResolver(
     52             net::HostResolver::CreateDefaultResolver(NULL)));
     53     host_resolver->SetRulesFromString(
     54         command_line.GetSwitchValueASCII(switches::kHostResolverRules));
     55     builder->set_host_resolver(host_resolver.release());
     56   }
     57 }
     58 
     59 void ApplyCmdlineOverridesToNetworkSessionParams(
     60     net::HttpNetworkSession::Params* params) {
     61   int value;
     62   const CommandLine& command_line = *CommandLine::ForCurrentProcess();
     63   if (command_line.HasSwitch(switches::kTestingFixedHttpPort)) {
     64     base::StringToInt(command_line.GetSwitchValueASCII(
     65         switches::kTestingFixedHttpPort), &value);
     66     params->testing_fixed_http_port = value;
     67   }
     68   if (command_line.HasSwitch(switches::kTestingFixedHttpsPort)) {
     69     base::StringToInt(command_line.GetSwitchValueASCII(
     70         switches::kTestingFixedHttpsPort), &value);
     71     params->testing_fixed_https_port = value;
     72   }
     73 }
     74 
     75 void PopulateNetworkSessionParams(
     76     net::URLRequestContext* context,
     77     net::HttpNetworkSession::Params* params) {
     78   params->host_resolver = context->host_resolver();
     79   params->cert_verifier = context->cert_verifier();
     80   params->server_bound_cert_service = context->server_bound_cert_service();
     81   params->transport_security_state = context->transport_security_state();
     82   params->proxy_service = context->proxy_service();
     83   params->ssl_config_service = context->ssl_config_service();
     84   params->http_auth_handler_factory = context->http_auth_handler_factory();
     85   params->network_delegate = context->network_delegate();
     86   params->http_server_properties = context->http_server_properties();
     87   params->net_log = context->net_log();
     88   ApplyCmdlineOverridesToNetworkSessionParams(params);
     89 }
     90 
     91 scoped_ptr<net::URLRequestJobFactory> CreateJobFactory(
     92     content::ProtocolHandlerMap* protocol_handlers) {
     93   scoped_ptr<AwURLRequestJobFactory> aw_job_factory(new AwURLRequestJobFactory);
     94   bool set_protocol = aw_job_factory->SetProtocolHandler(
     95       chrome::kFileScheme,
     96       new net::FileProtocolHandler(
     97           content::BrowserThread::GetBlockingPool()->
     98               GetTaskRunnerWithShutdownBehavior(
     99                   base::SequencedWorkerPool::SKIP_ON_SHUTDOWN)));
    100   DCHECK(set_protocol);
    101   set_protocol = aw_job_factory->SetProtocolHandler(
    102       chrome::kDataScheme, new net::DataProtocolHandler());
    103   DCHECK(set_protocol);
    104   set_protocol = aw_job_factory->SetProtocolHandler(
    105       chrome::kBlobScheme, (*protocol_handlers)[chrome::kBlobScheme].release());
    106   DCHECK(set_protocol);
    107   set_protocol = aw_job_factory->SetProtocolHandler(
    108       chrome::kFileSystemScheme,
    109       (*protocol_handlers)[chrome::kFileSystemScheme].release());
    110   DCHECK(set_protocol);
    111   set_protocol = aw_job_factory->SetProtocolHandler(
    112       chrome::kChromeUIScheme,
    113       (*protocol_handlers)[chrome::kChromeUIScheme].release());
    114   DCHECK(set_protocol);
    115   set_protocol = aw_job_factory->SetProtocolHandler(
    116       chrome::kChromeDevToolsScheme,
    117       (*protocol_handlers)[chrome::kChromeDevToolsScheme].release());
    118   DCHECK(set_protocol);
    119   protocol_handlers->clear();
    120 
    121   // Create a chain of URLRequestJobFactories. The handlers will be invoked
    122   // in the order in which they appear in the protocol_handlers vector.
    123   typedef std::vector<net::URLRequestJobFactory::ProtocolHandler*>
    124       ProtocolHandlerVector;
    125   ProtocolHandlerVector protocol_interceptors;
    126 
    127   // Note that even though the content:// scheme handler is created here,
    128   // it cannot be used by child processes until access to it is granted via
    129   // ChildProcessSecurityPolicy::GrantScheme(). This is done in
    130   // AwContentBrowserClient.
    131   protocol_interceptors.push_back(
    132       CreateAndroidContentProtocolHandler().release());
    133   protocol_interceptors.push_back(
    134       CreateAndroidAssetFileProtocolHandler().release());
    135   // The AwRequestInterceptor must come after the content and asset file job
    136   // factories. This for WebViewClassic compatibility where it was not
    137   // possible to intercept resource loads to resolvable content:// and
    138   // file:// URIs.
    139   // This logical dependency is also the reason why the Content
    140   // ProtocolHandler has to be added as a ProtocolInterceptJobFactory rather
    141   // than via SetProtocolHandler.
    142   protocol_interceptors.push_back(new AwRequestInterceptor());
    143 
    144   // The chain of responsibility will execute the handlers in reverse to the
    145   // order in which the elements of the chain are created.
    146   scoped_ptr<net::URLRequestJobFactory> job_factory(aw_job_factory.Pass());
    147   for (ProtocolHandlerVector::reverse_iterator
    148            i = protocol_interceptors.rbegin();
    149        i != protocol_interceptors.rend();
    150        ++i) {
    151     job_factory.reset(new net::ProtocolInterceptJobFactory(
    152         job_factory.Pass(), make_scoped_ptr(*i)));
    153   }
    154 
    155   return job_factory.Pass();
    156 }
    157 
    158 }  // namespace
    159 
    160 AwURLRequestContextGetter::AwURLRequestContextGetter(
    161     const base::FilePath& partition_path, net::CookieStore* cookie_store)
    162     : partition_path_(partition_path),
    163       cookie_store_(cookie_store),
    164       proxy_config_service_(net::ProxyService::CreateSystemProxyConfigService(
    165           GetNetworkTaskRunner(),
    166           NULL /* Ignored on Android */)) {
    167   // CreateSystemProxyConfigService for Android must be called on main thread.
    168   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
    169 }
    170 
    171 AwURLRequestContextGetter::~AwURLRequestContextGetter() {
    172 }
    173 
    174 void AwURLRequestContextGetter::InitializeURLRequestContext() {
    175   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
    176   DCHECK(!url_request_context_);
    177 
    178   net::URLRequestContextBuilder builder;
    179   builder.set_user_agent(content::GetUserAgent(GURL()));
    180   builder.set_network_delegate(new AwNetworkDelegate());
    181 #if !defined(DISABLE_FTP_SUPPORT)
    182   builder.set_ftp_enabled(false);  // Android WebView does not support ftp yet.
    183 #endif
    184   builder.set_proxy_config_service(proxy_config_service_.release());
    185   builder.set_accept_language(net::HttpUtil::GenerateAcceptLanguageHeader(
    186       AwContentBrowserClient::GetAcceptLangsImpl()));
    187   ApplyCmdlineOverridesToURLRequestContextBuilder(&builder);
    188 
    189   url_request_context_.reset(builder.Build());
    190   // TODO(mnaganov): Fix URLRequestContextBuilder to use proper threads.
    191   net::HttpNetworkSession::Params network_session_params;
    192 
    193   net::BackendType cache_type = net::CACHE_BACKEND_SIMPLE;
    194   if (CommandLine::ForCurrentProcess()->HasSwitch(
    195         switches::kDisableSimpleCache)) {
    196     cache_type = net::CACHE_BACKEND_BLOCKFILE;
    197   }
    198   PopulateNetworkSessionParams(url_request_context_.get(),
    199                                &network_session_params);
    200 
    201   net::HttpCache* main_cache = new net::HttpCache(
    202       network_session_params,
    203       new net::HttpCache::DefaultBackend(
    204           net::DISK_CACHE,
    205           cache_type,
    206           partition_path_.Append(FILE_PATH_LITERAL("Cache")),
    207           20 * 1024 * 1024,  // 20M
    208           BrowserThread::GetMessageLoopProxyForThread(BrowserThread::CACHE)));
    209   main_http_factory_.reset(main_cache);
    210   url_request_context_->set_http_transaction_factory(main_cache);
    211   url_request_context_->set_cookie_store(cookie_store_);
    212 
    213   job_factory_ = CreateJobFactory(&protocol_handlers_);
    214   url_request_context_->set_job_factory(job_factory_.get());
    215 
    216   // TODO(sgurun) remove once crbug.com/329681 is fixed. Should be
    217   // called only once.
    218   net::HttpStreamFactory::EnableNpnSpdy31();
    219 }
    220 
    221 net::URLRequestContext* AwURLRequestContextGetter::GetURLRequestContext() {
    222   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
    223   if (!url_request_context_)
    224     InitializeURLRequestContext();
    225 
    226   return url_request_context_.get();
    227 }
    228 
    229 scoped_refptr<base::SingleThreadTaskRunner>
    230 AwURLRequestContextGetter::GetNetworkTaskRunner() const {
    231   return BrowserThread::GetMessageLoopProxyForThread(BrowserThread::IO);
    232 }
    233 
    234 void AwURLRequestContextGetter::SetProtocolHandlers(
    235     content::ProtocolHandlerMap* protocol_handlers) {
    236   std::swap(protocol_handlers_, *protocol_handlers);
    237 }
    238 
    239 }  // namespace android_webview
    240