use anyhow::{Context as _, Error};
use dhcpv4::configuration;
use dhcpv4::protocol::{Message, CLIENT_PORT, SERVER_PORT};
use dhcpv4::server::{
DataStore, ResponseTarget, Server, ServerAction, ServerDispatcher, ServerError,
DEFAULT_STASH_ID,
};
use dhcpv4::stash::Stash;
use fuchsia_async::net::UdpSocket;
use fuchsia_async::{self as fasync};
use fuchsia_component::server::{ServiceFs, ServiceFsDir};
use futures::{Future, SinkExt as _, StreamExt as _, TryFutureExt as _, TryStreamExt as _};
use net_declare::net::prefix_length_v4;
use net_types::ethernet::Mac;
use packet::serialize::InnerPacketBuilder;
use packet::Serializer;
use packet_formats::ipv4::Ipv4PacketBuilder;
use packet_formats::udp::UdpPacketBuilder;
use sockaddr::IntoSockAddr as _;
use std::cell::RefCell;
use std::collections::HashMap;
use std::convert::{Infallible, TryInto as _};
use std::net::{IpAddr, Ipv4Addr, SocketAddr};
use tracing::{debug, error, info, warn};
const BUF_SZ: usize = 1024;
enum IncomingService {
Server(fidl_fuchsia_net_dhcp::Server_RequestStream),
}
const DEFAULT_LEASE_DURATION_SECONDS: u32 = 24 * 60 * 60;
fn default_parameters() -> configuration::ServerParameters {
configuration::ServerParameters {
server_ips: vec![],
lease_length: dhcpv4::configuration::LeaseLength {
default_seconds: DEFAULT_LEASE_DURATION_SECONDS,
max_seconds: DEFAULT_LEASE_DURATION_SECONDS,
},
managed_addrs: dhcpv4::configuration::ManagedAddresses {
mask: configuration::SubnetMask::new(prefix_length_v4!(0)),
pool_range_start: Ipv4Addr::UNSPECIFIED,
pool_range_stop: Ipv4Addr::UNSPECIFIED,
},
permitted_macs: dhcpv4::configuration::PermittedMacs(vec![]),
static_assignments: dhcpv4::configuration::StaticAssignments(
std::collections::hash_map::HashMap::new(),
),
arp_probe: false,
bound_device_names: vec![],
}
}
#[derive(argh::FromArgs)]
struct Args {
#[argh(switch)]
persistent: bool,
}
#[fuchsia::main()]
pub async fn main() -> Result<(), Error> {
info!("starting");
let Args { persistent } = argh::from_env();
info!("persistent={}", persistent);
if persistent {
let stash = Stash::new(DEFAULT_STASH_ID).context("failed to instantiate stash")?;
let (params, options, records) = match stash.load_parameters().await {
Ok(params) => {
let options = stash.load_options().await.unwrap_or_else(|e| {
warn!("failed to load options from stash: {:?}", e);
HashMap::new()
});
let records = stash.load_client_records().await.unwrap_or_else(|e| {
warn!("failed to load client records from stash: {:?}", e);
HashMap::new()
});
(params, options, records)
}
Err(e) => {
warn!("failed to load parameters from stash: {:?}", e);
(default_parameters(), HashMap::new(), HashMap::new())
}
};
let server = match Server::new_from_state(stash.clone(), params, options, records) {
Ok(v) => v,
Err(e) => {
warn!("failed to create server from persistent state: {}", e);
Server::new(Some(stash), default_parameters())
}
};
Ok(run(server).await?)
} else {
Ok(run(Server::<Stash>::new(None, default_parameters())).await?)
}
}
async fn run<DS: DataStore>(server: Server<DS>) -> Result<(), Error> {
let server = RefCell::new(ServerDispatcherRuntime::new(server));
let mut fs = ServiceFs::new_local();
let _: &mut ServiceFsDir<'_, _> = fs.dir("svc").add_fidl_service(IncomingService::Server);
let _: &mut ServiceFs<_> = fs
.take_and_serve_directory_handle()
.context("service fs failed to take and serve directory handle")?;
let (mut socket_sink, socket_stream) =
futures::channel::mpsc::channel::<ServerSocketCollection<UdpSocket>>(1);
match server.borrow_mut().enable() {
Ok(None) => unreachable!("server can't be enabled already"),
Ok(Some(socket_collection)) => {
let () = socket_sink.try_send(socket_collection)?;
}
Err(e @ zx::Status::INVALID_ARGS) => {
info!("server not configured for serving leases: {:?}", e)
}
Err(e) => warn!("could not enable server on startup: {:?}", e),
}
let admin_fut =
fs.then(futures::future::ok).try_for_each_concurrent(None, |incoming_service| async {
match incoming_service {
IncomingService::Server(stream) => {
run_server(stream, &server, &default_parameters(), socket_sink.clone())
.inspect_err(|e| warn!("run_server failed: {:?}", e))
.await?;
Ok(())
}
}
});
let server_fut = define_running_server_fut(&server, socket_stream);
info!("running");
let ((), ()) = futures::try_join!(server_fut, admin_fut)?;
Ok(())
}
trait SocketServerDispatcher: ServerDispatcher {
type Socket;
fn create_socket(name: &str, src: Ipv4Addr) -> std::io::Result<Self::Socket>;
fn dispatch_message(&mut self, msg: Message) -> Result<ServerAction, ServerError>;
fn create_sockets(
params: &configuration::ServerParameters,
) -> std::io::Result<Vec<SocketWithId<Self::Socket>>>;
}
impl<DS: DataStore> SocketServerDispatcher for Server<DS> {
type Socket = UdpSocket;
fn create_socket(name: &str, src: Ipv4Addr) -> std::io::Result<Self::Socket> {
let socket = socket2::Socket::new(
socket2::Domain::IPV4,
socket2::Type::DGRAM,
Some(socket2::Protocol::UDP),
)?;
let () = socket.set_reuse_port(true)?;
let () = socket.bind_device(Some(name.as_bytes()))?;
info!("socket bound to device {}", name);
let () = socket.set_broadcast(true)?;
let () = socket.bind(&SocketAddr::new(IpAddr::V4(src), SERVER_PORT.into()).into())?;
Ok(UdpSocket::from_socket(socket.into())?)
}
fn dispatch_message(&mut self, msg: Message) -> Result<ServerAction, ServerError> {
self.dispatch(msg)
}
fn create_sockets(
params: &configuration::ServerParameters,
) -> std::io::Result<Vec<SocketWithId<Self::Socket>>> {
let configuration::ServerParameters { bound_device_names, .. } = params;
bound_device_names
.iter()
.map(|name| {
let iface_id =
fuchsia_nix::net::if_::if_nametoindex(name.as_str()).map_err(|e| {
let e: std::io::Error = e.into();
e
})?;
let socket = Self::create_socket(name, Ipv4Addr::UNSPECIFIED)?;
Ok(SocketWithId { socket, iface_id: iface_id.into() })
})
.collect()
}
}
struct ServerDispatcherRuntime<S> {
abort_handle: Option<futures::future::AbortHandle>,
server: S,
}
impl<S> std::ops::Deref for ServerDispatcherRuntime<S> {
type Target = S;
fn deref(&self) -> &Self::Target {
&self.server
}
}
impl<S> std::ops::DerefMut for ServerDispatcherRuntime<S> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.server
}
}
impl<S: SocketServerDispatcher> ServerDispatcherRuntime<S> {
fn new(server: S) -> Self {
Self { abort_handle: None, server }
}
fn disable(&mut self) {
if let Some(abort_handle) = self.abort_handle.take() {
let () = abort_handle.abort();
}
}
fn enable(&mut self) -> Result<Option<ServerSocketCollection<S::Socket>>, zx::Status> {
if self.abort_handle.is_some() {
return Ok(None);
}
let params = self.server.try_validate_parameters()?;
let (abort_handle, abort_registration) = futures::future::AbortHandle::new_pair();
let sockets = S::create_sockets(params).map_err(|e| {
let () = match e.raw_os_error() {
Some(libc::ENODEV) => {
warn!("Failed to create server sockets: {}", e)
}
Some(_) | None => error!("Failed to create server sockets: {}", e),
};
zx::Status::IO
})?;
if sockets.is_empty() {
error!("No sockets to run server on");
return Err(zx::Status::INVALID_ARGS);
}
self.abort_handle = Some(abort_handle);
Ok(Some(ServerSocketCollection { sockets, abort_registration }))
}
fn enabled(&self) -> bool {
self.abort_handle.is_some()
}
fn if_disabled<R, F: FnOnce(&mut S) -> Result<R, zx::Status>>(
&mut self,
f: F,
) -> Result<R, zx::Status> {
if self.abort_handle.is_none() {
f(&mut self.server)
} else {
Err(zx::Status::BAD_STATE)
}
}
}
#[derive(Debug, PartialEq)]
struct SocketWithId<S> {
socket: S,
iface_id: u64,
}
struct MessageHandler<'a, S: SocketServerDispatcher> {
server: &'a RefCell<ServerDispatcherRuntime<S>>,
}
impl<'a, S: SocketServerDispatcher> MessageHandler<'a, S> {
fn new(server: &'a RefCell<ServerDispatcherRuntime<S>>) -> Self {
Self { server }
}
fn handle_from_sender(
&mut self,
buf: &[u8],
mut sender: std::net::SocketAddrV4,
) -> Result<Option<(std::net::SocketAddrV4, Message, Option<Mac>)>, Error> {
let msg = match Message::from_buffer(buf) {
Ok(msg) => {
debug!("parsed message from {}: {:?}", sender, msg);
msg
}
Err(e) => {
warn!("failed to parse message from {}: {}", sender, e);
return Ok(None);
}
};
let typ = msg.get_dhcp_type();
if sender.ip().is_unspecified() {
info!("processing {:?} from {}", typ, msg.chaddr);
} else {
info!("processing {:?} from {}", typ, sender);
}
let result = self.server.borrow_mut().dispatch_message(msg);
match result {
Err(e) => {
warn!("error processing client message: {:?}", e);
Ok(None)
}
Ok(ServerAction::AddressRelease(addr)) => {
info!("released address: {}", addr);
Ok(None)
}
Ok(ServerAction::AddressDecline(addr)) => {
info!("allocated address: {}", addr);
Ok(None)
}
Ok(ServerAction::SendResponse(message, dst)) => {
debug!("generated response: {:?}", message);
let typ = message.get_dhcp_type();
let (addr, chaddr) = match dst {
ResponseTarget::Broadcast => {
info!("sending {:?} to {}", typ, Ipv4Addr::BROADCAST);
(Ipv4Addr::BROADCAST, None)
}
ResponseTarget::Unicast(addr, None) => {
info!("sending {:?} to {}", typ, addr);
(addr, None)
}
ResponseTarget::Unicast(addr, Some(chaddr)) => {
info!("sending {:?} to ip {} chaddr {}", typ, addr, chaddr);
(addr, Some(chaddr))
}
};
sender.set_ip(addr);
Ok(Some((sender, message, chaddr)))
}
}
}
}
async fn define_msg_handling_loop_future<DS: DataStore>(
sock: SocketWithId<<Server<DS> as SocketServerDispatcher>::Socket>,
server: &RefCell<ServerDispatcherRuntime<Server<DS>>>,
) -> Result<Infallible, Error> {
let SocketWithId { socket, iface_id } = sock;
let mut handler = MessageHandler::new(server);
let mut buf = vec![0u8; BUF_SZ];
loop {
let (received, sender) =
socket.recv_from(&mut buf).await.context("failed to read from socket")?;
let sender = match sender {
std::net::SocketAddr::V4(sender) => sender,
std::net::SocketAddr::V6(sender) => {
return Err(anyhow::anyhow!(
"IPv4 socket received datagram from IPv6 sender: {}",
sender
))
}
};
if let Some((dst, msg, chaddr)) = handler
.handle_from_sender(&buf[..received], sender)
.context("failed to handle buffer")?
{
let chaddr = if let Some(chaddr) = chaddr {
chaddr
} else {
let response = msg.serialize();
let sent = socket
.send_to(&response, SocketAddr::V4(dst))
.await
.context("unable to send response")?;
if sent != response.len() {
return Err(anyhow::anyhow!(
"sent {} bytes for a message of size {}",
sent,
response.len()
));
}
info!("response sent to {}: {} bytes", dst, sent);
continue;
};
let dst_ip: net_types::ip::Ipv4Addr = (*dst.ip()).into();
let src_ip: net_types::ip::Ipv4Addr = msg
.options
.iter()
.find_map(|opt| match opt {
dhcpv4::protocol::DhcpOption::ServerIdentifier(addr) => Some(addr.clone()),
_ => None,
})
.expect("expect server identifier is always present")
.into();
let response = msg.serialize();
let udp_builder = UdpPacketBuilder::new(src_ip, dst_ip, Some(SERVER_PORT), CLIENT_PORT);
const TTL: u8 = 64;
let ipv4_builder = Ipv4PacketBuilder::new(
src_ip,
dst_ip,
TTL,
packet_formats::ip::Ipv4Proto::Proto(packet_formats::ip::IpProto::Udp),
);
let packet = response
.into_serializer()
.encapsulate(udp_builder)
.encapsulate(ipv4_builder)
.serialize_vec_outer()
.expect("serialize packet failed")
.unwrap_b();
let mut sll_addr = [0; 8];
(&mut sll_addr[..chaddr.bytes().len()]).copy_from_slice(&chaddr.bytes());
let sockaddr_ll = libc::sockaddr_ll {
sll_family: libc::AF_PACKET.try_into().expect("convert sll_family failed"),
sll_ifindex: iface_id.try_into().expect("convert sll_ifindex failed"),
sll_protocol: u16::try_from(libc::ETH_P_IP)
.expect("convert ETH_P_IP failed")
.to_be(),
sll_halen: chaddr.bytes().len().try_into().expect("convert chaddr size failed"),
sll_addr: sll_addr,
sll_hatype: 0,
sll_pkttype: 0,
};
let socket = socket2::Socket::new(
socket2::Domain::PACKET,
socket2::Type::DGRAM,
None, )
.context("create packet socket failed")?;
let socket = fasync::net::DatagramSocket::new_from_socket(socket)
.context("failed to wrap into fuchsia-async DatagramSocket")?;
let sent = socket
.send_to(packet.as_ref(), sockaddr_ll.into_sockaddr())
.await
.context("unable to send response")?;
if sent != packet.as_ref().len() {
return Err(anyhow::anyhow!(
"sent {} bytes for a packet of size {}",
sent,
packet.as_ref().len()
));
}
info!("response sent to {}: {} bytes", dst, sent);
}
}
}
fn define_running_server_fut<'a, S, DS>(
server: &'a RefCell<ServerDispatcherRuntime<Server<DS>>>,
socket_stream: S,
) -> impl Future<Output = Result<(), Error>> + 'a
where
S: futures::Stream<
Item = ServerSocketCollection<<Server<Stash> as SocketServerDispatcher>::Socket>,
> + 'static,
DS: DataStore,
{
socket_stream.map(Ok).try_for_each(move |socket_collection| async move {
let ServerSocketCollection { sockets, abort_registration } = socket_collection;
let msg_loops = futures::future::try_join_all(
sockets.into_iter().map(|sock| define_msg_handling_loop_future(sock, server)),
);
info!("Server starting");
match futures::future::Abortable::new(msg_loops, abort_registration).await {
Ok(Ok(v)) => {
let _: Vec<Infallible> = v;
Err(anyhow::anyhow!("Server futures finished unexpectedly"))
}
Ok(Err(error)) => {
error!("Server encountered an error: {:?}. Stopping server.", error);
let () = server.borrow_mut().disable();
Ok(())
}
Err(futures::future::Aborted {}) => {
info!("Server stopped");
Ok(())
}
}
})
}
struct ServerSocketCollection<S> {
sockets: Vec<SocketWithId<S>>,
abort_registration: futures::future::AbortRegistration,
}
async fn run_server<S, C>(
stream: fidl_fuchsia_net_dhcp::Server_RequestStream,
server: &RefCell<ServerDispatcherRuntime<S>>,
default_params: &dhcpv4::configuration::ServerParameters,
socket_sink: C,
) -> Result<(), fidl::Error>
where
S: SocketServerDispatcher,
C: futures::sink::Sink<ServerSocketCollection<S::Socket>> + Unpin,
C::Error: std::fmt::Debug,
{
stream
.try_fold(socket_sink, |mut socket_sink, request| async move {
match request {
fidl_fuchsia_net_dhcp::Server_Request::StartServing { responder } => {
responder.send(
match server.borrow_mut().enable() {
Ok(Some(socket_collection)) => {
socket_sink.send(socket_collection).await.map_err(|e| {
error!("Failed to send sockets to sink: {:?}", e);
let () = server.borrow_mut().disable();
zx::Status::INTERNAL
})
}
Ok(None) => {
info!("Server already running");
Ok(())
}
Err(status) => Err(status),
}
.map_err(zx::Status::into_raw),
)
}
fidl_fuchsia_net_dhcp::Server_Request::StopServing { responder } => {
let () = server.borrow_mut().disable();
responder.send()
}
fidl_fuchsia_net_dhcp::Server_Request::IsServing { responder } => {
responder.send(server.borrow().enabled())
}
fidl_fuchsia_net_dhcp::Server_Request::GetOption { code: c, responder: r } => r
.send(
server.borrow().dispatch_get_option(c).as_ref().map_err(|e| e.into_raw()),
),
fidl_fuchsia_net_dhcp::Server_Request::GetParameter { name: n, responder: r } => {
let response = server.borrow().dispatch_get_parameter(n);
r.send(response.as_ref().map_err(|e| e.into_raw()))
}
fidl_fuchsia_net_dhcp::Server_Request::SetOption { value: v, responder: r } => {
r.send(server.borrow_mut().dispatch_set_option(v).map_err(|e| e.into_raw()))
}
fidl_fuchsia_net_dhcp::Server_Request::SetParameter { value: v, responder: r } => r
.send(
server
.borrow_mut()
.if_disabled(|s| s.dispatch_set_parameter(v))
.map_err(|e| e.into_raw()),
),
fidl_fuchsia_net_dhcp::Server_Request::ListOptions { responder: r } => r.send(
server.borrow().dispatch_list_options().as_deref().map_err(|e| e.into_raw()),
),
fidl_fuchsia_net_dhcp::Server_Request::ListParameters { responder: r } => r.send(
server.borrow().dispatch_list_parameters().as_deref().map_err(|e| e.into_raw()),
),
fidl_fuchsia_net_dhcp::Server_Request::ResetOptions { responder: r } => {
r.send(server.borrow_mut().dispatch_reset_options().map_err(|e| e.into_raw()))
}
fidl_fuchsia_net_dhcp::Server_Request::ResetParameters { responder: r } => r.send(
server
.borrow_mut()
.if_disabled(|s| s.dispatch_reset_parameters(&default_params))
.map_err(|e| e.into_raw()),
),
fidl_fuchsia_net_dhcp::Server_Request::ClearLeases { responder: r } => r.send(
server.borrow_mut().dispatch_clear_leases().map_err(zx::Status::into_raw),
),
}
.map(|()| socket_sink)
})
.await
.map(|_socket_sink: C| ())
}
#[cfg(test)]
mod tests {
use super::*;
use dhcpv4::configuration::ServerParameters;
use fuchsia_async as fasync;
use futures::sink::drain;
use futures::FutureExt;
use net_declare::{fidl_ip_v4, std_ip_v4};
#[derive(Debug, Eq, PartialEq)]
struct CannedSocket {
name: String,
src: Ipv4Addr,
}
struct CannedDispatcher {
params: Option<ServerParameters>,
mock_leases: u32,
}
impl CannedDispatcher {
fn new() -> Self {
Self { params: None, mock_leases: 0 }
}
}
impl SocketServerDispatcher for CannedDispatcher {
type Socket = CannedSocket;
fn create_socket(name: &str, src: Ipv4Addr) -> std::io::Result<Self::Socket> {
let name = name.to_string();
Ok(CannedSocket { name, src })
}
fn dispatch_message(&mut self, mut msg: Message) -> Result<ServerAction, ServerError> {
msg.op = dhcpv4::protocol::OpCode::BOOTREPLY;
Ok(ServerAction::SendResponse(msg, ResponseTarget::Broadcast))
}
fn create_sockets(
params: &configuration::ServerParameters,
) -> std::io::Result<Vec<SocketWithId<Self::Socket>>> {
let configuration::ServerParameters { bound_device_names, .. } = params;
bound_device_names
.iter()
.map(String::as_str)
.enumerate()
.map(|(iface_id, name)| {
let iface_id = std::convert::TryInto::try_into(iface_id).map_err(|e| {
std::io::Error::new(
std::io::ErrorKind::InvalidInput,
format!("interface id {} out of range: {}", iface_id, e),
)
})?;
let socket = Self::create_socket(name, Ipv4Addr::UNSPECIFIED)?;
Ok(SocketWithId { socket, iface_id })
})
.collect()
}
}
impl ServerDispatcher for CannedDispatcher {
fn try_validate_parameters(&self) -> Result<&ServerParameters, zx::Status> {
self.params.as_ref().ok_or(zx::Status::INVALID_ARGS)
}
fn dispatch_get_option(
&self,
_code: fidl_fuchsia_net_dhcp::OptionCode,
) -> Result<fidl_fuchsia_net_dhcp::Option_, zx::Status> {
Ok(fidl_fuchsia_net_dhcp::Option_::SubnetMask(fidl_ip_v4!("0.0.0.0")))
}
fn dispatch_get_parameter(
&self,
_name: fidl_fuchsia_net_dhcp::ParameterName,
) -> Result<fidl_fuchsia_net_dhcp::Parameter, zx::Status> {
Ok(fidl_fuchsia_net_dhcp::Parameter::Lease(fidl_fuchsia_net_dhcp::LeaseLength {
default: None,
max: None,
..Default::default()
}))
}
fn dispatch_set_option(
&mut self,
_value: fidl_fuchsia_net_dhcp::Option_,
) -> Result<(), zx::Status> {
Ok(())
}
fn dispatch_set_parameter(
&mut self,
_value: fidl_fuchsia_net_dhcp::Parameter,
) -> Result<(), zx::Status> {
Ok(())
}
fn dispatch_list_options(&self) -> Result<Vec<fidl_fuchsia_net_dhcp::Option_>, zx::Status> {
Ok(vec![])
}
fn dispatch_list_parameters(
&self,
) -> Result<Vec<fidl_fuchsia_net_dhcp::Parameter>, zx::Status> {
Ok(vec![])
}
fn dispatch_reset_options(&mut self) -> Result<(), zx::Status> {
Ok(())
}
fn dispatch_reset_parameters(
&mut self,
_defaults: &dhcpv4::configuration::ServerParameters,
) -> Result<(), zx::Status> {
Ok(())
}
fn dispatch_clear_leases(&mut self) -> Result<(), zx::Status> {
self.mock_leases = 0;
Ok(())
}
}
const DEFAULT_DEVICE_NAME: &str = "foo13";
fn default_params() -> dhcpv4::configuration::ServerParameters {
dhcpv4::configuration::ServerParameters {
server_ips: vec![std_ip_v4!("192.168.0.1")],
lease_length: dhcpv4::configuration::LeaseLength {
default_seconds: 86400,
max_seconds: 86400,
},
managed_addrs: dhcpv4::configuration::ManagedAddresses {
mask: dhcpv4::configuration::SubnetMask::new(prefix_length_v4!(25)),
pool_range_start: std_ip_v4!("192.168.0.0"),
pool_range_stop: std_ip_v4!("192.168.0.0"),
},
permitted_macs: dhcpv4::configuration::PermittedMacs(vec![]),
static_assignments: dhcpv4::configuration::StaticAssignments(HashMap::new()),
arp_probe: false,
bound_device_names: vec![DEFAULT_DEVICE_NAME.to_string()],
}
}
async fn run_with_server<T, F, Fut>(f: F) -> T
where
F: Fn(fidl_fuchsia_net_dhcp::Server_Proxy) -> Fut,
Fut: Future<Output = T>,
{
let (proxy, stream) =
fidl::endpoints::create_proxy_and_stream::<fidl_fuchsia_net_dhcp::Server_Marker>();
let server = RefCell::new(ServerDispatcherRuntime::new(CannedDispatcher::new()));
let defaults = default_params();
futures::select! {
res = f(proxy).fuse() => res,
res = run_server(stream, &server, &defaults, drain()).fuse() => {
unreachable!("server finished before request: {:?}", res)
},
}
}
#[fasync::run_singlethreaded(test)]
async fn get_option_with_subnet_mask_returns_subnet_mask() {
run_with_server(|proxy| async move {
assert_eq!(
proxy
.get_option(fidl_fuchsia_net_dhcp::OptionCode::SubnetMask)
.await
.expect("get_option failed"),
Ok(fidl_fuchsia_net_dhcp::Option_::SubnetMask(fidl_ip_v4!("0.0.0.0")))
);
})
.await
}
#[fasync::run_until_stalled(test)]
async fn get_parameter_with_lease_length_returns_lease_length() {
run_with_server(|proxy| async move {
assert_eq!(
proxy
.get_parameter(fidl_fuchsia_net_dhcp::ParameterName::LeaseLength)
.await
.expect("get_parameter failed"),
Ok(fidl_fuchsia_net_dhcp::Parameter::Lease(fidl_fuchsia_net_dhcp::LeaseLength {
default: None,
max: None,
..Default::default()
}))
);
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn set_option_with_subnet_mask_returns_unit() {
run_with_server(|proxy| async move {
assert_eq!(
proxy
.set_option(&fidl_fuchsia_net_dhcp::Option_::SubnetMask(fidl_ip_v4!("0.0.0.0")))
.await
.expect("set_option failed"),
Ok(())
);
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn set_parameter_with_lease_length_returns_unit() {
run_with_server(|proxy| async move {
assert_eq!(
proxy
.set_parameter(&fidl_fuchsia_net_dhcp::Parameter::Lease(
fidl_fuchsia_net_dhcp::LeaseLength {
default: None,
max: None,
..Default::default()
},
))
.await
.expect("set_parameter failed"),
Ok(())
);
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn list_options_returns_empty_vec() {
run_with_server(|proxy| async move {
assert_eq!(proxy.list_options().await.expect("list_options failed"), Ok(Vec::new()));
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn list_parameters_returns_empty_vec() {
run_with_server(|proxy| async move {
assert_eq!(
proxy.list_parameters().await.expect("list_parameters failed"),
Ok(Vec::new())
);
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn reset_options_returns_unit() {
run_with_server(|proxy| async move {
assert_eq!(proxy.reset_options().await.expect("reset_options failed"), Ok(()));
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn reset_parameters_returns_unit() {
run_with_server(|proxy| async move {
assert_eq!(proxy.reset_parameters().await.expect("reset_parameters failed"), Ok(()));
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn clear_leases_returns_unit() {
run_with_server(|proxy| async move {
assert_eq!(proxy.clear_leases().await.expect("clear_leases failed"), Ok(()));
})
.await
}
#[fasync::run_singlethreaded(test)]
async fn start_stop_server() {
let (proxy, stream) =
fidl::endpoints::create_proxy_and_stream::<fidl_fuchsia_net_dhcp::Server_Marker>();
let (socket_sink, mut socket_stream) =
futures::channel::mpsc::channel::<ServerSocketCollection<CannedSocket>>(1);
let server = RefCell::new(ServerDispatcherRuntime::new(CannedDispatcher::new()));
server.borrow_mut().params = Some(default_params());
let defaults = default_params();
server.borrow_mut().mock_leases = 1;
let test_fut = async {
for () in std::iter::repeat(()).take(3) {
assert!(
!proxy.is_serving().await.expect("query server status request"),
"server should not be serving"
);
let () = proxy
.start_serving()
.await
.expect("start_serving failed")
.map_err(zx::Status::from_raw)
.expect("start_serving returned an error");
let ServerSocketCollection { sockets, abort_registration } =
socket_stream.next().await.expect("Socket stream ended unexpectedly");
assert_eq!(
sockets,
vec![SocketWithId {
socket: CannedSocket {
name: DEFAULT_DEVICE_NAME.to_string(),
src: Ipv4Addr::UNSPECIFIED
},
iface_id: 0
}]
);
let dummy_fut = futures::future::Abortable::new(
futures::future::pending::<()>(),
abort_registration,
);
assert!(
proxy.is_serving().await.expect("query server status request"),
"server should be serving"
);
let () = proxy.stop_serving().await.expect("stop_serving failed");
assert_eq!(dummy_fut.await, Err(futures::future::Aborted {}));
assert_eq!(server.borrow().mock_leases, 1);
assert!(
!proxy.is_serving().await.expect("query server status request"),
"server should no longer be serving"
);
}
};
let () = futures::select! {
res = test_fut.fuse() => res,
res = run_server(stream, &server, &defaults, socket_sink).fuse() => {
unreachable!("server finished before request: {:?}", res)
},
};
}
#[fasync::run_singlethreaded(test)]
async fn start_server_fails_on_bad_params() {
let (proxy, stream) =
fidl::endpoints::create_proxy_and_stream::<fidl_fuchsia_net_dhcp::Server_Marker>();
let server = RefCell::new(ServerDispatcherRuntime::new(CannedDispatcher::new()));
let defaults = default_params();
let res = futures::select! {
res = proxy.start_serving().fuse() => res.expect("start_serving failed"),
res = run_server(stream, &server, &defaults, drain()).fuse() => {
unreachable!("server finished before request: {:?}", res)
},
}
.map_err(zx::Status::from_raw);
assert_eq!(res, Err(zx::Status::INVALID_ARGS));
assert!(server.borrow().abort_handle.is_none());
}
#[fasync::run_singlethreaded(test)]
async fn start_server_fails_on_missing_interface_names() {
let (proxy, stream) =
fidl::endpoints::create_proxy_and_stream::<fidl_fuchsia_net_dhcp::Server_Marker>();
let server = RefCell::new(ServerDispatcherRuntime::new(CannedDispatcher::new()));
let defaults = dhcpv4::configuration::ServerParameters {
bound_device_names: Vec::new(),
..default_params()
};
server.borrow_mut().params = Some(defaults.clone());
let res = futures::select! {
res = proxy.start_serving().fuse() => res.expect("start_serving failed"),
res = run_server(stream, &server, &defaults, drain()).fuse() => {
unreachable!("server finished before request: {:?}", res)
},
}
.map_err(zx::Status::from_raw);
assert_eq!(res, Err(zx::Status::INVALID_ARGS));
assert!(server.borrow().abort_handle.is_none());
}
#[fasync::run_singlethreaded(test)]
async fn disallow_change_parameters_if_enabled() {
let (proxy, stream) =
fidl::endpoints::create_proxy_and_stream::<fidl_fuchsia_net_dhcp::Server_Marker>();
let server = RefCell::new(ServerDispatcherRuntime::new(CannedDispatcher::new()));
server.borrow_mut().params = Some(default_params());
let defaults = default_params();
let test_fut = async {
let () = proxy
.start_serving()
.await
.expect("start_serving failed")
.map_err(zx::Status::from_raw)
.expect("start_serving returned an error");
assert_eq!(
proxy
.set_parameter(&fidl_fuchsia_net_dhcp::Parameter::Lease(
fidl_fuchsia_net_dhcp::LeaseLength {
default: None,
max: None,
..Default::default()
},
))
.await
.expect("set_parameter FIDL failure")
.map_err(zx::Status::from_raw),
Err(zx::Status::BAD_STATE)
);
assert_eq!(
proxy
.reset_parameters()
.await
.expect("reset_parameters FIDL failure")
.map_err(zx::Status::from_raw),
Err(zx::Status::BAD_STATE)
);
};
let () = futures::select! {
res = test_fut.fuse() => res,
res = run_server(stream, &server, &defaults, drain()).fuse() => {
unreachable!("server finished before request: {:?}", res)
},
};
}
#[test]
fn handle_failed_parse() {
let server = RefCell::new(ServerDispatcherRuntime::new(CannedDispatcher::new()));
let mut handler = MessageHandler::new(&server);
assert_matches::assert_matches!(
handler.handle_from_sender(
&[0xFF, 0x00, 0xBA, 0x03],
std::net::SocketAddrV4::new(Ipv4Addr::UNSPECIFIED.into(), 0),
),
Ok(None)
);
}
}