1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker
5*6777b538SAndroid Build Coastguard Worker #include "net/dns/host_cache.h"
6*6777b538SAndroid Build Coastguard Worker
7*6777b538SAndroid Build Coastguard Worker #include <algorithm>
8*6777b538SAndroid Build Coastguard Worker #include <iterator>
9*6777b538SAndroid Build Coastguard Worker #include <map>
10*6777b538SAndroid Build Coastguard Worker #include <memory>
11*6777b538SAndroid Build Coastguard Worker #include <optional>
12*6777b538SAndroid Build Coastguard Worker #include <ostream>
13*6777b538SAndroid Build Coastguard Worker #include <set>
14*6777b538SAndroid Build Coastguard Worker #include <string>
15*6777b538SAndroid Build Coastguard Worker #include <string_view>
16*6777b538SAndroid Build Coastguard Worker #include <type_traits>
17*6777b538SAndroid Build Coastguard Worker #include <unordered_set>
18*6777b538SAndroid Build Coastguard Worker #include <utility>
19*6777b538SAndroid Build Coastguard Worker #include <vector>
20*6777b538SAndroid Build Coastguard Worker
21*6777b538SAndroid Build Coastguard Worker #include "base/check_op.h"
22*6777b538SAndroid Build Coastguard Worker #include "base/containers/contains.h"
23*6777b538SAndroid Build Coastguard Worker #include "base/functional/bind.h"
24*6777b538SAndroid Build Coastguard Worker #include "base/metrics/field_trial.h"
25*6777b538SAndroid Build Coastguard Worker #include "base/metrics/histogram_macros.h"
26*6777b538SAndroid Build Coastguard Worker #include "base/numerics/safe_conversions.h"
27*6777b538SAndroid Build Coastguard Worker #include "base/strings/string_number_conversions.h"
28*6777b538SAndroid Build Coastguard Worker #include "base/time/default_tick_clock.h"
29*6777b538SAndroid Build Coastguard Worker #include "base/types/optional_util.h"
30*6777b538SAndroid Build Coastguard Worker #include "base/value_iterators.h"
31*6777b538SAndroid Build Coastguard Worker #include "net/base/address_family.h"
32*6777b538SAndroid Build Coastguard Worker #include "net/base/ip_endpoint.h"
33*6777b538SAndroid Build Coastguard Worker #include "net/base/network_anonymization_key.h"
34*6777b538SAndroid Build Coastguard Worker #include "net/base/trace_constants.h"
35*6777b538SAndroid Build Coastguard Worker #include "net/base/tracing.h"
36*6777b538SAndroid Build Coastguard Worker #include "net/dns/host_resolver.h"
37*6777b538SAndroid Build Coastguard Worker #include "net/dns/host_resolver_internal_result.h"
38*6777b538SAndroid Build Coastguard Worker #include "net/dns/https_record_rdata.h"
39*6777b538SAndroid Build Coastguard Worker #include "net/dns/public/dns_protocol.h"
40*6777b538SAndroid Build Coastguard Worker #include "net/dns/public/host_resolver_source.h"
41*6777b538SAndroid Build Coastguard Worker #include "net/log/net_log.h"
42*6777b538SAndroid Build Coastguard Worker #include "third_party/abseil-cpp/absl/types/variant.h"
43*6777b538SAndroid Build Coastguard Worker #include "url/scheme_host_port.h"
44*6777b538SAndroid Build Coastguard Worker
45*6777b538SAndroid Build Coastguard Worker namespace net {
46*6777b538SAndroid Build Coastguard Worker
47*6777b538SAndroid Build Coastguard Worker namespace {
48*6777b538SAndroid Build Coastguard Worker
49*6777b538SAndroid Build Coastguard Worker #define CACHE_HISTOGRAM_TIME(name, time) \
50*6777b538SAndroid Build Coastguard Worker UMA_HISTOGRAM_LONG_TIMES("DNS.HostCache." name, time)
51*6777b538SAndroid Build Coastguard Worker
52*6777b538SAndroid Build Coastguard Worker #define CACHE_HISTOGRAM_COUNT(name, count) \
53*6777b538SAndroid Build Coastguard Worker UMA_HISTOGRAM_COUNTS_1000("DNS.HostCache." name, count)
54*6777b538SAndroid Build Coastguard Worker
55*6777b538SAndroid Build Coastguard Worker #define CACHE_HISTOGRAM_ENUM(name, value, max) \
56*6777b538SAndroid Build Coastguard Worker UMA_HISTOGRAM_ENUMERATION("DNS.HostCache." name, value, max)
57*6777b538SAndroid Build Coastguard Worker
58*6777b538SAndroid Build Coastguard Worker // String constants for dictionary keys.
59*6777b538SAndroid Build Coastguard Worker const char kSchemeKey[] = "scheme";
60*6777b538SAndroid Build Coastguard Worker const char kHostnameKey[] = "hostname";
61*6777b538SAndroid Build Coastguard Worker const char kPortKey[] = "port";
62*6777b538SAndroid Build Coastguard Worker const char kDnsQueryTypeKey[] = "dns_query_type";
63*6777b538SAndroid Build Coastguard Worker const char kFlagsKey[] = "flags";
64*6777b538SAndroid Build Coastguard Worker const char kHostResolverSourceKey[] = "host_resolver_source";
65*6777b538SAndroid Build Coastguard Worker const char kSecureKey[] = "secure";
66*6777b538SAndroid Build Coastguard Worker const char kNetworkAnonymizationKey[] = "network_anonymization_key";
67*6777b538SAndroid Build Coastguard Worker const char kExpirationKey[] = "expiration";
68*6777b538SAndroid Build Coastguard Worker const char kTtlKey[] = "ttl";
69*6777b538SAndroid Build Coastguard Worker const char kPinnedKey[] = "pinned";
70*6777b538SAndroid Build Coastguard Worker const char kNetworkChangesKey[] = "network_changes";
71*6777b538SAndroid Build Coastguard Worker const char kNetErrorKey[] = "net_error";
72*6777b538SAndroid Build Coastguard Worker const char kIpEndpointsKey[] = "ip_endpoints";
73*6777b538SAndroid Build Coastguard Worker const char kEndpointAddressKey[] = "endpoint_address";
74*6777b538SAndroid Build Coastguard Worker const char kEndpointPortKey[] = "endpoint_port";
75*6777b538SAndroid Build Coastguard Worker const char kEndpointMetadatasKey[] = "endpoint_metadatas";
76*6777b538SAndroid Build Coastguard Worker const char kEndpointMetadataWeightKey[] = "endpoint_metadata_weight";
77*6777b538SAndroid Build Coastguard Worker const char kEndpointMetadataValueKey[] = "endpoint_metadata_value";
78*6777b538SAndroid Build Coastguard Worker const char kAliasesKey[] = "aliases";
79*6777b538SAndroid Build Coastguard Worker const char kAddressesKey[] = "addresses";
80*6777b538SAndroid Build Coastguard Worker const char kTextRecordsKey[] = "text_records";
81*6777b538SAndroid Build Coastguard Worker const char kHostnameResultsKey[] = "hostname_results";
82*6777b538SAndroid Build Coastguard Worker const char kHostPortsKey[] = "host_ports";
83*6777b538SAndroid Build Coastguard Worker const char kCanonicalNamesKey[] = "canonical_names";
84*6777b538SAndroid Build Coastguard Worker
IpEndpointToValue(const IPEndPoint & endpoint)85*6777b538SAndroid Build Coastguard Worker base::Value IpEndpointToValue(const IPEndPoint& endpoint) {
86*6777b538SAndroid Build Coastguard Worker base::Value::Dict dictionary;
87*6777b538SAndroid Build Coastguard Worker dictionary.Set(kEndpointAddressKey, endpoint.ToStringWithoutPort());
88*6777b538SAndroid Build Coastguard Worker dictionary.Set(kEndpointPortKey, endpoint.port());
89*6777b538SAndroid Build Coastguard Worker return base::Value(std::move(dictionary));
90*6777b538SAndroid Build Coastguard Worker }
91*6777b538SAndroid Build Coastguard Worker
IpEndpointFromValue(const base::Value & value)92*6777b538SAndroid Build Coastguard Worker std::optional<IPEndPoint> IpEndpointFromValue(const base::Value& value) {
93*6777b538SAndroid Build Coastguard Worker if (!value.is_dict())
94*6777b538SAndroid Build Coastguard Worker return std::nullopt;
95*6777b538SAndroid Build Coastguard Worker
96*6777b538SAndroid Build Coastguard Worker const base::Value::Dict& dict = value.GetDict();
97*6777b538SAndroid Build Coastguard Worker const std::string* ip_str = dict.FindString(kEndpointAddressKey);
98*6777b538SAndroid Build Coastguard Worker std::optional<int> port = dict.FindInt(kEndpointPortKey);
99*6777b538SAndroid Build Coastguard Worker
100*6777b538SAndroid Build Coastguard Worker if (!ip_str || !port ||
101*6777b538SAndroid Build Coastguard Worker !base::IsValueInRangeForNumericType<uint16_t>(port.value())) {
102*6777b538SAndroid Build Coastguard Worker return std::nullopt;
103*6777b538SAndroid Build Coastguard Worker }
104*6777b538SAndroid Build Coastguard Worker
105*6777b538SAndroid Build Coastguard Worker IPAddress ip;
106*6777b538SAndroid Build Coastguard Worker if (!ip.AssignFromIPLiteral(*ip_str))
107*6777b538SAndroid Build Coastguard Worker return std::nullopt;
108*6777b538SAndroid Build Coastguard Worker
109*6777b538SAndroid Build Coastguard Worker return IPEndPoint(ip, base::checked_cast<uint16_t>(port.value()));
110*6777b538SAndroid Build Coastguard Worker }
111*6777b538SAndroid Build Coastguard Worker
EndpointMetadataPairToValue(const std::pair<HttpsRecordPriority,ConnectionEndpointMetadata> & pair)112*6777b538SAndroid Build Coastguard Worker base::Value EndpointMetadataPairToValue(
113*6777b538SAndroid Build Coastguard Worker const std::pair<HttpsRecordPriority, ConnectionEndpointMetadata>& pair) {
114*6777b538SAndroid Build Coastguard Worker base::Value::Dict dictionary;
115*6777b538SAndroid Build Coastguard Worker dictionary.Set(kEndpointMetadataWeightKey, pair.first);
116*6777b538SAndroid Build Coastguard Worker dictionary.Set(kEndpointMetadataValueKey, pair.second.ToValue());
117*6777b538SAndroid Build Coastguard Worker return base::Value(std::move(dictionary));
118*6777b538SAndroid Build Coastguard Worker }
119*6777b538SAndroid Build Coastguard Worker
120*6777b538SAndroid Build Coastguard Worker std::optional<std::pair<HttpsRecordPriority, ConnectionEndpointMetadata>>
EndpointMetadataPairFromValue(const base::Value & value)121*6777b538SAndroid Build Coastguard Worker EndpointMetadataPairFromValue(const base::Value& value) {
122*6777b538SAndroid Build Coastguard Worker if (!value.is_dict())
123*6777b538SAndroid Build Coastguard Worker return std::nullopt;
124*6777b538SAndroid Build Coastguard Worker
125*6777b538SAndroid Build Coastguard Worker const base::Value::Dict& dict = value.GetDict();
126*6777b538SAndroid Build Coastguard Worker std::optional<int> priority = dict.FindInt(kEndpointMetadataWeightKey);
127*6777b538SAndroid Build Coastguard Worker const base::Value* metadata_value = dict.Find(kEndpointMetadataValueKey);
128*6777b538SAndroid Build Coastguard Worker
129*6777b538SAndroid Build Coastguard Worker if (!priority || !base::IsValueInRangeForNumericType<HttpsRecordPriority>(
130*6777b538SAndroid Build Coastguard Worker priority.value())) {
131*6777b538SAndroid Build Coastguard Worker return std::nullopt;
132*6777b538SAndroid Build Coastguard Worker }
133*6777b538SAndroid Build Coastguard Worker
134*6777b538SAndroid Build Coastguard Worker if (!metadata_value)
135*6777b538SAndroid Build Coastguard Worker return std::nullopt;
136*6777b538SAndroid Build Coastguard Worker std::optional<ConnectionEndpointMetadata> metadata =
137*6777b538SAndroid Build Coastguard Worker ConnectionEndpointMetadata::FromValue(*metadata_value);
138*6777b538SAndroid Build Coastguard Worker if (!metadata)
139*6777b538SAndroid Build Coastguard Worker return std::nullopt;
140*6777b538SAndroid Build Coastguard Worker
141*6777b538SAndroid Build Coastguard Worker return std::pair(base::checked_cast<HttpsRecordPriority>(priority.value()),
142*6777b538SAndroid Build Coastguard Worker std::move(metadata).value());
143*6777b538SAndroid Build Coastguard Worker }
144*6777b538SAndroid Build Coastguard Worker
IPEndPointsFromLegacyAddressListValue(const base::Value::List & value,std::vector<IPEndPoint> & ip_endpoints)145*6777b538SAndroid Build Coastguard Worker bool IPEndPointsFromLegacyAddressListValue(
146*6777b538SAndroid Build Coastguard Worker const base::Value::List& value,
147*6777b538SAndroid Build Coastguard Worker std::vector<IPEndPoint>& ip_endpoints) {
148*6777b538SAndroid Build Coastguard Worker DCHECK(ip_endpoints.empty());
149*6777b538SAndroid Build Coastguard Worker for (const auto& it : value) {
150*6777b538SAndroid Build Coastguard Worker IPAddress address;
151*6777b538SAndroid Build Coastguard Worker const std::string* addr_string = it.GetIfString();
152*6777b538SAndroid Build Coastguard Worker if (!addr_string || !address.AssignFromIPLiteral(*addr_string)) {
153*6777b538SAndroid Build Coastguard Worker return false;
154*6777b538SAndroid Build Coastguard Worker }
155*6777b538SAndroid Build Coastguard Worker ip_endpoints.emplace_back(address, 0);
156*6777b538SAndroid Build Coastguard Worker }
157*6777b538SAndroid Build Coastguard Worker return true;
158*6777b538SAndroid Build Coastguard Worker }
159*6777b538SAndroid Build Coastguard Worker
160*6777b538SAndroid Build Coastguard Worker template <typename T>
MergeLists(T & target,const T & source)161*6777b538SAndroid Build Coastguard Worker void MergeLists(T& target, const T& source) {
162*6777b538SAndroid Build Coastguard Worker target.insert(target.end(), source.begin(), source.end());
163*6777b538SAndroid Build Coastguard Worker }
164*6777b538SAndroid Build Coastguard Worker
165*6777b538SAndroid Build Coastguard Worker template <typename T>
MergeContainers(T & target,const T & source)166*6777b538SAndroid Build Coastguard Worker void MergeContainers(T& target, const T& source) {
167*6777b538SAndroid Build Coastguard Worker target.insert(source.begin(), source.end());
168*6777b538SAndroid Build Coastguard Worker }
169*6777b538SAndroid Build Coastguard Worker
170*6777b538SAndroid Build Coastguard Worker // Used to reject empty and IP literal (whether or not surrounded by brackets)
171*6777b538SAndroid Build Coastguard Worker // hostnames.
IsValidHostname(std::string_view hostname)172*6777b538SAndroid Build Coastguard Worker bool IsValidHostname(std::string_view hostname) {
173*6777b538SAndroid Build Coastguard Worker if (hostname.empty())
174*6777b538SAndroid Build Coastguard Worker return false;
175*6777b538SAndroid Build Coastguard Worker
176*6777b538SAndroid Build Coastguard Worker IPAddress ip_address;
177*6777b538SAndroid Build Coastguard Worker if (ip_address.AssignFromIPLiteral(hostname) ||
178*6777b538SAndroid Build Coastguard Worker ParseURLHostnameToAddress(hostname, &ip_address)) {
179*6777b538SAndroid Build Coastguard Worker return false;
180*6777b538SAndroid Build Coastguard Worker }
181*6777b538SAndroid Build Coastguard Worker
182*6777b538SAndroid Build Coastguard Worker return true;
183*6777b538SAndroid Build Coastguard Worker }
184*6777b538SAndroid Build Coastguard Worker
GetHostname(const absl::variant<url::SchemeHostPort,std::string> & host)185*6777b538SAndroid Build Coastguard Worker const std::string& GetHostname(
186*6777b538SAndroid Build Coastguard Worker const absl::variant<url::SchemeHostPort, std::string>& host) {
187*6777b538SAndroid Build Coastguard Worker const std::string* hostname;
188*6777b538SAndroid Build Coastguard Worker if (absl::holds_alternative<url::SchemeHostPort>(host)) {
189*6777b538SAndroid Build Coastguard Worker hostname = &absl::get<url::SchemeHostPort>(host).host();
190*6777b538SAndroid Build Coastguard Worker } else {
191*6777b538SAndroid Build Coastguard Worker DCHECK(absl::holds_alternative<std::string>(host));
192*6777b538SAndroid Build Coastguard Worker hostname = &absl::get<std::string>(host);
193*6777b538SAndroid Build Coastguard Worker }
194*6777b538SAndroid Build Coastguard Worker
195*6777b538SAndroid Build Coastguard Worker DCHECK(IsValidHostname(*hostname));
196*6777b538SAndroid Build Coastguard Worker return *hostname;
197*6777b538SAndroid Build Coastguard Worker }
198*6777b538SAndroid Build Coastguard Worker
GetDnsQueryType(int dns_query_type)199*6777b538SAndroid Build Coastguard Worker std::optional<DnsQueryType> GetDnsQueryType(int dns_query_type) {
200*6777b538SAndroid Build Coastguard Worker for (const auto& type : kDnsQueryTypes) {
201*6777b538SAndroid Build Coastguard Worker if (base::strict_cast<int>(type.first) == dns_query_type)
202*6777b538SAndroid Build Coastguard Worker return type.first;
203*6777b538SAndroid Build Coastguard Worker }
204*6777b538SAndroid Build Coastguard Worker return std::nullopt;
205*6777b538SAndroid Build Coastguard Worker }
206*6777b538SAndroid Build Coastguard Worker
207*6777b538SAndroid Build Coastguard Worker } // namespace
208*6777b538SAndroid Build Coastguard Worker
209*6777b538SAndroid Build Coastguard Worker // Used in histograms; do not modify existing values.
210*6777b538SAndroid Build Coastguard Worker enum HostCache::SetOutcome : int {
211*6777b538SAndroid Build Coastguard Worker SET_INSERT = 0,
212*6777b538SAndroid Build Coastguard Worker SET_UPDATE_VALID = 1,
213*6777b538SAndroid Build Coastguard Worker SET_UPDATE_STALE = 2,
214*6777b538SAndroid Build Coastguard Worker MAX_SET_OUTCOME
215*6777b538SAndroid Build Coastguard Worker };
216*6777b538SAndroid Build Coastguard Worker
217*6777b538SAndroid Build Coastguard Worker // Used in histograms; do not modify existing values.
218*6777b538SAndroid Build Coastguard Worker enum HostCache::LookupOutcome : int {
219*6777b538SAndroid Build Coastguard Worker LOOKUP_MISS_ABSENT = 0,
220*6777b538SAndroid Build Coastguard Worker LOOKUP_MISS_STALE = 1,
221*6777b538SAndroid Build Coastguard Worker LOOKUP_HIT_VALID = 2,
222*6777b538SAndroid Build Coastguard Worker LOOKUP_HIT_STALE = 3,
223*6777b538SAndroid Build Coastguard Worker MAX_LOOKUP_OUTCOME
224*6777b538SAndroid Build Coastguard Worker };
225*6777b538SAndroid Build Coastguard Worker
226*6777b538SAndroid Build Coastguard Worker // Used in histograms; do not modify existing values.
227*6777b538SAndroid Build Coastguard Worker enum HostCache::EraseReason : int {
228*6777b538SAndroid Build Coastguard Worker ERASE_EVICT = 0,
229*6777b538SAndroid Build Coastguard Worker ERASE_CLEAR = 1,
230*6777b538SAndroid Build Coastguard Worker ERASE_DESTRUCT = 2,
231*6777b538SAndroid Build Coastguard Worker MAX_ERASE_REASON
232*6777b538SAndroid Build Coastguard Worker };
233*6777b538SAndroid Build Coastguard Worker
Key(absl::variant<url::SchemeHostPort,std::string> host,DnsQueryType dns_query_type,HostResolverFlags host_resolver_flags,HostResolverSource host_resolver_source,const NetworkAnonymizationKey & network_anonymization_key)234*6777b538SAndroid Build Coastguard Worker HostCache::Key::Key(absl::variant<url::SchemeHostPort, std::string> host,
235*6777b538SAndroid Build Coastguard Worker DnsQueryType dns_query_type,
236*6777b538SAndroid Build Coastguard Worker HostResolverFlags host_resolver_flags,
237*6777b538SAndroid Build Coastguard Worker HostResolverSource host_resolver_source,
238*6777b538SAndroid Build Coastguard Worker const NetworkAnonymizationKey& network_anonymization_key)
239*6777b538SAndroid Build Coastguard Worker : host(std::move(host)),
240*6777b538SAndroid Build Coastguard Worker dns_query_type(dns_query_type),
241*6777b538SAndroid Build Coastguard Worker host_resolver_flags(host_resolver_flags),
242*6777b538SAndroid Build Coastguard Worker host_resolver_source(host_resolver_source),
243*6777b538SAndroid Build Coastguard Worker network_anonymization_key(network_anonymization_key) {
244*6777b538SAndroid Build Coastguard Worker DCHECK(IsValidHostname(GetHostname(this->host)));
245*6777b538SAndroid Build Coastguard Worker if (absl::holds_alternative<url::SchemeHostPort>(this->host))
246*6777b538SAndroid Build Coastguard Worker DCHECK(absl::get<url::SchemeHostPort>(this->host).IsValid());
247*6777b538SAndroid Build Coastguard Worker }
248*6777b538SAndroid Build Coastguard Worker
249*6777b538SAndroid Build Coastguard Worker HostCache::Key::Key() = default;
250*6777b538SAndroid Build Coastguard Worker HostCache::Key::Key(const Key& key) = default;
251*6777b538SAndroid Build Coastguard Worker HostCache::Key::Key(Key&& key) = default;
252*6777b538SAndroid Build Coastguard Worker
253*6777b538SAndroid Build Coastguard Worker HostCache::Key::~Key() = default;
254*6777b538SAndroid Build Coastguard Worker
Entry(int error,Source source,std::optional<base::TimeDelta> ttl)255*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Entry(int error,
256*6777b538SAndroid Build Coastguard Worker Source source,
257*6777b538SAndroid Build Coastguard Worker std::optional<base::TimeDelta> ttl)
258*6777b538SAndroid Build Coastguard Worker : error_(error), source_(source), ttl_(ttl.value_or(kUnknownTtl)) {
259*6777b538SAndroid Build Coastguard Worker // If |ttl| has a value, must not be negative.
260*6777b538SAndroid Build Coastguard Worker DCHECK_GE(ttl.value_or(base::TimeDelta()), base::TimeDelta());
261*6777b538SAndroid Build Coastguard Worker DCHECK_NE(OK, error_);
262*6777b538SAndroid Build Coastguard Worker
263*6777b538SAndroid Build Coastguard Worker // host_cache.h defines its own `HttpsRecordPriority` due to
264*6777b538SAndroid Build Coastguard Worker // https_record_rdata.h not being allowed in the same places, but the types
265*6777b538SAndroid Build Coastguard Worker // should still be the same thing.
266*6777b538SAndroid Build Coastguard Worker static_assert(std::is_same<net::HttpsRecordPriority,
267*6777b538SAndroid Build Coastguard Worker HostCache::Entry::HttpsRecordPriority>::value,
268*6777b538SAndroid Build Coastguard Worker "`net::HttpsRecordPriority` and "
269*6777b538SAndroid Build Coastguard Worker "`HostCache::Entry::HttpsRecordPriority` must be same type");
270*6777b538SAndroid Build Coastguard Worker }
271*6777b538SAndroid Build Coastguard Worker
Entry(const std::set<std::unique_ptr<HostResolverInternalResult>> & results,base::Time now,base::TimeTicks now_ticks,Source empty_source)272*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Entry(
273*6777b538SAndroid Build Coastguard Worker const std::set<std::unique_ptr<HostResolverInternalResult>>& results,
274*6777b538SAndroid Build Coastguard Worker base::Time now,
275*6777b538SAndroid Build Coastguard Worker base::TimeTicks now_ticks,
276*6777b538SAndroid Build Coastguard Worker Source empty_source) {
277*6777b538SAndroid Build Coastguard Worker const HostResolverInternalResult* data_result = nullptr;
278*6777b538SAndroid Build Coastguard Worker const HostResolverInternalResult* metadata_result = nullptr;
279*6777b538SAndroid Build Coastguard Worker const HostResolverInternalResult* error_result = nullptr;
280*6777b538SAndroid Build Coastguard Worker std::vector<const HostResolverInternalResult*> alias_results;
281*6777b538SAndroid Build Coastguard Worker
282*6777b538SAndroid Build Coastguard Worker std::optional<base::TimeDelta> smallest_ttl =
283*6777b538SAndroid Build Coastguard Worker TtlFromInternalResults(results, now, now_ticks);
284*6777b538SAndroid Build Coastguard Worker std::optional<Source> source;
285*6777b538SAndroid Build Coastguard Worker for (auto it = results.cbegin(); it != results.cend();) {
286*6777b538SAndroid Build Coastguard Worker // Increment iterator now to allow extracting `result` (std::set::extract()
287*6777b538SAndroid Build Coastguard Worker // is guaranteed to not invalidate any iterators except those pointing to
288*6777b538SAndroid Build Coastguard Worker // the extracted value).
289*6777b538SAndroid Build Coastguard Worker const std::unique_ptr<HostResolverInternalResult>& result = *it++;
290*6777b538SAndroid Build Coastguard Worker
291*6777b538SAndroid Build Coastguard Worker Source result_source;
292*6777b538SAndroid Build Coastguard Worker switch (result->source()) {
293*6777b538SAndroid Build Coastguard Worker case HostResolverInternalResult::Source::kDns:
294*6777b538SAndroid Build Coastguard Worker result_source = SOURCE_DNS;
295*6777b538SAndroid Build Coastguard Worker break;
296*6777b538SAndroid Build Coastguard Worker case HostResolverInternalResult::Source::kHosts:
297*6777b538SAndroid Build Coastguard Worker result_source = SOURCE_HOSTS;
298*6777b538SAndroid Build Coastguard Worker break;
299*6777b538SAndroid Build Coastguard Worker case HostResolverInternalResult::Source::kUnknown:
300*6777b538SAndroid Build Coastguard Worker result_source = SOURCE_UNKNOWN;
301*6777b538SAndroid Build Coastguard Worker break;
302*6777b538SAndroid Build Coastguard Worker }
303*6777b538SAndroid Build Coastguard Worker
304*6777b538SAndroid Build Coastguard Worker switch (result->type()) {
305*6777b538SAndroid Build Coastguard Worker case HostResolverInternalResult::Type::kData:
306*6777b538SAndroid Build Coastguard Worker DCHECK(!data_result); // Expect at most one data result.
307*6777b538SAndroid Build Coastguard Worker data_result = result.get();
308*6777b538SAndroid Build Coastguard Worker break;
309*6777b538SAndroid Build Coastguard Worker case HostResolverInternalResult::Type::kMetadata:
310*6777b538SAndroid Build Coastguard Worker DCHECK(!metadata_result); // Expect at most one metadata result.
311*6777b538SAndroid Build Coastguard Worker metadata_result = result.get();
312*6777b538SAndroid Build Coastguard Worker break;
313*6777b538SAndroid Build Coastguard Worker case HostResolverInternalResult::Type::kError:
314*6777b538SAndroid Build Coastguard Worker DCHECK(!error_result); // Expect at most one error result.
315*6777b538SAndroid Build Coastguard Worker error_result = result.get();
316*6777b538SAndroid Build Coastguard Worker break;
317*6777b538SAndroid Build Coastguard Worker case HostResolverInternalResult::Type::kAlias:
318*6777b538SAndroid Build Coastguard Worker alias_results.emplace_back(result.get());
319*6777b538SAndroid Build Coastguard Worker break;
320*6777b538SAndroid Build Coastguard Worker }
321*6777b538SAndroid Build Coastguard Worker
322*6777b538SAndroid Build Coastguard Worker // Expect all results to have the same source.
323*6777b538SAndroid Build Coastguard Worker DCHECK(!source.has_value() || source.value() == result_source);
324*6777b538SAndroid Build Coastguard Worker source = result_source;
325*6777b538SAndroid Build Coastguard Worker }
326*6777b538SAndroid Build Coastguard Worker
327*6777b538SAndroid Build Coastguard Worker ttl_ = smallest_ttl.value_or(kUnknownTtl);
328*6777b538SAndroid Build Coastguard Worker source_ = source.value_or(empty_source);
329*6777b538SAndroid Build Coastguard Worker
330*6777b538SAndroid Build Coastguard Worker if (error_result) {
331*6777b538SAndroid Build Coastguard Worker DCHECK(!data_result);
332*6777b538SAndroid Build Coastguard Worker DCHECK(!metadata_result);
333*6777b538SAndroid Build Coastguard Worker
334*6777b538SAndroid Build Coastguard Worker error_ = error_result->AsError().error();
335*6777b538SAndroid Build Coastguard Worker
336*6777b538SAndroid Build Coastguard Worker // For error results, should not create entry with a TTL unless it is a
337*6777b538SAndroid Build Coastguard Worker // cacheable error.
338*6777b538SAndroid Build Coastguard Worker if (!error_result->expiration().has_value() &&
339*6777b538SAndroid Build Coastguard Worker !error_result->timed_expiration().has_value()) {
340*6777b538SAndroid Build Coastguard Worker ttl_ = kUnknownTtl;
341*6777b538SAndroid Build Coastguard Worker }
342*6777b538SAndroid Build Coastguard Worker } else if (!data_result && !metadata_result) {
343*6777b538SAndroid Build Coastguard Worker // Only alias results (or completely empty results). Never cacheable due to
344*6777b538SAndroid Build Coastguard Worker // being equivalent to an error result without TTL.
345*6777b538SAndroid Build Coastguard Worker error_ = ERR_NAME_NOT_RESOLVED;
346*6777b538SAndroid Build Coastguard Worker ttl_ = kUnknownTtl;
347*6777b538SAndroid Build Coastguard Worker } else {
348*6777b538SAndroid Build Coastguard Worker error_ = OK;
349*6777b538SAndroid Build Coastguard Worker }
350*6777b538SAndroid Build Coastguard Worker
351*6777b538SAndroid Build Coastguard Worker if (data_result) {
352*6777b538SAndroid Build Coastguard Worker DCHECK(!error_result);
353*6777b538SAndroid Build Coastguard Worker DCHECK(!data_result->AsData().endpoints().empty() ||
354*6777b538SAndroid Build Coastguard Worker !data_result->AsData().strings().empty() ||
355*6777b538SAndroid Build Coastguard Worker !data_result->AsData().hosts().empty());
356*6777b538SAndroid Build Coastguard Worker // Data results should always be cacheable.
357*6777b538SAndroid Build Coastguard Worker DCHECK(data_result->expiration().has_value() ||
358*6777b538SAndroid Build Coastguard Worker data_result->timed_expiration().has_value());
359*6777b538SAndroid Build Coastguard Worker
360*6777b538SAndroid Build Coastguard Worker ip_endpoints_ = data_result->AsData().endpoints();
361*6777b538SAndroid Build Coastguard Worker text_records_ = data_result->AsData().strings();
362*6777b538SAndroid Build Coastguard Worker hostnames_ = data_result->AsData().hosts();
363*6777b538SAndroid Build Coastguard Worker canonical_names_ = {data_result->domain_name()};
364*6777b538SAndroid Build Coastguard Worker
365*6777b538SAndroid Build Coastguard Worker for (const auto* alias_result : alias_results) {
366*6777b538SAndroid Build Coastguard Worker aliases_.insert(alias_result->domain_name());
367*6777b538SAndroid Build Coastguard Worker aliases_.insert(alias_result->AsAlias().alias_target());
368*6777b538SAndroid Build Coastguard Worker }
369*6777b538SAndroid Build Coastguard Worker aliases_.insert(data_result->domain_name());
370*6777b538SAndroid Build Coastguard Worker }
371*6777b538SAndroid Build Coastguard Worker if (metadata_result) {
372*6777b538SAndroid Build Coastguard Worker DCHECK(!error_result);
373*6777b538SAndroid Build Coastguard Worker // Metadata results should always be cacheable.
374*6777b538SAndroid Build Coastguard Worker DCHECK(metadata_result->expiration().has_value() ||
375*6777b538SAndroid Build Coastguard Worker metadata_result->timed_expiration().has_value());
376*6777b538SAndroid Build Coastguard Worker
377*6777b538SAndroid Build Coastguard Worker endpoint_metadatas_ = metadata_result->AsMetadata().metadatas();
378*6777b538SAndroid Build Coastguard Worker
379*6777b538SAndroid Build Coastguard Worker // Even if otherwise empty, having the metadata result object signifies
380*6777b538SAndroid Build Coastguard Worker // receiving a compatible HTTPS record.
381*6777b538SAndroid Build Coastguard Worker https_record_compatibility_ = std::vector<bool>{true};
382*6777b538SAndroid Build Coastguard Worker
383*6777b538SAndroid Build Coastguard Worker if (endpoint_metadatas_.empty()) {
384*6777b538SAndroid Build Coastguard Worker error_ = ERR_NAME_NOT_RESOLVED;
385*6777b538SAndroid Build Coastguard Worker }
386*6777b538SAndroid Build Coastguard Worker }
387*6777b538SAndroid Build Coastguard Worker }
388*6777b538SAndroid Build Coastguard Worker
389*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Entry(const Entry& entry) = default;
390*6777b538SAndroid Build Coastguard Worker
391*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Entry(Entry&& entry) = default;
392*6777b538SAndroid Build Coastguard Worker
393*6777b538SAndroid Build Coastguard Worker HostCache::Entry::~Entry() = default;
394*6777b538SAndroid Build Coastguard Worker
GetEndpoints() const395*6777b538SAndroid Build Coastguard Worker std::vector<HostResolverEndpointResult> HostCache::Entry::GetEndpoints() const {
396*6777b538SAndroid Build Coastguard Worker std::vector<HostResolverEndpointResult> endpoints;
397*6777b538SAndroid Build Coastguard Worker
398*6777b538SAndroid Build Coastguard Worker if (ip_endpoints_.empty()) {
399*6777b538SAndroid Build Coastguard Worker return endpoints;
400*6777b538SAndroid Build Coastguard Worker }
401*6777b538SAndroid Build Coastguard Worker
402*6777b538SAndroid Build Coastguard Worker std::vector<ConnectionEndpointMetadata> metadatas = GetMetadatas();
403*6777b538SAndroid Build Coastguard Worker
404*6777b538SAndroid Build Coastguard Worker if (!metadatas.empty() && canonical_names_.size() == 1) {
405*6777b538SAndroid Build Coastguard Worker // Currently Chrome uses HTTPS records only when A and AAAA records are at
406*6777b538SAndroid Build Coastguard Worker // the same canonical name and that matches the HTTPS target name.
407*6777b538SAndroid Build Coastguard Worker for (ConnectionEndpointMetadata& metadata : metadatas) {
408*6777b538SAndroid Build Coastguard Worker if (!base::Contains(canonical_names_, metadata.target_name)) {
409*6777b538SAndroid Build Coastguard Worker continue;
410*6777b538SAndroid Build Coastguard Worker }
411*6777b538SAndroid Build Coastguard Worker endpoints.emplace_back();
412*6777b538SAndroid Build Coastguard Worker endpoints.back().ip_endpoints = ip_endpoints_;
413*6777b538SAndroid Build Coastguard Worker endpoints.back().metadata = std::move(metadata);
414*6777b538SAndroid Build Coastguard Worker }
415*6777b538SAndroid Build Coastguard Worker }
416*6777b538SAndroid Build Coastguard Worker
417*6777b538SAndroid Build Coastguard Worker // Add a final non-alternative endpoint at the end.
418*6777b538SAndroid Build Coastguard Worker endpoints.emplace_back();
419*6777b538SAndroid Build Coastguard Worker endpoints.back().ip_endpoints = ip_endpoints_;
420*6777b538SAndroid Build Coastguard Worker
421*6777b538SAndroid Build Coastguard Worker return endpoints;
422*6777b538SAndroid Build Coastguard Worker }
423*6777b538SAndroid Build Coastguard Worker
GetMetadatas() const424*6777b538SAndroid Build Coastguard Worker std::vector<ConnectionEndpointMetadata> HostCache::Entry::GetMetadatas() const {
425*6777b538SAndroid Build Coastguard Worker std::vector<ConnectionEndpointMetadata> metadatas;
426*6777b538SAndroid Build Coastguard Worker HttpsRecordPriority last_priority = 0;
427*6777b538SAndroid Build Coastguard Worker for (const auto& metadata : endpoint_metadatas_) {
428*6777b538SAndroid Build Coastguard Worker // Ensure metadatas are iterated in priority order.
429*6777b538SAndroid Build Coastguard Worker DCHECK_GE(metadata.first, last_priority);
430*6777b538SAndroid Build Coastguard Worker last_priority = metadata.first;
431*6777b538SAndroid Build Coastguard Worker
432*6777b538SAndroid Build Coastguard Worker metadatas.push_back(metadata.second);
433*6777b538SAndroid Build Coastguard Worker }
434*6777b538SAndroid Build Coastguard Worker
435*6777b538SAndroid Build Coastguard Worker return metadatas;
436*6777b538SAndroid Build Coastguard Worker }
437*6777b538SAndroid Build Coastguard Worker
GetOptionalTtl() const438*6777b538SAndroid Build Coastguard Worker std::optional<base::TimeDelta> HostCache::Entry::GetOptionalTtl() const {
439*6777b538SAndroid Build Coastguard Worker if (has_ttl())
440*6777b538SAndroid Build Coastguard Worker return ttl();
441*6777b538SAndroid Build Coastguard Worker else
442*6777b538SAndroid Build Coastguard Worker return std::nullopt;
443*6777b538SAndroid Build Coastguard Worker }
444*6777b538SAndroid Build Coastguard Worker
445*6777b538SAndroid Build Coastguard Worker // static
MergeEntries(Entry front,Entry back)446*6777b538SAndroid Build Coastguard Worker HostCache::Entry HostCache::Entry::MergeEntries(Entry front, Entry back) {
447*6777b538SAndroid Build Coastguard Worker // Only expected to merge OK or ERR_NAME_NOT_RESOLVED results.
448*6777b538SAndroid Build Coastguard Worker DCHECK(front.error() == OK || front.error() == ERR_NAME_NOT_RESOLVED);
449*6777b538SAndroid Build Coastguard Worker DCHECK(back.error() == OK || back.error() == ERR_NAME_NOT_RESOLVED);
450*6777b538SAndroid Build Coastguard Worker
451*6777b538SAndroid Build Coastguard Worker // Build results in |front| to preserve unmerged fields.
452*6777b538SAndroid Build Coastguard Worker
453*6777b538SAndroid Build Coastguard Worker front.error_ =
454*6777b538SAndroid Build Coastguard Worker front.error() == OK || back.error() == OK ? OK : ERR_NAME_NOT_RESOLVED;
455*6777b538SAndroid Build Coastguard Worker
456*6777b538SAndroid Build Coastguard Worker MergeLists(front.ip_endpoints_, back.ip_endpoints_);
457*6777b538SAndroid Build Coastguard Worker MergeContainers(front.endpoint_metadatas_, back.endpoint_metadatas_);
458*6777b538SAndroid Build Coastguard Worker MergeContainers(front.aliases_, back.aliases_);
459*6777b538SAndroid Build Coastguard Worker MergeLists(front.text_records_, back.text_records());
460*6777b538SAndroid Build Coastguard Worker MergeLists(front.hostnames_, back.hostnames());
461*6777b538SAndroid Build Coastguard Worker MergeLists(front.https_record_compatibility_,
462*6777b538SAndroid Build Coastguard Worker back.https_record_compatibility_);
463*6777b538SAndroid Build Coastguard Worker MergeContainers(front.canonical_names_, back.canonical_names_);
464*6777b538SAndroid Build Coastguard Worker
465*6777b538SAndroid Build Coastguard Worker // Only expected to merge entries from same source.
466*6777b538SAndroid Build Coastguard Worker DCHECK_EQ(front.source(), back.source());
467*6777b538SAndroid Build Coastguard Worker
468*6777b538SAndroid Build Coastguard Worker if (front.has_ttl() && back.has_ttl()) {
469*6777b538SAndroid Build Coastguard Worker front.ttl_ = std::min(front.ttl(), back.ttl());
470*6777b538SAndroid Build Coastguard Worker } else if (back.has_ttl()) {
471*6777b538SAndroid Build Coastguard Worker front.ttl_ = back.ttl();
472*6777b538SAndroid Build Coastguard Worker }
473*6777b538SAndroid Build Coastguard Worker
474*6777b538SAndroid Build Coastguard Worker front.expires_ = std::min(front.expires(), back.expires());
475*6777b538SAndroid Build Coastguard Worker front.network_changes_ =
476*6777b538SAndroid Build Coastguard Worker std::max(front.network_changes(), back.network_changes());
477*6777b538SAndroid Build Coastguard Worker
478*6777b538SAndroid Build Coastguard Worker front.total_hits_ = front.total_hits_ + back.total_hits_;
479*6777b538SAndroid Build Coastguard Worker front.stale_hits_ = front.stale_hits_ + back.stale_hits_;
480*6777b538SAndroid Build Coastguard Worker
481*6777b538SAndroid Build Coastguard Worker return front;
482*6777b538SAndroid Build Coastguard Worker }
483*6777b538SAndroid Build Coastguard Worker
CopyWithDefaultPort(uint16_t port) const484*6777b538SAndroid Build Coastguard Worker HostCache::Entry HostCache::Entry::CopyWithDefaultPort(uint16_t port) const {
485*6777b538SAndroid Build Coastguard Worker Entry copy(*this);
486*6777b538SAndroid Build Coastguard Worker
487*6777b538SAndroid Build Coastguard Worker for (IPEndPoint& endpoint : copy.ip_endpoints_) {
488*6777b538SAndroid Build Coastguard Worker if (endpoint.port() == 0) {
489*6777b538SAndroid Build Coastguard Worker endpoint = IPEndPoint(endpoint.address(), port);
490*6777b538SAndroid Build Coastguard Worker }
491*6777b538SAndroid Build Coastguard Worker }
492*6777b538SAndroid Build Coastguard Worker
493*6777b538SAndroid Build Coastguard Worker for (HostPortPair& hostname : copy.hostnames_) {
494*6777b538SAndroid Build Coastguard Worker if (hostname.port() == 0) {
495*6777b538SAndroid Build Coastguard Worker hostname = HostPortPair(hostname.host(), port);
496*6777b538SAndroid Build Coastguard Worker }
497*6777b538SAndroid Build Coastguard Worker }
498*6777b538SAndroid Build Coastguard Worker
499*6777b538SAndroid Build Coastguard Worker return copy;
500*6777b538SAndroid Build Coastguard Worker }
501*6777b538SAndroid Build Coastguard Worker
502*6777b538SAndroid Build Coastguard Worker HostCache::Entry& HostCache::Entry::operator=(const Entry& entry) = default;
503*6777b538SAndroid Build Coastguard Worker
504*6777b538SAndroid Build Coastguard Worker HostCache::Entry& HostCache::Entry::operator=(Entry&& entry) = default;
505*6777b538SAndroid Build Coastguard Worker
Entry(int error,std::vector<IPEndPoint> ip_endpoints,std::set<std::string> aliases,Source source,std::optional<base::TimeDelta> ttl)506*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Entry(int error,
507*6777b538SAndroid Build Coastguard Worker std::vector<IPEndPoint> ip_endpoints,
508*6777b538SAndroid Build Coastguard Worker std::set<std::string> aliases,
509*6777b538SAndroid Build Coastguard Worker Source source,
510*6777b538SAndroid Build Coastguard Worker std::optional<base::TimeDelta> ttl)
511*6777b538SAndroid Build Coastguard Worker : error_(error),
512*6777b538SAndroid Build Coastguard Worker ip_endpoints_(std::move(ip_endpoints)),
513*6777b538SAndroid Build Coastguard Worker aliases_(std::move(aliases)),
514*6777b538SAndroid Build Coastguard Worker source_(source),
515*6777b538SAndroid Build Coastguard Worker ttl_(ttl ? ttl.value() : kUnknownTtl) {
516*6777b538SAndroid Build Coastguard Worker DCHECK(!ttl || ttl.value() >= base::TimeDelta());
517*6777b538SAndroid Build Coastguard Worker }
518*6777b538SAndroid Build Coastguard Worker
Entry(const HostCache::Entry & entry,base::TimeTicks now,base::TimeDelta ttl,int network_changes)519*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Entry(const HostCache::Entry& entry,
520*6777b538SAndroid Build Coastguard Worker base::TimeTicks now,
521*6777b538SAndroid Build Coastguard Worker base::TimeDelta ttl,
522*6777b538SAndroid Build Coastguard Worker int network_changes)
523*6777b538SAndroid Build Coastguard Worker : error_(entry.error()),
524*6777b538SAndroid Build Coastguard Worker ip_endpoints_(entry.ip_endpoints_),
525*6777b538SAndroid Build Coastguard Worker endpoint_metadatas_(entry.endpoint_metadatas_),
526*6777b538SAndroid Build Coastguard Worker aliases_(entry.aliases()),
527*6777b538SAndroid Build Coastguard Worker text_records_(entry.text_records()),
528*6777b538SAndroid Build Coastguard Worker hostnames_(entry.hostnames()),
529*6777b538SAndroid Build Coastguard Worker https_record_compatibility_(entry.https_record_compatibility_),
530*6777b538SAndroid Build Coastguard Worker source_(entry.source()),
531*6777b538SAndroid Build Coastguard Worker pinning_(entry.pinning()),
532*6777b538SAndroid Build Coastguard Worker canonical_names_(entry.canonical_names()),
533*6777b538SAndroid Build Coastguard Worker ttl_(entry.ttl()),
534*6777b538SAndroid Build Coastguard Worker expires_(now + ttl),
535*6777b538SAndroid Build Coastguard Worker network_changes_(network_changes) {}
536*6777b538SAndroid Build Coastguard Worker
Entry(int error,std::vector<IPEndPoint> ip_endpoints,std::multimap<HttpsRecordPriority,ConnectionEndpointMetadata> endpoint_metadatas,std::set<std::string> aliases,std::vector<std::string> && text_records,std::vector<HostPortPair> && hostnames,std::vector<bool> && https_record_compatibility,Source source,base::TimeTicks expires,int network_changes)537*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Entry(
538*6777b538SAndroid Build Coastguard Worker int error,
539*6777b538SAndroid Build Coastguard Worker std::vector<IPEndPoint> ip_endpoints,
540*6777b538SAndroid Build Coastguard Worker std::multimap<HttpsRecordPriority, ConnectionEndpointMetadata>
541*6777b538SAndroid Build Coastguard Worker endpoint_metadatas,
542*6777b538SAndroid Build Coastguard Worker std::set<std::string> aliases,
543*6777b538SAndroid Build Coastguard Worker std::vector<std::string>&& text_records,
544*6777b538SAndroid Build Coastguard Worker std::vector<HostPortPair>&& hostnames,
545*6777b538SAndroid Build Coastguard Worker std::vector<bool>&& https_record_compatibility,
546*6777b538SAndroid Build Coastguard Worker Source source,
547*6777b538SAndroid Build Coastguard Worker base::TimeTicks expires,
548*6777b538SAndroid Build Coastguard Worker int network_changes)
549*6777b538SAndroid Build Coastguard Worker : error_(error),
550*6777b538SAndroid Build Coastguard Worker ip_endpoints_(std::move(ip_endpoints)),
551*6777b538SAndroid Build Coastguard Worker endpoint_metadatas_(std::move(endpoint_metadatas)),
552*6777b538SAndroid Build Coastguard Worker aliases_(std::move(aliases)),
553*6777b538SAndroid Build Coastguard Worker text_records_(std::move(text_records)),
554*6777b538SAndroid Build Coastguard Worker hostnames_(std::move(hostnames)),
555*6777b538SAndroid Build Coastguard Worker https_record_compatibility_(std::move(https_record_compatibility)),
556*6777b538SAndroid Build Coastguard Worker source_(source),
557*6777b538SAndroid Build Coastguard Worker expires_(expires),
558*6777b538SAndroid Build Coastguard Worker network_changes_(network_changes) {}
559*6777b538SAndroid Build Coastguard Worker
PrepareForCacheInsertion()560*6777b538SAndroid Build Coastguard Worker void HostCache::Entry::PrepareForCacheInsertion() {
561*6777b538SAndroid Build Coastguard Worker https_record_compatibility_.clear();
562*6777b538SAndroid Build Coastguard Worker }
563*6777b538SAndroid Build Coastguard Worker
IsStale(base::TimeTicks now,int network_changes) const564*6777b538SAndroid Build Coastguard Worker bool HostCache::Entry::IsStale(base::TimeTicks now, int network_changes) const {
565*6777b538SAndroid Build Coastguard Worker EntryStaleness stale;
566*6777b538SAndroid Build Coastguard Worker stale.expired_by = now - expires_;
567*6777b538SAndroid Build Coastguard Worker stale.network_changes = network_changes - network_changes_;
568*6777b538SAndroid Build Coastguard Worker stale.stale_hits = stale_hits_;
569*6777b538SAndroid Build Coastguard Worker return stale.is_stale();
570*6777b538SAndroid Build Coastguard Worker }
571*6777b538SAndroid Build Coastguard Worker
CountHit(bool hit_is_stale)572*6777b538SAndroid Build Coastguard Worker void HostCache::Entry::CountHit(bool hit_is_stale) {
573*6777b538SAndroid Build Coastguard Worker ++total_hits_;
574*6777b538SAndroid Build Coastguard Worker if (hit_is_stale)
575*6777b538SAndroid Build Coastguard Worker ++stale_hits_;
576*6777b538SAndroid Build Coastguard Worker }
577*6777b538SAndroid Build Coastguard Worker
GetStaleness(base::TimeTicks now,int network_changes,EntryStaleness * out) const578*6777b538SAndroid Build Coastguard Worker void HostCache::Entry::GetStaleness(base::TimeTicks now,
579*6777b538SAndroid Build Coastguard Worker int network_changes,
580*6777b538SAndroid Build Coastguard Worker EntryStaleness* out) const {
581*6777b538SAndroid Build Coastguard Worker DCHECK(out);
582*6777b538SAndroid Build Coastguard Worker out->expired_by = now - expires_;
583*6777b538SAndroid Build Coastguard Worker out->network_changes = network_changes - network_changes_;
584*6777b538SAndroid Build Coastguard Worker out->stale_hits = stale_hits_;
585*6777b538SAndroid Build Coastguard Worker }
586*6777b538SAndroid Build Coastguard Worker
NetLogParams() const587*6777b538SAndroid Build Coastguard Worker base::Value HostCache::Entry::NetLogParams() const {
588*6777b538SAndroid Build Coastguard Worker return base::Value(GetAsValue(false /* include_staleness */));
589*6777b538SAndroid Build Coastguard Worker }
590*6777b538SAndroid Build Coastguard Worker
GetAsValue(bool include_staleness) const591*6777b538SAndroid Build Coastguard Worker base::Value::Dict HostCache::Entry::GetAsValue(bool include_staleness) const {
592*6777b538SAndroid Build Coastguard Worker base::Value::Dict entry_dict;
593*6777b538SAndroid Build Coastguard Worker
594*6777b538SAndroid Build Coastguard Worker if (include_staleness) {
595*6777b538SAndroid Build Coastguard Worker // The kExpirationKey value is using TimeTicks instead of Time used if
596*6777b538SAndroid Build Coastguard Worker // |include_staleness| is false, so it cannot be used to deserialize.
597*6777b538SAndroid Build Coastguard Worker // This is ok as it is used only for netlog.
598*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kExpirationKey, NetLog::TickCountToString(expires()));
599*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kTtlKey, base::saturated_cast<int>(ttl().InMilliseconds()));
600*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kNetworkChangesKey, network_changes());
601*6777b538SAndroid Build Coastguard Worker // The "pinned" status is meaningful only if "network_changes" is also
602*6777b538SAndroid Build Coastguard Worker // preserved.
603*6777b538SAndroid Build Coastguard Worker if (pinning())
604*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kPinnedKey, *pinning());
605*6777b538SAndroid Build Coastguard Worker } else {
606*6777b538SAndroid Build Coastguard Worker // Convert expiration time in TimeTicks to Time for serialization, using a
607*6777b538SAndroid Build Coastguard Worker // string because base::Value doesn't handle 64-bit integers.
608*6777b538SAndroid Build Coastguard Worker base::Time expiration_time =
609*6777b538SAndroid Build Coastguard Worker base::Time::Now() - (base::TimeTicks::Now() - expires());
610*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kExpirationKey,
611*6777b538SAndroid Build Coastguard Worker base::NumberToString(expiration_time.ToInternalValue()));
612*6777b538SAndroid Build Coastguard Worker }
613*6777b538SAndroid Build Coastguard Worker
614*6777b538SAndroid Build Coastguard Worker if (error() != OK) {
615*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kNetErrorKey, error());
616*6777b538SAndroid Build Coastguard Worker } else {
617*6777b538SAndroid Build Coastguard Worker base::Value::List ip_endpoints_list;
618*6777b538SAndroid Build Coastguard Worker for (const IPEndPoint& ip_endpoint : ip_endpoints_) {
619*6777b538SAndroid Build Coastguard Worker ip_endpoints_list.Append(IpEndpointToValue(ip_endpoint));
620*6777b538SAndroid Build Coastguard Worker }
621*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kIpEndpointsKey, std::move(ip_endpoints_list));
622*6777b538SAndroid Build Coastguard Worker
623*6777b538SAndroid Build Coastguard Worker base::Value::List endpoint_metadatas_list;
624*6777b538SAndroid Build Coastguard Worker for (const auto& endpoint_metadata_pair : endpoint_metadatas_) {
625*6777b538SAndroid Build Coastguard Worker endpoint_metadatas_list.Append(
626*6777b538SAndroid Build Coastguard Worker EndpointMetadataPairToValue(endpoint_metadata_pair));
627*6777b538SAndroid Build Coastguard Worker }
628*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kEndpointMetadatasKey, std::move(endpoint_metadatas_list));
629*6777b538SAndroid Build Coastguard Worker
630*6777b538SAndroid Build Coastguard Worker base::Value::List alias_list;
631*6777b538SAndroid Build Coastguard Worker for (const std::string& alias : aliases()) {
632*6777b538SAndroid Build Coastguard Worker alias_list.Append(alias);
633*6777b538SAndroid Build Coastguard Worker }
634*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kAliasesKey, std::move(alias_list));
635*6777b538SAndroid Build Coastguard Worker
636*6777b538SAndroid Build Coastguard Worker // Append all resolved text records.
637*6777b538SAndroid Build Coastguard Worker base::Value::List text_list_value;
638*6777b538SAndroid Build Coastguard Worker for (const std::string& text_record : text_records()) {
639*6777b538SAndroid Build Coastguard Worker text_list_value.Append(text_record);
640*6777b538SAndroid Build Coastguard Worker }
641*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kTextRecordsKey, std::move(text_list_value));
642*6777b538SAndroid Build Coastguard Worker
643*6777b538SAndroid Build Coastguard Worker // Append all the resolved hostnames.
644*6777b538SAndroid Build Coastguard Worker base::Value::List hostnames_value;
645*6777b538SAndroid Build Coastguard Worker base::Value::List host_ports_value;
646*6777b538SAndroid Build Coastguard Worker for (const HostPortPair& hostname : hostnames()) {
647*6777b538SAndroid Build Coastguard Worker hostnames_value.Append(hostname.host());
648*6777b538SAndroid Build Coastguard Worker host_ports_value.Append(hostname.port());
649*6777b538SAndroid Build Coastguard Worker }
650*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kHostnameResultsKey, std::move(hostnames_value));
651*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kHostPortsKey, std::move(host_ports_value));
652*6777b538SAndroid Build Coastguard Worker
653*6777b538SAndroid Build Coastguard Worker base::Value::List canonical_names_list;
654*6777b538SAndroid Build Coastguard Worker for (const std::string& canonical_name : canonical_names()) {
655*6777b538SAndroid Build Coastguard Worker canonical_names_list.Append(canonical_name);
656*6777b538SAndroid Build Coastguard Worker }
657*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kCanonicalNamesKey, std::move(canonical_names_list));
658*6777b538SAndroid Build Coastguard Worker }
659*6777b538SAndroid Build Coastguard Worker
660*6777b538SAndroid Build Coastguard Worker return entry_dict;
661*6777b538SAndroid Build Coastguard Worker }
662*6777b538SAndroid Build Coastguard Worker
663*6777b538SAndroid Build Coastguard Worker // static
TtlFromInternalResults(const std::set<std::unique_ptr<HostResolverInternalResult>> & results,base::Time now,base::TimeTicks now_ticks)664*6777b538SAndroid Build Coastguard Worker std::optional<base::TimeDelta> HostCache::Entry::TtlFromInternalResults(
665*6777b538SAndroid Build Coastguard Worker const std::set<std::unique_ptr<HostResolverInternalResult>>& results,
666*6777b538SAndroid Build Coastguard Worker base::Time now,
667*6777b538SAndroid Build Coastguard Worker base::TimeTicks now_ticks) {
668*6777b538SAndroid Build Coastguard Worker std::optional<base::TimeDelta> smallest_ttl;
669*6777b538SAndroid Build Coastguard Worker for (const std::unique_ptr<HostResolverInternalResult>& result : results) {
670*6777b538SAndroid Build Coastguard Worker if (result->expiration().has_value()) {
671*6777b538SAndroid Build Coastguard Worker smallest_ttl = std::min(smallest_ttl.value_or(base::TimeDelta::Max()),
672*6777b538SAndroid Build Coastguard Worker result->expiration().value() - now_ticks);
673*6777b538SAndroid Build Coastguard Worker }
674*6777b538SAndroid Build Coastguard Worker if (result->timed_expiration().has_value()) {
675*6777b538SAndroid Build Coastguard Worker smallest_ttl = std::min(smallest_ttl.value_or(base::TimeDelta::Max()),
676*6777b538SAndroid Build Coastguard Worker result->timed_expiration().value() - now);
677*6777b538SAndroid Build Coastguard Worker }
678*6777b538SAndroid Build Coastguard Worker }
679*6777b538SAndroid Build Coastguard Worker return smallest_ttl;
680*6777b538SAndroid Build Coastguard Worker }
681*6777b538SAndroid Build Coastguard Worker
682*6777b538SAndroid Build Coastguard Worker // static
683*6777b538SAndroid Build Coastguard Worker const HostCache::EntryStaleness HostCache::kNotStale = {base::Seconds(-1), 0,
684*6777b538SAndroid Build Coastguard Worker 0};
685*6777b538SAndroid Build Coastguard Worker
HostCache(size_t max_entries)686*6777b538SAndroid Build Coastguard Worker HostCache::HostCache(size_t max_entries)
687*6777b538SAndroid Build Coastguard Worker : max_entries_(max_entries),
688*6777b538SAndroid Build Coastguard Worker tick_clock_(base::DefaultTickClock::GetInstance()) {}
689*6777b538SAndroid Build Coastguard Worker
~HostCache()690*6777b538SAndroid Build Coastguard Worker HostCache::~HostCache() {
691*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
692*6777b538SAndroid Build Coastguard Worker }
693*6777b538SAndroid Build Coastguard Worker
694*6777b538SAndroid Build Coastguard Worker const std::pair<const HostCache::Key, HostCache::Entry>*
Lookup(const Key & key,base::TimeTicks now,bool ignore_secure)695*6777b538SAndroid Build Coastguard Worker HostCache::Lookup(const Key& key, base::TimeTicks now, bool ignore_secure) {
696*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
697*6777b538SAndroid Build Coastguard Worker if (caching_is_disabled())
698*6777b538SAndroid Build Coastguard Worker return nullptr;
699*6777b538SAndroid Build Coastguard Worker
700*6777b538SAndroid Build Coastguard Worker auto* result = LookupInternalIgnoringFields(key, now, ignore_secure);
701*6777b538SAndroid Build Coastguard Worker if (!result)
702*6777b538SAndroid Build Coastguard Worker return nullptr;
703*6777b538SAndroid Build Coastguard Worker
704*6777b538SAndroid Build Coastguard Worker auto* entry = &result->second;
705*6777b538SAndroid Build Coastguard Worker if (entry->IsStale(now, network_changes_))
706*6777b538SAndroid Build Coastguard Worker return nullptr;
707*6777b538SAndroid Build Coastguard Worker
708*6777b538SAndroid Build Coastguard Worker entry->CountHit(/* hit_is_stale= */ false);
709*6777b538SAndroid Build Coastguard Worker return result;
710*6777b538SAndroid Build Coastguard Worker }
711*6777b538SAndroid Build Coastguard Worker
LookupStale(const Key & key,base::TimeTicks now,HostCache::EntryStaleness * stale_out,bool ignore_secure)712*6777b538SAndroid Build Coastguard Worker const std::pair<const HostCache::Key, HostCache::Entry>* HostCache::LookupStale(
713*6777b538SAndroid Build Coastguard Worker const Key& key,
714*6777b538SAndroid Build Coastguard Worker base::TimeTicks now,
715*6777b538SAndroid Build Coastguard Worker HostCache::EntryStaleness* stale_out,
716*6777b538SAndroid Build Coastguard Worker bool ignore_secure) {
717*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
718*6777b538SAndroid Build Coastguard Worker if (caching_is_disabled())
719*6777b538SAndroid Build Coastguard Worker return nullptr;
720*6777b538SAndroid Build Coastguard Worker
721*6777b538SAndroid Build Coastguard Worker auto* result = LookupInternalIgnoringFields(key, now, ignore_secure);
722*6777b538SAndroid Build Coastguard Worker if (!result)
723*6777b538SAndroid Build Coastguard Worker return nullptr;
724*6777b538SAndroid Build Coastguard Worker
725*6777b538SAndroid Build Coastguard Worker auto* entry = &result->second;
726*6777b538SAndroid Build Coastguard Worker bool is_stale = entry->IsStale(now, network_changes_);
727*6777b538SAndroid Build Coastguard Worker entry->CountHit(/* hit_is_stale= */ is_stale);
728*6777b538SAndroid Build Coastguard Worker
729*6777b538SAndroid Build Coastguard Worker if (stale_out)
730*6777b538SAndroid Build Coastguard Worker entry->GetStaleness(now, network_changes_, stale_out);
731*6777b538SAndroid Build Coastguard Worker return result;
732*6777b538SAndroid Build Coastguard Worker }
733*6777b538SAndroid Build Coastguard Worker
734*6777b538SAndroid Build Coastguard Worker // static
735*6777b538SAndroid Build Coastguard Worker std::pair<const HostCache::Key, HostCache::Entry>*
GetLessStaleMoreSecureResult(base::TimeTicks now,std::pair<const HostCache::Key,HostCache::Entry> * result1,std::pair<const HostCache::Key,HostCache::Entry> * result2)736*6777b538SAndroid Build Coastguard Worker HostCache::GetLessStaleMoreSecureResult(
737*6777b538SAndroid Build Coastguard Worker base::TimeTicks now,
738*6777b538SAndroid Build Coastguard Worker std::pair<const HostCache::Key, HostCache::Entry>* result1,
739*6777b538SAndroid Build Coastguard Worker std::pair<const HostCache::Key, HostCache::Entry>* result2) {
740*6777b538SAndroid Build Coastguard Worker // Prefer a non-null result if possible.
741*6777b538SAndroid Build Coastguard Worker if (!result1 && !result2)
742*6777b538SAndroid Build Coastguard Worker return nullptr;
743*6777b538SAndroid Build Coastguard Worker if (result1 && !result2)
744*6777b538SAndroid Build Coastguard Worker return result1;
745*6777b538SAndroid Build Coastguard Worker if (!result1 && result2)
746*6777b538SAndroid Build Coastguard Worker return result2;
747*6777b538SAndroid Build Coastguard Worker
748*6777b538SAndroid Build Coastguard Worker // Both result1 are result2 are non-null.
749*6777b538SAndroid Build Coastguard Worker EntryStaleness staleness1, staleness2;
750*6777b538SAndroid Build Coastguard Worker result1->second.GetStaleness(now, 0, &staleness1);
751*6777b538SAndroid Build Coastguard Worker result2->second.GetStaleness(now, 0, &staleness2);
752*6777b538SAndroid Build Coastguard Worker if (staleness1.network_changes == staleness2.network_changes) {
753*6777b538SAndroid Build Coastguard Worker // Exactly one of the results should be secure.
754*6777b538SAndroid Build Coastguard Worker DCHECK(result1->first.secure != result2->first.secure);
755*6777b538SAndroid Build Coastguard Worker // If the results have the same number of network changes, prefer a
756*6777b538SAndroid Build Coastguard Worker // non-expired result.
757*6777b538SAndroid Build Coastguard Worker if (staleness1.expired_by.is_negative() &&
758*6777b538SAndroid Build Coastguard Worker staleness2.expired_by >= base::TimeDelta()) {
759*6777b538SAndroid Build Coastguard Worker return result1;
760*6777b538SAndroid Build Coastguard Worker }
761*6777b538SAndroid Build Coastguard Worker if (staleness1.expired_by >= base::TimeDelta() &&
762*6777b538SAndroid Build Coastguard Worker staleness2.expired_by.is_negative()) {
763*6777b538SAndroid Build Coastguard Worker return result2;
764*6777b538SAndroid Build Coastguard Worker }
765*6777b538SAndroid Build Coastguard Worker // Both results are equally stale, so prefer a secure result.
766*6777b538SAndroid Build Coastguard Worker return (result1->first.secure) ? result1 : result2;
767*6777b538SAndroid Build Coastguard Worker }
768*6777b538SAndroid Build Coastguard Worker // Prefer the result with the fewest network changes.
769*6777b538SAndroid Build Coastguard Worker return (staleness1.network_changes < staleness2.network_changes) ? result1
770*6777b538SAndroid Build Coastguard Worker : result2;
771*6777b538SAndroid Build Coastguard Worker }
772*6777b538SAndroid Build Coastguard Worker
773*6777b538SAndroid Build Coastguard Worker std::pair<const HostCache::Key, HostCache::Entry>*
LookupInternalIgnoringFields(const Key & initial_key,base::TimeTicks now,bool ignore_secure)774*6777b538SAndroid Build Coastguard Worker HostCache::LookupInternalIgnoringFields(const Key& initial_key,
775*6777b538SAndroid Build Coastguard Worker base::TimeTicks now,
776*6777b538SAndroid Build Coastguard Worker bool ignore_secure) {
777*6777b538SAndroid Build Coastguard Worker std::pair<const HostCache::Key, HostCache::Entry>* preferred_result =
778*6777b538SAndroid Build Coastguard Worker LookupInternal(initial_key);
779*6777b538SAndroid Build Coastguard Worker
780*6777b538SAndroid Build Coastguard Worker if (ignore_secure) {
781*6777b538SAndroid Build Coastguard Worker Key effective_key = initial_key;
782*6777b538SAndroid Build Coastguard Worker effective_key.secure = !initial_key.secure;
783*6777b538SAndroid Build Coastguard Worker preferred_result = GetLessStaleMoreSecureResult(
784*6777b538SAndroid Build Coastguard Worker now, preferred_result, LookupInternal(effective_key));
785*6777b538SAndroid Build Coastguard Worker }
786*6777b538SAndroid Build Coastguard Worker
787*6777b538SAndroid Build Coastguard Worker return preferred_result;
788*6777b538SAndroid Build Coastguard Worker }
789*6777b538SAndroid Build Coastguard Worker
LookupInternal(const Key & key)790*6777b538SAndroid Build Coastguard Worker std::pair<const HostCache::Key, HostCache::Entry>* HostCache::LookupInternal(
791*6777b538SAndroid Build Coastguard Worker const Key& key) {
792*6777b538SAndroid Build Coastguard Worker auto it = entries_.find(key);
793*6777b538SAndroid Build Coastguard Worker return (it != entries_.end()) ? &*it : nullptr;
794*6777b538SAndroid Build Coastguard Worker }
795*6777b538SAndroid Build Coastguard Worker
Set(const Key & key,const Entry & entry,base::TimeTicks now,base::TimeDelta ttl)796*6777b538SAndroid Build Coastguard Worker void HostCache::Set(const Key& key,
797*6777b538SAndroid Build Coastguard Worker const Entry& entry,
798*6777b538SAndroid Build Coastguard Worker base::TimeTicks now,
799*6777b538SAndroid Build Coastguard Worker base::TimeDelta ttl) {
800*6777b538SAndroid Build Coastguard Worker TRACE_EVENT0(NetTracingCategory(), "HostCache::Set");
801*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
802*6777b538SAndroid Build Coastguard Worker if (caching_is_disabled())
803*6777b538SAndroid Build Coastguard Worker return;
804*6777b538SAndroid Build Coastguard Worker
805*6777b538SAndroid Build Coastguard Worker bool has_active_pin = false;
806*6777b538SAndroid Build Coastguard Worker bool result_changed = false;
807*6777b538SAndroid Build Coastguard Worker auto it = entries_.find(key);
808*6777b538SAndroid Build Coastguard Worker if (it != entries_.end()) {
809*6777b538SAndroid Build Coastguard Worker has_active_pin = HasActivePin(it->second);
810*6777b538SAndroid Build Coastguard Worker
811*6777b538SAndroid Build Coastguard Worker // TODO(juliatuttle): Remember some old metadata (hit count or frequency or
812*6777b538SAndroid Build Coastguard Worker // something like that) if it's useful for better eviction algorithms?
813*6777b538SAndroid Build Coastguard Worker result_changed = entry.error() == OK && !it->second.ContentsEqual(entry);
814*6777b538SAndroid Build Coastguard Worker entries_.erase(it);
815*6777b538SAndroid Build Coastguard Worker } else {
816*6777b538SAndroid Build Coastguard Worker result_changed = true;
817*6777b538SAndroid Build Coastguard Worker // This loop almost always runs at most once, for total runtime
818*6777b538SAndroid Build Coastguard Worker // O(max_entries_). It only runs more than once if the cache was over-full
819*6777b538SAndroid Build Coastguard Worker // due to pinned entries, and this is the first call to Set() after
820*6777b538SAndroid Build Coastguard Worker // Invalidate(). The amortized cost remains O(size()) per call to Set().
821*6777b538SAndroid Build Coastguard Worker while (size() >= max_entries_ && EvictOneEntry(now)) {
822*6777b538SAndroid Build Coastguard Worker }
823*6777b538SAndroid Build Coastguard Worker }
824*6777b538SAndroid Build Coastguard Worker
825*6777b538SAndroid Build Coastguard Worker Entry entry_for_cache(entry, now, ttl, network_changes_);
826*6777b538SAndroid Build Coastguard Worker entry_for_cache.set_pinning(entry.pinning().value_or(has_active_pin));
827*6777b538SAndroid Build Coastguard Worker entry_for_cache.PrepareForCacheInsertion();
828*6777b538SAndroid Build Coastguard Worker AddEntry(key, std::move(entry_for_cache));
829*6777b538SAndroid Build Coastguard Worker
830*6777b538SAndroid Build Coastguard Worker if (delegate_ && result_changed)
831*6777b538SAndroid Build Coastguard Worker delegate_->ScheduleWrite();
832*6777b538SAndroid Build Coastguard Worker }
833*6777b538SAndroid Build Coastguard Worker
GetMatchingKeyForTesting(std::string_view hostname,HostCache::Entry::Source * source_out,HostCache::EntryStaleness * stale_out) const834*6777b538SAndroid Build Coastguard Worker const HostCache::Key* HostCache::GetMatchingKeyForTesting(
835*6777b538SAndroid Build Coastguard Worker std::string_view hostname,
836*6777b538SAndroid Build Coastguard Worker HostCache::Entry::Source* source_out,
837*6777b538SAndroid Build Coastguard Worker HostCache::EntryStaleness* stale_out) const {
838*6777b538SAndroid Build Coastguard Worker for (const EntryMap::value_type& entry : entries_) {
839*6777b538SAndroid Build Coastguard Worker if (GetHostname(entry.first.host) == hostname) {
840*6777b538SAndroid Build Coastguard Worker if (source_out != nullptr)
841*6777b538SAndroid Build Coastguard Worker *source_out = entry.second.source();
842*6777b538SAndroid Build Coastguard Worker if (stale_out != nullptr) {
843*6777b538SAndroid Build Coastguard Worker entry.second.GetStaleness(tick_clock_->NowTicks(), network_changes_,
844*6777b538SAndroid Build Coastguard Worker stale_out);
845*6777b538SAndroid Build Coastguard Worker }
846*6777b538SAndroid Build Coastguard Worker return &entry.first;
847*6777b538SAndroid Build Coastguard Worker }
848*6777b538SAndroid Build Coastguard Worker }
849*6777b538SAndroid Build Coastguard Worker
850*6777b538SAndroid Build Coastguard Worker return nullptr;
851*6777b538SAndroid Build Coastguard Worker }
852*6777b538SAndroid Build Coastguard Worker
AddEntry(const Key & key,Entry && entry)853*6777b538SAndroid Build Coastguard Worker void HostCache::AddEntry(const Key& key, Entry&& entry) {
854*6777b538SAndroid Build Coastguard Worker DCHECK_EQ(0u, entries_.count(key));
855*6777b538SAndroid Build Coastguard Worker DCHECK(entry.pinning().has_value());
856*6777b538SAndroid Build Coastguard Worker entries_.emplace(key, std::move(entry));
857*6777b538SAndroid Build Coastguard Worker }
858*6777b538SAndroid Build Coastguard Worker
Invalidate()859*6777b538SAndroid Build Coastguard Worker void HostCache::Invalidate() {
860*6777b538SAndroid Build Coastguard Worker ++network_changes_;
861*6777b538SAndroid Build Coastguard Worker }
862*6777b538SAndroid Build Coastguard Worker
set_persistence_delegate(PersistenceDelegate * delegate)863*6777b538SAndroid Build Coastguard Worker void HostCache::set_persistence_delegate(PersistenceDelegate* delegate) {
864*6777b538SAndroid Build Coastguard Worker // A PersistenceDelegate shouldn't be added if there already was one, and
865*6777b538SAndroid Build Coastguard Worker // shouldn't be removed (by setting to nullptr) if it wasn't previously there.
866*6777b538SAndroid Build Coastguard Worker DCHECK_NE(delegate == nullptr, delegate_ == nullptr);
867*6777b538SAndroid Build Coastguard Worker delegate_ = delegate;
868*6777b538SAndroid Build Coastguard Worker }
869*6777b538SAndroid Build Coastguard Worker
clear()870*6777b538SAndroid Build Coastguard Worker void HostCache::clear() {
871*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
872*6777b538SAndroid Build Coastguard Worker
873*6777b538SAndroid Build Coastguard Worker // Don't bother scheduling a write if there's nothing to clear.
874*6777b538SAndroid Build Coastguard Worker if (size() == 0)
875*6777b538SAndroid Build Coastguard Worker return;
876*6777b538SAndroid Build Coastguard Worker
877*6777b538SAndroid Build Coastguard Worker entries_.clear();
878*6777b538SAndroid Build Coastguard Worker if (delegate_)
879*6777b538SAndroid Build Coastguard Worker delegate_->ScheduleWrite();
880*6777b538SAndroid Build Coastguard Worker }
881*6777b538SAndroid Build Coastguard Worker
ClearForHosts(const base::RepeatingCallback<bool (const std::string &)> & host_filter)882*6777b538SAndroid Build Coastguard Worker void HostCache::ClearForHosts(
883*6777b538SAndroid Build Coastguard Worker const base::RepeatingCallback<bool(const std::string&)>& host_filter) {
884*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
885*6777b538SAndroid Build Coastguard Worker
886*6777b538SAndroid Build Coastguard Worker if (host_filter.is_null()) {
887*6777b538SAndroid Build Coastguard Worker clear();
888*6777b538SAndroid Build Coastguard Worker return;
889*6777b538SAndroid Build Coastguard Worker }
890*6777b538SAndroid Build Coastguard Worker
891*6777b538SAndroid Build Coastguard Worker bool changed = false;
892*6777b538SAndroid Build Coastguard Worker for (auto it = entries_.begin(); it != entries_.end();) {
893*6777b538SAndroid Build Coastguard Worker auto next_it = std::next(it);
894*6777b538SAndroid Build Coastguard Worker
895*6777b538SAndroid Build Coastguard Worker if (host_filter.Run(GetHostname(it->first.host))) {
896*6777b538SAndroid Build Coastguard Worker entries_.erase(it);
897*6777b538SAndroid Build Coastguard Worker changed = true;
898*6777b538SAndroid Build Coastguard Worker }
899*6777b538SAndroid Build Coastguard Worker
900*6777b538SAndroid Build Coastguard Worker it = next_it;
901*6777b538SAndroid Build Coastguard Worker }
902*6777b538SAndroid Build Coastguard Worker
903*6777b538SAndroid Build Coastguard Worker if (delegate_ && changed)
904*6777b538SAndroid Build Coastguard Worker delegate_->ScheduleWrite();
905*6777b538SAndroid Build Coastguard Worker }
906*6777b538SAndroid Build Coastguard Worker
GetList(base::Value::List & entry_list,bool include_staleness,SerializationType serialization_type) const907*6777b538SAndroid Build Coastguard Worker void HostCache::GetList(base::Value::List& entry_list,
908*6777b538SAndroid Build Coastguard Worker bool include_staleness,
909*6777b538SAndroid Build Coastguard Worker SerializationType serialization_type) const {
910*6777b538SAndroid Build Coastguard Worker entry_list.clear();
911*6777b538SAndroid Build Coastguard Worker
912*6777b538SAndroid Build Coastguard Worker for (const auto& pair : entries_) {
913*6777b538SAndroid Build Coastguard Worker const Key& key = pair.first;
914*6777b538SAndroid Build Coastguard Worker const Entry& entry = pair.second;
915*6777b538SAndroid Build Coastguard Worker
916*6777b538SAndroid Build Coastguard Worker base::Value network_anonymization_key_value;
917*6777b538SAndroid Build Coastguard Worker if (serialization_type == SerializationType::kRestorable) {
918*6777b538SAndroid Build Coastguard Worker // Don't save entries associated with ephemeral NetworkAnonymizationKeys.
919*6777b538SAndroid Build Coastguard Worker if (!key.network_anonymization_key.ToValue(
920*6777b538SAndroid Build Coastguard Worker &network_anonymization_key_value)) {
921*6777b538SAndroid Build Coastguard Worker continue;
922*6777b538SAndroid Build Coastguard Worker }
923*6777b538SAndroid Build Coastguard Worker } else {
924*6777b538SAndroid Build Coastguard Worker // ToValue() fails for transient NAKs, since they should never be
925*6777b538SAndroid Build Coastguard Worker // serialized to disk in a restorable format, so use ToDebugString() when
926*6777b538SAndroid Build Coastguard Worker // serializing for debugging instead of for restoring from disk.
927*6777b538SAndroid Build Coastguard Worker network_anonymization_key_value =
928*6777b538SAndroid Build Coastguard Worker base::Value(key.network_anonymization_key.ToDebugString());
929*6777b538SAndroid Build Coastguard Worker }
930*6777b538SAndroid Build Coastguard Worker
931*6777b538SAndroid Build Coastguard Worker base::Value::Dict entry_dict = entry.GetAsValue(include_staleness);
932*6777b538SAndroid Build Coastguard Worker
933*6777b538SAndroid Build Coastguard Worker const auto* host = absl::get_if<url::SchemeHostPort>(&key.host);
934*6777b538SAndroid Build Coastguard Worker if (host) {
935*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kSchemeKey, host->scheme());
936*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kHostnameKey, host->host());
937*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kPortKey, host->port());
938*6777b538SAndroid Build Coastguard Worker } else {
939*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kHostnameKey, absl::get<std::string>(key.host));
940*6777b538SAndroid Build Coastguard Worker }
941*6777b538SAndroid Build Coastguard Worker
942*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kDnsQueryTypeKey,
943*6777b538SAndroid Build Coastguard Worker base::strict_cast<int>(key.dns_query_type));
944*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kFlagsKey, key.host_resolver_flags);
945*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kHostResolverSourceKey,
946*6777b538SAndroid Build Coastguard Worker base::strict_cast<int>(key.host_resolver_source));
947*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kNetworkAnonymizationKey,
948*6777b538SAndroid Build Coastguard Worker std::move(network_anonymization_key_value));
949*6777b538SAndroid Build Coastguard Worker entry_dict.Set(kSecureKey, key.secure);
950*6777b538SAndroid Build Coastguard Worker
951*6777b538SAndroid Build Coastguard Worker entry_list.Append(std::move(entry_dict));
952*6777b538SAndroid Build Coastguard Worker }
953*6777b538SAndroid Build Coastguard Worker }
954*6777b538SAndroid Build Coastguard Worker
RestoreFromListValue(const base::Value::List & old_cache)955*6777b538SAndroid Build Coastguard Worker bool HostCache::RestoreFromListValue(const base::Value::List& old_cache) {
956*6777b538SAndroid Build Coastguard Worker // Reset the restore size to 0.
957*6777b538SAndroid Build Coastguard Worker restore_size_ = 0;
958*6777b538SAndroid Build Coastguard Worker
959*6777b538SAndroid Build Coastguard Worker for (const auto& entry : old_cache) {
960*6777b538SAndroid Build Coastguard Worker // If the cache is already full, don't bother prioritizing what to evict,
961*6777b538SAndroid Build Coastguard Worker // just stop restoring.
962*6777b538SAndroid Build Coastguard Worker if (size() == max_entries_)
963*6777b538SAndroid Build Coastguard Worker break;
964*6777b538SAndroid Build Coastguard Worker
965*6777b538SAndroid Build Coastguard Worker if (!entry.is_dict())
966*6777b538SAndroid Build Coastguard Worker return false;
967*6777b538SAndroid Build Coastguard Worker
968*6777b538SAndroid Build Coastguard Worker const base::Value::Dict& entry_dict = entry.GetDict();
969*6777b538SAndroid Build Coastguard Worker const std::string* hostname_ptr = entry_dict.FindString(kHostnameKey);
970*6777b538SAndroid Build Coastguard Worker if (!hostname_ptr || !IsValidHostname(*hostname_ptr)) {
971*6777b538SAndroid Build Coastguard Worker return false;
972*6777b538SAndroid Build Coastguard Worker }
973*6777b538SAndroid Build Coastguard Worker
974*6777b538SAndroid Build Coastguard Worker // Use presence of scheme to determine host type.
975*6777b538SAndroid Build Coastguard Worker const std::string* scheme_ptr = entry_dict.FindString(kSchemeKey);
976*6777b538SAndroid Build Coastguard Worker absl::variant<url::SchemeHostPort, std::string> host;
977*6777b538SAndroid Build Coastguard Worker if (scheme_ptr) {
978*6777b538SAndroid Build Coastguard Worker std::optional<int> port = entry_dict.FindInt(kPortKey);
979*6777b538SAndroid Build Coastguard Worker if (!port || !base::IsValueInRangeForNumericType<uint16_t>(port.value()))
980*6777b538SAndroid Build Coastguard Worker return false;
981*6777b538SAndroid Build Coastguard Worker
982*6777b538SAndroid Build Coastguard Worker url::SchemeHostPort scheme_host_port(*scheme_ptr, *hostname_ptr,
983*6777b538SAndroid Build Coastguard Worker port.value());
984*6777b538SAndroid Build Coastguard Worker if (!scheme_host_port.IsValid())
985*6777b538SAndroid Build Coastguard Worker return false;
986*6777b538SAndroid Build Coastguard Worker host = std::move(scheme_host_port);
987*6777b538SAndroid Build Coastguard Worker } else {
988*6777b538SAndroid Build Coastguard Worker host = *hostname_ptr;
989*6777b538SAndroid Build Coastguard Worker }
990*6777b538SAndroid Build Coastguard Worker
991*6777b538SAndroid Build Coastguard Worker const std::string* expiration_ptr = entry_dict.FindString(kExpirationKey);
992*6777b538SAndroid Build Coastguard Worker std::optional<int> maybe_flags = entry_dict.FindInt(kFlagsKey);
993*6777b538SAndroid Build Coastguard Worker if (expiration_ptr == nullptr || !maybe_flags.has_value())
994*6777b538SAndroid Build Coastguard Worker return false;
995*6777b538SAndroid Build Coastguard Worker std::string expiration(*expiration_ptr);
996*6777b538SAndroid Build Coastguard Worker HostResolverFlags flags = maybe_flags.value();
997*6777b538SAndroid Build Coastguard Worker
998*6777b538SAndroid Build Coastguard Worker std::optional<int> maybe_dns_query_type =
999*6777b538SAndroid Build Coastguard Worker entry_dict.FindInt(kDnsQueryTypeKey);
1000*6777b538SAndroid Build Coastguard Worker if (!maybe_dns_query_type.has_value())
1001*6777b538SAndroid Build Coastguard Worker return false;
1002*6777b538SAndroid Build Coastguard Worker std::optional<DnsQueryType> dns_query_type =
1003*6777b538SAndroid Build Coastguard Worker GetDnsQueryType(maybe_dns_query_type.value());
1004*6777b538SAndroid Build Coastguard Worker if (!dns_query_type.has_value())
1005*6777b538SAndroid Build Coastguard Worker return false;
1006*6777b538SAndroid Build Coastguard Worker // HostResolverSource is optional.
1007*6777b538SAndroid Build Coastguard Worker int host_resolver_source =
1008*6777b538SAndroid Build Coastguard Worker entry_dict.FindInt(kHostResolverSourceKey)
1009*6777b538SAndroid Build Coastguard Worker .value_or(base::strict_cast<int>(HostResolverSource::ANY));
1010*6777b538SAndroid Build Coastguard Worker
1011*6777b538SAndroid Build Coastguard Worker const base::Value* network_anonymization_key_value =
1012*6777b538SAndroid Build Coastguard Worker entry_dict.Find(kNetworkAnonymizationKey);
1013*6777b538SAndroid Build Coastguard Worker NetworkAnonymizationKey network_anonymization_key;
1014*6777b538SAndroid Build Coastguard Worker if (!network_anonymization_key_value ||
1015*6777b538SAndroid Build Coastguard Worker network_anonymization_key_value->type() == base::Value::Type::STRING ||
1016*6777b538SAndroid Build Coastguard Worker !NetworkAnonymizationKey::FromValue(*network_anonymization_key_value,
1017*6777b538SAndroid Build Coastguard Worker &network_anonymization_key)) {
1018*6777b538SAndroid Build Coastguard Worker return false;
1019*6777b538SAndroid Build Coastguard Worker }
1020*6777b538SAndroid Build Coastguard Worker
1021*6777b538SAndroid Build Coastguard Worker bool secure = entry_dict.FindBool(kSecureKey).value_or(false);
1022*6777b538SAndroid Build Coastguard Worker
1023*6777b538SAndroid Build Coastguard Worker int error = OK;
1024*6777b538SAndroid Build Coastguard Worker const base::Value::List* ip_endpoints_list = nullptr;
1025*6777b538SAndroid Build Coastguard Worker const base::Value::List* endpoint_metadatas_list = nullptr;
1026*6777b538SAndroid Build Coastguard Worker const base::Value::List* aliases_list = nullptr;
1027*6777b538SAndroid Build Coastguard Worker const base::Value::List* legacy_addresses_list = nullptr;
1028*6777b538SAndroid Build Coastguard Worker const base::Value::List* text_records_list = nullptr;
1029*6777b538SAndroid Build Coastguard Worker const base::Value::List* hostname_records_list = nullptr;
1030*6777b538SAndroid Build Coastguard Worker const base::Value::List* host_ports_list = nullptr;
1031*6777b538SAndroid Build Coastguard Worker const base::Value::List* canonical_names_list = nullptr;
1032*6777b538SAndroid Build Coastguard Worker std::optional<int> maybe_error = entry_dict.FindInt(kNetErrorKey);
1033*6777b538SAndroid Build Coastguard Worker std::optional<bool> maybe_pinned = entry_dict.FindBool(kPinnedKey);
1034*6777b538SAndroid Build Coastguard Worker if (maybe_error.has_value()) {
1035*6777b538SAndroid Build Coastguard Worker error = maybe_error.value();
1036*6777b538SAndroid Build Coastguard Worker } else {
1037*6777b538SAndroid Build Coastguard Worker ip_endpoints_list = entry_dict.FindList(kIpEndpointsKey);
1038*6777b538SAndroid Build Coastguard Worker endpoint_metadatas_list = entry_dict.FindList(kEndpointMetadatasKey);
1039*6777b538SAndroid Build Coastguard Worker aliases_list = entry_dict.FindList(kAliasesKey);
1040*6777b538SAndroid Build Coastguard Worker legacy_addresses_list = entry_dict.FindList(kAddressesKey);
1041*6777b538SAndroid Build Coastguard Worker text_records_list = entry_dict.FindList(kTextRecordsKey);
1042*6777b538SAndroid Build Coastguard Worker hostname_records_list = entry_dict.FindList(kHostnameResultsKey);
1043*6777b538SAndroid Build Coastguard Worker host_ports_list = entry_dict.FindList(kHostPortsKey);
1044*6777b538SAndroid Build Coastguard Worker canonical_names_list = entry_dict.FindList(kCanonicalNamesKey);
1045*6777b538SAndroid Build Coastguard Worker
1046*6777b538SAndroid Build Coastguard Worker if ((hostname_records_list == nullptr && host_ports_list != nullptr) ||
1047*6777b538SAndroid Build Coastguard Worker (hostname_records_list != nullptr && host_ports_list == nullptr)) {
1048*6777b538SAndroid Build Coastguard Worker return false;
1049*6777b538SAndroid Build Coastguard Worker }
1050*6777b538SAndroid Build Coastguard Worker }
1051*6777b538SAndroid Build Coastguard Worker
1052*6777b538SAndroid Build Coastguard Worker int64_t time_internal;
1053*6777b538SAndroid Build Coastguard Worker if (!base::StringToInt64(expiration, &time_internal))
1054*6777b538SAndroid Build Coastguard Worker return false;
1055*6777b538SAndroid Build Coastguard Worker
1056*6777b538SAndroid Build Coastguard Worker base::TimeTicks expiration_time =
1057*6777b538SAndroid Build Coastguard Worker tick_clock_->NowTicks() -
1058*6777b538SAndroid Build Coastguard Worker (base::Time::Now() - base::Time::FromInternalValue(time_internal));
1059*6777b538SAndroid Build Coastguard Worker
1060*6777b538SAndroid Build Coastguard Worker std::vector<IPEndPoint> ip_endpoints;
1061*6777b538SAndroid Build Coastguard Worker if (ip_endpoints_list) {
1062*6777b538SAndroid Build Coastguard Worker for (const base::Value& ip_endpoint_value : *ip_endpoints_list) {
1063*6777b538SAndroid Build Coastguard Worker std::optional<IPEndPoint> ip_endpoint =
1064*6777b538SAndroid Build Coastguard Worker IpEndpointFromValue(ip_endpoint_value);
1065*6777b538SAndroid Build Coastguard Worker if (!ip_endpoint)
1066*6777b538SAndroid Build Coastguard Worker return false;
1067*6777b538SAndroid Build Coastguard Worker ip_endpoints.push_back(std::move(ip_endpoint).value());
1068*6777b538SAndroid Build Coastguard Worker }
1069*6777b538SAndroid Build Coastguard Worker }
1070*6777b538SAndroid Build Coastguard Worker
1071*6777b538SAndroid Build Coastguard Worker std::multimap<HttpsRecordPriority, ConnectionEndpointMetadata>
1072*6777b538SAndroid Build Coastguard Worker endpoint_metadatas;
1073*6777b538SAndroid Build Coastguard Worker if (endpoint_metadatas_list) {
1074*6777b538SAndroid Build Coastguard Worker for (const base::Value& endpoint_metadata_value :
1075*6777b538SAndroid Build Coastguard Worker *endpoint_metadatas_list) {
1076*6777b538SAndroid Build Coastguard Worker std::optional<
1077*6777b538SAndroid Build Coastguard Worker std::pair<HttpsRecordPriority, ConnectionEndpointMetadata>>
1078*6777b538SAndroid Build Coastguard Worker pair = EndpointMetadataPairFromValue(endpoint_metadata_value);
1079*6777b538SAndroid Build Coastguard Worker if (!pair)
1080*6777b538SAndroid Build Coastguard Worker return false;
1081*6777b538SAndroid Build Coastguard Worker endpoint_metadatas.insert(std::move(pair).value());
1082*6777b538SAndroid Build Coastguard Worker }
1083*6777b538SAndroid Build Coastguard Worker }
1084*6777b538SAndroid Build Coastguard Worker
1085*6777b538SAndroid Build Coastguard Worker std::set<std::string> aliases;
1086*6777b538SAndroid Build Coastguard Worker if (aliases_list) {
1087*6777b538SAndroid Build Coastguard Worker for (const base::Value& alias_value : *aliases_list) {
1088*6777b538SAndroid Build Coastguard Worker if (!alias_value.is_string())
1089*6777b538SAndroid Build Coastguard Worker return false;
1090*6777b538SAndroid Build Coastguard Worker aliases.insert(alias_value.GetString());
1091*6777b538SAndroid Build Coastguard Worker }
1092*6777b538SAndroid Build Coastguard Worker }
1093*6777b538SAndroid Build Coastguard Worker
1094*6777b538SAndroid Build Coastguard Worker // `addresses` field was supported until M105. We keep reading this field
1095*6777b538SAndroid Build Coastguard Worker // for backward compatibility for several milestones.
1096*6777b538SAndroid Build Coastguard Worker if (legacy_addresses_list) {
1097*6777b538SAndroid Build Coastguard Worker if (!ip_endpoints.empty()) {
1098*6777b538SAndroid Build Coastguard Worker return false;
1099*6777b538SAndroid Build Coastguard Worker }
1100*6777b538SAndroid Build Coastguard Worker if (!IPEndPointsFromLegacyAddressListValue(*legacy_addresses_list,
1101*6777b538SAndroid Build Coastguard Worker ip_endpoints)) {
1102*6777b538SAndroid Build Coastguard Worker return false;
1103*6777b538SAndroid Build Coastguard Worker }
1104*6777b538SAndroid Build Coastguard Worker }
1105*6777b538SAndroid Build Coastguard Worker
1106*6777b538SAndroid Build Coastguard Worker std::vector<std::string> text_records;
1107*6777b538SAndroid Build Coastguard Worker if (text_records_list) {
1108*6777b538SAndroid Build Coastguard Worker for (const base::Value& value : *text_records_list) {
1109*6777b538SAndroid Build Coastguard Worker if (!value.is_string())
1110*6777b538SAndroid Build Coastguard Worker return false;
1111*6777b538SAndroid Build Coastguard Worker text_records.push_back(value.GetString());
1112*6777b538SAndroid Build Coastguard Worker }
1113*6777b538SAndroid Build Coastguard Worker }
1114*6777b538SAndroid Build Coastguard Worker
1115*6777b538SAndroid Build Coastguard Worker std::vector<HostPortPair> hostname_records;
1116*6777b538SAndroid Build Coastguard Worker if (hostname_records_list) {
1117*6777b538SAndroid Build Coastguard Worker DCHECK(host_ports_list);
1118*6777b538SAndroid Build Coastguard Worker if (hostname_records_list->size() != host_ports_list->size()) {
1119*6777b538SAndroid Build Coastguard Worker return false;
1120*6777b538SAndroid Build Coastguard Worker }
1121*6777b538SAndroid Build Coastguard Worker
1122*6777b538SAndroid Build Coastguard Worker for (size_t i = 0; i < hostname_records_list->size(); ++i) {
1123*6777b538SAndroid Build Coastguard Worker if (!(*hostname_records_list)[i].is_string() ||
1124*6777b538SAndroid Build Coastguard Worker !(*host_ports_list)[i].is_int() ||
1125*6777b538SAndroid Build Coastguard Worker !base::IsValueInRangeForNumericType<uint16_t>(
1126*6777b538SAndroid Build Coastguard Worker (*host_ports_list)[i].GetInt())) {
1127*6777b538SAndroid Build Coastguard Worker return false;
1128*6777b538SAndroid Build Coastguard Worker }
1129*6777b538SAndroid Build Coastguard Worker hostname_records.emplace_back(
1130*6777b538SAndroid Build Coastguard Worker (*hostname_records_list)[i].GetString(),
1131*6777b538SAndroid Build Coastguard Worker base::checked_cast<uint16_t>((*host_ports_list)[i].GetInt()));
1132*6777b538SAndroid Build Coastguard Worker }
1133*6777b538SAndroid Build Coastguard Worker }
1134*6777b538SAndroid Build Coastguard Worker
1135*6777b538SAndroid Build Coastguard Worker std::set<std::string> canonical_names;
1136*6777b538SAndroid Build Coastguard Worker if (canonical_names_list) {
1137*6777b538SAndroid Build Coastguard Worker for (const auto& item : *canonical_names_list) {
1138*6777b538SAndroid Build Coastguard Worker const std::string* name = item.GetIfString();
1139*6777b538SAndroid Build Coastguard Worker if (!name)
1140*6777b538SAndroid Build Coastguard Worker return false;
1141*6777b538SAndroid Build Coastguard Worker canonical_names.insert(*name);
1142*6777b538SAndroid Build Coastguard Worker }
1143*6777b538SAndroid Build Coastguard Worker }
1144*6777b538SAndroid Build Coastguard Worker
1145*6777b538SAndroid Build Coastguard Worker // We do not intend to serialize experimental results with the host cache.
1146*6777b538SAndroid Build Coastguard Worker std::vector<bool> experimental_results;
1147*6777b538SAndroid Build Coastguard Worker
1148*6777b538SAndroid Build Coastguard Worker Key key(std::move(host), dns_query_type.value(), flags,
1149*6777b538SAndroid Build Coastguard Worker static_cast<HostResolverSource>(host_resolver_source),
1150*6777b538SAndroid Build Coastguard Worker network_anonymization_key);
1151*6777b538SAndroid Build Coastguard Worker key.secure = secure;
1152*6777b538SAndroid Build Coastguard Worker
1153*6777b538SAndroid Build Coastguard Worker // If the key is already in the cache, assume it's more recent and don't
1154*6777b538SAndroid Build Coastguard Worker // replace the entry.
1155*6777b538SAndroid Build Coastguard Worker auto found = entries_.find(key);
1156*6777b538SAndroid Build Coastguard Worker if (found == entries_.end()) {
1157*6777b538SAndroid Build Coastguard Worker Entry new_entry(error, std::move(ip_endpoints),
1158*6777b538SAndroid Build Coastguard Worker std::move(endpoint_metadatas), std::move(aliases),
1159*6777b538SAndroid Build Coastguard Worker std::move(text_records), std::move(hostname_records),
1160*6777b538SAndroid Build Coastguard Worker std::move(experimental_results), Entry::SOURCE_UNKNOWN,
1161*6777b538SAndroid Build Coastguard Worker expiration_time, network_changes_ - 1);
1162*6777b538SAndroid Build Coastguard Worker new_entry.set_pinning(maybe_pinned.value_or(false));
1163*6777b538SAndroid Build Coastguard Worker new_entry.set_canonical_names(std::move(canonical_names));
1164*6777b538SAndroid Build Coastguard Worker AddEntry(key, std::move(new_entry));
1165*6777b538SAndroid Build Coastguard Worker restore_size_++;
1166*6777b538SAndroid Build Coastguard Worker }
1167*6777b538SAndroid Build Coastguard Worker }
1168*6777b538SAndroid Build Coastguard Worker return true;
1169*6777b538SAndroid Build Coastguard Worker }
1170*6777b538SAndroid Build Coastguard Worker
size() const1171*6777b538SAndroid Build Coastguard Worker size_t HostCache::size() const {
1172*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
1173*6777b538SAndroid Build Coastguard Worker return entries_.size();
1174*6777b538SAndroid Build Coastguard Worker }
1175*6777b538SAndroid Build Coastguard Worker
max_entries() const1176*6777b538SAndroid Build Coastguard Worker size_t HostCache::max_entries() const {
1177*6777b538SAndroid Build Coastguard Worker DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
1178*6777b538SAndroid Build Coastguard Worker return max_entries_;
1179*6777b538SAndroid Build Coastguard Worker }
1180*6777b538SAndroid Build Coastguard Worker
EvictOneEntry(base::TimeTicks now)1181*6777b538SAndroid Build Coastguard Worker bool HostCache::EvictOneEntry(base::TimeTicks now) {
1182*6777b538SAndroid Build Coastguard Worker DCHECK_LT(0u, entries_.size());
1183*6777b538SAndroid Build Coastguard Worker
1184*6777b538SAndroid Build Coastguard Worker std::optional<net::HostCache::EntryMap::iterator> oldest_it;
1185*6777b538SAndroid Build Coastguard Worker for (auto it = entries_.begin(); it != entries_.end(); ++it) {
1186*6777b538SAndroid Build Coastguard Worker const Entry& entry = it->second;
1187*6777b538SAndroid Build Coastguard Worker if (HasActivePin(entry)) {
1188*6777b538SAndroid Build Coastguard Worker continue;
1189*6777b538SAndroid Build Coastguard Worker }
1190*6777b538SAndroid Build Coastguard Worker
1191*6777b538SAndroid Build Coastguard Worker if (!oldest_it) {
1192*6777b538SAndroid Build Coastguard Worker oldest_it = it;
1193*6777b538SAndroid Build Coastguard Worker continue;
1194*6777b538SAndroid Build Coastguard Worker }
1195*6777b538SAndroid Build Coastguard Worker
1196*6777b538SAndroid Build Coastguard Worker const Entry& oldest = (*oldest_it)->second;
1197*6777b538SAndroid Build Coastguard Worker if ((entry.expires() < oldest.expires()) &&
1198*6777b538SAndroid Build Coastguard Worker (entry.IsStale(now, network_changes_) ||
1199*6777b538SAndroid Build Coastguard Worker !oldest.IsStale(now, network_changes_))) {
1200*6777b538SAndroid Build Coastguard Worker oldest_it = it;
1201*6777b538SAndroid Build Coastguard Worker }
1202*6777b538SAndroid Build Coastguard Worker }
1203*6777b538SAndroid Build Coastguard Worker
1204*6777b538SAndroid Build Coastguard Worker if (oldest_it) {
1205*6777b538SAndroid Build Coastguard Worker entries_.erase(*oldest_it);
1206*6777b538SAndroid Build Coastguard Worker return true;
1207*6777b538SAndroid Build Coastguard Worker }
1208*6777b538SAndroid Build Coastguard Worker return false;
1209*6777b538SAndroid Build Coastguard Worker }
1210*6777b538SAndroid Build Coastguard Worker
HasActivePin(const Entry & entry)1211*6777b538SAndroid Build Coastguard Worker bool HostCache::HasActivePin(const Entry& entry) {
1212*6777b538SAndroid Build Coastguard Worker return entry.pinning().value_or(false) &&
1213*6777b538SAndroid Build Coastguard Worker entry.network_changes() == network_changes();
1214*6777b538SAndroid Build Coastguard Worker }
1215*6777b538SAndroid Build Coastguard Worker
1216*6777b538SAndroid Build Coastguard Worker } // namespace net
1217*6777b538SAndroid Build Coastguard Worker
1218*6777b538SAndroid Build Coastguard Worker // Debug logging support
operator <<(std::ostream & out,const net::HostCache::EntryStaleness & s)1219*6777b538SAndroid Build Coastguard Worker std::ostream& operator<<(std::ostream& out,
1220*6777b538SAndroid Build Coastguard Worker const net::HostCache::EntryStaleness& s) {
1221*6777b538SAndroid Build Coastguard Worker return out << "EntryStaleness{" << s.expired_by << ", " << s.network_changes
1222*6777b538SAndroid Build Coastguard Worker << ", " << s.stale_hits << "}";
1223*6777b538SAndroid Build Coastguard Worker }
1224