blob: 31d28ef3e77084d24287a9296d62ca4cef8a8c4b [file] [log] [blame]
Torne (Richard Coles)58218062012-11-14 11:43:16 +00001// 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_URL_REQUEST_URL_REQUEST_THROTTLER_MANAGER_H_
6#define NET_URL_REQUEST_URL_REQUEST_THROTTLER_MANAGER_H_
7
8#include <map>
9#include <set>
10#include <string>
11
12#include "base/basictypes.h"
13#include "base/memory/ref_counted.h"
14#include "base/threading/non_thread_safe.h"
15#include "base/threading/platform_thread.h"
Torne (Richard Coles)58218062012-11-14 11:43:16 +000016#include "net/base/net_export.h"
17#include "net/base/network_change_notifier.h"
18#include "net/url_request/url_request_throttler_entry.h"
Ben Murdoch7dbb3d52013-07-17 14:55:54 +010019#include "url/gurl.h"
Torne (Richard Coles)58218062012-11-14 11:43:16 +000020
21namespace net {
22
23class BoundNetLog;
24class NetLog;
25
26// Class that registers URL request throttler entries for URLs being accessed
27// in order to supervise traffic. URL requests for HTTP contents should
28// register their URLs in this manager on each request.
29//
30// URLRequestThrottlerManager maintains a map of URL IDs to URL request
31// throttler entries. It creates URL request throttler entries when new URLs
32// are registered, and does garbage collection from time to time in order to
33// clean out outdated entries. URL ID consists of lowercased scheme, host, port
34// and path. All URLs converted to the same ID will share the same entry.
35class NET_EXPORT URLRequestThrottlerManager
36 : NON_EXPORTED_BASE(public base::NonThreadSafe),
37 public NetworkChangeNotifier::IPAddressObserver,
38 public NetworkChangeNotifier::ConnectionTypeObserver {
39 public:
40 URLRequestThrottlerManager();
41 virtual ~URLRequestThrottlerManager();
42
43 // Must be called for every request, returns the URL request throttler entry
44 // associated with the URL. The caller must inform this entry of some events.
45 // Please refer to url_request_throttler_entry_interface.h for further
46 // informations.
47 scoped_refptr<URLRequestThrottlerEntryInterface> RegisterRequestUrl(
48 const GURL& url);
49
50 // Adds the given host to a list of sites for which exponential back-off
51 // throttling will be disabled. Subdomains are not included, so they
52 // must be added separately.
53 void AddToOptOutList(const std::string& host);
54
55 // Registers a new entry in this service and overrides the existing entry (if
56 // any) for the URL. The service will hold a reference to the entry.
57 // It is only used by unit tests.
58 void OverrideEntryForTests(const GURL& url, URLRequestThrottlerEntry* entry);
59
60 // Explicitly erases an entry.
61 // This is useful to remove those entries which have got infinite lifetime and
62 // thus won't be garbage collected.
63 // It is only used by unit tests.
64 void EraseEntryForTests(const GURL& url);
65
66 // Turns threading model verification on or off. Any code that correctly
67 // uses the network stack should preferably call this function to enable
68 // verification of correct adherence to the network stack threading model.
69 void set_enable_thread_checks(bool enable);
70 bool enable_thread_checks() const;
71
72 // Whether throttling is enabled or not.
73 void set_enforce_throttling(bool enforce);
74 bool enforce_throttling();
75
76 // Sets the NetLog instance to use.
77 void set_net_log(NetLog* net_log);
78 NetLog* net_log() const;
79
80 // IPAddressObserver interface.
81 virtual void OnIPAddressChanged() OVERRIDE;
82
83 // ConnectionTypeObserver interface.
84 virtual void OnConnectionTypeChanged(
85 NetworkChangeNotifier::ConnectionType type) OVERRIDE;
86
87 // Method that allows us to transform a URL into an ID that can be used in our
88 // map. Resulting IDs will be lowercase and consist of the scheme, host, port
89 // and path (without query string, fragment, etc.).
90 // If the URL is invalid, the invalid spec will be returned, without any
91 // transformation.
92 std::string GetIdFromUrl(const GURL& url) const;
93
94 // Method that ensures the map gets cleaned from time to time. The period at
95 // which garbage collecting happens is adjustable with the
96 // kRequestBetweenCollecting constant.
97 void GarbageCollectEntriesIfNecessary();
98
99 // Method that does the actual work of garbage collecting.
100 void GarbageCollectEntries();
101
102 // When we switch from online to offline or change IP addresses, we
103 // clear all back-off history. This is a precaution in case the change in
104 // online state now lets us communicate without error with servers that
105 // we were previously getting 500 or 503 responses from (perhaps the
106 // responses are from a badly-written proxy that should have returned a
107 // 502 or 504 because it's upstream connection was down or it had no route
108 // to the server).
109 void OnNetworkChange();
110
111 // Used by tests.
112 int GetNumberOfEntriesForTests() const { return url_entries_.size(); }
113
114 private:
115 // From each URL we generate an ID composed of the scheme, host, port and path
116 // that allows us to uniquely map an entry to it.
117 typedef std::map<std::string, scoped_refptr<URLRequestThrottlerEntry> >
118 UrlEntryMap;
119
120 // We maintain a set of hosts that have opted out of exponential
121 // back-off throttling.
122 typedef std::set<std::string> OptOutHosts;
123
124 // Maximum number of entries that we are willing to collect in our map.
125 static const unsigned int kMaximumNumberOfEntries;
126 // Number of requests that will be made between garbage collection.
127 static const unsigned int kRequestsBetweenCollecting;
128
129 // Map that contains a list of URL ID and their matching
130 // URLRequestThrottlerEntry.
131 UrlEntryMap url_entries_;
132
133 // Set of hosts that have opted out.
134 OptOutHosts opt_out_hosts_;
135
136 // This keeps track of how many requests have been made. Used with
137 // GarbageCollectEntries.
138 unsigned int requests_since_last_gc_;
139
140 // Valid after construction.
141 GURL::Replacements url_id_replacements_;
142
143 // Certain tests do not obey the net component's threading policy, so we
144 // keep track of whether we're being used by tests, and turn off certain
145 // checks.
146 //
147 // TODO(joi): See if we can fix the offending unit tests and remove this
148 // workaround.
149 bool enable_thread_checks_;
150
151 // Initially false, switches to true once we have logged because of back-off
152 // being disabled for localhost.
153 bool logged_for_localhost_disabled_;
154
155 // NetLog to use, if configured.
156 BoundNetLog net_log_;
157
158 // Valid once we've registered for network notifications.
159 base::PlatformThreadId registered_from_thread_;
160
161 DISALLOW_COPY_AND_ASSIGN(URLRequestThrottlerManager);
162};
163
164} // namespace net
165
166#endif // NET_URL_REQUEST_URL_REQUEST_THROTTLER_MANAGER_H_