// 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. #ifndef NET_PROXY_DHCP_PAC_FILE_FETCHER_WIN_H_ #define NET_PROXY_DHCP_PAC_FILE_FETCHER_WIN_H_ #include #include #include #include #include "base/macros.h" #include "base/memory/ref_counted.h" #include "base/threading/thread_checker.h" #include "base/time/time.h" #include "base/timer/timer.h" #include "net/base/net_export.h" #include "net/log/net_log_with_source.h" #include "net/proxy_resolution/dhcp_pac_file_fetcher.h" #include "net/traffic_annotation/network_traffic_annotation.h" namespace base { class TaskRunner; } namespace net { struct DhcpAdapterNamesLoggingInfo; class DhcpPacFileAdapterFetcher; class URLRequestContext; // Windows-specific implementation. class NET_EXPORT_PRIVATE DhcpPacFileFetcherWin : public DhcpPacFileFetcher, public base::SupportsWeakPtr { public: // Creates a DhcpPacFileFetcherWin that issues requests through // |url_request_context|. |url_request_context| must remain valid for // the lifetime of DhcpPacFileFetcherWin. explicit DhcpPacFileFetcherWin(URLRequestContext* url_request_context); ~DhcpPacFileFetcherWin() override; // DhcpPacFileFetcher implementation. int Fetch(base::string16* utf16_text, const CompletionCallback& callback, const NetLogWithSource& net_log, const NetworkTrafficAnnotationTag traffic_annotation) override; void Cancel() override; void OnShutdown() override; const GURL& GetPacURL() const override; std::string GetFetcherName() const override; // Sets |adapter_names| to contain the name of each network adapter on // this machine that has DHCP enabled and is not a loop-back adapter. May // optionally update |info| (if non-null) with information for logging. // Returns false on error. static bool GetCandidateAdapterNames(std::set* adapter_names, DhcpAdapterNamesLoggingInfo* info); protected: int num_pending_fetchers() const; URLRequestContext* url_request_context() const; scoped_refptr GetTaskRunner(); // This inner class encapsulate work done on a worker pool thread. // The class calls GetCandidateAdapterNames, which can take a couple of // hundred milliseconds. class NET_EXPORT_PRIVATE AdapterQuery : public base::RefCountedThreadSafe { public: AdapterQuery(); // This is the method that runs on the worker pool thread. void GetCandidateAdapterNames(); // This set is valid after GetCandidateAdapterNames has // been run. Its lifetime is scoped by this object. const std::set& adapter_names() const; DhcpAdapterNamesLoggingInfo* logging_info() { return logging_info_.get(); } protected: // Virtual method introduced to allow unit testing. virtual bool ImplGetCandidateAdapterNames( std::set* adapter_names, DhcpAdapterNamesLoggingInfo* info); friend class base::RefCountedThreadSafe; virtual ~AdapterQuery(); private: // These are constructed on the originating thread, then used on the // worker thread, then used again on the originating thread only when // the task has completed on the worker thread. No locking required. std::set adapter_names_; std::unique_ptr logging_info_; DISALLOW_COPY_AND_ASSIGN(AdapterQuery); }; // Virtual methods introduced to allow unit testing. virtual DhcpPacFileAdapterFetcher* ImplCreateAdapterFetcher(); virtual AdapterQuery* ImplCreateAdapterQuery(); virtual base::TimeDelta ImplGetMaxWait(); virtual void ImplOnGetCandidateAdapterNamesDone() {} private: // Event/state transition handlers void CancelImpl(); void OnGetCandidateAdapterNamesDone( scoped_refptr query, const NetworkTrafficAnnotationTag traffic_annotation); void OnFetcherDone(size_t fetcher_i, int result); void OnWaitTimer(); void TransitionToDone(); // This is the outer state machine for fetching PAC configuration from // DHCP. It relies for sub-states on the state machine of the // DhcpPacFileAdapterFetcher class. // // The goal of the implementation is to the following work in parallel // for all network adapters that are using DHCP: // a) Try to get the PAC URL configured in DHCP; // b) If one is configured, try to fetch the PAC URL. // c) Once this is done for all adapters, or a timeout has passed after // it has completed for the fastest adapter, return the PAC file // available for the most preferred network adapter, if any. // // The state machine goes from START->WAIT_ADAPTERS when it starts a // worker thread to get the list of adapters with DHCP enabled. // It then goes from WAIT_ADAPTERS->NO_RESULTS when it creates // and starts an DhcpPacFileAdapterFetcher for each adapter. It goes // from NO_RESULTS->SOME_RESULTS when it gets the first result; at this // point a wait timer is started. It goes from SOME_RESULTS->DONE in // two cases: All results are known, or the wait timer expired. A call // to Cancel() will also go straight to DONE from any state. Any // way the DONE state is entered, we will at that point cancel any // outstanding work and return the best known PAC script or the empty // string. // // The state machine is reset for each Fetch(), a call to which is // only valid in states START and DONE, as only one Fetch() is // allowed to be outstanding at any given time. enum State { STATE_START, STATE_WAIT_ADAPTERS, STATE_NO_RESULTS, STATE_SOME_RESULTS, STATE_DONE, }; // Vector, in Windows' network adapter preference order, of // DhcpPacFileAdapterFetcher objects that are or were attempting // to fetch a PAC file based on DHCP configuration. using FetcherVector = std::vector>; FetcherVector fetchers_; // Current state of this state machine. State state_; // The following members are associated with the latest call to Fetch(). // Number of fetchers we are waiting for. int num_pending_fetchers_; // Lets our client know we're done. Not valid in states START or DONE. CompletionCallback callback_; // The NetLog to use for the current Fetch(). NetLogWithSource net_log_; // Pointer to string we will write results to. Not valid in states // START and DONE. base::string16* destination_string_; // PAC URL retrieved from DHCP, if any. Valid only in state STATE_DONE. GURL pac_url_; base::OneShotTimer wait_timer_; // Set to nullptr on cancellation. URLRequestContext* url_request_context_; // NULL or the AdapterQuery currently in flight. scoped_refptr last_query_; // TaskRunner used for all DHCP lookup tasks. const scoped_refptr task_runner_; THREAD_CHECKER(thread_checker_); DISALLOW_IMPLICIT_CONSTRUCTORS(DhcpPacFileFetcherWin); }; } // namespace net #endif // NET_PROXY_DHCP_PAC_FILE_FETCHER_WIN_H_