// Copyright (c) 2006-2008 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. #ifndef NET_PROXY_PROXY_SERVICE_H_ #define NET_PROXY_PROXY_SERVICE_H_ #include #include #include "base/ref_counted.h" #include "base/scoped_ptr.h" #include "base/string_util.h" #include "base/thread.h" #include "base/time.h" #include "net/base/completion_callback.h" #if defined(OS_WIN) typedef LPVOID HINTERNET; // From winhttp.h #endif class GURL; namespace net { class ProxyInfo; class ProxyResolver; // Proxy configuration used to by the ProxyService. class ProxyConfig { public: typedef int ID; // Indicates an invalid proxy config. enum { INVALID_ID = 0 }; ProxyConfig(); // Default copy-constructor an assignment operator are OK! // Used to numerically identify this configuration. ID id() const { return id_; } // True if the proxy configuration should be auto-detected. bool auto_detect; // If non-empty, indicates the URL of the proxy auto-config file to use. std::string pac_url; // If non-empty, indicates the proxy server to use (of the form host:port). std::string proxy_server; // If non-empty, indicates a comma-delimited list of hosts that should bypass // any proxy configuration. For these hosts, a direct connection should // always be used. std::string proxy_bypass; // Returns true if the given config is equivalent to this config. bool Equals(const ProxyConfig& other) const; private: static int last_id_; int id_; }; // Contains the information about when to retry a proxy server. struct ProxyRetryInfo { // We should not retry until this time. TimeTicks bad_until; // This is the current delay. If the proxy is still bad, we need to increase // this delay. TimeDelta current_delay; }; // Map of proxy servers with the associated RetryInfo structures. typedef std::map ProxyRetryInfoMap; // This class can be used to resolve the proxy server to use when loading a // HTTP(S) URL. It uses to the given ProxyResolver to handle the actual proxy // resolution. See ProxyResolverWinHttp for example. The consumer of this // class is responsible for ensuring that the ProxyResolver instance remains // valid for the lifetime of the ProxyService. class ProxyService { public: explicit ProxyService(ProxyResolver* resolver); // Used internally to handle PAC queries. class PacRequest; // Returns OK if proxy information could be provided synchronously. Else, // ERR_IO_PENDING is returned to indicate that the result will be available // when the callback is run. The callback is run on the thread that calls // ResolveProxy. // // The caller is responsible for ensuring that |results| and |callback| // remain valid until the callback is run or until |pac_request| is cancelled // via CancelPacRequest. |pac_request| is only valid while the completion // callback is still pending. // // We use the three possible proxy access types in the following order, and // we only use one of them (no falling back to other access types if the // chosen one doesn't work). // 1. named proxy // 2. PAC URL // 3. WPAD auto-detection // int ResolveProxy(const GURL& url, ProxyInfo* results, CompletionCallback* callback, PacRequest** pac_request); // This method is called after a failure to connect or resolve a host name. // It gives the proxy service an opportunity to reconsider the proxy to use. // The |results| parameter contains the results returned by an earlier call // to ResolveProxy. The semantics of this call are otherwise similar to // ResolveProxy. // // Returns ERR_FAILED if there is not another proxy config to try. // int ReconsiderProxyAfterError(const GURL& url, ProxyInfo* results, CompletionCallback* callback, PacRequest** pac_request); // Call this method with a non-null |pac_request| to cancel the PAC request. void CancelPacRequest(PacRequest* pac_request); private: friend class PacRequest; ProxyResolver* resolver() { return resolver_; } base::Thread* pac_thread() { return pac_thread_.get(); } // Identifies the proxy configuration. ProxyConfig::ID config_id() const { return config_.id(); } // Checks to see if the proxy configuration changed, and then updates config_ // to reference the new configuration. void UpdateConfig(); // Called to indicate that a PacRequest completed. The |config_id| parameter // indicates the proxy configuration that was queried. |result_code| is OK // if the PAC file could be downloaded and executed. Otherwise, it is an // error code, indicating a bad proxy configuration. void DidCompletePacRequest(int config_id, int result_code); // Returns true if the URL passed in should not go through the proxy server. // 1. If the bypass proxy list contains the string and the URL // passed in is a local URL, i.e. a URL without a DOT (.) // 2. The URL matches one of the entities in the proxy bypass list. bool ShouldBypassProxyForURL(const GURL& url); ProxyResolver* resolver_; scoped_ptr pac_thread_; // We store the IE proxy config and a counter that is incremented each time // the config changes. ProxyConfig config_; // Indicates that the configuration is bad and should be ignored. bool config_is_bad_; // The time when the proxy configuration was last read from the system. TimeTicks config_last_update_time_; // Map of the known bad proxies and the information about the retry time. ProxyRetryInfoMap proxy_retry_info_; DISALLOW_COPY_AND_ASSIGN(ProxyService); }; // This class is used to hold a list of proxies returned by GetProxyForUrl or // manually configured. It handles proxy fallback if multiple servers are // specified. class ProxyList { public: // Initializes the proxy list to a string containing one or more proxy servers // delimited by a semicolon. void Set(const std::string& proxy_list); // Initializes the proxy list to a vector containing one or more proxy // servers. void SetVector(const std::vector& proxy_list); // Remove all proxies known to be bad from the proxy list. void RemoveBadProxies(const ProxyRetryInfoMap& proxy_retry_info); // Returns the first valid proxy server in the list. std::string Get() const; // Returns all the valid proxies, delimited by a semicolon. std::string GetList() const; // Marks the current proxy server as bad and deletes it from the list. The // list of known bad proxies is given by proxy_retry_info. Returns true if // there is another server available in the list. bool Fallback(ProxyRetryInfoMap* proxy_retry_info); private: // List of proxies. std::vector proxies_; }; // This object holds proxy information returned by ResolveProxy. class ProxyInfo { public: ProxyInfo(); // Use the same proxy server as the given |proxy_info|. void Use(const ProxyInfo& proxy_info); // Use a direct connection. void UseDirect(); // Use a specific proxy server, of the form: [":" ] // This may optionally be a semi-colon delimited list of proxy servers. void UseNamedProxy(const std::string& proxy_server); #if defined(OS_WIN) // Apply this proxy information to the given WinHTTP request handle. void Apply(HINTERNET request_handle); #endif // Returns true if this proxy info specifies a direct connection. bool is_direct() const { return proxy_list_.Get().empty(); } // Returns the first valid proxy server. std::string proxy_server() const { return proxy_list_.Get(); } // Marks the current proxy as bad. Returns true if there is another proxy // available to try in proxy list_. bool Fallback(ProxyRetryInfoMap* proxy_retry_info) { return proxy_list_.Fallback(proxy_retry_info); } // Remove all proxies known to be bad from the proxy list. void RemoveBadProxies(const ProxyRetryInfoMap& proxy_retry_info) { proxy_list_.RemoveBadProxies(proxy_retry_info); } private: friend class ProxyService; // If proxy_list_ is set to empty, then a "direct" connection is indicated. ProxyList proxy_list_; // This value identifies the proxy config used to initialize this object. ProxyConfig::ID config_id_; // This flag is false when the proxy configuration was known to be bad when // this proxy info was initialized. In such cases, we know that if this // proxy info does not yield a connection that we might want to reconsider // the proxy config given by config_id_. bool config_was_tried_; DISALLOW_COPY_AND_ASSIGN(ProxyInfo); }; // This interface provides the low-level functions to access the proxy // configuration and resolve proxies for given URLs synchronously. class ProxyResolver { public: virtual ~ProxyResolver() {} // Get the proxy configuration. Returns OK if successful or an error code if // otherwise. |config| should be in its initial state when this method is // called. virtual int GetProxyConfig(ProxyConfig* config) = 0; // Query the proxy auto-config file (specified by |pac_url|) for the proxy to // use to load the given |query_url|. Returns OK if successful or an error // code if otherwise. virtual int GetProxyForURL(const std::string& query_url, const std::string& pac_url, ProxyInfo* results) = 0; }; } // namespace net #endif // NET_PROXY_PROXY_SERVICE_H_