mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 22:36:09 +03:00
250 lines
7.8 KiB
C++
250 lines
7.8 KiB
C++
// 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 <utility>
|
|
|
|
#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<base::Value> DnsConfig::ToValue() const {
|
|
auto dict = std::make_unique<base::DictionaryValue>();
|
|
|
|
auto list = std::make_unique<base::ListValue>();
|
|
for (size_t i = 0; i < nameservers.size(); ++i)
|
|
list->AppendString(nameservers[i].ToString());
|
|
dict->Set("nameservers", std::move(list));
|
|
|
|
list = std::make_unique<base::ListValue>();
|
|
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<base::ListValue>();
|
|
for (auto& server : dns_over_https_servers) {
|
|
base::Value val(base::Value::Type::DICTIONARY);
|
|
base::DictionaryValue* dict;
|
|
val.GetAsDictionary(&dict);
|
|
dict->SetString("server_template", server.server_template);
|
|
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 std::string& server_template,
|
|
bool use_post)
|
|
: server_template(server_template), 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
|