// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "net/dns/dns_config_service.h" #include #include "base/logging.h" #include "base/metrics/histogram_macros.h" #include "base/values.h" #include "net/base/ip_endpoint.h" #include "net/base/ip_pattern.h" namespace net { // Default values are taken from glibc resolv.h except timeout which is set to // |kDnsDefaultTimeoutMs|. DnsConfig::DnsConfig() : unhandled_options(false), append_to_multi_label_name(true), randomize_ports(false), ndots(1), timeout(base::TimeDelta::FromMilliseconds(kDnsDefaultTimeoutMs)), attempts(2), rotate(false), use_local_ipv6(false) {} DnsConfig::DnsConfig(const DnsConfig& other) = default; DnsConfig::~DnsConfig() = default; bool DnsConfig::Equals(const DnsConfig& d) const { return EqualsIgnoreHosts(d) && (hosts == d.hosts); } bool DnsConfig::EqualsIgnoreHosts(const DnsConfig& d) const { return (nameservers == d.nameservers) && (search == d.search) && (unhandled_options == d.unhandled_options) && (append_to_multi_label_name == d.append_to_multi_label_name) && (ndots == d.ndots) && (timeout == d.timeout) && (attempts == d.attempts) && (rotate == d.rotate) && (use_local_ipv6 == d.use_local_ipv6); } void DnsConfig::CopyIgnoreHosts(const DnsConfig& d) { nameservers = d.nameservers; search = d.search; unhandled_options = d.unhandled_options; append_to_multi_label_name = d.append_to_multi_label_name; ndots = d.ndots; timeout = d.timeout; attempts = d.attempts; rotate = d.rotate; use_local_ipv6 = d.use_local_ipv6; } std::unique_ptr DnsConfig::ToValue() const { auto dict = std::make_unique(); auto list = std::make_unique(); for (size_t i = 0; i < nameservers.size(); ++i) list->AppendString(nameservers[i].ToString()); dict->Set("nameservers", std::move(list)); list = std::make_unique(); for (size_t i = 0; i < search.size(); ++i) list->AppendString(search[i]); dict->Set("search", std::move(list)); dict->SetBoolean("unhandled_options", unhandled_options); dict->SetBoolean("append_to_multi_label_name", append_to_multi_label_name); dict->SetInteger("ndots", ndots); dict->SetDouble("timeout", timeout.InSecondsF()); dict->SetInteger("attempts", attempts); dict->SetBoolean("rotate", rotate); dict->SetBoolean("use_local_ipv6", use_local_ipv6); dict->SetInteger("num_hosts", hosts.size()); list = std::make_unique(); for (auto& server : dns_over_https_servers) { base::Value val(base::Value::Type::DICTIONARY); base::DictionaryValue* dict; val.GetAsDictionary(&dict); dict->SetString("server", server.server.spec()); dict->SetBoolean("use_post", server.use_post); list->GetList().push_back(std::move(val)); } dict->Set("doh_servers", std::move(list)); return std::move(dict); } DnsConfig::DnsOverHttpsServerConfig::DnsOverHttpsServerConfig( const GURL& server, bool use_post) : server(server), use_post(use_post) {} DnsConfigService::DnsConfigService() : watch_failed_(false), have_config_(false), have_hosts_(false), need_update_(false), last_sent_empty_(true) {} DnsConfigService::~DnsConfigService() { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); } void DnsConfigService::ReadConfig(const CallbackType& callback) { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); DCHECK(!callback.is_null()); DCHECK(callback_.is_null()); callback_ = callback; ReadNow(); } void DnsConfigService::WatchConfig(const CallbackType& callback) { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); DCHECK(!callback.is_null()); DCHECK(callback_.is_null()); callback_ = callback; watch_failed_ = !StartWatching(); ReadNow(); } void DnsConfigService::InvalidateConfig() { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); base::TimeTicks now = base::TimeTicks::Now(); if (!last_invalidate_config_time_.is_null()) { UMA_HISTOGRAM_LONG_TIMES("AsyncDNS.ConfigNotifyInterval", now - last_invalidate_config_time_); } last_invalidate_config_time_ = now; if (!have_config_) return; have_config_ = false; StartTimer(); } void DnsConfigService::InvalidateHosts() { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); base::TimeTicks now = base::TimeTicks::Now(); if (!last_invalidate_hosts_time_.is_null()) { UMA_HISTOGRAM_LONG_TIMES("AsyncDNS.HostsNotifyInterval", now - last_invalidate_hosts_time_); } last_invalidate_hosts_time_ = now; if (!have_hosts_) return; have_hosts_ = false; StartTimer(); } void DnsConfigService::OnConfigRead(const DnsConfig& config) { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); DCHECK(config.IsValid()); bool changed = false; if (!config.EqualsIgnoreHosts(dns_config_)) { dns_config_.CopyIgnoreHosts(config); need_update_ = true; changed = true; } if (!changed && !last_sent_empty_time_.is_null()) { UMA_HISTOGRAM_LONG_TIMES("AsyncDNS.UnchangedConfigInterval", base::TimeTicks::Now() - last_sent_empty_time_); } UMA_HISTOGRAM_BOOLEAN("AsyncDNS.ConfigChange", changed); have_config_ = true; if (have_hosts_ || watch_failed_) OnCompleteConfig(); } void DnsConfigService::OnHostsRead(const DnsHosts& hosts) { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); bool changed = false; if (hosts != dns_config_.hosts) { dns_config_.hosts = hosts; need_update_ = true; changed = true; } if (!changed && !last_sent_empty_time_.is_null()) { UMA_HISTOGRAM_LONG_TIMES("AsyncDNS.UnchangedHostsInterval", base::TimeTicks::Now() - last_sent_empty_time_); } UMA_HISTOGRAM_BOOLEAN("AsyncDNS.HostsChange", changed); have_hosts_ = true; if (have_config_ || watch_failed_) OnCompleteConfig(); } void DnsConfigService::StartTimer() { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); if (last_sent_empty_) { DCHECK(!timer_.IsRunning()); return; // No need to withdraw again. } timer_.Stop(); // Give it a short timeout to come up with a valid config. Otherwise withdraw // the config from the receiver. The goal is to avoid perceivable network // outage (when using the wrong config) but at the same time avoid // unnecessary Job aborts in HostResolverImpl. The signals come from multiple // sources so it might receive multiple events during a config change. // DHCP and user-induced changes are on the order of seconds, so 150ms should // not add perceivable delay. On the other hand, config readers should finish // within 150ms with the rare exception of I/O block or extra large HOSTS. const base::TimeDelta kTimeout = base::TimeDelta::FromMilliseconds(150); timer_.Start(FROM_HERE, kTimeout, this, &DnsConfigService::OnTimeout); } void DnsConfigService::OnTimeout() { DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); DCHECK(!last_sent_empty_); // Indicate that even if there is no change in On*Read, we will need to // update the receiver when the config becomes complete. need_update_ = true; // Empty config is considered invalid. last_sent_empty_ = true; last_sent_empty_time_ = base::TimeTicks::Now(); callback_.Run(DnsConfig()); } void DnsConfigService::OnCompleteConfig() { timer_.Stop(); if (!need_update_) return; need_update_ = false; last_sent_empty_ = false; if (watch_failed_) { // If a watch failed, the config may not be accurate, so report empty. callback_.Run(DnsConfig()); } else { callback_.Run(dns_config_); } } } // namespace net