use anyhow::{Context as _, Error};
use async_trait::async_trait;
use dns::async_resolver::{Resolver, Spawner};
use dns::config::{ServerList, UpdateServersResult};
use fidl_fuchsia_net_name::{
self as fname, LookupAdminRequest, LookupAdminRequestStream, LookupRequest, LookupRequestStream,
};
use fuchsia_component::server::{ServiceFs, ServiceFsDir};
use fuchsia_sync::RwLock;
use futures::channel::mpsc;
use futures::lock::Mutex;
use futures::{FutureExt as _, SinkExt as _, StreamExt as _, TryFutureExt as _, TryStreamExt as _};
use log::{debug, error, info, warn};
use net_declare::fidl_ip_v6;
use net_types::ip::IpAddress;
use std::collections::{BTreeMap, HashMap, VecDeque};
use std::convert::TryFrom as _;
use std::hash::{Hash, Hasher};
use std::net::IpAddr;
use std::num::NonZeroUsize;
use std::rc::Rc;
use std::str::FromStr as _;
use std::sync::Arc;
use trust_dns_proto::op::ResponseCode;
use trust_dns_proto::rr::domain::IntoName;
use trust_dns_proto::rr::{RData, RecordType};
use trust_dns_resolver::config::{
LookupIpStrategy, NameServerConfig, NameServerConfigGroup, Protocol, ResolverConfig,
ResolverOpts, ServerOrderingStrategy,
};
use trust_dns_resolver::error::{ResolveError, ResolveErrorKind};
use trust_dns_resolver::lookup;
use unicode_xid::UnicodeXID as _;
use {
fidl_fuchsia_net as fnet, fidl_fuchsia_net_ext as net_ext,
fidl_fuchsia_net_routes as fnet_routes, fuchsia_async as fasync,
};
struct SharedResolver<T>(RwLock<Rc<T>>);
impl<T> SharedResolver<T> {
fn new(resolver: T) -> Self {
SharedResolver(RwLock::new(Rc::new(resolver)))
}
fn read(&self) -> Rc<T> {
let Self(inner) = self;
inner.read().clone()
}
fn write(&self, other: Rc<T>) {
let Self(inner) = self;
*inner.write() = other;
}
}
const STAT_WINDOW_DURATION: zx::MonotonicDuration = zx::MonotonicDuration::from_seconds(60);
const STAT_WINDOW_COUNT: usize = 30;
struct QueryStats {
inner: Mutex<VecDeque<QueryWindow>>,
}
type QueryResult<'a> = Result<NonZeroUsize, &'a ResolveErrorKind>;
impl QueryStats {
fn new() -> Self {
Self { inner: Mutex::new(VecDeque::new()) }
}
async fn finish_query(&self, start_time: fasync::MonotonicInstant, result: QueryResult<'_>) {
let end_time = fasync::MonotonicInstant::now();
let finish = move |window: &mut QueryWindow| {
let elapsed_time = end_time - start_time;
match result {
Ok(num_addrs) => window.succeed(elapsed_time, num_addrs),
Err(e) => window.fail(elapsed_time, e),
}
};
let Self { inner } = self;
let past_queries = &mut *inner.lock().await;
let current_window = past_queries.back_mut().and_then(|window| {
let QueryWindow { start, .. } = window;
(end_time - *start < STAT_WINDOW_DURATION).then_some(window)
});
match current_window {
Some(window) => finish(window),
None => {
if past_queries.len() == STAT_WINDOW_COUNT {
let _: QueryWindow = past_queries
.pop_front()
.expect("there should be at least one element in `past_queries`");
}
let mut window = QueryWindow::new(end_time);
finish(&mut window);
past_queries.push_back(window);
}
}
}
}
#[derive(Debug)]
struct HashableResponseCode {
response_code: ResponseCode,
}
impl Hash for HashableResponseCode {
fn hash<H: Hasher>(&self, state: &mut H) {
let HashableResponseCode { response_code } = self;
u16::from(*response_code).hash(state)
}
}
impl PartialEq for HashableResponseCode {
fn eq(&self, other: &Self) -> bool {
let HashableResponseCode { response_code } = self;
let HashableResponseCode { response_code: other } = other;
response_code.eq(other)
}
}
impl Eq for HashableResponseCode {}
impl From<ResponseCode> for HashableResponseCode {
fn from(response_code: ResponseCode) -> Self {
HashableResponseCode { response_code }
}
}
#[derive(Default, Debug, PartialEq)]
struct NoRecordsFoundStats {
response_code_counts: HashMap<HashableResponseCode, u64>,
}
impl NoRecordsFoundStats {
fn increment(&mut self, response_code: &ResponseCode) {
let NoRecordsFoundStats { response_code_counts } = self;
let count = response_code_counts.entry((*response_code).into()).or_insert(0);
*count += 1
}
}
#[derive(Default, Debug, PartialEq)]
struct UnhandledResolveErrorKindStats {
resolve_error_kind_counts: HashMap<String, u64>,
}
impl UnhandledResolveErrorKindStats {
fn increment(&mut self, resolve_error_kind: &ResolveErrorKind) -> String {
let Self { resolve_error_kind_counts } = self;
let truncated_debug = enum_variant_string(resolve_error_kind);
let count = resolve_error_kind_counts.entry(truncated_debug.clone()).or_insert(0);
*count += 1;
truncated_debug
}
}
#[derive(Default, Debug, PartialEq)]
struct FailureStats {
message: u64,
no_connections: u64,
no_records_found: NoRecordsFoundStats,
io: u64,
proto: u64,
timeout: u64,
unhandled_resolve_error_kind: UnhandledResolveErrorKindStats,
}
impl FailureStats {
fn increment(&mut self, kind: &ResolveErrorKind) {
let FailureStats {
message,
no_connections,
no_records_found,
io,
proto,
timeout,
unhandled_resolve_error_kind,
} = self;
match kind {
ResolveErrorKind::Message(error) => {
let _: &str = error;
*message += 1
}
ResolveErrorKind::Msg(error) => {
let _: &String = error;
*message += 1
}
ResolveErrorKind::NoConnections => *no_connections += 1,
ResolveErrorKind::NoRecordsFound {
query: _,
soa: _,
negative_ttl: _,
response_code,
trusted: _,
} => no_records_found.increment(response_code),
ResolveErrorKind::Io(error) => {
let _: &std::io::Error = error;
*io += 1
}
ResolveErrorKind::Proto(error) => {
let _: &trust_dns_proto::error::ProtoError = error;
*proto += 1
}
ResolveErrorKind::Timeout => *timeout += 1,
kind => {
let variant = unhandled_resolve_error_kind.increment(kind);
error!("unhandled variant: {variant}");
}
}
}
}
struct QueryWindow {
start: fasync::MonotonicInstant,
success_count: u64,
failure_count: u64,
success_elapsed_time: zx::MonotonicDuration,
failure_elapsed_time: zx::MonotonicDuration,
failure_stats: FailureStats,
address_counts_histogram: BTreeMap<NonZeroUsize, u64>,
}
impl QueryWindow {
fn new(start: fasync::MonotonicInstant) -> Self {
Self {
start,
success_count: 0,
failure_count: 0,
success_elapsed_time: zx::MonotonicDuration::from_nanos(0),
failure_elapsed_time: zx::MonotonicDuration::from_nanos(0),
failure_stats: FailureStats::default(),
address_counts_histogram: Default::default(),
}
}
fn succeed(&mut self, elapsed_time: zx::MonotonicDuration, num_addrs: NonZeroUsize) {
let QueryWindow {
success_count,
success_elapsed_time,
address_counts_histogram: address_counts,
start: _,
failure_count: _,
failure_elapsed_time: _,
failure_stats: _,
} = self;
*success_count += 1;
*success_elapsed_time += elapsed_time;
*address_counts.entry(num_addrs).or_default() += 1;
}
fn fail(&mut self, elapsed_time: zx::MonotonicDuration, error: &ResolveErrorKind) {
let QueryWindow {
failure_count,
failure_elapsed_time,
failure_stats,
start: _,
success_count: _,
success_elapsed_time: _,
address_counts_histogram: _,
} = self;
*failure_count += 1;
*failure_elapsed_time += elapsed_time;
failure_stats.increment(error)
}
}
fn enum_variant_string(variant: &impl std::fmt::Debug) -> String {
let debug = format!("{:?}", variant);
match debug.find(|c: char| !c.is_xid_continue() && !c.is_xid_start()) {
Some(i) => debug[..i].to_string(),
None => debug,
}
}
fn update_resolver<T: ResolverLookup>(resolver: &SharedResolver<T>, servers: ServerList) {
let mut resolver_opts = ResolverOpts::default();
resolver_opts.num_concurrent_reqs = 10;
resolver_opts.server_ordering_strategy = ServerOrderingStrategy::UserProvidedOrder;
let mut name_servers = NameServerConfigGroup::with_capacity(servers.len() * 2);
name_servers.extend(servers.into_iter().flat_map(|server| {
let net_ext::SocketAddress(socket_addr) = server.into();
std::iter::once(NameServerConfig {
socket_addr,
protocol: Protocol::Udp,
tls_dns_name: None,
trust_nx_responses: false,
bind_addr: None,
})
.chain(std::iter::once(NameServerConfig {
socket_addr,
protocol: Protocol::Tcp,
tls_dns_name: None,
trust_nx_responses: false,
bind_addr: None,
}))
}));
let new_resolver =
T::new(ResolverConfig::from_parts(None, Vec::new(), name_servers), resolver_opts);
let () = resolver.write(Rc::new(new_resolver));
}
enum IncomingRequest {
Lookup(LookupRequestStream),
LookupAdmin(LookupAdminRequestStream),
}
#[async_trait]
trait ResolverLookup {
fn new(config: ResolverConfig, options: ResolverOpts) -> Self;
async fn lookup<N: IntoName + Send>(
&self,
name: N,
record_type: RecordType,
) -> Result<lookup::Lookup, ResolveError>;
async fn reverse_lookup(&self, addr: IpAddr) -> Result<lookup::ReverseLookup, ResolveError>;
}
#[async_trait]
impl ResolverLookup for Resolver {
fn new(config: ResolverConfig, options: ResolverOpts) -> Self {
Resolver::new(config, options, Spawner).expect("failed to create resolver")
}
async fn lookup<N: IntoName + Send>(
&self,
name: N,
record_type: RecordType,
) -> Result<lookup::Lookup, ResolveError> {
self.lookup(name, record_type).await
}
async fn reverse_lookup(&self, addr: IpAddr) -> Result<lookup::ReverseLookup, ResolveError> {
self.reverse_lookup(addr).await
}
}
#[derive(Debug)]
enum LookupIpErrorSource {
Ipv4,
Ipv6,
CanonicalName,
}
#[derive(Default)]
struct LookupIpErrorsFromSource {
ipv4: Option<ResolveError>,
ipv6: Option<ResolveError>,
canonical_name: Option<ResolveError>,
}
impl LookupIpErrorsFromSource {
fn any_error(&self) -> Option<&ResolveError> {
let Self { ipv4, ipv6, canonical_name } = self;
ipv4.as_ref().or_else(|| ipv6.as_ref()).or_else(|| canonical_name.as_ref())
}
fn accumulate(&mut self, src: LookupIpErrorSource, error: ResolveError) {
let Self { ipv4, ipv6, canonical_name } = self;
let target = match src {
LookupIpErrorSource::Ipv4 => ipv4,
LookupIpErrorSource::Ipv6 => ipv6,
LookupIpErrorSource::CanonicalName => canonical_name,
};
debug_assert!(target.is_none(), "multiple errors observed for {src:?}");
*target = Some(error)
}
fn handle(self) -> fname::LookupError {
let Self { ipv4, ipv6, canonical_name } = self;
let mut ret = None;
for (src, err) in [
("LookupIp(IPv4)", ipv4),
("LookupIp(IPv6)", ipv6),
("LookupIp(CanonicalName)", canonical_name),
]
.into_iter()
.filter_map(|(src, err)| err.map(|e| (src, e)))
{
let err = handle_err(src, err);
if ret.is_none() {
ret = Some(err)
}
}
ret.unwrap_or(fname::LookupError::InternalError)
}
}
fn handle_err(source: &str, err: ResolveError) -> fname::LookupError {
use trust_dns_proto::error::ProtoErrorKind;
let (lookup_err, ioerr): (_, Option<(std::io::ErrorKind, _)>) = match err.kind() {
ResolveErrorKind::NoRecordsFound {
query: _,
soa: _,
negative_ttl: _,
response_code: _,
trusted: _,
} => (fname::LookupError::NotFound, None),
ResolveErrorKind::Proto(err) => match err.kind() {
ProtoErrorKind::DomainNameTooLong(_) | ProtoErrorKind::EdnsNameNotRoot(_) => {
(fname::LookupError::InvalidArgs, None)
}
ProtoErrorKind::Busy | ProtoErrorKind::Canceled(_) | ProtoErrorKind::Timeout => {
(fname::LookupError::Transient, None)
}
ProtoErrorKind::Io(inner) => {
(fname::LookupError::Transient, Some((inner.kind(), inner.raw_os_error())))
}
ProtoErrorKind::BadQueryCount(_)
| ProtoErrorKind::CharacterDataTooLong { max: _, len: _ }
| ProtoErrorKind::LabelOverlapsWithOther { label: _, other: _ }
| ProtoErrorKind::DnsKeyProtocolNot3(_)
| ProtoErrorKind::FormError { header: _, error: _ }
| ProtoErrorKind::HmacInvalid()
| ProtoErrorKind::IncorrectRDataLengthRead { read: _, len: _ }
| ProtoErrorKind::LabelBytesTooLong(_)
| ProtoErrorKind::PointerNotPriorToLabel { idx: _, ptr: _ }
| ProtoErrorKind::MaxBufferSizeExceeded(_)
| ProtoErrorKind::Message(_)
| ProtoErrorKind::Msg(_)
| ProtoErrorKind::NoError
| ProtoErrorKind::NotAllRecordsWritten { count: _ }
| ProtoErrorKind::RrsigsNotPresent { name: _, record_type: _ }
| ProtoErrorKind::UnknownAlgorithmTypeValue(_)
| ProtoErrorKind::UnknownDnsClassStr(_)
| ProtoErrorKind::UnknownDnsClassValue(_)
| ProtoErrorKind::UnknownRecordTypeStr(_)
| ProtoErrorKind::UnknownRecordTypeValue(_)
| ProtoErrorKind::UnrecognizedLabelCode(_)
| ProtoErrorKind::UnrecognizedNsec3Flags(_)
| ProtoErrorKind::UnrecognizedCsyncFlags(_)
| ProtoErrorKind::Poisoned
| ProtoErrorKind::Ring(_)
| ProtoErrorKind::SSL(_)
| ProtoErrorKind::Timer
| ProtoErrorKind::UrlParsing(_)
| ProtoErrorKind::Utf8(_)
| ProtoErrorKind::FromUtf8(_)
| ProtoErrorKind::ParseInt(_) => (fname::LookupError::InternalError, None),
kind => {
error!("unhandled variant {:?}", enum_variant_string(kind));
(fname::LookupError::InternalError, None)
}
},
ResolveErrorKind::Io(inner) => {
(fname::LookupError::Transient, Some((inner.kind(), inner.raw_os_error())))
}
ResolveErrorKind::Timeout => (fname::LookupError::Transient, None),
ResolveErrorKind::Msg(_)
| ResolveErrorKind::Message(_)
| ResolveErrorKind::NoConnections => (fname::LookupError::InternalError, None),
kind => {
error!("unhandled variant {:?}", enum_variant_string(kind));
(fname::LookupError::InternalError, None)
}
};
if let Some((ioerr, raw_os_error)) = ioerr {
match raw_os_error {
Some(libc::EHOSTUNREACH | libc::ENETUNREACH) => {
debug!("{} error: {:?}; (IO error {:?})", source, lookup_err, ioerr)
}
_ => warn!("{} error: {:?}; (IO error {:?})", source, lookup_err, ioerr),
}
} else {
warn!("{} error: {:?}", source, lookup_err);
}
lookup_err
}
async fn sort_preferred_addresses(
mut addrs: Vec<fnet::IpAddress>,
routes: &fnet_routes::StateProxy,
) -> Result<Vec<fnet::IpAddress>, fname::LookupError> {
let mut addrs_info = futures::future::try_join_all(
addrs
.drain(..)
.map(|addr| async move {
let source_addr = match routes.resolve(&addr).await? {
Ok(fnet_routes::Resolved::Direct(fnet_routes::Destination {
source_address,
..
}))
| Ok(fnet_routes::Resolved::Gateway(fnet_routes::Destination {
source_address,
..
})) => source_address,
Err(e) => {
debug!(
"fuchsia.net.routes/State.resolve({}) failed {}",
net_ext::IpAddress::from(addr),
zx::Status::from_raw(e)
);
None
}
};
Ok((addr, DasCmpInfo::from_addrs(&addr, source_addr.as_ref())))
}),
)
.await
.map_err(|e: fidl::Error| {
warn!("fuchsia.net.routes/State.resolve FIDL error {:?}", e);
fname::LookupError::InternalError
})?;
let () = addrs_info.sort_by(|(_laddr, left), (_raddr, right)| left.cmp(right));
let () = addrs.extend(addrs_info.into_iter().map(|(addr, _)| addr));
Ok(addrs)
}
#[derive(Debug)]
struct Policy {
prefix: net_types::ip::Subnet<net_types::ip::Ipv6Addr>,
precedence: usize,
label: usize,
}
macro_rules! decl_policy {
($ip:tt/$prefix:expr => $precedence:expr, $label:expr) => {
Policy {
prefix: unsafe {
net_types::ip::Subnet::new_unchecked(
net_types::ip::Ipv6Addr::from_bytes(fidl_ip_v6!($ip).addr),
$prefix,
)
},
precedence: $precedence,
label: $label,
}
};
}
const POLICY_TABLE: [Policy; 6] = [
decl_policy!("::1"/128 => 50, 0),
decl_policy!("::ffff:0:0"/96 => 35, 4),
decl_policy!("2001::"/32 => 5, 5),
decl_policy!("2002::"/16 => 30, 2),
decl_policy!("fc00::"/7 => 3, 13),
decl_policy!("::"/0 => 40, 1),
];
fn policy_lookup(addr: &net_types::ip::Ipv6Addr) -> &'static Policy {
POLICY_TABLE
.iter()
.find(|policy| policy.prefix.contains(addr))
.expect("policy table MUST contain the all addresses subnet")
}
#[derive(Debug)]
struct DasCmpInfo {
usable: bool,
matching_scope: bool,
matching_label: bool,
precedence: usize,
scope: net_types::ip::Ipv6Scope,
common_prefix_len: u8,
}
impl DasCmpInfo {
fn convert_addr(fidl: &fnet::IpAddress) -> net_types::ip::Ipv6Addr {
match fidl {
fnet::IpAddress::Ipv4(fnet::Ipv4Address { addr }) => {
net_types::ip::Ipv6Addr::from(net_types::ip::Ipv4Addr::new(*addr))
}
fnet::IpAddress::Ipv6(fnet::Ipv6Address { addr }) => {
net_types::ip::Ipv6Addr::from_bytes(*addr)
}
}
}
fn from_addrs(dst_addr: &fnet::IpAddress, src_addr: Option<&fnet::IpAddress>) -> Self {
use net_types::ScopeableAddress;
let dst_addr = Self::convert_addr(dst_addr);
let Policy { prefix: _, precedence, label: dst_label } = policy_lookup(&dst_addr);
let (usable, matching_scope, matching_label, common_prefix_len) = match src_addr {
Some(src_addr) => {
let src_addr = Self::convert_addr(src_addr);
let Policy { prefix: _, precedence: _, label: src_label } =
policy_lookup(&src_addr);
(
true,
dst_addr.scope() == src_addr.scope(),
dst_label == src_label,
dst_addr.common_prefix_len(&src_addr),
)
}
None => (false, false, false, 0),
};
DasCmpInfo {
usable,
matching_scope,
matching_label,
precedence: *precedence,
scope: dst_addr.scope(),
common_prefix_len,
}
}
}
impl std::cmp::Ord for DasCmpInfo {
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
use std::cmp::Ordering;
let DasCmpInfo {
usable: self_usable,
matching_scope: self_matching_scope,
matching_label: self_matching_label,
precedence: self_precedence,
scope: self_scope,
common_prefix_len: self_common_prefix_len,
} = self;
let DasCmpInfo {
usable: other_usable,
matching_scope: other_matching_scope,
matching_label: other_matching_label,
precedence: other_precedence,
scope: other_scope,
common_prefix_len: other_common_prefix_len,
} = other;
fn prefer_true(left: bool, right: bool) -> Ordering {
match (left, right) {
(true, false) => Ordering::Less,
(false, true) => Ordering::Greater,
(false, false) | (true, true) => Ordering::Equal,
}
}
prefer_true(*self_usable, *other_usable)
.then(
prefer_true(*self_matching_scope, *other_matching_scope),
)
.then(
prefer_true(*self_matching_label, *other_matching_label),
)
.then(
self_precedence.cmp(other_precedence).reverse(),
)
.then(
self_scope.multicast_scope_id().cmp(&other_scope.multicast_scope_id()),
)
.then(
self_common_prefix_len.cmp(other_common_prefix_len).reverse(),
)
}
}
impl std::cmp::PartialOrd for DasCmpInfo {
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
Some(self.cmp(other))
}
}
impl std::cmp::PartialEq for DasCmpInfo {
fn eq(&self, other: &Self) -> bool {
self.cmp(other) == std::cmp::Ordering::Equal
}
}
impl std::cmp::Eq for DasCmpInfo {}
async fn handle_lookup_hostname<T: ResolverLookup>(
resolver: &SharedResolver<T>,
addr: fnet::IpAddress,
) -> Result<String, fname::LookupError> {
let net_ext::IpAddress(addr) = addr.into();
let resolver = resolver.read();
match resolver.reverse_lookup(addr).await {
Ok(response) => {
response.iter().next().ok_or(fname::LookupError::NotFound).map(ToString::to_string)
}
Err(error) => Err(handle_err("LookupHostname", error)),
}
}
struct IpLookupRequest {
hostname: String,
options: fname::LookupIpOptions,
responder: fname::LookupLookupIpResponder,
}
async fn run_lookup<T: ResolverLookup>(
resolver: &SharedResolver<T>,
stream: LookupRequestStream,
sender: mpsc::Sender<IpLookupRequest>,
) -> Result<(), fidl::Error> {
stream
.try_for_each_concurrent(None, |request| async {
match request {
LookupRequest::LookupIp { hostname, options, responder } => {
let () = sender
.clone()
.send(IpLookupRequest { hostname, options, responder })
.await
.expect("receiver should not be closed");
Ok(())
}
LookupRequest::LookupHostname { addr, responder } => responder
.send(handle_lookup_hostname(&resolver, addr).await.as_deref().map_err(|e| *e)),
}
})
.await
}
const MAX_PARALLEL_REQUESTS: usize = 256;
fn create_ip_lookup_fut<T: ResolverLookup>(
resolver: &SharedResolver<T>,
stats: Arc<QueryStats>,
routes: fnet_routes::StateProxy,
recv: mpsc::Receiver<IpLookupRequest>,
) -> impl futures::Future<Output = ()> + '_ {
recv.for_each_concurrent(
MAX_PARALLEL_REQUESTS,
move |IpLookupRequest { hostname, options, responder }| {
let stats = stats.clone();
let routes = routes.clone();
async move {
let fname::LookupIpOptions {
ipv4_lookup,
ipv6_lookup,
sort_addresses,
canonical_name_lookup,
..
} = options;
let ipv4_lookup = ipv4_lookup.unwrap_or(false);
let ipv6_lookup = ipv6_lookup.unwrap_or(false);
let sort_addresses = sort_addresses.unwrap_or(false);
let canonical_name_lookup = canonical_name_lookup.unwrap_or(false);
let lookup_result = (|| async {
let hostname = hostname.as_str();
match IpAddr::from_str(hostname) {
Ok(addr) => {
let _: IpAddr = addr;
return Err(fname::LookupError::InvalidArgs);
}
Err(std::net::AddrParseError { .. }) => {}
};
let resolver = resolver.read();
let start_time = fasync::MonotonicInstant::now();
let (ret1, ret2, ret3) = futures::future::join3(
futures::future::OptionFuture::from(
ipv4_lookup.then(|| {
resolver
.lookup(hostname, RecordType::A)
.map_err(|e| (LookupIpErrorSource::Ipv4, e))
}),
),
futures::future::OptionFuture::from(
ipv6_lookup.then(|| {
resolver
.lookup(hostname, RecordType::AAAA)
.map_err(|e| (LookupIpErrorSource::Ipv6, e))
}),
),
futures::future::OptionFuture::from(
canonical_name_lookup
.then(|| {
resolver
.lookup(hostname, RecordType::CNAME)
.map_err(|e| (LookupIpErrorSource::CanonicalName, e))
}),
),
)
.await;
let result = [ret1, ret2, ret3];
if result.iter().all(Option::is_none) {
return Err(fname::LookupError::InvalidArgs);
}
let (addrs, cnames, error) =
result.into_iter().filter_map(std::convert::identity).fold(
(Vec::new(), Vec::new(), LookupIpErrorsFromSource::default()),
|(mut addrs, mut cnames, mut error), result| {
let () = match result {
Err((src, err)) => {
error.accumulate(src, err);
},
Ok(lookup) => lookup.iter().for_each(|rdata| match rdata {
RData::A(addr) if ipv4_lookup => addrs
.push(net_ext::IpAddress(IpAddr::V4(*addr)).into()),
RData::AAAA(addr) if ipv6_lookup => addrs
.push(net_ext::IpAddress(IpAddr::V6(*addr)).into()),
RData::CNAME(name) => {
if canonical_name_lookup {
cnames.push(name.to_utf8())
}
}
rdata => {
error!(
"Lookup(_, {:?}) yielded unexpected record type: {}",
options, rdata.to_record_type(),
)
}
}),
};
(addrs, cnames, error)
});
let count = match NonZeroUsize::try_from(addrs.len() + cnames.len()) {
Ok(count) => Ok(count),
Err(std::num::TryFromIntError { .. }) => match error.any_error() {
None => {
error!("resolver response unexpectedly contained no records \
and no error. See https://fxbug.dev/42062388.");
return Err(fname::LookupError::NotFound);
},
Some(any_err) => {
Err(any_err)
}
}
};
let () = stats
.finish_query(
start_time,
count.as_ref().copied().map_err(|e| e.kind()),
)
.await;
match count {
Ok(_) => {},
Err(_any_err) => {
return Err(error.handle());
}
}
let addrs = if sort_addresses {
sort_preferred_addresses(addrs, &routes).await?
} else {
addrs
};
let addrs = if addrs.len() > fname::MAX_ADDRESSES.into() {
warn!(
"Lookup(_, {:?}): {} addresses, truncating to {}",
options, addrs.len(), fname::MAX_ADDRESSES
);
let mut addrs = addrs;
addrs.truncate(fname::MAX_ADDRESSES.into());
addrs
} else {
addrs
};
if cnames.len() > 1 {
let cnames =
cnames.iter().fold(HashMap::<&str, usize>::new(), |mut acc, cname| {
*acc.entry(cname).or_default() += 1;
acc
});
warn!(
"Lookup(_, {:?}): multiple CNAMEs: {:?}",
options, cnames
)
}
let cname = {
let mut cnames = cnames;
cnames.pop()
};
Ok(fname::LookupResult {
addresses: Some(addrs),
canonical_name: cname,
..Default::default()
})
})()
.await;
responder.send(lookup_result.as_ref().map_err(|e| *e)).unwrap_or_else(|e|
warn!(
"failed to send IP lookup result due to FIDL error: {}",
e
)
)
}
},
)
}
async fn run_lookup_admin<T: ResolverLookup>(
resolver: &SharedResolver<T>,
state: &dns::config::ServerConfigState,
stream: LookupAdminRequestStream,
) -> Result<(), fidl::Error> {
stream
.try_for_each(|req| async {
match req {
LookupAdminRequest::SetDnsServers { servers, responder } => {
let response = match state.update_servers(servers) {
UpdateServersResult::Updated(servers) => {
let () = update_resolver(resolver, servers);
Ok(())
}
UpdateServersResult::NoChange => Ok(()),
UpdateServersResult::InvalidsServers => {
Err(zx::Status::INVALID_ARGS.into_raw())
}
};
let () = responder.send(response)?;
}
LookupAdminRequest::GetDnsServers { responder } => {
let () = responder.send(&state.servers())?;
}
}
Ok(())
})
.await
}
fn add_config_state_inspect(
parent: &fuchsia_inspect::Node,
config_state: Arc<dns::config::ServerConfigState>,
) -> fuchsia_inspect::LazyNode {
parent.create_lazy_child("servers", move || {
let config_state = config_state.clone();
async move {
let srv = fuchsia_inspect::Inspector::default();
let server_list = config_state.servers();
for (i, server) in server_list.into_iter().enumerate() {
let child = srv.root().create_child(format!("{}", i));
let net_ext::SocketAddress(addr) = server.into();
let () = child.record_string("address", format!("{}", addr));
let () = srv.root().record(child);
}
Ok(srv)
}
.boxed()
})
}
fn add_query_stats_inspect(
parent: &fuchsia_inspect::Node,
stats: Arc<QueryStats>,
) -> fuchsia_inspect::LazyNode {
parent.create_lazy_child("query_stats", move || {
let stats = stats.clone();
async move {
let past_queries = &*stats.inner.lock().await;
let node = fuchsia_inspect::Inspector::default();
for (
i,
QueryWindow {
start,
success_count,
failure_count,
success_elapsed_time,
failure_elapsed_time,
failure_stats,
address_counts_histogram,
},
) in past_queries.iter().enumerate()
{
let child = node.root().create_child(format!("window {}", i + 1));
match u64::try_from(start.into_nanos()) {
Ok(nanos) => {
let () = child.record_uint("start_time_nanos", nanos);
},
Err(e) => warn!(
"error computing `start_time_nanos`: {:?}.into_nanos() from i64 -> u64 failed: {}",
start, e
),
}
let () = child.record_uint("successful_queries", *success_count);
let () = child.record_uint("failed_queries", *failure_count);
let record_average = |name: &str, total: zx::MonotonicDuration, count: u64| {
if count == 0 {
return;
}
match u64::try_from(total.into_micros()) {
Ok(micros) => child.record_uint(name, micros / count),
Err(e) => warn!(
"error computing `{}`: {:?}.into_micros() from i64 -> u64 failed: {}",
name, success_elapsed_time, e
),
}
};
let () = record_average(
"average_success_duration_micros",
*success_elapsed_time,
*success_count,
);
let () = record_average(
"average_failure_duration_micros",
*failure_elapsed_time,
*failure_count,
);
let FailureStats {
message,
no_connections,
no_records_found: NoRecordsFoundStats {
response_code_counts,
},
io,
proto,
timeout,
unhandled_resolve_error_kind: UnhandledResolveErrorKindStats {
resolve_error_kind_counts,
},
} = failure_stats;
let errors = child.create_child("errors");
let () = errors.record_uint("Message", *message);
let () = errors.record_uint("NoConnections", *no_connections);
let () = errors.record_uint("Io", *io);
let () = errors.record_uint("Proto", *proto);
let () = errors.record_uint("Timeout", *timeout);
let no_records_found_response_codes =
errors.create_child("NoRecordsFoundResponseCodeCounts");
for (HashableResponseCode { response_code }, count) in response_code_counts {
let () = no_records_found_response_codes.record_uint(
format!("{:?}", response_code),
*count,
);
}
let () = errors.record(no_records_found_response_codes);
let unhandled_resolve_error_kinds =
errors.create_child("UnhandledResolveErrorKindCounts");
for (error_kind, count) in resolve_error_kind_counts {
let () = unhandled_resolve_error_kinds.record_uint(error_kind, *count);
}
let () = errors.record(unhandled_resolve_error_kinds);
let () = child.record(errors);
let address_counts_node = child.create_child("address_counts");
for (count, occurrences) in address_counts_histogram {
address_counts_node.record_uint(count.to_string(), *occurrences);
}
child.record(address_counts_node);
let () = node.root().record(child);
}
Ok(node)
}
.boxed()
})
}
#[fuchsia::main(logging_tags = ["dns"])]
pub async fn main() -> Result<(), Error> {
info!("starting");
let mut resolver_opts = ResolverOpts::default();
resolver_opts.ip_strategy = LookupIpStrategy::Ipv4AndIpv6;
let resolver = SharedResolver::new(
Resolver::new(ResolverConfig::default(), resolver_opts, Spawner)
.expect("failed to create resolver"),
);
let config_state = Arc::new(dns::config::ServerConfigState::new());
let stats = Arc::new(QueryStats::new());
let mut fs = ServiceFs::new_local();
let inspector = fuchsia_inspect::component::inspector();
let _state_inspect_node = add_config_state_inspect(inspector.root(), config_state.clone());
let _query_stats_inspect_node = add_query_stats_inspect(inspector.root(), stats.clone());
let _inspect_server_task =
inspect_runtime::publish(inspector, inspect_runtime::PublishOptions::default())
.context("publish Inspect task")?;
let routes = fuchsia_component::client::connect_to_protocol::<fnet_routes::StateMarker>()
.context("failed to connect to fuchsia.net.routes/State")?;
let _: &mut ServiceFsDir<'_, _> = fs
.dir("svc")
.add_fidl_service(IncomingRequest::Lookup)
.add_fidl_service(IncomingRequest::LookupAdmin);
let _: &mut ServiceFs<_> =
fs.take_and_serve_directory_handle().context("failed to serve directory")?;
let (sender, recv) = mpsc::channel(MAX_PARALLEL_REQUESTS);
let serve_fut = fs.for_each_concurrent(None, |incoming_service| async {
match incoming_service {
IncomingRequest::Lookup(stream) => run_lookup(&resolver, stream, sender.clone())
.await
.unwrap_or_else(|e| warn!("run_lookup finished with error: {}", e)),
IncomingRequest::LookupAdmin(stream) => {
run_lookup_admin(&resolver, &config_state, stream)
.await
.unwrap_or_else(|e| error!("run_lookup_admin finished with error: {}", e))
}
}
});
let ip_lookup_fut = create_ip_lookup_fut(&resolver, stats.clone(), routes, recv);
match fuchsia_scheduler::set_role_for_this_thread("fuchsia.networking.dns.resolver.main") {
Ok(_) => info!("Applied scheduling role"),
Err(err) => warn!("Failed to apply scheduling role: {}", err),
};
let ((), ()) = futures::future::join(serve_fut, ip_lookup_fut).await;
Ok(())
}
#[cfg(test)]
mod tests {
use std::net::{Ipv4Addr, Ipv6Addr, SocketAddr};
use std::pin::pin;
use std::str::FromStr;
use assert_matches::assert_matches;
use diagnostics_assertions::{assert_data_tree, tree_assertion, NonZeroUintProperty};
use dns::test_util::*;
use dns::DEFAULT_PORT;
use futures::future::TryFutureExt as _;
use itertools::Itertools as _;
use net_declare::{fidl_ip, std_ip, std_ip_v4, std_ip_v6};
use net_types::ip::Ip as _;
use test_case::test_case;
use trust_dns_proto::op::Query;
use trust_dns_proto::rr::{Name, Record};
use trust_dns_resolver::lookup::{Lookup, ReverseLookup};
use super::*;
const IPV4_LOOPBACK: fnet::IpAddress = fidl_ip!("127.0.0.1");
const IPV6_LOOPBACK: fnet::IpAddress = fidl_ip!("::1");
const LOCAL_HOST: &str = "localhost.";
const IPV4_HOST: Ipv4Addr = std_ip_v4!("240.0.0.2");
const IPV6_HOST: Ipv6Addr = std_ip_v6!("abcd::2");
const REMOTE_IPV4_HOST: &str = "www.foo.com";
const REMOTE_IPV6_HOST: &str = "www.bar.com";
const REMOTE_IPV4_HOST_ALIAS: &str = "www.alsofoo.com";
const REMOTE_IPV6_HOST_ALIAS: &str = "www.alsobar.com";
const REMOTE_IPV6_HOST_EXTRA: &str = "www.bar2.com";
const REMOTE_IPV4_IPV6_HOST: &str = "www.foobar.com";
const NO_RECORDS_AND_NO_ERROR_HOST: &str = "www.no-records-and-no-error.com";
async fn setup_namelookup_service() -> (fname::LookupProxy, impl futures::Future<Output = ()>) {
let (name_lookup_proxy, stream) =
fidl::endpoints::create_proxy_and_stream::<fname::LookupMarker>();
let mut resolver_opts = ResolverOpts::default();
resolver_opts.ip_strategy = LookupIpStrategy::Ipv4AndIpv6;
let resolver = SharedResolver::new(
Resolver::new(ResolverConfig::default(), resolver_opts, Spawner)
.expect("failed to create resolver"),
);
let stats = Arc::new(QueryStats::new());
let (routes_proxy, routes_stream) =
fidl::endpoints::create_proxy_and_stream::<fnet_routes::StateMarker>();
let routes_fut =
routes_stream.try_for_each(|req| -> futures::future::Ready<Result<(), fidl::Error>> {
panic!("Should not call routes/State. Received request {:?}", req)
});
let (sender, recv) = mpsc::channel(MAX_PARALLEL_REQUESTS);
(name_lookup_proxy, async move {
futures::future::try_join3(
run_lookup(&resolver, stream, sender),
routes_fut,
create_ip_lookup_fut(&resolver, stats.clone(), routes_proxy, recv).map(Ok),
)
.map(|r| match r {
Ok(((), (), ())) => (),
Err(e) => panic!("namelookup service error {:?}", e),
})
.await
})
}
#[fasync::run_singlethreaded(test)]
async fn test_lookupip_localhost() {
let (proxy, fut) = setup_namelookup_service().await;
let ((), ()) = futures::future::join(fut, async move {
assert_eq!(
proxy
.lookup_ip(
LOCAL_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![IPV4_LOOPBACK, IPV6_LOOPBACK]),
..Default::default()
}),
);
assert_eq!(
proxy
.lookup_ip(
LOCAL_HOST,
&fname::LookupIpOptions { ipv4_lookup: Some(true), ..Default::default() }
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![IPV4_LOOPBACK]),
..Default::default()
}),
);
assert_eq!(
proxy
.lookup_ip(
LOCAL_HOST,
&fname::LookupIpOptions { ipv6_lookup: Some(true), ..Default::default() }
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![IPV6_LOOPBACK]),
..Default::default()
}),
);
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_lookuphostname_localhost() {
let (proxy, fut) = setup_namelookup_service().await;
let ((), ()) = futures::future::join(fut, async move {
let hostname = IPV4_LOOPBACK;
assert_eq!(
proxy.lookup_hostname(&hostname).await.expect("lookup_hostname").as_deref(),
Ok(LOCAL_HOST)
);
})
.await;
}
struct MockResolver {
config: ResolverConfig,
repeat: u16,
}
#[async_trait]
impl ResolverLookup for MockResolver {
fn new(config: ResolverConfig, _options: ResolverOpts) -> Self {
Self { config, repeat: 1 }
}
async fn lookup<N: IntoName + Send>(
&self,
name: N,
record_type: RecordType,
) -> Result<lookup::Lookup, ResolveError> {
let Self { config: _, repeat } = self;
let name = name.into_name()?;
let host_name = name.to_utf8();
if host_name == NO_RECORDS_AND_NO_ERROR_HOST {
return Ok(Lookup::new_with_max_ttl(Query::default(), Arc::new([])));
}
let rdatas = match record_type {
RecordType::A => [REMOTE_IPV4_HOST, REMOTE_IPV4_IPV6_HOST]
.contains(&host_name.as_str())
.then_some(RData::A(IPV4_HOST)),
RecordType::AAAA => [REMOTE_IPV6_HOST, REMOTE_IPV4_IPV6_HOST]
.contains(&host_name.as_str())
.then_some(RData::AAAA(IPV6_HOST)),
RecordType::CNAME => match host_name.as_str() {
REMOTE_IPV4_HOST_ALIAS => Some(REMOTE_IPV4_HOST),
REMOTE_IPV6_HOST_ALIAS => Some(REMOTE_IPV6_HOST),
_ => None,
}
.map(Name::from_str)
.transpose()
.unwrap()
.map(RData::CNAME),
record_type => {
panic!("unexpected record type {:?}", record_type)
}
}
.into_iter();
let len = rdatas.len() * usize::from(*repeat);
let records: Vec<Record> = rdatas
.map(|rdata| {
Record::from_rdata(
Name::new(),
60,
rdata,
)
})
.cycle()
.take(len)
.collect();
if records.is_empty() {
let mut response = trust_dns_proto::op::Message::new();
let _: &mut trust_dns_proto::op::Message =
response.set_response_code(ResponseCode::NoError);
let error = ResolveError::from_response(response.into(), false)
.expect_err("response with no records should be a NoRecordsFound error");
return Err(error);
}
Ok(Lookup::new_with_max_ttl(Query::default(), records.into()))
}
async fn reverse_lookup(
&self,
addr: IpAddr,
) -> Result<lookup::ReverseLookup, ResolveError> {
let lookup = if addr == IPV4_HOST {
Lookup::from_rdata(
Query::default(),
RData::PTR(Name::from_str(REMOTE_IPV4_HOST).unwrap()),
)
} else if addr == IPV6_HOST {
Lookup::new_with_max_ttl(
Query::default(),
Arc::new([
Record::from_rdata(
Name::new(),
60, RData::PTR(Name::from_str(REMOTE_IPV6_HOST).unwrap()),
),
Record::from_rdata(
Name::new(),
60, RData::PTR(Name::from_str(REMOTE_IPV6_HOST_EXTRA).unwrap()),
),
]),
)
} else {
Lookup::new_with_max_ttl(Query::default(), Arc::new([]))
};
Ok(ReverseLookup::from(lookup))
}
}
struct TestEnvironment {
shared_resolver: SharedResolver<MockResolver>,
config_state: Arc<dns::config::ServerConfigState>,
stats: Arc<QueryStats>,
}
impl Default for TestEnvironment {
fn default() -> Self {
Self::new(1)
}
}
impl TestEnvironment {
fn new(repeat: u16) -> Self {
Self {
shared_resolver: SharedResolver::new(MockResolver {
config: ResolverConfig::from_parts(
None,
vec![],
NameServerConfigGroup::with_capacity(0),
),
repeat,
}),
config_state: Arc::new(dns::config::ServerConfigState::new()),
stats: Arc::new(QueryStats::new()),
}
}
async fn run_lookup<F, Fut>(&self, f: F)
where
Fut: futures::Future<Output = ()>,
F: FnOnce(fname::LookupProxy) -> Fut,
{
self.run_lookup_with_routes_handler(f, |req| {
panic!("Should not call routes/State. Received request {:?}", req)
})
.await
}
async fn run_lookup_with_routes_handler<F, Fut, R>(&self, f: F, handle_routes: R)
where
Fut: futures::Future<Output = ()>,
F: FnOnce(fname::LookupProxy) -> Fut,
R: Fn(fnet_routes::StateRequest),
{
let (name_lookup_proxy, name_lookup_stream) =
fidl::endpoints::create_proxy_and_stream::<fname::LookupMarker>();
let (routes_proxy, routes_stream) =
fidl::endpoints::create_proxy_and_stream::<fnet_routes::StateMarker>();
let (sender, recv) = mpsc::channel(MAX_PARALLEL_REQUESTS);
let Self { shared_resolver, config_state: _, stats } = self;
let ((), (), (), ()) = futures::future::try_join4(
run_lookup(shared_resolver, name_lookup_stream, sender),
f(name_lookup_proxy).map(Ok),
routes_stream.try_for_each(|req| futures::future::ok(handle_routes(req))),
create_ip_lookup_fut(shared_resolver, stats.clone(), routes_proxy, recv).map(Ok),
)
.await
.expect("Error running lookup future");
}
async fn run_admin<F, Fut>(&self, f: F)
where
Fut: futures::Future<Output = ()>,
F: FnOnce(fname::LookupAdminProxy) -> Fut,
{
let (lookup_admin_proxy, lookup_admin_stream) =
fidl::endpoints::create_proxy_and_stream::<fname::LookupAdminMarker>();
let Self { shared_resolver, config_state, stats: _ } = self;
let ((), ()) = futures::future::try_join(
run_lookup_admin(shared_resolver, config_state, lookup_admin_stream)
.map_err(anyhow::Error::from),
f(lookup_admin_proxy).map(Ok),
)
.await
.expect("Error running admin future");
}
}
fn map_ip<T: Into<IpAddr>>(addr: T) -> fnet::IpAddress {
net_ext::IpAddress(addr.into()).into()
}
#[fasync::run_singlethreaded(test)]
async fn test_no_records_and_no_error() {
TestEnvironment::default()
.run_lookup(|proxy| async move {
let proxy = &proxy;
futures::stream::iter([(true, true), (true, false), (false, true)])
.for_each_concurrent(None, move |(ipv4_lookup, ipv6_lookup)| async move {
assert_eq!(
proxy
.lookup_ip(
NO_RECORDS_AND_NO_ERROR_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(ipv4_lookup),
ipv6_lookup: Some(ipv6_lookup),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::NotFound),
);
})
.await
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_lookupip_remotehost_overflow() {
const REPEAT: u16 = fname::MAX_ADDRESSES / 2 + 1;
let expected = std::iter::empty()
.chain(std::iter::repeat(map_ip(IPV4_HOST)).take(REPEAT.into()))
.chain(std::iter::repeat(map_ip(IPV6_HOST)).take(REPEAT.into()))
.take(fname::MAX_ADDRESSES.into())
.collect::<Vec<_>>();
assert_eq!(expected.len(), usize::from(fname::MAX_ADDRESSES));
TestEnvironment::new(REPEAT)
.run_lookup(|proxy| async move {
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_IPV6_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult { addresses: Some(expected), ..Default::default() })
);
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_lookupip_remotehost_ipv4() {
TestEnvironment::default()
.run_lookup(|proxy| async move {
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV4_HOST)]),
..Default::default()
}),
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV4_HOST)]),
..Default::default()
}),
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::NotFound),
);
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_lookupip_remotehost_ipv6() {
TestEnvironment::default()
.run_lookup(|proxy| async move {
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV6_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV6_HOST)]),
..Default::default()
}),
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV6_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::NotFound),
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV6_HOST,
&fname::LookupIpOptions {
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV6_HOST)]),
..Default::default()
}),
);
})
.await;
}
#[test_case(REMOTE_IPV4_HOST_ALIAS, REMOTE_IPV4_HOST; "ipv4")]
#[test_case(REMOTE_IPV6_HOST_ALIAS, REMOTE_IPV6_HOST; "ipv6")]
#[fasync::run_singlethreaded(test)]
async fn test_lookupip_remotehost_canonical_name(hostname: &str, expected: &str) {
TestEnvironment::default()
.run_lookup(|proxy| async move {
assert_matches!(
proxy
.lookup_ip(
hostname,
&fname::LookupIpOptions {
canonical_name_lookup: Some(true),
..Default::default()
}
)
.await,
Ok(Ok(fname::LookupResult {
canonical_name: Some(cname),
..
})) => assert_eq!(cname, expected)
);
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_lookupip_ip_literal() {
TestEnvironment::default()
.run_lookup(|proxy| async move {
let proxy = &proxy;
let range = || [true, false].into_iter();
futures::stream::iter(range().cartesian_product(range()))
.for_each_concurrent(None, move |(ipv4_lookup, ipv6_lookup)| async move {
assert_eq!(
proxy
.lookup_ip(
"240.0.0.2",
&fname::LookupIpOptions {
ipv4_lookup: Some(ipv4_lookup),
ipv6_lookup: Some(ipv6_lookup),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::InvalidArgs),
"ipv4_lookup={},ipv6_lookup={}",
ipv4_lookup,
ipv6_lookup,
);
assert_eq!(
proxy
.lookup_ip(
"abcd::2",
&fname::LookupIpOptions {
ipv4_lookup: Some(ipv4_lookup),
ipv6_lookup: Some(ipv6_lookup),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::InvalidArgs),
"ipv4_lookup={},ipv6_lookup={}",
ipv4_lookup,
ipv6_lookup,
);
})
.await
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn test_lookup_hostname() {
TestEnvironment::default()
.run_lookup(|proxy| async move {
assert_eq!(
proxy
.lookup_hostname(&map_ip(IPV4_HOST))
.await
.expect("lookup_hostname")
.as_deref(),
Ok(REMOTE_IPV4_HOST)
);
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_lookup_hostname_multi() {
TestEnvironment::default()
.run_lookup(|proxy| async move {
assert_eq!(
proxy
.lookup_hostname(&map_ip(IPV6_HOST))
.await
.expect("lookup_hostname")
.as_deref(),
Ok(REMOTE_IPV6_HOST)
);
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_set_server_names() {
let env = TestEnvironment::default();
let to_server_configs = |socket_addr: SocketAddr| -> [NameServerConfig; 2] {
[
NameServerConfig {
socket_addr,
protocol: Protocol::Udp,
tls_dns_name: None,
trust_nx_responses: false,
bind_addr: None,
},
NameServerConfig {
socket_addr,
protocol: Protocol::Tcp,
tls_dns_name: None,
trust_nx_responses: false,
bind_addr: None,
},
]
};
assert_eq!(env.shared_resolver.read().config.name_servers().to_vec(), vec![]);
env.run_admin(|proxy| async move {
let () = proxy
.set_dns_servers(&[DHCP_SERVER, NDP_SERVER, DHCPV6_SERVER])
.await
.expect("Failed to call SetDnsServers")
.expect("SetDnsServers error");
})
.await;
assert_eq!(
env.shared_resolver.read().config.name_servers().to_vec(),
vec![DHCP_SERVER, NDP_SERVER, DHCPV6_SERVER]
.into_iter()
.map(|s| {
let net_ext::SocketAddress(s) = s.into();
s
})
.flat_map(|x| to_server_configs(x).to_vec().into_iter())
.collect::<Vec<_>>()
);
env.run_admin(|proxy| async move {
let () = proxy
.set_dns_servers(&[])
.await
.expect("Failed to call SetDnsServers")
.expect("SetDnsServers error");
})
.await;
assert_eq!(env.shared_resolver.read().config.name_servers().to_vec(), Vec::new());
}
#[fasync::run_singlethreaded(test)]
async fn test_set_server_names_error() {
let env = TestEnvironment::default();
assert_eq!(env.shared_resolver.read().config.name_servers().to_vec(), vec![]);
env.run_admin(|proxy| async move {
let status = proxy
.set_dns_servers(&[fnet::SocketAddress::Ipv4(fnet::Ipv4SocketAddress {
address: fnet::Ipv4Address { addr: [224, 0, 0, 1] },
port: DEFAULT_PORT,
})])
.await
.expect("Failed to call SetDnsServers")
.expect_err("SetDnsServers should fail for multicast address");
assert_eq!(zx::Status::from_raw(status), zx::Status::INVALID_ARGS);
let status = proxy
.set_dns_servers(&[fnet::SocketAddress::Ipv6(fnet::Ipv6SocketAddress {
address: fnet::Ipv6Address { addr: [0; 16] },
port: DEFAULT_PORT,
zone_index: 0,
})])
.await
.expect("Failed to call SetDnsServers")
.expect_err("SetDnsServers should fail for unspecified address");
assert_eq!(zx::Status::from_raw(status), zx::Status::INVALID_ARGS);
})
.await;
assert_eq!(env.shared_resolver.read().config.name_servers().to_vec(), vec![]);
}
#[fasync::run_singlethreaded(test)]
async fn test_get_servers() {
let env = TestEnvironment::default();
env.run_admin(|proxy| async move {
let expect = &[NDP_SERVER, DHCP_SERVER, DHCPV6_SERVER, STATIC_SERVER];
let () = proxy
.set_dns_servers(expect)
.await
.expect("FIDL error")
.expect("set_servers failed");
assert_matches!(proxy.get_dns_servers().await, Ok(got) if got == expect);
})
.await;
}
#[fasync::run_singlethreaded(test)]
async fn test_config_inspect() {
let env = TestEnvironment::default();
let inspector = fuchsia_inspect::Inspector::default();
let _config_state_node =
add_config_state_inspect(inspector.root(), env.config_state.clone());
assert_data_tree!(inspector, root:{
servers: {}
});
env.run_admin(|proxy| async move {
let servers = &[NDP_SERVER, DHCP_SERVER, DHCPV6_SERVER, STATIC_SERVER];
let () = proxy
.set_dns_servers(servers)
.await
.expect("FIDL error")
.expect("set_servers failed");
})
.await;
assert_data_tree!(inspector, root:{
servers: {
"0": {
address: "[2001:4860:4860::4444%2]:53",
},
"1": {
address: "8.8.4.4:53",
},
"2": {
address: "[2002:4860:4860::4444%3]:53",
},
"3": {
address: "8.8.8.8:53",
},
}
});
}
#[test]
fn test_unhandled_resolve_error_kind_stats() {
use ResolveErrorKind::{Msg, Timeout};
let mut unhandled_resolve_error_kind_stats = UnhandledResolveErrorKindStats::default();
assert_eq!(
unhandled_resolve_error_kind_stats.increment(&Msg(String::from("abcdefgh"))),
"Msg"
);
assert_eq!(
unhandled_resolve_error_kind_stats.increment(&Msg(String::from("ijklmn"))),
"Msg"
);
assert_eq!(unhandled_resolve_error_kind_stats.increment(&Timeout), "Timeout");
assert_eq!(
unhandled_resolve_error_kind_stats,
UnhandledResolveErrorKindStats {
resolve_error_kind_counts: [(String::from("Msg"), 2), (String::from("Timeout"), 1)]
.into()
}
)
}
#[fasync::run_singlethreaded(test)]
async fn test_query_stats_updated() {
let env = TestEnvironment::default();
let inspector = fuchsia_inspect::Inspector::default();
let _query_stats_inspect_node =
add_query_stats_inspect(inspector.root(), env.stats.clone());
assert_data_tree!(inspector, root:{
query_stats: {}
});
let () = env
.run_lookup(|proxy| async move {
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV4_HOST)]),
..Default::default()
}),
);
})
.await;
let () = env
.run_lookup(|proxy| async move {
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::NotFound),
);
})
.await;
assert_data_tree!(inspector, root:{
query_stats: {
"window 1": {
start_time_nanos: NonZeroUintProperty,
successful_queries: 1u64,
failed_queries: 1u64,
average_success_duration_micros: NonZeroUintProperty,
average_failure_duration_micros: NonZeroUintProperty,
errors: {
Message: 0u64,
NoConnections: 0u64,
NoRecordsFoundResponseCodeCounts: {
NoError: 1u64,
},
Io: 0u64,
Proto: 0u64,
Timeout: 0u64,
UnhandledResolveErrorKindCounts: {},
},
address_counts: {
"1": 1u64,
},
},
}
});
}
fn run_fake_lookup(
exec: &mut fasync::TestExecutor,
stats: Arc<QueryStats>,
result: QueryResult<'_>,
delay: zx::MonotonicDuration,
) {
let start_time = fasync::MonotonicInstant::now();
let () = exec.set_fake_time(fasync::MonotonicInstant::after(delay));
let update_stats = stats.finish_query(start_time, result);
let mut update_stats = pin!(update_stats);
assert!(exec.run_until_stalled(&mut update_stats).is_ready());
}
const NON_ZERO_USIZE_ONE: NonZeroUsize = NonZeroUsize::new(1).unwrap();
#[test]
fn test_query_stats_inspect_average() {
let mut exec = fasync::TestExecutor::new_with_fake_time();
const START_NANOS: i64 = 1_234_567;
let () = exec.set_fake_time(fasync::MonotonicInstant::from_nanos(START_NANOS));
let stats = Arc::new(QueryStats::new());
let inspector = fuchsia_inspect::Inspector::default();
let _query_stats_inspect_node = add_query_stats_inspect(inspector.root(), stats.clone());
const SUCCESSFUL_QUERY_COUNT: u64 = 10;
const SUCCESSFUL_QUERY_DURATION: zx::MonotonicDuration =
zx::MonotonicDuration::from_seconds(30);
for _ in 0..SUCCESSFUL_QUERY_COUNT / 2 {
let () = run_fake_lookup(
&mut exec,
stats.clone(),
Ok(NON_ZERO_USIZE_ONE),
zx::MonotonicDuration::from_nanos(0),
);
let () = run_fake_lookup(
&mut exec,
stats.clone(),
Ok(NON_ZERO_USIZE_ONE),
SUCCESSFUL_QUERY_DURATION,
);
let () = exec.set_fake_time(fasync::MonotonicInstant::after(
STAT_WINDOW_DURATION - SUCCESSFUL_QUERY_DURATION,
));
}
let mut expected = tree_assertion!(query_stats: {});
for i in 0..SUCCESSFUL_QUERY_COUNT / 2 {
let name = &format!("window {}", i + 1);
let child = tree_assertion!(var name: {
start_time_nanos: u64::try_from(
START_NANOS + STAT_WINDOW_DURATION.into_nanos() * i64::try_from(i).unwrap()
).unwrap(),
successful_queries: 2u64,
failed_queries: 0u64,
average_success_duration_micros: u64::try_from(
SUCCESSFUL_QUERY_DURATION.into_micros()
).unwrap() / 2,
errors: {
Message: 0u64,
NoConnections: 0u64,
NoRecordsFoundResponseCodeCounts: {},
Io: 0u64,
Proto: 0u64,
Timeout: 0u64,
UnhandledResolveErrorKindCounts: {},
},
address_counts: {
"1": 2u64,
},
});
expected.add_child_assertion(child);
}
assert_data_tree!(inspector, root: {
expected,
});
}
#[test]
fn test_query_stats_inspect_error_counters() {
let mut exec = fasync::TestExecutor::new_with_fake_time();
const START_NANOS: i64 = 1_234_567;
let () = exec.set_fake_time(fasync::MonotonicInstant::from_nanos(START_NANOS));
let stats = Arc::new(QueryStats::new());
let inspector = fuchsia_inspect::Inspector::default();
let _query_stats_inspect_node = add_query_stats_inspect(inspector.root(), stats.clone());
const FAILED_QUERY_COUNT: u64 = 10;
const FAILED_QUERY_DURATION: zx::MonotonicDuration =
zx::MonotonicDuration::from_millis(500);
for _ in 0..FAILED_QUERY_COUNT {
let () = run_fake_lookup(
&mut exec,
stats.clone(),
Err(&ResolveErrorKind::Timeout),
FAILED_QUERY_DURATION,
);
}
assert_data_tree!(inspector, root:{
query_stats: {
"window 1": {
start_time_nanos: u64::try_from(
START_NANOS + FAILED_QUERY_DURATION.into_nanos()
).unwrap(),
successful_queries: 0u64,
failed_queries: FAILED_QUERY_COUNT,
average_failure_duration_micros: u64::try_from(
FAILED_QUERY_DURATION.into_micros()
).unwrap(),
errors: {
Message: 0u64,
NoConnections: 0u64,
NoRecordsFoundResponseCodeCounts: {},
Io: 0u64,
Proto: 0u64,
Timeout: FAILED_QUERY_COUNT,
UnhandledResolveErrorKindCounts: {},
},
address_counts: {},
},
}
});
}
#[test]
fn test_query_stats_inspect_no_records_found() {
let mut exec = fasync::TestExecutor::new_with_fake_time();
const START_NANOS: i64 = 1_234_567;
let () = exec.set_fake_time(fasync::MonotonicInstant::from_nanos(START_NANOS));
let stats = Arc::new(QueryStats::new());
let inspector = fuchsia_inspect::Inspector::default();
let _query_stats_inspect_node = add_query_stats_inspect(inspector.root(), stats.clone());
const FAILED_QUERY_COUNT: u64 = 10;
const FAILED_QUERY_DURATION: zx::MonotonicDuration =
zx::MonotonicDuration::from_millis(500);
let mut run_fake_no_records_lookup = |response_code: ResponseCode| {
run_fake_lookup(
&mut exec,
stats.clone(),
Err(&ResolveErrorKind::NoRecordsFound {
query: Box::new(Query::default()),
soa: None,
negative_ttl: None,
response_code,
trusted: false,
}),
FAILED_QUERY_DURATION,
)
};
for _ in 0..FAILED_QUERY_COUNT {
let () = run_fake_no_records_lookup(ResponseCode::NXDomain);
let () = run_fake_no_records_lookup(ResponseCode::Refused);
let () = run_fake_no_records_lookup(4096.into());
let () = run_fake_no_records_lookup(4097.into());
}
assert_data_tree!(inspector, root:{
query_stats: {
"window 1": {
start_time_nanos: u64::try_from(
START_NANOS + FAILED_QUERY_DURATION.into_nanos()
).unwrap(),
successful_queries: 0u64,
failed_queries: FAILED_QUERY_COUNT * 4,
average_failure_duration_micros: u64::try_from(
FAILED_QUERY_DURATION.into_micros()
).unwrap(),
errors: {
Message: 0u64,
NoConnections: 0u64,
NoRecordsFoundResponseCodeCounts: {
NXDomain: FAILED_QUERY_COUNT,
Refused: FAILED_QUERY_COUNT,
"Unknown(4096)": FAILED_QUERY_COUNT,
"Unknown(4097)": FAILED_QUERY_COUNT,
},
Io: 0u64,
Proto: 0u64,
Timeout: 0u64,
UnhandledResolveErrorKindCounts: {},
},
address_counts: {},
},
}
});
}
#[test]
fn test_query_stats_resolved_address_counts() {
let mut exec = fasync::TestExecutor::new_with_fake_time();
const START_NANOS: i64 = 1_234_567;
exec.set_fake_time(fasync::MonotonicInstant::from_nanos(START_NANOS));
let stats = Arc::new(QueryStats::new());
let inspector = fuchsia_inspect::Inspector::default();
let _query_stats_inspect_node = add_query_stats_inspect(inspector.root(), stats.clone());
let address_counts: HashMap<usize, _> = (1..100).zip((1..100).rev()).collect();
const QUERY_DURATION: zx::MonotonicDuration = zx::MonotonicDuration::from_millis(10);
for (count, occurrences) in address_counts.iter() {
for _ in 0..*occurrences {
run_fake_lookup(
&mut exec,
stats.clone(),
Ok(NonZeroUsize::new(*count).expect("address count must be greater than zero")),
QUERY_DURATION,
);
}
}
let mut expected_address_counts = tree_assertion!(address_counts: {});
for (count, occurrences) in address_counts.iter() {
expected_address_counts
.add_property_assertion(&count.to_string(), Box::new(*occurrences));
}
assert_data_tree!(inspector, root: {
query_stats: {
"window 1": {
start_time_nanos: u64::try_from(
START_NANOS + QUERY_DURATION.into_nanos()
).unwrap(),
successful_queries: address_counts.values().sum::<u64>(),
failed_queries: 0u64,
average_success_duration_micros: u64::try_from(
QUERY_DURATION.into_micros()
).unwrap(),
errors: {
Message: 0u64,
NoConnections: 0u64,
NoRecordsFoundResponseCodeCounts: {},
Io: 0u64,
Proto: 0u64,
Timeout: 0u64,
UnhandledResolveErrorKindCounts: {},
},
expected_address_counts,
},
},
});
}
#[test]
fn test_query_stats_inspect_oldest_stats_erased() {
let mut exec = fasync::TestExecutor::new_with_fake_time();
const START_NANOS: i64 = 1_234_567;
let () = exec.set_fake_time(fasync::MonotonicInstant::from_nanos(START_NANOS));
let stats = Arc::new(QueryStats::new());
let inspector = fuchsia_inspect::Inspector::default();
let _query_stats_inspect_node = add_query_stats_inspect(inspector.root(), stats.clone());
const DELAY: zx::MonotonicDuration = zx::MonotonicDuration::from_millis(100);
for _ in 0..STAT_WINDOW_COUNT {
let () =
run_fake_lookup(&mut exec, stats.clone(), Err(&ResolveErrorKind::Timeout), DELAY);
let () =
exec.set_fake_time(fasync::MonotonicInstant::after(STAT_WINDOW_DURATION - DELAY));
}
for _ in 0..STAT_WINDOW_COUNT {
let () = run_fake_lookup(
&mut exec,
stats.clone(),
Ok(NON_ZERO_USIZE_ONE),
DELAY,
);
let () =
exec.set_fake_time(fasync::MonotonicInstant::after(STAT_WINDOW_DURATION - DELAY));
}
let mut expected = tree_assertion!(query_stats: {});
let start_offset = START_NANOS
+ DELAY.into_nanos()
+ STAT_WINDOW_DURATION.into_nanos() * i64::try_from(STAT_WINDOW_COUNT).unwrap();
for i in 0..STAT_WINDOW_COUNT {
let name = &format!("window {}", i + 1);
let child = tree_assertion!(var name: {
start_time_nanos: u64::try_from(
start_offset + STAT_WINDOW_DURATION.into_nanos() * i64::try_from(i).unwrap()
).unwrap(),
successful_queries: 1u64,
failed_queries: 0u64,
average_success_duration_micros: u64::try_from(DELAY.into_micros()).unwrap(),
errors: {
Message: 0u64,
NoConnections: 0u64,
NoRecordsFoundResponseCodeCounts: {},
Io: 0u64,
Proto: 0u64,
Timeout: 0u64,
UnhandledResolveErrorKindCounts: {},
},
address_counts: {
"1": 1u64,
},
});
expected.add_child_assertion(child);
}
assert_data_tree!(inspector, root: {
expected,
});
}
struct BlockingResolver {}
#[async_trait]
impl ResolverLookup for BlockingResolver {
fn new(_config: ResolverConfig, _options: ResolverOpts) -> Self {
BlockingResolver {}
}
async fn lookup<N: IntoName + Send>(
&self,
_name: N,
_record_type: RecordType,
) -> Result<lookup::Lookup, ResolveError> {
futures::future::pending().await
}
async fn reverse_lookup(
&self,
_addr: IpAddr,
) -> Result<lookup::ReverseLookup, ResolveError> {
panic!("BlockingResolver does not handle reverse lookup")
}
}
#[fasync::run_singlethreaded(test)]
async fn test_parallel_query_limit() {
let requests = {
let (name_lookup_proxy, name_lookup_stream) =
fidl::endpoints::create_proxy_and_stream::<fname::LookupMarker>();
const NUM_REQUESTS: usize = MAX_PARALLEL_REQUESTS * 2 + 2;
for _ in 0..NUM_REQUESTS {
let _: fidl::client::QueryResponseFut<fname::LookupLookupIpResult> =
name_lookup_proxy.lookup_ip(
LOCAL_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
..Default::default()
},
);
}
drop(name_lookup_proxy);
let requests = name_lookup_stream
.map(|request| match request.expect("channel error") {
LookupRequest::LookupIp { hostname, options, responder } => {
IpLookupRequest { hostname, options, responder }
}
req => panic!("Expected LookupRequest::LookupIp request, found {:?}", req),
})
.collect::<Vec<_>>()
.await;
assert_eq!(requests.len(), NUM_REQUESTS);
requests
};
let (mut sender, recv) = mpsc::channel(MAX_PARALLEL_REQUESTS);
const BEFORE_LAST_INDEX: usize = MAX_PARALLEL_REQUESTS * 2;
const LAST_INDEX: usize = MAX_PARALLEL_REQUESTS * 2 + 1;
let mut send_fut = pin!(async {
for (i, req) in requests.into_iter().enumerate() {
match i {
BEFORE_LAST_INDEX => assert_matches!(sender.try_send(req), Ok(())),
LAST_INDEX => assert_matches!(sender.try_send(req), Err(e) if e.is_full()),
_ => assert_matches!(sender.send(req).await, Ok(())),
}
}
}
.fuse());
let mut recv_fut = pin!({
let resolver = SharedResolver::new(BlockingResolver::new(
ResolverConfig::default(),
ResolverOpts::default(),
));
let stats = Arc::new(QueryStats::new());
let (routes_proxy, _routes_stream) =
fidl::endpoints::create_proxy_and_stream::<fnet_routes::StateMarker>();
async move { create_ip_lookup_fut(&resolver, stats.clone(), routes_proxy, recv).await }
.fuse()
});
futures::select! {
() = send_fut => {},
() = recv_fut => panic!("recv_fut should never complete"),
};
}
#[test]
fn test_failure_stats() {
use anyhow::anyhow;
use trust_dns_proto::error::ProtoError;
use trust_dns_proto::op::Query;
let mut stats = FailureStats::default();
for (error_kind, expected) in &[
(ResolveErrorKind::Message("foo"), FailureStats { message: 1, ..Default::default() }),
(
ResolveErrorKind::Msg("foo".to_string()),
FailureStats { message: 2, ..Default::default() },
),
(
ResolveErrorKind::NoRecordsFound {
query: Box::new(Query::default()),
soa: None,
negative_ttl: None,
response_code: ResponseCode::Refused,
trusted: false,
},
FailureStats {
message: 2,
no_records_found: NoRecordsFoundStats {
response_code_counts: [(ResponseCode::Refused.into(), 1)].into(),
},
..Default::default()
},
),
(
ResolveErrorKind::Io(std::io::Error::new(
std::io::ErrorKind::NotFound,
anyhow!("foo"),
)),
FailureStats {
message: 2,
no_records_found: NoRecordsFoundStats {
response_code_counts: [(ResponseCode::Refused.into(), 1)].into(),
},
io: 1,
..Default::default()
},
),
(
ResolveErrorKind::Proto(ProtoError::from("foo")),
FailureStats {
message: 2,
no_records_found: NoRecordsFoundStats {
response_code_counts: [(ResponseCode::Refused.into(), 1)].into(),
},
io: 1,
proto: 1,
..Default::default()
},
),
(
ResolveErrorKind::NoConnections,
FailureStats {
message: 2,
no_connections: 1,
no_records_found: NoRecordsFoundStats {
response_code_counts: [(ResponseCode::Refused.into(), 1)].into(),
},
io: 1,
proto: 1,
..Default::default()
},
),
(
ResolveErrorKind::Timeout,
FailureStats {
message: 2,
no_connections: 1,
no_records_found: NoRecordsFoundStats {
response_code_counts: [(ResponseCode::Refused.into(), 1)].into(),
},
io: 1,
proto: 1,
timeout: 1,
unhandled_resolve_error_kind: Default::default(),
},
),
(
ResolveErrorKind::NoRecordsFound {
query: Box::new(Query::default()),
soa: None,
negative_ttl: None,
response_code: ResponseCode::NXDomain,
trusted: false,
},
FailureStats {
message: 2,
no_connections: 1,
no_records_found: NoRecordsFoundStats {
response_code_counts: [
(ResponseCode::NXDomain.into(), 1),
(ResponseCode::Refused.into(), 1),
]
.into(),
},
io: 1,
proto: 1,
timeout: 1,
unhandled_resolve_error_kind: Default::default(),
},
),
(
ResolveErrorKind::NoRecordsFound {
query: Box::new(Query::default()),
soa: None,
negative_ttl: None,
response_code: ResponseCode::NXDomain,
trusted: false,
},
FailureStats {
message: 2,
no_connections: 1,
no_records_found: NoRecordsFoundStats {
response_code_counts: [
(ResponseCode::NXDomain.into(), 2),
(ResponseCode::Refused.into(), 1),
]
.into(),
},
io: 1,
proto: 1,
timeout: 1,
unhandled_resolve_error_kind: Default::default(),
},
),
][..]
{
let () = stats.increment(error_kind);
assert_eq!(&stats, expected, "invalid stats after incrementing with {:?}", error_kind);
}
}
fn test_das_helper(
l_addr: fnet::IpAddress,
l_src: Option<fnet::IpAddress>,
r_addr: fnet::IpAddress,
r_src: Option<fnet::IpAddress>,
want: std::cmp::Ordering,
) {
let left = DasCmpInfo::from_addrs(&l_addr, l_src.as_ref());
let right = DasCmpInfo::from_addrs(&r_addr, r_src.as_ref());
assert_eq!(
left.cmp(&right),
want,
"want = {:?}\n left = {:?}({:?}) DAS={:?}\n right = {:?}({:?}) DAS={:?}",
want,
l_addr,
l_src,
left,
r_addr,
r_src,
right
);
}
macro_rules! add_das_test {
($name:ident, preferred: $pref_dst:expr => $pref_src:expr, other: $other_dst:expr => $other_src:expr) => {
#[test]
fn $name() {
test_das_helper(
$pref_dst,
$pref_src,
$other_dst,
$other_src,
std::cmp::Ordering::Less,
)
}
};
}
add_das_test!(
prefer_reachable,
preferred: fidl_ip!("198.51.100.121") => Some(fidl_ip!("198.51.100.117")),
other: fidl_ip!("2001:db8:1::1") => Option::<fnet::IpAddress>::None
);
add_das_test!(
prefer_matching_scope,
preferred: fidl_ip!("198.51.100.121") => Some(fidl_ip!("198.51.100.117")),
other: fidl_ip!("2001:db8:1::1") => Some(fidl_ip!("fe80::1"))
);
add_das_test!(
prefer_matching_label,
preferred: fidl_ip!("2002:c633:6401::1") => Some(fidl_ip!("2002:c633:6401::2")),
other: fidl_ip!("2001:db8:1::1") => Some(fidl_ip!("2002:c633:6401::2"))
);
add_das_test!(
prefer_higher_precedence_1,
preferred: fidl_ip!("2001:db8:1::1") => Some(fidl_ip!("2001:db8:1::2")),
other: fidl_ip!("10.1.2.3") => Some(fidl_ip!("10.1.2.4"))
);
add_das_test!(
prefer_higher_precedence_2,
preferred: fidl_ip!("2001:db8:1::1") => Some(fidl_ip!("2001:db8:1::2")),
other: fidl_ip!("2002:c633:6401::1") => Some(fidl_ip!("2002:c633:6401::2"))
);
add_das_test!(
prefer_smaller_scope,
preferred: fidl_ip!("fe80::1") => Some(fidl_ip!("fe80::2")),
other: fidl_ip!("2001:db8:1::1") => Some(fidl_ip!("2001:db8:1::2"))
);
add_das_test!(
prefer_longest_matching_prefix,
preferred: fidl_ip!("2001:db8:1::1") => Some(fidl_ip!("2001:db8:1::2")),
other: fidl_ip!("2001:db8:3ffe::1") => Some(fidl_ip!("2001:db8:3f44::2"))
);
#[test]
fn test_das_equals() {
for (dst, src) in [
(fidl_ip!("192.168.0.1"), fidl_ip!("192.168.0.2")),
(fidl_ip!("2001:db8::1"), fidl_ip!("2001:db8::2")),
]
.iter()
{
let () = test_das_helper(*dst, None, *dst, None, std::cmp::Ordering::Equal);
let () = test_das_helper(*dst, Some(*src), *dst, Some(*src), std::cmp::Ordering::Equal);
}
}
#[test]
fn test_valid_policy_table() {
assert_eq!(
POLICY_TABLE.iter().last().expect("empty policy table").prefix,
net_types::ip::Subnet::new(net_types::ip::Ipv6::UNSPECIFIED_ADDRESS, 0)
.expect("invalid subnet")
);
let () = POLICY_TABLE.windows(2).for_each(|w| {
let Policy { prefix: cur, precedence: _, label: _ } = w[0];
let Policy { prefix: nxt, precedence: _, label: _ } = w[1];
assert!(
cur.prefix() >= nxt.prefix(),
"bad ordering of prefixes, {} must come after {}",
cur,
nxt
)
});
for policy in POLICY_TABLE.iter() {
assert!(policy.prefix.prefix() <= 128, "Invalid subnet in policy {:?}", policy);
}
}
#[fasync::run_singlethreaded(test)]
async fn test_sort_preferred_addresses() {
const TEST_IPS: [(fnet::IpAddress, Option<fnet::IpAddress>); 5] = [
(fidl_ip!("127.0.0.1"), Some(fidl_ip!("127.0.0.1"))),
(fidl_ip!("::1"), Some(fidl_ip!("::1"))),
(fidl_ip!("192.168.50.22"), None),
(fidl_ip!("2001::2"), None),
(fidl_ip!("2001:db8:1::1"), Some(fidl_ip!("2001:db8:1::2"))),
];
const SORTED: [IpAddr; 5] = [
std_ip!("::1"),
std_ip!("2001:db8:1::1"),
std_ip!("127.0.0.1"),
std_ip!("192.168.50.22"),
std_ip!("2001::2"),
];
let (routes_proxy, routes_stream) =
fidl::endpoints::create_proxy_and_stream::<fnet_routes::StateMarker>();
let routes_fut =
routes_stream.map(|r| r.context("stream FIDL error")).try_for_each(|req| {
let (destination, responder) = assert_matches!(
req,
fnet_routes::StateRequest::Resolve { destination, responder }
=> (destination, responder)
);
let result = TEST_IPS
.iter()
.enumerate()
.find_map(|(i, (dst, src))| {
if *dst == destination && src.is_some() {
let inner = fnet_routes::Destination {
address: Some(*dst),
source_address: *src,
..Default::default()
};
if i % 2 == 0 {
Some(fnet_routes::Resolved::Direct(inner))
} else {
Some(fnet_routes::Resolved::Gateway(inner))
}
} else {
None
}
})
.ok_or(zx::Status::ADDRESS_UNREACHABLE.into_raw());
futures::future::ready(
responder
.send(result.as_ref().map_err(|e| *e))
.context("failed to send Resolve response"),
)
});
let ((), ()) = futures::future::try_join(routes_fut, async move {
let addrs = TEST_IPS.iter().map(|(dst, _src)| *dst).collect();
let addrs = sort_preferred_addresses(addrs, &routes_proxy)
.await
.expect("failed to sort addresses");
let addrs = addrs
.into_iter()
.map(|a| {
let net_ext::IpAddress(a) = a.into();
a
})
.collect::<Vec<_>>();
assert_eq!(&addrs[..], &SORTED[..]);
Ok(())
})
.await
.expect("error running futures");
}
#[fasync::run_singlethreaded(test)]
async fn test_lookupip() {
let routes_handler = |req| {
let (destination, responder) = assert_matches!(
req,
fnet_routes::StateRequest::Resolve { destination, responder }
=> (destination, responder)
);
let resolved;
let response = if destination == map_ip(IPV6_HOST) {
resolved = fnet_routes::Resolved::Direct(fnet_routes::Destination {
address: Some(destination),
source_address: Some(destination),
..Default::default()
});
Ok(&resolved)
} else {
Err(zx::Status::ADDRESS_UNREACHABLE.into_raw())
};
let () = responder.send(response).expect("failed to send Resolve FIDL response");
};
TestEnvironment::default()
.run_lookup_with_routes_handler(
|proxy| async move {
assert_eq!(
proxy
.lookup_ip(REMOTE_IPV4_HOST, &fname::LookupIpOptions::default())
.await
.expect("lookup_ip"),
Err(fname::LookupError::InvalidArgs)
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(false),
ipv6_lookup: Some(false),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::InvalidArgs)
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(false),
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Err(fname::LookupError::NotFound)
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV4_HOST)]),
..Default::default()
})
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_IPV6_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV4_HOST), map_ip(IPV6_HOST)]),
..Default::default()
})
);
assert_eq!(
proxy
.lookup_ip(
REMOTE_IPV4_IPV6_HOST,
&fname::LookupIpOptions {
ipv4_lookup: Some(true),
ipv6_lookup: Some(true),
sort_addresses: Some(true),
..Default::default()
}
)
.await
.expect("lookup_ip"),
Ok(fname::LookupResult {
addresses: Some(vec![map_ip(IPV6_HOST), map_ip(IPV4_HOST)]),
..Default::default()
})
);
},
routes_handler,
)
.await
}
}