use std::collections::hash_map::DefaultHasher;
use std::collections::{HashMap, HashSet};
use std::hash::{Hash, Hasher};
use std::net::{IpAddr, SocketAddr};
use std::ops::Add;
use std::str::FromStr as _;
use std::time::Duration;
use fidl::endpoints::{ControlHandle as _, ServerEnd};
use fidl_fuchsia_net_dhcpv6::{
ClientMarker, ClientRequest, ClientRequestStream, ClientWatchAddressResponder,
ClientWatchPrefixesResponder, ClientWatchServersResponder, Duid, Empty, Lifetimes,
LinkLayerAddress, LinkLayerAddressPlusTime, Prefix, PrefixDelegationConfig,
RELAY_AGENT_AND_SERVER_LINK_LOCAL_MULTICAST_ADDRESS, RELAY_AGENT_AND_SERVER_PORT,
};
use fidl_fuchsia_net_dhcpv6_ext::{
AddressConfig, ClientConfig, InformationConfig, NewClientParams,
};
use futures::future::{AbortHandle, Abortable, Aborted};
use futures::stream::futures_unordered::FuturesUnordered;
use futures::{select, stream, Future, FutureExt as _, StreamExt as _, TryStreamExt as _};
use {
fidl_fuchsia_net as fnet, fidl_fuchsia_net_ext as fnet_ext, fidl_fuchsia_net_name as fnet_name,
fuchsia_async as fasync,
};
use anyhow::{Context as _, Result};
use assert_matches::assert_matches;
use async_utils::futures::{FutureExt as _, ReplaceValue};
use byteorder::{NetworkEndian, WriteBytesExt as _};
use dns_server_watcher::DEFAULT_DNS_PORT;
use log::{debug, error, warn};
use net_types::ip::{Ip as _, Ipv6, Ipv6Addr, Subnet, SubnetError};
use net_types::MulticastAddress as _;
use packet::ParsablePacket;
use packet_formats_dhcp::v6;
use rand::rngs::StdRng;
use rand::SeedableRng;
#[derive(PartialEq, Eq, PartialOrd, Ord, Copy, Clone, Debug)]
pub(crate) struct MonotonicInstant(zx::MonotonicInstant);
impl MonotonicInstant {
fn now() -> MonotonicInstant {
MonotonicInstant(zx::MonotonicInstant::get())
}
}
impl dhcpv6_core::Instant for MonotonicInstant {
fn duration_since(&self, MonotonicInstant(earlier): MonotonicInstant) -> Duration {
let Self(this) = *self;
let diff: zx::MonotonicDuration = this - earlier;
Duration::from_nanos(diff.into_nanos().try_into().unwrap_or_else(|e| {
panic!(
"failed to calculate duration since {:?} with instant {:?}: {}",
earlier, this, e,
)
}))
}
fn checked_add(&self, duration: Duration) -> Option<MonotonicInstant> {
Some(self.add(duration))
}
}
impl Add<Duration> for MonotonicInstant {
type Output = MonotonicInstant;
fn add(self, duration: Duration) -> MonotonicInstant {
let MonotonicInstant(this) = self;
MonotonicInstant(this + duration.into())
}
}
#[derive(Debug, thiserror::Error)]
pub enum ClientError {
#[error("fidl error")]
Fidl(#[source] fidl::Error),
#[error("got watch request while the previous one is pending")]
DoubleWatch,
#[error("unsupported DHCPv6 configuration")]
UnsupportedConfigs,
#[error("socket create error")]
SocketCreate(std::io::Error),
#[error("socket receive error")]
SocketRecv(std::io::Error),
#[error("unimplemented DHCPv6 functionality: {:?}()", _0)]
Unimplemented(String),
}
const MAX_UDP_DATAGRAM_SIZE: usize = 65_535;
type TimerFut = ReplaceValue<fasync::Timer, dhcpv6_core::client::ClientTimerType>;
pub(crate) struct Client<S: for<'a> AsyncSocket<'a>> {
interface_id: u64,
last_observed_dns_hash: u64,
dns_responder: Option<ClientWatchServersResponder>,
address_responder: Option<ClientWatchAddressResponder>,
prefixes: HashMap<fnet::Ipv6AddressWithPrefix, Lifetimes>,
prefixes_changed: bool,
prefixes_responder: Option<ClientWatchPrefixesResponder>,
state_machine: dhcpv6_core::client::ClientStateMachine<MonotonicInstant, StdRng>,
socket: S,
server_addr: SocketAddr,
timer_abort_handles: HashMap<dhcpv6_core::client::ClientTimerType, AbortHandle>,
timer_futs: FuturesUnordered<Abortable<TimerFut>>,
request_stream: ClientRequestStream,
}
pub(crate) trait AsyncSocket<'a> {
type RecvFromFut: Future<Output = Result<(usize, SocketAddr), std::io::Error>> + 'a;
type SendToFut: Future<Output = Result<usize, std::io::Error>> + 'a;
fn recv_from(&'a self, buf: &'a mut [u8]) -> Self::RecvFromFut;
fn send_to(&'a self, buf: &'a [u8], addr: SocketAddr) -> Self::SendToFut;
}
impl<'a> AsyncSocket<'a> for fasync::net::UdpSocket {
type RecvFromFut = fasync::net::UdpRecvFrom<'a>;
type SendToFut = fasync::net::SendTo<'a>;
fn recv_from(&'a self, buf: &'a mut [u8]) -> Self::RecvFromFut {
self.recv_from(buf)
}
fn send_to(&'a self, buf: &'a [u8], addr: SocketAddr) -> Self::SendToFut {
self.send_to(buf, addr)
}
}
fn to_dhcpv6_option_codes(
InformationConfig { dns_servers }: InformationConfig,
) -> Vec<v6::OptionCode> {
dns_servers.then_some(v6::OptionCode::DnsServers).into_iter().collect()
}
fn to_configured_addresses(
AddressConfig { address_count, preferred_addresses }: AddressConfig,
) -> Result<HashMap<v6::IAID, HashSet<Ipv6Addr>>, ClientError> {
let preferred_addresses = preferred_addresses.unwrap_or(Vec::new());
if preferred_addresses.len() > address_count.into() {
return Err(ClientError::UnsupportedConfigs);
}
Ok((0..)
.map(v6::IAID::new)
.zip(
preferred_addresses
.into_iter()
.map(|fnet::Ipv6Address { addr, .. }| HashSet::from([Ipv6Addr::from(addr)]))
.chain(std::iter::repeat_with(HashSet::new)),
)
.take(address_count.into())
.collect())
}
const IA_PD_IAID: v6::IAID = v6::IAID::new(0);
fn create_state_machine(
duid: Option<dhcpv6_core::ClientDuid>,
transaction_id: [u8; 3],
ClientConfig {
information_config,
non_temporary_address_config,
prefix_delegation_config,
}: ClientConfig,
) -> Result<
(
dhcpv6_core::client::ClientStateMachine<MonotonicInstant, StdRng>,
dhcpv6_core::client::Actions<MonotonicInstant>,
),
ClientError,
> {
let information_option_codes = to_dhcpv6_option_codes(information_config);
let configured_non_temporary_addresses = to_configured_addresses(non_temporary_address_config)?;
let configured_delegated_prefixes = prefix_delegation_config
.map(|prefix_delegation_config| {
let prefix = match prefix_delegation_config {
PrefixDelegationConfig::Empty(Empty {}) => Ok(None),
PrefixDelegationConfig::PrefixLength(prefix_len) => {
if prefix_len == 0 {
return Err(ClientError::UnsupportedConfigs);
}
Subnet::new(Ipv6::UNSPECIFIED_ADDRESS, prefix_len).map(Some)
}
PrefixDelegationConfig::Prefix(fnet::Ipv6AddressWithPrefix {
addr: fnet::Ipv6Address { addr, .. },
prefix_len,
}) => {
let addr = Ipv6Addr::from_bytes(addr);
if addr == Ipv6::UNSPECIFIED_ADDRESS {
return Err(ClientError::UnsupportedConfigs);
}
Subnet::new(addr, prefix_len).map(Some)
}
};
match prefix {
Ok(o) => Ok(HashMap::from([(IA_PD_IAID, HashSet::from_iter(o.into_iter()))])),
Err(SubnetError::PrefixTooLong | SubnetError::HostBitsSet) => {
Err(ClientError::UnsupportedConfigs)
}
}
})
.transpose()?;
let now = MonotonicInstant::now();
match (
information_option_codes.is_empty(),
configured_non_temporary_addresses.is_empty(),
configured_delegated_prefixes,
) {
(true, true, None) => Err(ClientError::UnsupportedConfigs),
(false, true, None) => {
if duid.is_some() {
Err(ClientError::UnsupportedConfigs)
} else {
Ok(dhcpv6_core::client::ClientStateMachine::start_stateless(
transaction_id,
information_option_codes,
StdRng::from_entropy(),
now,
))
}
}
(
_request_information,
_configure_non_temporary_addresses,
configured_delegated_prefixes,
) => Ok(dhcpv6_core::client::ClientStateMachine::start_stateful(
transaction_id,
if let Some(duid) = duid {
duid
} else {
return Err(ClientError::UnsupportedConfigs);
},
configured_non_temporary_addresses,
configured_delegated_prefixes.unwrap_or_else(Default::default),
information_option_codes,
StdRng::from_entropy(),
now,
)),
}
}
fn hash<H: Hash>(h: &H) -> u64 {
let mut dh = DefaultHasher::new();
let () = h.hash(&mut dh);
dh.finish()
}
fn subnet_to_address_with_prefix(prefix: Subnet<Ipv6Addr>) -> fnet::Ipv6AddressWithPrefix {
fnet::Ipv6AddressWithPrefix {
addr: fnet::Ipv6Address { addr: prefix.network().ipv6_bytes() },
prefix_len: prefix.prefix(),
}
}
impl<S: for<'a> AsyncSocket<'a>> Client<S> {
pub(crate) async fn start(
duid: Option<dhcpv6_core::ClientDuid>,
transaction_id: [u8; 3],
config: ClientConfig,
interface_id: u64,
socket_fn: impl FnOnce() -> std::io::Result<S>,
server_addr: SocketAddr,
request_stream: ClientRequestStream,
) -> Result<Self, ClientError> {
let (state_machine, actions) = create_state_machine(duid, transaction_id, config)?;
let mut client = Self {
state_machine,
interface_id,
socket: socket_fn().map_err(ClientError::SocketCreate)?,
server_addr,
request_stream,
timer_abort_handles: HashMap::new(),
timer_futs: FuturesUnordered::new(),
last_observed_dns_hash: hash(&Vec::<Ipv6Addr>::new()),
dns_responder: None,
address_responder: None,
prefixes: Default::default(),
prefixes_changed: false,
prefixes_responder: None,
};
let () = client.run_actions(actions).await?;
Ok(client)
}
async fn run_actions(
&mut self,
actions: dhcpv6_core::client::Actions<MonotonicInstant>,
) -> Result<(), ClientError> {
stream::iter(actions)
.map(Ok)
.try_fold(self, |client, action| async move {
match action {
dhcpv6_core::client::Action::SendMessage(buf) => {
let () = match client.socket.send_to(&buf, client.server_addr).await {
Ok(size) => assert_eq!(size, buf.len()),
Err(e) => warn!(
"failed to send message to {}: {}; will retransmit later",
client.server_addr, e
),
};
}
dhcpv6_core::client::Action::ScheduleTimer(timer_type, timeout) => {
client.schedule_timer(timer_type, timeout)
}
dhcpv6_core::client::Action::CancelTimer(timer_type) => {
client.cancel_timer(timer_type)
}
dhcpv6_core::client::Action::UpdateDnsServers(servers) => {
let () = client.maybe_send_dns_server_updates(servers)?;
}
dhcpv6_core::client::Action::IaNaUpdates(_) => {
}
dhcpv6_core::client::Action::IaPdUpdates(mut updates) => {
let updates = {
let ret =
updates.remove(&IA_PD_IAID).expect("Update missing for IAID");
debug_assert_eq!(updates, HashMap::new());
ret
};
let Self { prefixes, prefixes_changed, .. } = client;
let now = zx::MonotonicInstant::get();
let nonzero_timevalue_to_zx_time = |tv| match tv {
v6::NonZeroTimeValue::Finite(tv) => {
now + zx::MonotonicDuration::from_seconds(tv.get().into())
}
v6::NonZeroTimeValue::Infinity => zx::MonotonicInstant::INFINITE,
};
let calculate_lifetimes = |dhcpv6_core::client::Lifetimes {
preferred_lifetime,
valid_lifetime,
}| {
Lifetimes {
preferred_until: match preferred_lifetime {
v6::TimeValue::Zero => zx::MonotonicInstant::ZERO,
v6::TimeValue::NonZero(preferred_lifetime) => {
nonzero_timevalue_to_zx_time(preferred_lifetime)
},
}.into_nanos(),
valid_until: nonzero_timevalue_to_zx_time(valid_lifetime)
.into_nanos(),
}
};
for (prefix, update) in updates.into_iter() {
let fidl_prefix = subnet_to_address_with_prefix(prefix);
match update {
dhcpv6_core::client::IaValueUpdateKind::Added(lifetimes) => {
assert_matches!(
prefixes.insert(
fidl_prefix,
calculate_lifetimes(lifetimes)
),
None,
"must not know about prefix {} to add it with lifetimes {:?}",
prefix, lifetimes,
);
}
dhcpv6_core::client::IaValueUpdateKind::UpdatedLifetimes(updated_lifetimes) => {
assert_matches!(
prefixes.get_mut(&fidl_prefix),
Some(lifetimes) => {
*lifetimes = calculate_lifetimes(updated_lifetimes);
},
"must know about prefix {} to update lifetimes with {:?}",
prefix, updated_lifetimes,
);
}
dhcpv6_core::client::IaValueUpdateKind::Removed => {
assert_matches!(
prefixes.remove(&fidl_prefix),
Some(_),
"must know about prefix {} to remove it",
prefix
);
}
}
}
*prefixes_changed = true;
client.maybe_send_prefixes()?;
}
};
Ok(client)
})
.await
.map(|_: &mut Client<S>| ())
}
fn maybe_send_dns_server_updates(&mut self, servers: Vec<Ipv6Addr>) -> Result<(), ClientError> {
let servers_hash = hash(&servers);
if servers_hash == self.last_observed_dns_hash {
Ok(())
} else {
Ok(match self.dns_responder.take() {
Some(responder) => {
self.send_dns_server_updates(responder, servers, servers_hash)?
}
None => (),
})
}
}
fn maybe_send_prefixes(&mut self) -> Result<(), ClientError> {
let Self { prefixes, prefixes_changed, prefixes_responder, .. } = self;
if !*prefixes_changed {
return Ok(());
}
let responder = if let Some(responder) = prefixes_responder.take() {
responder
} else {
return Ok(());
};
let prefixes = prefixes
.iter()
.map(|(prefix, lifetimes)| Prefix { prefix: *prefix, lifetimes: *lifetimes })
.collect::<Vec<_>>();
responder.send(&prefixes).map_err(ClientError::Fidl)?;
*prefixes_changed = false;
Ok(())
}
fn send_dns_server_updates(
&mut self,
responder: ClientWatchServersResponder,
servers: Vec<Ipv6Addr>,
hash: u64,
) -> Result<(), ClientError> {
let response: Vec<_> = servers
.iter()
.map(|addr| {
let address = fnet::Ipv6Address { addr: addr.ipv6_bytes() };
let zone_index =
if is_unicast_link_local_strict(&address) { self.interface_id } else { 0 };
fnet_name::DnsServer_ {
address: Some(fnet::SocketAddress::Ipv6(fnet::Ipv6SocketAddress {
address,
zone_index,
port: DEFAULT_DNS_PORT,
})),
source: Some(fnet_name::DnsServerSource::Dhcpv6(
fnet_name::Dhcpv6DnsServerSource {
source_interface: Some(self.interface_id),
..Default::default()
},
)),
..Default::default()
}
})
.collect();
let () = responder
.send(&response)
.map_err(ClientError::Fidl)?;
self.last_observed_dns_hash = hash;
Ok(())
}
fn schedule_timer(
&mut self,
timer_type: dhcpv6_core::client::ClientTimerType,
MonotonicInstant(instant): MonotonicInstant,
) {
let (handle, reg) = AbortHandle::new_pair();
match self.timer_abort_handles.insert(timer_type, handle) {
Some(handle) => handle.abort(),
None => {}
}
self.timer_futs.push(Abortable::new(
fasync::Timer::new(fasync::MonotonicInstant::from_zx(instant))
.replace_value(timer_type),
reg,
))
}
fn cancel_timer(&mut self, timer_type: dhcpv6_core::client::ClientTimerType) {
match self.timer_abort_handles.remove(&timer_type) {
Some(handle) => handle.abort(),
None => {}
}
}
async fn handle_timeout(
&mut self,
timer_type: dhcpv6_core::client::ClientTimerType,
) -> Result<(), ClientError> {
self.cancel_timer(timer_type);
let actions = self.state_machine.handle_timeout(timer_type, MonotonicInstant::now());
self.run_actions(actions).await
}
async fn handle_message_recv(&mut self, mut msg: &[u8]) -> Result<(), ClientError> {
let msg = match v6::Message::parse(&mut msg, ()) {
Ok(msg) => msg,
Err(e) => {
warn!("failed to parse received message: {}", e);
return Ok(());
}
};
let actions = self.state_machine.handle_message_receive(msg, MonotonicInstant::now());
self.run_actions(actions).await
}
fn handle_client_request(&mut self, request: ClientRequest) -> Result<(), ClientError> {
debug!("handling client request: {:?}", request);
match request {
ClientRequest::WatchServers { responder } => match self.dns_responder {
Some(_) => {
self.dns_responder = None;
Err(ClientError::DoubleWatch)
}
None => {
let dns_servers = self.state_machine.get_dns_servers();
let servers_hash = hash(&dns_servers);
if servers_hash != self.last_observed_dns_hash {
let () =
self.send_dns_server_updates(responder, dns_servers, servers_hash)?;
} else {
self.dns_responder = Some(responder);
}
Ok(())
}
},
ClientRequest::WatchAddress { responder } => match self.address_responder.take() {
Some(ClientWatchAddressResponder { .. }) => Err(ClientError::DoubleWatch),
None => {
warn!("WatchAddress call will block forever as it is unimplemented");
self.address_responder = Some(responder);
Ok(())
}
},
ClientRequest::WatchPrefixes { responder } => match self.prefixes_responder.take() {
Some(ClientWatchPrefixesResponder { .. }) => Err(ClientError::DoubleWatch),
None => {
self.prefixes_responder = Some(responder);
self.maybe_send_prefixes()
}
},
ClientRequest::Shutdown { responder: _ } => {
Err(ClientError::Unimplemented("Shutdown".to_string()))
}
}
}
async fn handle_next_event(&mut self, buf: &mut [u8]) -> Result<Option<()>, ClientError> {
select! {
timer_res = self.timer_futs.select_next_some() => {
match timer_res {
Ok(timer_type) => {
let () = self.handle_timeout(timer_type).await?;
Ok(Some(()))
},
Err(Aborted) => {
debug!("timer aborted");
Ok(Some(()))
}
}
},
recv_from_res = self.socket.recv_from(buf).fuse() => {
let (size, _addr) = recv_from_res.map_err(ClientError::SocketRecv)?;
let () = self.handle_message_recv(&buf[..size]).await?;
Ok(Some(()))
},
request = self.request_stream.try_next() => {
match request {
Ok(request) => {
request.map(|request| self.handle_client_request(request)).transpose()
}
Err(e) => {
Err(ClientError::Fidl(e))
}
}
}
}
}
}
fn create_socket(addr: SocketAddr) -> std::io::Result<fasync::net::UdpSocket> {
let socket = socket2::Socket::new(
socket2::Domain::IPV6,
socket2::Type::DGRAM,
Some(socket2::Protocol::UDP),
)?;
let () = socket.set_reuse_port(true)?;
let () = socket.bind(&addr.into())?;
fasync::net::UdpSocket::from_socket(socket.into())
}
fn is_unicast_link_local_strict(addr: &fnet::Ipv6Address) -> bool {
addr.addr[..8] == [0xfe, 0x80, 0, 0, 0, 0, 0, 0]
}
fn duid_from_fidl(duid: Duid) -> Result<dhcpv6_core::ClientDuid, ()> {
const DUID_TYPE_LLT: [u8; 2] = [0, 1];
const DUID_TYPE_LL: [u8; 2] = [0, 3];
const DUID_TYPE_UUID: [u8; 2] = [0, 4];
const HARDWARE_TYPE_ETHERNET: [u8; 2] = [0, 1];
match duid {
Duid::LinkLayerAddressPlusTime(LinkLayerAddressPlusTime {
time,
link_layer_address: LinkLayerAddress::Ethernet(mac),
}) => {
let mut duid = dhcpv6_core::ClientDuid::new();
duid.try_extend_from_slice(&DUID_TYPE_LLT).unwrap();
duid.try_extend_from_slice(&HARDWARE_TYPE_ETHERNET).unwrap();
duid.write_u32::<NetworkEndian>(time).unwrap();
duid.try_extend_from_slice(&mac.octets).unwrap();
Ok(duid)
}
Duid::LinkLayerAddress(LinkLayerAddress::Ethernet(mac)) => Ok(DUID_TYPE_LL
.into_iter()
.chain(HARDWARE_TYPE_ETHERNET.into_iter())
.chain(mac.octets.into_iter())
.collect()),
Duid::Uuid(uuid) => Ok(DUID_TYPE_UUID.into_iter().chain(uuid.into_iter()).collect()),
_ => Err(()),
}
}
pub(crate) async fn serve_client(
NewClientParams { interface_id, address, duid, config }: NewClientParams,
request: ServerEnd<ClientMarker>,
) -> Result<()> {
if Ipv6Addr::from(address.address.addr).is_multicast()
|| (is_unicast_link_local_strict(&address.address) && address.zone_index != interface_id)
{
return request
.close_with_epitaph(zx::Status::INVALID_ARGS)
.context("closing request channel with epitaph");
}
let fnet_ext::SocketAddress(addr) = fnet::SocketAddress::Ipv6(address).into();
let servers_addr = IpAddr::from_str(RELAY_AGENT_AND_SERVER_LINK_LOCAL_MULTICAST_ADDRESS)
.with_context(|| {
format!(
"{} should be a valid IPv6 address",
RELAY_AGENT_AND_SERVER_LINK_LOCAL_MULTICAST_ADDRESS,
)
})?;
let duid = match duid.map(|fidl| duid_from_fidl(fidl)).transpose() {
Ok(duid) => duid,
Err(()) => {
return request
.close_with_epitaph(zx::Status::INVALID_ARGS)
.context("closing request channel with epitaph")
}
};
let (request_stream, control_handle) = request.into_stream_and_control_handle();
let mut client = match Client::<fasync::net::UdpSocket>::start(
duid,
dhcpv6_core::client::transaction_id(),
config,
interface_id,
|| create_socket(addr),
SocketAddr::new(servers_addr, RELAY_AGENT_AND_SERVER_PORT),
request_stream,
)
.await
{
Ok(client) => client,
Err(ClientError::UnsupportedConfigs) => {
control_handle.shutdown_with_epitaph(zx::Status::INVALID_ARGS);
return Ok(());
}
Err(e) => {
return Err(e.into());
}
};
let mut buf = vec![0u8; MAX_UDP_DATAGRAM_SIZE];
loop {
match client.handle_next_event(&mut buf).await? {
Some(()) => (),
None => break Ok(()),
}
}
}
#[cfg(test)]
mod tests {
use std::pin::{pin, Pin};
use std::task::Poll;
use fidl::endpoints::{
create_proxy, create_proxy_and_stream, create_request_stream, ClientEnd,
};
use fidl_fuchsia_net_dhcpv6::{self as fnet_dhcpv6, ClientProxy, DEFAULT_CLIENT_PORT};
use fuchsia_async as fasync;
use futures::{join, poll, TryFutureExt as _};
use assert_matches::assert_matches;
use net_declare::{
fidl_ip_v6, fidl_ip_v6_with_prefix, fidl_mac, fidl_socket_addr, fidl_socket_addr_v6,
net_ip_v6, net_subnet_v6, std_socket_addr,
};
use net_types::ip::IpAddress as _;
use packet::serialize::InnerPacketBuilder;
use test_case::test_case;
use super::*;
fn create_test_socket() -> (fasync::net::UdpSocket, SocketAddr) {
let addr: SocketAddr = std_socket_addr!("[::1]:0");
let socket = std::net::UdpSocket::bind(addr).expect("failed to create test socket");
let addr = socket.local_addr().expect("failed to get address of test socket");
(fasync::net::UdpSocket::from_socket(socket).expect("failed to create test socket"), addr)
}
struct ReceivedMessage {
transaction_id: [u8; 3],
client_id: Option<Vec<u8>>,
}
async fn assert_received_message(
socket: &fasync::net::UdpSocket,
want_from_addr: SocketAddr,
msg_type: v6::MessageType,
) -> ReceivedMessage {
let mut buf = vec![0u8; MAX_UDP_DATAGRAM_SIZE];
let (size, from_addr) =
socket.recv_from(&mut buf).await.expect("failed to receive on test server socket");
assert_eq!(from_addr, want_from_addr);
let buf = &mut &buf[..size]; let msg = v6::Message::parse(buf, ()).expect("failed to parse message");
assert_eq!(msg.msg_type(), msg_type);
let mut client_id = None;
for opt in msg.options() {
match opt {
v6::ParsedDhcpOption::ClientId(id) => {
assert_eq!(core::mem::replace(&mut client_id, Some(id.to_vec())), None)
}
_ => {}
}
}
ReceivedMessage { transaction_id: *msg.transaction_id(), client_id: client_id }
}
const TEST_MAC: fnet::MacAddress = fidl_mac!("00:01:02:03:04:05");
#[test_case(
Duid::LinkLayerAddress(LinkLayerAddress::Ethernet(TEST_MAC)),
&[0, 3, 0, 1, 0, 1, 2, 3, 4, 5];
"ll"
)]
#[test_case(
Duid::LinkLayerAddressPlusTime(LinkLayerAddressPlusTime {
time: 0,
link_layer_address: LinkLayerAddress::Ethernet(TEST_MAC),
}),
&[0, 1, 0, 1, 0, 0, 0, 0, 0, 1, 2, 3, 4, 5];
"llt"
)]
#[test_case(
Duid::Uuid([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]),
&[0, 4, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15];
"uuid"
)]
#[fuchsia::test]
fn test_duid_from_fidl(duid: Duid, want: &[u8]) {
assert_eq!(duid_from_fidl(duid), Ok(dhcpv6_core::ClientDuid::try_from(want).unwrap()));
}
#[fuchsia::test]
fn test_create_client_with_unsupported_config() {
let prefix_delegation_configs = [
None,
Some(PrefixDelegationConfig::PrefixLength(0)),
Some(PrefixDelegationConfig::PrefixLength(Ipv6Addr::BYTES * 8 + 1)),
Some(PrefixDelegationConfig::Prefix(fidl_ip_v6_with_prefix!("::/64"))),
Some(PrefixDelegationConfig::Prefix(fidl_ip_v6_with_prefix!("a::1/64"))),
];
for prefix_delegation_config in prefix_delegation_configs.iter() {
assert_matches!(
create_state_machine(
prefix_delegation_config.is_some().then(|| CLIENT_ID.into()),
[1, 2, 3],
ClientConfig {
information_config: Default::default(),
non_temporary_address_config: Default::default(),
prefix_delegation_config: prefix_delegation_config.clone(),
}
),
Err(ClientError::UnsupportedConfigs),
"prefix_delegation_config={:?}",
prefix_delegation_config
);
}
}
const STATELESS_CLIENT_CONFIG: ClientConfig = ClientConfig {
information_config: InformationConfig { dns_servers: true },
non_temporary_address_config: AddressConfig { address_count: 0, preferred_addresses: None },
prefix_delegation_config: None,
};
#[fuchsia::test]
async fn test_client_stops_on_channel_close() {
let (client_proxy, server_end) = create_proxy::<ClientMarker>();
let ((), client_res) = join!(
async { drop(client_proxy) },
serve_client(
NewClientParams {
interface_id: 1,
address: fidl_socket_addr_v6!("[::1]:546"),
config: STATELESS_CLIENT_CONFIG,
duid: None,
},
server_end,
),
);
client_res.expect("client future should return with Ok");
}
fn client_proxy_watch_servers(
client_proxy: &fnet_dhcpv6::ClientProxy,
) -> impl Future<Output = Result<(), fidl::Error>> {
client_proxy.watch_servers().map_ok(|_: Vec<fidl_fuchsia_net_name::DnsServer_>| ())
}
fn client_proxy_watch_address(
client_proxy: &fnet_dhcpv6::ClientProxy,
) -> impl Future<Output = Result<(), fidl::Error>> {
client_proxy.watch_address().map_ok(
|_: (
fnet::Subnet,
fidl_fuchsia_net_interfaces_admin::AddressParameters,
fidl::endpoints::ServerEnd<
fidl_fuchsia_net_interfaces_admin::AddressStateProviderMarker,
>,
)| (),
)
}
fn client_proxy_watch_prefixes(
client_proxy: &fnet_dhcpv6::ClientProxy,
) -> impl Future<Output = Result<(), fidl::Error>> {
client_proxy.watch_prefixes().map_ok(|_: Vec<fnet_dhcpv6::Prefix>| ())
}
#[test_case(client_proxy_watch_servers; "watch_servers")]
#[test_case(client_proxy_watch_address; "watch_address")]
#[test_case(client_proxy_watch_prefixes; "watch_prefixes")]
#[fuchsia::test]
async fn test_client_should_return_error_on_double_watch<Fut, F>(watch: F)
where
Fut: Future<Output = Result<(), fidl::Error>>,
F: Fn(&fnet_dhcpv6::ClientProxy) -> Fut,
{
let (client_proxy, server_end) = create_proxy::<ClientMarker>();
let (caller1_res, caller2_res, client_res) = join!(
watch(&client_proxy),
watch(&client_proxy),
serve_client(
NewClientParams {
interface_id: 1,
address: fidl_socket_addr_v6!("[::1]:546"),
config: STATELESS_CLIENT_CONFIG,
duid: None,
},
server_end,
)
);
assert_matches!(
caller1_res,
Err(fidl::Error::ClientChannelClosed { status: zx::Status::PEER_CLOSED, .. })
);
assert_matches!(
caller2_res,
Err(fidl::Error::ClientChannelClosed { status: zx::Status::PEER_CLOSED, .. })
);
assert!(client_res
.expect_err("client should fail with double watch error")
.to_string()
.contains("got watch request while the previous one is pending"));
}
const VALID_INFORMATION_CONFIGS: [InformationConfig; 2] =
[InformationConfig { dns_servers: false }, InformationConfig { dns_servers: true }];
const VALID_DELEGATED_PREFIX_CONFIGS: [Option<PrefixDelegationConfig>; 4] = [
Some(PrefixDelegationConfig::Empty(Empty {})),
Some(PrefixDelegationConfig::PrefixLength(1)),
Some(PrefixDelegationConfig::PrefixLength(127)),
Some(PrefixDelegationConfig::Prefix(fidl_ip_v6_with_prefix!("a::/64"))),
];
fn get_valid_non_temporary_address_configs() -> [AddressConfig; 5] {
[
Default::default(),
AddressConfig { address_count: 1, preferred_addresses: None },
AddressConfig { address_count: 1, preferred_addresses: Some(Vec::new()) },
AddressConfig {
address_count: 1,
preferred_addresses: Some(vec![fidl_ip_v6!("a::1")]),
},
AddressConfig {
address_count: 2,
preferred_addresses: Some(vec![fidl_ip_v6!("a::2")]),
},
]
}
#[fuchsia::test]
fn test_client_starts_with_valid_args() {
for information_config in VALID_INFORMATION_CONFIGS {
for non_temporary_address_config in get_valid_non_temporary_address_configs() {
for prefix_delegation_config in VALID_DELEGATED_PREFIX_CONFIGS {
let mut exec = fasync::TestExecutor::new();
let (client_proxy, server_end) = create_proxy::<ClientMarker>();
let test_fut = async {
join!(
client_proxy.watch_servers(),
serve_client(
NewClientParams {
interface_id: 1,
address: fidl_socket_addr_v6!("[::1]:546"),
config: ClientConfig {
information_config: information_config.clone(),
non_temporary_address_config: non_temporary_address_config
.clone(),
prefix_delegation_config: prefix_delegation_config.clone(),
},
duid: (non_temporary_address_config.address_count != 0
|| prefix_delegation_config.is_some())
.then(|| fnet_dhcpv6::Duid::LinkLayerAddress(
fnet_dhcpv6::LinkLayerAddress::Ethernet(fidl_mac!(
"00:11:22:33:44:55"
))
)),
},
server_end
)
)
};
let mut test_fut = pin!(test_fut);
assert_matches!(
exec.run_until_stalled(&mut test_fut),
Poll::Pending,
"information_config={:?}, non_temporary_address_config={:?}, prefix_delegation_config={:?}",
information_config, non_temporary_address_config, prefix_delegation_config
);
}
}
}
}
const CLIENT_ID: [u8; 18] = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17];
#[fuchsia::test]
async fn test_client_starts_in_correct_mode() {
for information_config @ InformationConfig { dns_servers } in VALID_INFORMATION_CONFIGS {
for non_temporary_address_config @ AddressConfig {
address_count,
preferred_addresses: _,
} in get_valid_non_temporary_address_configs()
{
for prefix_delegation_config in VALID_DELEGATED_PREFIX_CONFIGS {
let (stateful, want_msg_type) =
if address_count == 0 && prefix_delegation_config.is_none() {
if !dns_servers {
continue;
} else {
(false, v6::MessageType::InformationRequest)
}
} else {
(true, v6::MessageType::Solicit)
};
let (_, client_stream): (ClientEnd<ClientMarker>, _) =
create_request_stream::<ClientMarker>();
let (client_socket, client_addr) = create_test_socket();
let (server_socket, server_addr) = create_test_socket();
println!(
"{:?} {:?} {:?}",
information_config, non_temporary_address_config, prefix_delegation_config
);
let _: Client<fasync::net::UdpSocket> = Client::start(
stateful.then(|| CLIENT_ID.into()),
[1, 2, 3], ClientConfig {
information_config: information_config.clone(),
non_temporary_address_config: non_temporary_address_config.clone(),
prefix_delegation_config: prefix_delegation_config.clone(),
},
1, || Ok(client_socket),
server_addr,
client_stream,
)
.await
.unwrap_or_else(|e| panic!(
"failed to create test client: {}; information_config={:?}, non_temporary_address_config={:?}, prefix_delegation_config={:?}",
e, information_config, non_temporary_address_config, prefix_delegation_config
));
let _: ReceivedMessage =
assert_received_message(&server_socket, client_addr, want_msg_type).await;
}
}
}
}
#[fuchsia::test]
async fn test_client_fails_to_start_with_invalid_args() {
for params in vec![
NewClientParams {
interface_id: 2,
address: fnet::Ipv6SocketAddress {
address: fidl_ip_v6!("fe80::1"),
port: DEFAULT_CLIENT_PORT,
zone_index: 1,
},
config: STATELESS_CLIENT_CONFIG,
duid: None,
},
NewClientParams {
interface_id: 1,
address: fnet::Ipv6SocketAddress {
address: fidl_ip_v6!("ff01::1"),
port: DEFAULT_CLIENT_PORT,
zone_index: 1,
},
config: STATELESS_CLIENT_CONFIG,
duid: None,
},
NewClientParams {
interface_id: 1,
address: fidl_socket_addr_v6!("[2001:db8::1]:12345"),
config: STATELESS_CLIENT_CONFIG,
duid: Some(fnet_dhcpv6::Duid::LinkLayerAddress(
fnet_dhcpv6::LinkLayerAddress::Ethernet(fidl_mac!("00:11:22:33:44:55")),
)),
},
NewClientParams {
interface_id: 1,
address: fidl_socket_addr_v6!("[2001:db8::1]:12345"),
config: ClientConfig {
information_config: InformationConfig { dns_servers: true },
non_temporary_address_config: AddressConfig {
address_count: 1,
preferred_addresses: None,
},
prefix_delegation_config: None,
},
duid: None,
},
] {
let (client_proxy, server_end) = create_proxy::<ClientMarker>();
let () =
serve_client(params, server_end).await.expect("start server failed unexpectedly");
assert_matches!(
client_proxy.watch_servers().await,
Err(fidl::Error::ClientChannelClosed { status: zx::Status::INVALID_ARGS, .. })
);
}
}
#[test]
fn test_is_unicast_link_local_strict() {
assert_eq!(is_unicast_link_local_strict(&fidl_ip_v6!("fe80::")), true);
assert_eq!(is_unicast_link_local_strict(&fidl_ip_v6!("fe80::1")), true);
assert_eq!(is_unicast_link_local_strict(&fidl_ip_v6!("fe80::ffff:1:2:3")), true);
assert_eq!(is_unicast_link_local_strict(&fidl_ip_v6!("fe80::1:0:0:0:0")), false);
assert_eq!(is_unicast_link_local_strict(&fidl_ip_v6!("fe81::")), false);
}
fn create_test_dns_server(
address: fnet::Ipv6Address,
source_interface: u64,
zone_index: u64,
) -> fnet_name::DnsServer_ {
fnet_name::DnsServer_ {
address: Some(fnet::SocketAddress::Ipv6(fnet::Ipv6SocketAddress {
address,
zone_index,
port: DEFAULT_DNS_PORT,
})),
source: Some(fnet_name::DnsServerSource::Dhcpv6(fnet_name::Dhcpv6DnsServerSource {
source_interface: Some(source_interface),
..Default::default()
})),
..Default::default()
}
}
async fn send_msg_with_options(
socket: &fasync::net::UdpSocket,
to_addr: SocketAddr,
transaction_id: [u8; 3],
msg_type: v6::MessageType,
options: &[v6::DhcpOption<'_>],
) -> Result<()> {
let builder = v6::MessageBuilder::new(msg_type, transaction_id, options);
let mut buf = vec![0u8; builder.bytes_len()];
let () = builder.serialize(&mut buf);
let size = socket.send_to(&buf, to_addr).await?;
assert_eq!(size, buf.len());
Ok(())
}
#[fuchsia::test]
fn test_client_should_respond_to_dns_watch_requests() {
let mut exec = fasync::TestExecutor::new();
let transaction_id = [1, 2, 3];
let (client_proxy, client_stream) = create_proxy_and_stream::<ClientMarker>();
let (client_socket, client_addr) = create_test_socket();
let (server_socket, server_addr) = create_test_socket();
let mut client = exec
.run_singlethreaded(Client::<fasync::net::UdpSocket>::start(
None,
transaction_id,
STATELESS_CLIENT_CONFIG,
1, || Ok(client_socket),
server_addr,
client_stream,
))
.expect("failed to create test client");
type WatchServersResponseFut = <fnet_dhcpv6::ClientProxy as fnet_dhcpv6::ClientProxyInterface>::WatchServersResponseFut;
type WatchServersResponse = <WatchServersResponseFut as Future>::Output;
struct Test<'a> {
client: &'a mut Client<fasync::net::UdpSocket>,
buf: Vec<u8>,
watcher_fut: WatchServersResponseFut,
}
impl<'a> Test<'a> {
fn new(
client: &'a mut Client<fasync::net::UdpSocket>,
client_proxy: &ClientProxy,
) -> Self {
Self {
client,
buf: vec![0u8; MAX_UDP_DATAGRAM_SIZE],
watcher_fut: client_proxy.watch_servers(),
}
}
async fn handle_next_event(&mut self) {
self.client
.handle_next_event(&mut self.buf)
.await
.expect("test client failed to handle next event")
.expect("request stream closed");
}
async fn refresh_client(&mut self) {
if self
.client
.timer_abort_handles
.contains_key(&dhcpv6_core::client::ClientTimerType::Refresh)
{
self.client
.handle_timeout(dhcpv6_core::client::ClientTimerType::Refresh)
.await
.expect("test client failed to handle timeout");
} else {
panic!("no refresh timer is scheduled and refresh is requested in test");
}
self.handle_next_event().await;
}
fn run(&mut self) -> Pin<Box<dyn Future<Output = WatchServersResponse> + '_>> {
let Self { client, buf, watcher_fut } = self;
Box::pin(async move {
let client_fut = async {
loop {
client
.handle_next_event(buf)
.await
.expect("test client failed to handle next event")
.expect("request stream closed");
}
}
.fuse();
let mut client_fut = pin!(client_fut);
let mut watcher_fut = watcher_fut.fuse();
select! {
() = client_fut => panic!("test client returned unexpectedly"),
r = watcher_fut => r,
}
})
}
}
{
let mut test = Test::new(&mut client, &client_proxy);
exec.run_singlethreaded(test.handle_next_event());
assert!(
test.client.dns_responder.is_some(),
"WatchServers responder should be present"
);
let () = exec
.run_singlethreaded(send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[v6::DhcpOption::ServerId(&[1, 2, 3]), v6::DhcpOption::DnsServers(&[])],
))
.expect("failed to send test reply");
exec.run_singlethreaded(test.handle_next_event());
assert_matches!(exec.run_until_stalled(&mut test.run()), Poll::Pending);
exec.run_singlethreaded(test.refresh_client());
let dns_servers = [net_ip_v6!("fe80::1:2")];
let () = exec
.run_singlethreaded(send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[
v6::DhcpOption::ServerId(&[1, 2, 3]),
v6::DhcpOption::DnsServers(&dns_servers),
],
))
.expect("failed to send test reply");
let want_servers = vec![create_test_dns_server(
fidl_ip_v6!("fe80::1:2"),
1, 1, )];
let servers = exec.run_singlethreaded(test.run()).expect("get servers");
assert_eq!(servers, want_servers);
} {
let mut test = Test::new(&mut client, &client_proxy);
exec.run_singlethreaded(test.handle_next_event());
assert!(
test.client.dns_responder.is_some(),
"WatchServers responder should be present"
);
exec.run_singlethreaded(test.refresh_client());
let dns_servers = [net_ip_v6!("fe80::1:2")];
let () = exec
.run_singlethreaded(send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[
v6::DhcpOption::ServerId(&[1, 2, 3]),
v6::DhcpOption::DnsServers(&dns_servers),
],
))
.expect("failed to send test reply");
exec.run_singlethreaded(test.handle_next_event());
assert_matches!(exec.run_until_stalled(&mut test.run()), Poll::Pending);
exec.run_singlethreaded(test.refresh_client());
let dns_servers = [net_ip_v6!("fe80::1:2"), net_ip_v6!("1234::5:6")];
let () = exec
.run_singlethreaded(send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[
v6::DhcpOption::ServerId(&[1, 2, 3]),
v6::DhcpOption::DnsServers(&dns_servers),
],
))
.expect("failed to send test reply");
let want_servers = vec![
create_test_dns_server(
fidl_ip_v6!("fe80::1:2"),
1, 1, ),
create_test_dns_server(
fidl_ip_v6!("1234::5:6"),
1, 0, ),
];
let servers = exec.run_singlethreaded(test.run()).expect("get servers");
assert_eq!(servers, want_servers);
} {
let mut test = Test::new(&mut client, &client_proxy);
exec.run_singlethreaded(test.refresh_client());
let () = exec
.run_singlethreaded(send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[v6::DhcpOption::ServerId(&[1, 2, 3]), v6::DhcpOption::DnsServers(&[])],
))
.expect("failed to send test reply");
let want_servers = Vec::<fnet_name::DnsServer_>::new();
assert_eq!(exec.run_singlethreaded(test.run()).expect("get servers"), want_servers);
} }
#[fuchsia::test]
async fn test_client_should_respond_with_dns_servers_on_first_watch_if_non_empty() {
let transaction_id = [1, 2, 3];
let (client_proxy, client_stream) = create_proxy_and_stream::<ClientMarker>();
let (client_socket, client_addr) = create_test_socket();
let (server_socket, server_addr) = create_test_socket();
let client = Client::<fasync::net::UdpSocket>::start(
None,
transaction_id,
STATELESS_CLIENT_CONFIG,
1, || Ok(client_socket),
server_addr,
client_stream,
)
.await
.expect("failed to create test client");
let dns_servers = [net_ip_v6!("fe80::1:2"), net_ip_v6!("1234::5:6")];
let () = send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[v6::DhcpOption::ServerId(&[4, 5, 6]), v6::DhcpOption::DnsServers(&dns_servers)],
)
.await
.expect("failed to send test message");
let buf = vec![0u8; MAX_UDP_DATAGRAM_SIZE];
let handle_client_events_fut =
futures::stream::try_unfold((client, buf), |(mut client, mut buf)| async {
client
.handle_next_event(&mut buf)
.await
.map(|res| res.map(|()| ((), (client, buf))))
})
.try_fold((), |(), ()| futures::future::ready(Ok(())))
.fuse();
let mut handle_client_events_fut = pin!(handle_client_events_fut);
let want_servers = vec![
create_test_dns_server(
fidl_ip_v6!("fe80::1:2"),
1, 1, ),
create_test_dns_server(
fidl_ip_v6!("1234::5:6"),
1, 0, ),
];
let found_servers = select!(
status = handle_client_events_fut => panic!("client unexpectedly exited: {status:?}"),
found_servers = client_proxy.watch_servers() => found_servers.expect(
"watch servers should succeed"),
);
assert_eq!(found_servers, want_servers);
}
#[fuchsia::test]
async fn watch_prefixes() {
const SERVER_ID: [u8; 3] = [3, 4, 5];
const PREFERRED_LIFETIME_SECS: u32 = 1000;
const VALID_LIFETIME_SECS: u32 = 2000;
const T1: u32 = 1;
const T2: u32 = 2000;
let (client_proxy, client_stream) = create_proxy_and_stream::<ClientMarker>();
let (client_socket, client_addr) = create_test_socket();
let (server_socket, server_addr) = create_test_socket();
let mut client = Client::<fasync::net::UdpSocket>::start(
Some(CLIENT_ID.into()),
[1, 2, 3],
ClientConfig {
information_config: Default::default(),
non_temporary_address_config: Default::default(),
prefix_delegation_config: Some(PrefixDelegationConfig::Empty(Empty {})),
},
1, || Ok(client_socket),
server_addr,
client_stream,
)
.await
.expect("failed to create test client");
let client_fut = async {
let mut buf = vec![0u8; MAX_UDP_DATAGRAM_SIZE];
loop {
select! {
res = client.handle_next_event(&mut buf).fuse() => {
match res.expect("test client failed to handle next event") {
Some(()) => (),
None => break (),
};
}
}
}
}
.fuse();
let mut client_fut = pin!(client_fut);
let update_prefix = net_subnet_v6!("a::/64");
let remove_prefix = net_subnet_v6!("b::/64");
let add_prefix = net_subnet_v6!("c::/64");
let client_id = {
let ReceivedMessage { client_id, transaction_id } =
assert_received_message(&server_socket, client_addr, v6::MessageType::Solicit)
.await;
let client_id = client_id.unwrap();
let ia_prefix = [
v6::DhcpOption::IaPrefix(v6::IaPrefixSerializer::new(
PREFERRED_LIFETIME_SECS,
VALID_LIFETIME_SECS,
update_prefix,
&[],
)),
v6::DhcpOption::IaPrefix(v6::IaPrefixSerializer::new(
PREFERRED_LIFETIME_SECS,
VALID_LIFETIME_SECS,
remove_prefix,
&[],
)),
];
let () = send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Advertise,
&[
v6::DhcpOption::ServerId(&SERVER_ID),
v6::DhcpOption::ClientId(&client_id),
v6::DhcpOption::Preference(u8::MAX),
v6::DhcpOption::IaPd(v6::IaPdSerializer::new(IA_PD_IAID, T1, T2, &ia_prefix)),
],
)
.await
.expect("failed to send adv message");
let transaction_id = select! {
() = client_fut => panic!("should never return"),
res = assert_received_message(
&server_socket,
client_addr,
v6::MessageType::Request,
).fuse() => {
let ReceivedMessage { client_id: req_client_id, transaction_id } = res;
assert_eq!(Some(&client_id), req_client_id.as_ref());
transaction_id
},
};
let () = send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[
v6::DhcpOption::ServerId(&SERVER_ID),
v6::DhcpOption::ClientId(&client_id),
v6::DhcpOption::IaPd(v6::IaPdSerializer::new(IA_PD_IAID, T1, T2, &ia_prefix)),
],
)
.await
.expect("failed to send reply message");
client_id
};
let check_watch_prefixes_result =
|res: Result<Vec<Prefix>, _>,
before_handling_reply,
preferred_lifetime_secs: u32,
valid_lifetime_secs: u32,
expected_prefixes| {
assert_matches!(
res.unwrap()[..],
[
Prefix {
prefix: got_prefix1,
lifetimes: Lifetimes {
preferred_until: preferred_until1,
valid_until: valid_until1,
},
},
Prefix {
prefix: got_prefix2,
lifetimes: Lifetimes {
preferred_until: preferred_until2,
valid_until: valid_until2,
},
},
] => {
let now = zx::MonotonicInstant::get();
let preferred_until = zx::MonotonicInstant::from_nanos(preferred_until1);
let valid_until = zx::MonotonicInstant::from_nanos(valid_until1);
let preferred_for = zx::MonotonicDuration::from_seconds(
preferred_lifetime_secs.into(),
);
let valid_for = zx::MonotonicDuration::from_seconds(valid_lifetime_secs.into());
assert_eq!(
HashSet::from([got_prefix1, got_prefix2]),
HashSet::from(expected_prefixes),
);
assert!(preferred_until >= before_handling_reply + preferred_for);
assert!(preferred_until <= now + preferred_for);
assert!(valid_until >= before_handling_reply + valid_for);
assert!(valid_until <= now + valid_for);
assert_eq!(preferred_until1, preferred_until2);
assert_eq!(valid_until1, valid_until2);
}
)
};
{
let mut watch_prefixes = client_proxy.watch_prefixes().fuse();
assert_matches!(poll!(&mut watch_prefixes), Poll::Pending);
let before_handling_reply = zx::MonotonicInstant::get();
select! {
() = client_fut => panic!("should never return"),
res = watch_prefixes => check_watch_prefixes_result(
res,
before_handling_reply,
PREFERRED_LIFETIME_SECS,
VALID_LIFETIME_SECS,
[
subnet_to_address_with_prefix(update_prefix),
subnet_to_address_with_prefix(remove_prefix),
],
),
}
}
{
let transaction_id = select! {
() = client_fut => panic!("should never return"),
res = assert_received_message(
&server_socket,
client_addr,
v6::MessageType::Renew,
).fuse() => {
let ReceivedMessage { client_id: ren_client_id, transaction_id } = res;
assert_eq!(ren_client_id.as_ref(), Some(&client_id));
transaction_id
},
};
const NEW_PREFERRED_LIFETIME_SECS: u32 = 2 * PREFERRED_LIFETIME_SECS;
const NEW_VALID_LIFETIME_SECS: u32 = 2 * VALID_LIFETIME_SECS;
let ia_prefix = [
v6::DhcpOption::IaPrefix(v6::IaPrefixSerializer::new(
NEW_PREFERRED_LIFETIME_SECS,
NEW_VALID_LIFETIME_SECS,
update_prefix,
&[],
)),
v6::DhcpOption::IaPrefix(v6::IaPrefixSerializer::new(
NEW_PREFERRED_LIFETIME_SECS,
NEW_VALID_LIFETIME_SECS,
add_prefix,
&[],
)),
v6::DhcpOption::IaPrefix(v6::IaPrefixSerializer::new(0, 0, remove_prefix, &[])),
];
let () = send_msg_with_options(
&server_socket,
client_addr,
transaction_id,
v6::MessageType::Reply,
&[
v6::DhcpOption::ServerId(&SERVER_ID),
v6::DhcpOption::ClientId(&client_id),
v6::DhcpOption::IaPd(v6::IaPdSerializer::new(
v6::IAID::new(0),
T1,
T2,
&ia_prefix,
)),
],
)
.await
.expect("failed to send reply message");
let before_handling_reply = zx::MonotonicInstant::get();
select! {
() = client_fut => panic!("should never return"),
res = client_proxy.watch_prefixes().fuse() => check_watch_prefixes_result(
res,
before_handling_reply,
NEW_PREFERRED_LIFETIME_SECS,
NEW_VALID_LIFETIME_SECS,
[
subnet_to_address_with_prefix(update_prefix),
subnet_to_address_with_prefix(add_prefix),
],
),
}
}
}
#[fuchsia::test]
async fn test_client_schedule_and_cancel_timers() {
let (_client_end, client_stream) = create_request_stream::<ClientMarker>();
let (client_socket, _client_addr) = create_test_socket();
let (_server_socket, server_addr) = create_test_socket();
let mut client = Client::<fasync::net::UdpSocket>::start(
None,
[1, 2, 3], STATELESS_CLIENT_CONFIG,
1, || Ok(client_socket),
server_addr,
client_stream,
)
.await
.expect("failed to create test client");
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let () = client.cancel_timer(dhcpv6_core::client::ClientTimerType::Retransmission);
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
Vec::<&dhcpv6_core::client::ClientTimerType>::new()
);
let now = MonotonicInstant::now();
let () = client.schedule_timer(
dhcpv6_core::client::ClientTimerType::Refresh,
now + Duration::from_nanos(1),
);
let () = client.schedule_timer(
dhcpv6_core::client::ClientTimerType::Retransmission,
now + Duration::from_nanos(2),
);
assert_eq!(
client.timer_abort_handles.keys().collect::<HashSet<_>>(),
vec![
&dhcpv6_core::client::ClientTimerType::Retransmission,
&dhcpv6_core::client::ClientTimerType::Refresh
]
.into_iter()
.collect()
);
let now = MonotonicInstant::now();
client.schedule_timer(
dhcpv6_core::client::ClientTimerType::Refresh,
now + Duration::from_nanos(1),
);
client.schedule_timer(
dhcpv6_core::client::ClientTimerType::Retransmission,
now + Duration::from_nanos(2),
);
let () = client.cancel_timer(dhcpv6_core::client::ClientTimerType::Refresh);
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
client.cancel_timer(dhcpv6_core::client::ClientTimerType::Refresh);
let () = client.cancel_timer(dhcpv6_core::client::ClientTimerType::Retransmission);
assert_eq!(
client
.timer_abort_handles
.keys()
.collect::<Vec<&dhcpv6_core::client::ClientTimerType>>(),
Vec::<&dhcpv6_core::client::ClientTimerType>::new()
);
client.cancel_timer(dhcpv6_core::client::ClientTimerType::Retransmission);
}
#[fuchsia::test]
async fn test_handle_next_event_on_stateless_client() {
let (client_proxy, client_stream) = create_proxy_and_stream::<ClientMarker>();
let (client_socket, client_addr) = create_test_socket();
let (server_socket, server_addr) = create_test_socket();
let mut client = Client::<fasync::net::UdpSocket>::start(
None,
[1, 2, 3], STATELESS_CLIENT_CONFIG,
1, || Ok(client_socket),
server_addr,
client_stream,
)
.await
.expect("failed to create test client");
let ReceivedMessage { client_id, transaction_id: _ } = assert_received_message(
&server_socket,
client_addr,
v6::MessageType::InformationRequest,
)
.await;
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let mut buf = vec![0u8; MAX_UDP_DATAGRAM_SIZE];
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
let ReceivedMessage { client_id: got_client_id, transaction_id: _ } =
assert_received_message(
&server_socket,
client_addr,
v6::MessageType::InformationRequest,
)
.await;
assert_eq!(got_client_id, client_id);
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let () = send_msg_with_options(
&server_socket,
client_addr,
[5, 6, 7],
v6::MessageType::Reply,
&[],
)
.await
.expect("failed to send test message");
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let size =
server_socket.send_to(&[], client_addr).await.expect("failed to send test message");
assert_eq!(size, 0);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let () = send_msg_with_options(
&server_socket,
client_addr,
[1, 2, 3],
v6::MessageType::Reply,
&[v6::DhcpOption::ServerId(&[4, 5, 6])],
)
.await
.expect("failed to send test message");
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Refresh]
);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
client.cancel_timer(dhcpv6_core::client::ClientTimerType::Refresh);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
client.schedule_timer(
dhcpv6_core::client::ClientTimerType::Refresh,
MonotonicInstant::now() + Duration::from_nanos(1),
);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
let ReceivedMessage { client_id, transaction_id: _ } = assert_received_message(
&server_socket,
client_addr,
v6::MessageType::InformationRequest,
)
.await;
assert_eq!(got_client_id, client_id,);
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let test_fut = async {
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
client
.dns_responder
.take()
.expect("test client did not get a channel responder")
.send(&[fnet_name::DnsServer_ {
address: Some(fidl_socket_addr!("[fe01::2:3]:42")),
source: Some(fnet_name::DnsServerSource::Dhcpv6(
fnet_name::Dhcpv6DnsServerSource {
source_interface: Some(42),
..Default::default()
},
)),
..Default::default()
}])
.expect("failed to send response on test channel");
};
let (watcher_res, ()) = join!(client_proxy.watch_servers(), test_fut);
let servers = watcher_res.expect("failed to watch servers");
assert_eq!(
servers,
vec![fnet_name::DnsServer_ {
address: Some(fidl_socket_addr!("[fe01::2:3]:42")),
source: Some(fnet_name::DnsServerSource::Dhcpv6(
fnet_name::Dhcpv6DnsServerSource {
source_interface: Some(42),
..Default::default()
},
)),
..Default::default()
}]
);
drop(client_proxy);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(None));
}
#[fuchsia::test]
async fn test_handle_next_event_on_stateful_client() {
let (client_proxy, client_stream) = create_proxy_and_stream::<ClientMarker>();
let (client_socket, client_addr) = create_test_socket();
let (server_socket, server_addr) = create_test_socket();
let mut client = Client::<fasync::net::UdpSocket>::start(
Some(CLIENT_ID.into()),
[1, 2, 3], ClientConfig {
information_config: Default::default(),
non_temporary_address_config: AddressConfig {
address_count: 1,
preferred_addresses: None,
},
prefix_delegation_config: None,
},
1, || Ok(client_socket),
server_addr,
client_stream,
)
.await
.expect("failed to create test client");
let _: ReceivedMessage =
assert_received_message(&server_socket, client_addr, v6::MessageType::Solicit).await;
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let mut buf = vec![0u8; MAX_UDP_DATAGRAM_SIZE];
drop(client_proxy);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(None));
}
#[fuchsia::test]
#[should_panic]
async fn test_handle_next_event_respects_timer_order() {
let (_client_end, client_stream) = create_request_stream::<ClientMarker>();
let (client_socket, client_addr) = create_test_socket();
let (server_socket, server_addr) = create_test_socket();
let mut client = Client::<fasync::net::UdpSocket>::start(
None,
[1, 2, 3], STATELESS_CLIENT_CONFIG,
1, || Ok(client_socket),
server_addr,
client_stream,
)
.await
.expect("failed to create test client");
let mut buf = vec![0u8; MAX_UDP_DATAGRAM_SIZE];
let () = client.cancel_timer(dhcpv6_core::client::ClientTimerType::Retransmission);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
let () = client.schedule_timer(
dhcpv6_core::client::ClientTimerType::Retransmission,
MonotonicInstant::now() + Duration::from_secs(1_000_000),
);
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let () = send_msg_with_options(
&server_socket,
client_addr,
[5, 6, 7],
v6::MessageType::Reply,
&[],
)
.await
.expect("failed to send test message");
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
assert_eq!(
client.timer_abort_handles.keys().collect::<Vec<_>>(),
vec![&dhcpv6_core::client::ClientTimerType::Retransmission]
);
let () = client.schedule_timer(
dhcpv6_core::client::ClientTimerType::Refresh,
MonotonicInstant::now() + Duration::from_nanos(1),
);
assert_eq!(
client.timer_abort_handles.keys().collect::<HashSet<_>>(),
vec![
&dhcpv6_core::client::ClientTimerType::Retransmission,
&dhcpv6_core::client::ClientTimerType::Refresh
]
.into_iter()
.collect()
);
assert_matches!(client.handle_next_event(&mut buf).await, Ok(Some(())));
}
#[fuchsia::test]
async fn test_handle_next_event_fails_on_recv_err() {
struct StubSocket {}
impl<'a> AsyncSocket<'a> for StubSocket {
type RecvFromFut = futures::future::Ready<Result<(usize, SocketAddr), std::io::Error>>;
type SendToFut = futures::future::Ready<Result<usize, std::io::Error>>;
fn recv_from(&'a self, _buf: &'a mut [u8]) -> Self::RecvFromFut {
futures::future::ready(Err(std::io::Error::new(
std::io::ErrorKind::Other,
"test recv error",
)))
}
fn send_to(&'a self, buf: &'a [u8], _addr: SocketAddr) -> Self::SendToFut {
futures::future::ready(Ok(buf.len()))
}
}
let (_client_end, client_stream) = create_request_stream::<ClientMarker>();
let mut client = Client::<StubSocket>::start(
None,
[1, 2, 3], STATELESS_CLIENT_CONFIG,
1, || Ok(StubSocket {}),
std_socket_addr!("[::1]:0"),
client_stream,
)
.await
.expect("failed to create test client");
assert_matches!(
client.handle_next_event(&mut [0u8]).await,
Err(ClientError::SocketRecv(err)) if err.kind() == std::io::ErrorKind::Other
);
}
}