• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 #ifndef NET_BASE_NETWORK_CHANGE_NOTIFIER_H_
6 #define NET_BASE_NETWORK_CHANGE_NOTIFIER_H_
7 
8 #include "base/basictypes.h"
9 #include "base/observer_list_threadsafe.h"
10 #include "base/time/time.h"
11 #include "net/base/net_export.h"
12 
13 class GURL;
14 
15 namespace net {
16 
17 struct DnsConfig;
18 class HistogramWatcher;
19 class NetworkChangeNotifierFactory;
20 class URLRequest;
21 
22 #if defined(OS_LINUX)
23 namespace internal {
24 class AddressTrackerLinux;
25 }
26 #endif
27 
28 // NetworkChangeNotifier monitors the system for network changes, and notifies
29 // registered observers of those events.  Observers may register on any thread,
30 // and will be called back on the thread from which they registered.
31 // NetworkChangeNotifiers are threadsafe, though they must be created and
32 // destroyed on the same thread.
33 class NET_EXPORT NetworkChangeNotifier {
34  public:
35   // Using the terminology of the Network Information API:
36   // http://www.w3.org/TR/netinfo-api.
37   enum ConnectionType {
38     CONNECTION_UNKNOWN = 0, // A connection exists, but its type is unknown.
39     CONNECTION_ETHERNET = 1,
40     CONNECTION_WIFI = 2,
41     CONNECTION_2G = 3,
42     CONNECTION_3G = 4,
43     CONNECTION_4G = 5,
44     CONNECTION_NONE = 6     // No connection.
45   };
46 
47   class NET_EXPORT IPAddressObserver {
48    public:
49     // Will be called when the IP address of the primary interface changes.
50     // This includes when the primary interface itself changes.
51     virtual void OnIPAddressChanged() = 0;
52 
53    protected:
IPAddressObserver()54     IPAddressObserver() {}
~IPAddressObserver()55     virtual ~IPAddressObserver() {}
56 
57    private:
58     DISALLOW_COPY_AND_ASSIGN(IPAddressObserver);
59   };
60 
61   class NET_EXPORT ConnectionTypeObserver {
62    public:
63     // Will be called when the connection type of the system has changed.
64     // See NetworkChangeNotifier::GetConnectionType() for important caveats
65     // about the unreliability of using this signal to infer the ability to
66     // reach remote sites.
67     virtual void OnConnectionTypeChanged(ConnectionType type) = 0;
68 
69    protected:
ConnectionTypeObserver()70     ConnectionTypeObserver() {}
~ConnectionTypeObserver()71     virtual ~ConnectionTypeObserver() {}
72 
73    private:
74     DISALLOW_COPY_AND_ASSIGN(ConnectionTypeObserver);
75   };
76 
77   class NET_EXPORT DNSObserver {
78    public:
79     // Will be called when the DNS settings of the system may have changed.
80     // Use GetDnsConfig to obtain the current settings.
81     virtual void OnDNSChanged() = 0;
82 
83    protected:
DNSObserver()84     DNSObserver() {}
~DNSObserver()85     virtual ~DNSObserver() {}
86 
87    private:
88     DISALLOW_COPY_AND_ASSIGN(DNSObserver);
89   };
90 
91   class NET_EXPORT NetworkChangeObserver {
92    public:
93     // OnNetworkChanged will be called when a change occurs to the host
94     // computer's hardware or software that affects the route network packets
95     // take to any network server. Some examples:
96     //   1. A network connection becoming available or going away. For example
97     //      plugging or unplugging an Ethernet cable, WiFi or cellular modem
98     //      connecting or disconnecting from a network, or a VPN tunnel being
99     //      established or taken down.
100     //   2. An active network connection's IP address changes.
101     //   3. A change to the local IP routing tables.
102     // The signal shall only be produced when the change is complete.  For
103     // example if a new network connection has become available, only give the
104     // signal once we think the O/S has finished establishing the connection
105     // (i.e. DHCP is done) to the point where the new connection is usable.
106     // The signal shall not be produced spuriously as it will be triggering some
107     // expensive operations, like socket pools closing all connections and
108     // sockets and then re-establishing them.
109     // |type| indicates the type of the active primary network connection after
110     // the change.  Observers performing "constructive" activities like trying
111     // to establish a connection to a server should only do so when
112     // |type != CONNECTION_NONE|.  Observers performing "destructive" activities
113     // like resetting already established server connections should only do so
114     // when |type == CONNECTION_NONE|.  OnNetworkChanged will always be called
115     // with CONNECTION_NONE immediately prior to being called with an online
116     // state; this is done to make sure that destructive actions take place
117     // prior to constructive actions.
118     virtual void OnNetworkChanged(ConnectionType type) = 0;
119 
120    protected:
NetworkChangeObserver()121     NetworkChangeObserver() {}
~NetworkChangeObserver()122     virtual ~NetworkChangeObserver() {}
123 
124    private:
125     DISALLOW_COPY_AND_ASSIGN(NetworkChangeObserver);
126   };
127 
128   virtual ~NetworkChangeNotifier();
129 
130   // See the description of NetworkChangeNotifier::GetConnectionType().
131   // Implementations must be thread-safe. Implementations must also be
132   // cheap as this could be called (repeatedly) from the network thread.
133   virtual ConnectionType GetCurrentConnectionType() const = 0;
134 
135   // Replaces the default class factory instance of NetworkChangeNotifier class.
136   // The method will take over the ownership of |factory| object.
137   static void SetFactory(NetworkChangeNotifierFactory* factory);
138 
139   // Creates the process-wide, platform-specific NetworkChangeNotifier.  The
140   // caller owns the returned pointer.  You may call this on any thread.  You
141   // may also avoid creating this entirely (in which case nothing will be
142   // monitored), but if you do create it, you must do so before any other
143   // threads try to access the API below, and it must outlive all other threads
144   // which might try to use it.
145   static NetworkChangeNotifier* Create();
146 
147   // Returns the connection type.
148   // A return value of |CONNECTION_NONE| is a pretty strong indicator that the
149   // user won't be able to connect to remote sites. However, another return
150   // value doesn't imply that the user will be able to connect to remote sites;
151   // even if some link is up, it is uncertain whether a particular connection
152   // attempt to a particular remote site will be successful.
153   // The returned value only describes the connection currently used by the
154   // device, and does not take into account other machines on the network. For
155   // example, if the device is connected using Wifi to a 3G gateway to access
156   // the internet, the connection type is CONNECTION_WIFI.
157   static ConnectionType GetConnectionType();
158 
159   // Retrieve the last read DnsConfig. This could be expensive if the system has
160   // a large HOSTS file.
161   static void GetDnsConfig(DnsConfig* config);
162 
163 #if defined(OS_LINUX)
164   // Returns the AddressTrackerLinux if present.
165   static const internal::AddressTrackerLinux* GetAddressTracker();
166 #endif
167 
168   // Convenience method to determine if the user is offline.
169   // Returns true if there is currently no internet connection.
170   //
171   // A return value of |true| is a pretty strong indicator that the user
172   // won't be able to connect to remote sites. However, a return value of
173   // |false| is inconclusive; even if some link is up, it is uncertain
174   // whether a particular connection attempt to a particular remote site
175   // will be successfully.
176   static bool IsOffline();
177 
178   // Returns true if |type| is a cellular connection.
179   // Returns false if |type| is CONNECTION_UNKNOWN, and thus, depending on the
180   // implementation of GetConnectionType(), it is possible that
181   // IsConnectionCellular(GetConnectionType()) returns false even if the
182   // current connection is cellular.
183   static bool IsConnectionCellular(ConnectionType type);
184 
185   // Like Create(), but for use in tests.  The mock object doesn't monitor any
186   // events, it merely rebroadcasts notifications when requested.
187   static NetworkChangeNotifier* CreateMock();
188 
189   // Registers |observer| to receive notifications of network changes.  The
190   // thread on which this is called is the thread on which |observer| will be
191   // called back with notifications.  This is safe to call if Create() has not
192   // been called (as long as it doesn't race the Create() call on another
193   // thread), in which case it will simply do nothing.
194   static void AddIPAddressObserver(IPAddressObserver* observer);
195   static void AddConnectionTypeObserver(ConnectionTypeObserver* observer);
196   static void AddDNSObserver(DNSObserver* observer);
197   static void AddNetworkChangeObserver(NetworkChangeObserver* observer);
198 
199   // Unregisters |observer| from receiving notifications.  This must be called
200   // on the same thread on which AddObserver() was called.  Like AddObserver(),
201   // this is safe to call if Create() has not been called (as long as it doesn't
202   // race the Create() call on another thread), in which case it will simply do
203   // nothing.  Technically, it's also safe to call after the notifier object has
204   // been destroyed, if the call doesn't race the notifier's destruction, but
205   // there's no reason to use the API in this risky way, so don't do it.
206   static void RemoveIPAddressObserver(IPAddressObserver* observer);
207   static void RemoveConnectionTypeObserver(ConnectionTypeObserver* observer);
208   static void RemoveDNSObserver(DNSObserver* observer);
209   static void RemoveNetworkChangeObserver(NetworkChangeObserver* observer);
210 
211   // Allow unit tests to trigger notifications.
NotifyObserversOfIPAddressChangeForTests()212   static void NotifyObserversOfIPAddressChangeForTests() {
213     NotifyObserversOfIPAddressChange();
214   }
215 
216   // Return a string equivalent to |type|.
217   static const char* ConnectionTypeToString(ConnectionType type);
218 
219   // Let the NetworkChangeNotifier know we received some data.
220   // This is used for producing histogram data about the accuracy of
221   // the NetworkChangenotifier's online detection and rough network
222   // connection measurements.
223   static void NotifyDataReceived(const URLRequest& request, int bytes_read);
224 
225   // Register the Observer callbacks for producing histogram data.  This
226   // should be called from the network thread to avoid race conditions.
227   static void InitHistogramWatcher();
228 
229   // Allows a second NetworkChangeNotifier to be created for unit testing, so
230   // the test suite can create a MockNetworkChangeNotifier, but platform
231   // specific NetworkChangeNotifiers can also be created for testing.  To use,
232   // create an DisableForTest object, and then create the new
233   // NetworkChangeNotifier object.  The NetworkChangeNotifier must be
234   // destroyed before the DisableForTest object, as its destruction will restore
235   // the original NetworkChangeNotifier.
236   class NET_EXPORT DisableForTest {
237    public:
238     DisableForTest();
239     ~DisableForTest();
240 
241    private:
242     // The original NetworkChangeNotifier to be restored on destruction.
243     NetworkChangeNotifier* network_change_notifier_;
244   };
245 
246  protected:
247   // NetworkChanged signal is calculated from the IPAddressChanged and
248   // ConnectionTypeChanged signals. Delay parameters control how long to delay
249   // producing NetworkChanged signal after particular input signals so as to
250   // combine duplicates.  In other words if an input signal is repeated within
251   // the corresponding delay period, only one resulting NetworkChange signal is
252   // produced.
253   struct NET_EXPORT NetworkChangeCalculatorParams {
254     NetworkChangeCalculatorParams();
255     // Controls delay after OnIPAddressChanged when transitioning from an
256     // offline state.
257     base::TimeDelta ip_address_offline_delay_;
258     // Controls delay after OnIPAddressChanged when transitioning from an
259     // online state.
260     base::TimeDelta ip_address_online_delay_;
261     // Controls delay after OnConnectionTypeChanged when transitioning from an
262     // offline state.
263     base::TimeDelta connection_type_offline_delay_;
264     // Controls delay after OnConnectionTypeChanged when transitioning from an
265     // online state.
266     base::TimeDelta connection_type_online_delay_;
267   };
268 
269   explicit NetworkChangeNotifier(
270       const NetworkChangeCalculatorParams& params =
271           NetworkChangeCalculatorParams());
272 
273 #if defined(OS_LINUX)
274   // Returns the AddressTrackerLinux if present.
275   // TODO(szym): Retrieve AddressMap from NetworkState. http://crbug.com/144212
276   virtual const internal::AddressTrackerLinux*
277       GetAddressTrackerInternal() const;
278 #endif
279 
280   // Broadcasts a notification to all registered observers.  Note that this
281   // happens asynchronously, even for observers on the current thread, even in
282   // tests.
283   static void NotifyObserversOfIPAddressChange();
284   static void NotifyObserversOfConnectionTypeChange();
285   static void NotifyObserversOfDNSChange();
286   static void NotifyObserversOfNetworkChange(ConnectionType type);
287 
288   // Stores |config| in NetworkState and notifies observers.
289   static void SetDnsConfig(const DnsConfig& config);
290 
291  private:
292   friend class HostResolverImplDnsTest;
293   friend class NetworkChangeNotifierAndroidTest;
294   friend class NetworkChangeNotifierLinuxTest;
295   friend class NetworkChangeNotifierWinTest;
296 
297   class NetworkState;
298   class NetworkChangeCalculator;
299 
300   const scoped_refptr<ObserverListThreadSafe<IPAddressObserver> >
301       ip_address_observer_list_;
302   const scoped_refptr<ObserverListThreadSafe<ConnectionTypeObserver> >
303       connection_type_observer_list_;
304   const scoped_refptr<ObserverListThreadSafe<DNSObserver> >
305       resolver_state_observer_list_;
306   const scoped_refptr<ObserverListThreadSafe<NetworkChangeObserver> >
307       network_change_observer_list_;
308 
309   // The current network state. Hosts DnsConfig, exposed via GetDnsConfig.
310   scoped_ptr<NetworkState> network_state_;
311 
312   // A little-piggy-back observer that simply logs UMA histogram data.
313   scoped_ptr<HistogramWatcher> histogram_watcher_;
314 
315   // Computes NetworkChange signal from IPAddress and ConnectionType signals.
316   scoped_ptr<NetworkChangeCalculator> network_change_calculator_;
317 
318   DISALLOW_COPY_AND_ASSIGN(NetworkChangeNotifier);
319 };
320 
321 }  // namespace net
322 
323 #endif  // NET_BASE_NETWORK_CHANGE_NOTIFIER_H_
324