use std::collections::HashSet;
use std::pin::{pin, Pin};
use fidl::endpoints::Proxy as _;
use {
fidl_fuchsia_hardware_network as fhardware_network,
fidl_fuchsia_net_interfaces as fnet_interfaces,
fidl_fuchsia_net_interfaces_admin as fnet_interfaces_admin,
fidl_fuchsia_net_interfaces_ext as fnet_interfaces_ext, fidl_fuchsia_net_stack as fnet_stack,
fidl_fuchsia_net_virtualization as fnet_virtualization,
};
use anyhow::{anyhow, Context as _};
use async_trait::async_trait;
use derivative::Derivative;
use futures::channel::oneshot;
use futures::{future, FutureExt as _, StreamExt as _, TryStreamExt as _};
use log::{debug, error, info, warn};
use crate::errors::{self, ContextExt as _};
use crate::{exit_with_fidl_error, DeviceClass};
#[derive(Debug, Clone)]
pub(super) enum NetworkId {
Bridged,
}
#[derive(Debug)]
pub(super) enum NetworkRequest {
Request(fnet_virtualization::NetworkRequest, future::Shared<oneshot::Receiver<()>>),
Finished(oneshot::Sender<()>),
}
#[derive(Derivative)]
#[derivative(Debug)]
pub(super) enum Event {
ControlRequestStream(#[derivative(Debug = "ignore")] fnet_virtualization::ControlRequestStream),
ControlRequest(fnet_virtualization::ControlRequest),
NetworkRequest(NetworkId, NetworkRequest),
InterfaceClose(NetworkId, u64),
}
pub(super) type EventStream = Pin<Box<dyn futures::stream::Stream<Item = Event>>>;
enum BridgeState {
Init,
WaitingForGuests {
upstream: u64,
upstream_candidates: HashSet<u64>,
},
WaitingForUpstream {
guests: HashSet<u64>,
},
Bridged {
handle: BridgeHandle,
upstream: u64,
upstream_candidates: HashSet<u64>,
guests: HashSet<u64>,
},
}
impl Default for BridgeState {
fn default() -> Self {
BridgeState::Init
}
}
#[async_trait(?Send)]
pub(super) trait Handler {
async fn handle_event(
&'async_trait mut self,
event: Event,
events: &'async_trait mut futures::stream::SelectAll<EventStream>,
) -> Result<(), errors::Error>;
async fn handle_interface_update_result(
&mut self,
update_result: &fnet_interfaces_ext::UpdateResult<
'_,
(),
fnet_interfaces_ext::DefaultInterest,
>,
) -> Result<(), errors::Error>;
}
fn take_any<T: std::marker::Copy + std::cmp::Eq + std::hash::Hash>(
set: &mut HashSet<T>,
) -> Option<T> {
set.iter().copied().next().map(|elem| {
assert!(set.remove(&elem));
elem
})
}
pub(super) struct Virtualization<'a, B: BridgeHandler> {
installer: fnet_interfaces_admin::InstallerProxy,
_allowed_upstream_device_classes: &'a HashSet<DeviceClass>,
bridge: Bridge<B>,
}
impl<'a, B: BridgeHandler> Virtualization<'a, B> {
pub fn new(
_allowed_upstream_device_classes: &'a HashSet<DeviceClass>,
allowed_bridge_upstream_device_classes: HashSet<DeviceClass>,
bridge_handler: B,
installer: fnet_interfaces_admin::InstallerProxy,
) -> Self {
Self {
installer,
_allowed_upstream_device_classes,
bridge: Bridge {
allowed_bridge_upstream_device_classes,
bridge_handler,
bridge_state: Default::default(),
},
}
}
async fn handle_network_request(
&mut self,
network_id: NetworkId,
request: NetworkRequest,
events: &mut futures::stream::SelectAll<EventStream>,
) -> Result<(), errors::Error> {
let Self { installer, bridge, _allowed_upstream_device_classes } = self;
match request {
NetworkRequest::Request(
fnet_virtualization::NetworkRequest::AddPort { port, interface, control_handle: _ },
mut network_close_rx,
) => {
let (device, server_end) =
fidl::endpoints::create_endpoints::<fhardware_network::DeviceMarker>();
let port = port.into_proxy();
port.get_device(server_end)
.context("call get device")
.map_err(errors::Error::NonFatal)?;
let (device_control, server_end) =
fidl::endpoints::create_proxy::<fnet_interfaces_admin::DeviceControlMarker>();
installer
.install_device(device, server_end)
.unwrap_or_else(|err| exit_with_fidl_error(err));
let fhardware_network::PortInfo { id: port_id, .. } = port
.get_info()
.await
.context("get port info")
.map_err(errors::Error::NonFatal)?;
let port_id = port_id
.context("port id not included in port info")
.map_err(errors::Error::NonFatal)?;
let (control, server_end) = fnet_interfaces_ext::admin::Control::create_endpoints()
.context("create Control endpoints")
.map_err(errors::Error::NonFatal)?;
device_control
.create_interface(
&port_id,
server_end,
&fnet_interfaces_admin::Options::default(),
)
.context("call create interface")
.map_err(errors::Error::NonFatal)?;
let id = control
.get_id()
.await
.context("call get id")
.map_err(errors::Error::NonFatal)?;
if !control
.enable()
.await
.context("call enable")
.map_err(errors::Error::NonFatal)?
.map_err(|e| anyhow!("failed to enable interface: {:?}", e))
.map_err(errors::Error::NonFatal)?
{
warn!("added interface {} was already enabled", id);
}
match network_id {
NetworkId::Bridged => {
bridge
.add_guest_to_bridge(id)
.await
.context("adding interface to bridge")?;
}
}
let shutdown_fut = async move {
let mut interface_closure = interface
.into_stream()
.map(|request| {
request.map(|_request: fnet_virtualization::InterfaceRequest| ())
})
.try_collect::<()>();
let mut device_control_closure = device_control.on_closed().fuse();
let control_termination = control.wait_termination().fuse();
let mut control_termination = pin!(control_termination);
let reason = futures::select! {
result = interface_closure => {
format!("interface channel closed by client: {:?}", result)
},
result = device_control_closure => {
match result {
Ok(zx::Signals::CHANNEL_PEER_CLOSED) => {},
result => error!(
"got unexpected result waiting for device control \
channel closure: {:?}",
result,
),
}
"device detached from netstack".to_string()
}
result = network_close_rx => {
result.expect("sender should not be dropped");
"network has been shut down".to_string()
},
terminal_error = control_termination => {
format!(
"interface control channel closed: {:?}",
terminal_error
)
}
};
info!("interface {}: {}, removing interface", id, reason);
id
};
events.push(
futures::stream::once(
shutdown_fut.map(|id| Event::InterfaceClose(network_id, id)),
)
.boxed(),
);
}
NetworkRequest::Finished(network_close_tx) => {
match network_close_tx.send(()) {
Ok(()) => {}
Err(()) => {
info!("removing virtualized network with no devices attached")
}
}
}
}
Ok(())
}
}
struct Bridge<B: BridgeHandler> {
allowed_bridge_upstream_device_classes: HashSet<DeviceClass>,
bridge_handler: B,
bridge_state: BridgeState,
}
impl<B: BridgeHandler> Bridge<B> {
fn is_device_class_allowed_for_bridge_upstream(
&self,
port_class: fnet_interfaces_ext::PortClass,
) -> bool {
let device_class = match port_class {
fnet_interfaces_ext::PortClass::Loopback
| fnet_interfaces_ext::PortClass::Blackhole => None,
fnet_interfaces_ext::PortClass::Virtual => Some(DeviceClass::Virtual),
fnet_interfaces_ext::PortClass::Ethernet => Some(DeviceClass::Ethernet),
fnet_interfaces_ext::PortClass::WlanClient => Some(DeviceClass::WlanClient),
fnet_interfaces_ext::PortClass::WlanAp => Some(DeviceClass::WlanAp),
fnet_interfaces_ext::PortClass::Ppp => Some(DeviceClass::Ppp),
fnet_interfaces_ext::PortClass::Bridge => Some(DeviceClass::Bridge),
fnet_interfaces_ext::PortClass::Lowpan => Some(DeviceClass::Lowpan),
};
device_class.is_some_and(|device_class| {
self.allowed_bridge_upstream_device_classes.contains(&device_class)
})
}
async fn add_guest_to_bridge(&mut self, id: u64) -> Result<(), errors::Error> {
info!("got a request to add interface {} to bridge", id);
let Self { bridge_state, bridge_handler, allowed_bridge_upstream_device_classes: _ } = self;
*bridge_state = match std::mem::take(bridge_state) {
BridgeState::Init => BridgeState::WaitingForUpstream { guests: HashSet::from([id]) },
BridgeState::WaitingForGuests { upstream, upstream_candidates } => {
let guests = HashSet::from([id]);
let handle = bridge_handler
.build_bridge(guests.iter().copied(), upstream)
.await
.context("building bridge")?;
BridgeState::Bridged { handle, upstream, upstream_candidates, guests }
}
BridgeState::WaitingForUpstream { mut guests } => {
assert!(guests.insert(id));
BridgeState::WaitingForUpstream { guests }
}
BridgeState::Bridged { handle, upstream, upstream_candidates, mut guests } => {
bridge_handler.destroy_bridge(handle).await.context("destroying bridge")?;
assert!(guests.insert(id));
let handle = bridge_handler
.build_bridge(guests.iter().copied(), upstream)
.await
.context("building bridge")?;
BridgeState::Bridged { handle, upstream, upstream_candidates, guests }
}
};
Ok(())
}
async fn remove_guest_from_bridge(&mut self, id: u64) -> Result<(), errors::Error> {
info!("got a request to remove interface {} from bridge", id);
let Self { bridge_state, bridge_handler, allowed_bridge_upstream_device_classes: _ } = self;
*bridge_state = match std::mem::take(bridge_state) {
BridgeState::Init | BridgeState::WaitingForGuests { .. } => {
panic!("cannot remove guest interface {} since it was not previously added", id)
}
BridgeState::WaitingForUpstream { mut guests } => {
assert!(guests.remove(&id));
if guests.is_empty() {
BridgeState::Init
} else {
BridgeState::WaitingForUpstream { guests }
}
}
BridgeState::Bridged { handle, upstream, upstream_candidates, mut guests } => {
bridge_handler.destroy_bridge(handle).await.context("destroying bridge")?;
assert!(guests.remove(&id));
if guests.is_empty() {
BridgeState::WaitingForGuests { upstream, upstream_candidates }
} else {
let handle = bridge_handler
.build_bridge(guests.iter().copied(), upstream)
.await
.context("building bridge")?;
BridgeState::Bridged { handle, upstream, upstream_candidates, guests }
}
}
};
Ok(())
}
async fn handle_interface_online(
&mut self,
id: u64,
allowed_for_bridge_upstream: bool,
) -> Result<(), errors::Error> {
info!("interface {} (allowed for upstream: {}) is online", id, allowed_for_bridge_upstream);
let Self { bridge_state, bridge_handler, allowed_bridge_upstream_device_classes: _ } = self;
*bridge_state = match std::mem::take(bridge_state) {
BridgeState::Init => {
if allowed_for_bridge_upstream {
BridgeState::WaitingForGuests {
upstream: id,
upstream_candidates: Default::default(),
}
} else {
BridgeState::Init
}
}
BridgeState::WaitingForGuests { upstream, mut upstream_candidates } => {
if allowed_for_bridge_upstream {
assert_ne!(
upstream, id,
"interface {} expected to provide upstream but was offline and came online",
id
);
assert!(
upstream_candidates.insert(id),
"upstream candidate {} already present",
id
);
}
BridgeState::WaitingForGuests { upstream, upstream_candidates }
}
BridgeState::WaitingForUpstream { guests } => {
if allowed_for_bridge_upstream && !guests.contains(&id) {
let handle = bridge_handler
.build_bridge(guests.iter().copied(), id)
.await
.context("building bridge")?;
BridgeState::Bridged {
handle,
upstream: id,
upstream_candidates: Default::default(),
guests,
}
} else {
BridgeState::WaitingForUpstream { guests }
}
}
BridgeState::Bridged { handle, upstream, mut upstream_candidates, guests } => {
if id == upstream {
info!("upstream-providing interface {} went online", id);
} else if id == handle.id {
info!("bridge interface {} went online", handle.id);
} else if !guests.contains(&id) && allowed_for_bridge_upstream {
assert!(
upstream_candidates.insert(id),
"upstream candidate {} already present",
id
);
}
BridgeState::Bridged { handle, upstream, upstream_candidates, guests }
}
};
Ok(())
}
async fn handle_interface_offline(
&mut self,
id: u64,
allowed_for_bridge_upstream: bool,
) -> Result<(), errors::Error> {
info!(
"interface {} (allowed for upstream: {}) is offline",
id, allowed_for_bridge_upstream
);
let Self { bridge_state, allowed_bridge_upstream_device_classes: _, bridge_handler: _ } =
self;
*bridge_state = match std::mem::take(bridge_state) {
BridgeState::Init => BridgeState::Init,
BridgeState::WaitingForUpstream { guests } => {
BridgeState::WaitingForUpstream { guests }
}
BridgeState::Bridged { handle, upstream, mut upstream_candidates, guests } => {
if id == handle.id {
warn!("bridge interface {} went offline", id);
} else if id == upstream {
warn!("upstream interface {} went offline", id);
} else if !guests.contains(&id) && allowed_for_bridge_upstream {
assert!(upstream_candidates.remove(&id), "upstream candidate {} not found", id);
}
BridgeState::Bridged { handle, upstream, upstream_candidates, guests }
}
BridgeState::WaitingForGuests { upstream, mut upstream_candidates } => {
if id == upstream {
match take_any(&mut upstream_candidates) {
Some(id) => {
BridgeState::WaitingForGuests { upstream: id, upstream_candidates }
}
None => BridgeState::Init,
}
} else {
if allowed_for_bridge_upstream {
assert!(
upstream_candidates.remove(&id),
"upstream candidate {} not found",
id
);
}
BridgeState::WaitingForGuests { upstream, upstream_candidates }
}
}
};
Ok(())
}
async fn handle_interface_removed(&mut self, removed_id: u64) -> Result<(), errors::Error> {
info!("interface {} removed", removed_id);
let Self { bridge_state, bridge_handler, allowed_bridge_upstream_device_classes: _ } = self;
*bridge_state = match std::mem::take(bridge_state) {
BridgeState::Init => BridgeState::Init,
BridgeState::WaitingForUpstream { guests } => {
if guests.contains(&removed_id) {
info!("guest interface {} removed", removed_id);
}
BridgeState::WaitingForUpstream { guests }
}
BridgeState::WaitingForGuests { upstream, mut upstream_candidates } => {
if upstream == removed_id {
match take_any(&mut upstream_candidates) {
Some(new_upstream_id) => BridgeState::WaitingForGuests {
upstream: new_upstream_id,
upstream_candidates,
},
None => BridgeState::Init,
}
} else {
let _: bool = upstream_candidates.remove(&removed_id);
BridgeState::WaitingForGuests { upstream, upstream_candidates }
}
}
BridgeState::Bridged { handle, upstream, mut upstream_candidates, guests } => {
if guests.contains(&removed_id) {
info!("guest interface {} removed", removed_id);
}
if handle.id == removed_id {
error!("bridge interface {} removed; rebuilding", handle.id);
let handle = bridge_handler
.build_bridge(guests.iter().copied(), upstream)
.await
.context("building bridge")?;
BridgeState::Bridged { handle, upstream, upstream_candidates, guests }
} else if upstream == removed_id {
bridge_handler.destroy_bridge(handle).await.context("destroying bridge")?;
match take_any(&mut upstream_candidates) {
Some(new_upstream_id) => {
let handle = bridge_handler
.build_bridge(guests.iter().copied(), new_upstream_id)
.await
.context("building bridge")?;
BridgeState::Bridged {
handle,
upstream: new_upstream_id,
upstream_candidates,
guests,
}
}
None => BridgeState::WaitingForUpstream { guests },
}
} else {
let _: bool = upstream_candidates.remove(&removed_id);
BridgeState::Bridged { handle, upstream, upstream_candidates, guests }
}
}
};
Ok(())
}
}
#[async_trait(?Send)]
impl<'a, B: BridgeHandler> Handler for Virtualization<'a, B> {
async fn handle_event(
&'async_trait mut self,
event: Event,
events: &'async_trait mut futures::stream::SelectAll<EventStream>,
) -> Result<(), errors::Error> {
match event {
Event::ControlRequestStream(stream) => {
events.push(
stream
.filter_map(|request| {
future::ready(match request {
Ok(request) => Some(Event::ControlRequest(request)),
Err(e) => {
error!("control request error: {:?}", e);
None
}
})
})
.boxed(),
);
}
Event::ControlRequest(fnet_virtualization::ControlRequest::CreateNetwork {
config,
network,
control_handle: _,
}) => {
let network_id = match config {
fnet_virtualization::Config::Bridged(fnet_virtualization::Bridged {
..
}) => {
info!("got a request to create a bridged network");
NetworkId::Bridged
}
config => {
panic!("unsupported network config type {:?}", config);
}
};
let (close_channel_tx, close_channel_rx) = oneshot::channel();
let close_channel_rx = close_channel_rx.shared();
let stream = network
.into_stream()
.filter_map(move |request| {
future::ready(match request {
Ok(request) => {
Some(NetworkRequest::Request(request, close_channel_rx.clone()))
}
Err(e) => {
error!("network request error: {:?}", e);
None
}
})
})
.chain(futures::stream::once(futures::future::ready(
NetworkRequest::Finished(close_channel_tx),
)));
events.push(
stream.map(move |r| Event::NetworkRequest(network_id.clone(), r)).boxed(),
);
}
Event::NetworkRequest(network_id, request) => self
.handle_network_request(network_id, request, events)
.await
.context("handle network request")?,
Event::InterfaceClose(network_id, id) => {
match network_id {
NetworkId::Bridged => {
self.bridge
.remove_guest_from_bridge(id)
.await
.context("removing interface from bridge")?;
}
}
}
}
Ok(())
}
async fn handle_interface_update_result(
&mut self,
update_result: &fnet_interfaces_ext::UpdateResult<
'_,
(),
fnet_interfaces_ext::DefaultInterest,
>,
) -> Result<(), errors::Error> {
let Self { bridge, installer: _, _allowed_upstream_device_classes } = self;
match update_result {
fnet_interfaces_ext::UpdateResult::Added { properties, state: _ }
| fnet_interfaces_ext::UpdateResult::Existing { properties, state: _ } => {
let fnet_interfaces_ext::Properties { id, online, port_class, .. } = **properties;
let allowed_for_bridge_upstream =
bridge.is_device_class_allowed_for_bridge_upstream(port_class);
if online {
bridge
.handle_interface_online(id.get(), allowed_for_bridge_upstream)
.await
.context("handle new interface online")?;
}
}
fnet_interfaces_ext::UpdateResult::Changed {
previous: fnet_interfaces::Properties { online: previously_online, .. },
current: current_properties,
state: _,
} => {
let fnet_interfaces_ext::Properties { id, online, port_class, .. } =
**current_properties;
let allowed_for_bridge_upstream =
bridge.is_device_class_allowed_for_bridge_upstream(port_class);
match (*previously_online, online) {
(Some(false), true) => {
bridge
.handle_interface_online(id.get(), allowed_for_bridge_upstream)
.await
.context("handle interface online")?;
}
(Some(true), false) => {
bridge
.handle_interface_offline(id.get(), allowed_for_bridge_upstream)
.await
.context("handle interface offline")?;
}
(Some(true), true) | (Some(false), false) => {
error!("interface {} changed event indicates no actual change to online ({} before and after)", id, online);
}
(None, true) => {}
(None, false) => {}
}
}
fnet_interfaces_ext::UpdateResult::Removed(
fnet_interfaces_ext::PropertiesAndState {
properties: fnet_interfaces_ext::Properties { id, .. },
state: _,
},
) => {
bridge
.handle_interface_removed(id.get())
.await
.context("handle interface removed")?;
}
fnet_interfaces_ext::UpdateResult::NoChange => {}
}
Ok(())
}
}
pub(super) struct Stub;
#[async_trait(?Send)]
impl Handler for Stub {
async fn handle_event(
&'async_trait mut self,
event: Event,
_events: &'async_trait mut futures::stream::SelectAll<EventStream>,
) -> Result<(), errors::Error> {
panic!("stub handler requested to handle a virtualization event: {:#?}", event)
}
async fn handle_interface_update_result(
&mut self,
_update_result: &fnet_interfaces_ext::UpdateResult<
'_,
(),
fnet_interfaces_ext::DefaultInterest,
>,
) -> Result<(), errors::Error> {
Ok(())
}
}
pub(super) struct BridgeHandle {
id: u64,
control: fnet_interfaces_ext::admin::Control,
}
#[async_trait(?Send)]
pub(super) trait BridgeHandler {
async fn build_bridge(
&self,
interfaces: impl Iterator<Item = u64> + 'async_trait,
upstream_interface: u64,
) -> Result<BridgeHandle, errors::Error>;
async fn destroy_bridge(&self, handle: BridgeHandle) -> Result<(), errors::Error>;
}
pub(super) struct BridgeHandlerImpl {
stack: fnet_stack::StackProxy,
}
impl BridgeHandlerImpl {
pub fn new(stack: fnet_stack::StackProxy) -> Self {
Self { stack }
}
async fn start_dhcpv4_client(&self, bridge_id: u64) -> Result<(), errors::Error> {
self.stack
.set_dhcp_client_enabled(bridge_id, true)
.await
.unwrap_or_else(|err| exit_with_fidl_error(err))
.map_err(|e| anyhow!("failed to start dhcp client: {:?}", e))
.map_err(errors::Error::NonFatal)
}
}
#[async_trait(?Send)]
impl BridgeHandler for BridgeHandlerImpl {
async fn build_bridge(
&self,
interfaces: impl Iterator<Item = u64> + 'async_trait,
upstream_interface: u64,
) -> Result<BridgeHandle, errors::Error> {
let bridge = {
let interfaces: Vec<_> =
interfaces.chain(std::iter::once(upstream_interface)).collect();
info!(
"building bridge with upstream={}, interfaces={:?}",
upstream_interface, interfaces
);
let (control, server_end) = fnet_interfaces_ext::admin::Control::create_endpoints()
.context("create bridge endpoints")
.map_err(errors::Error::Fatal)?;
self.stack
.bridge_interfaces(&interfaces[..], server_end)
.context("calling bridge interfaces")
.map_err(errors::Error::Fatal)?;
let id =
control.get_id().await.context("get bridge id").map_err(errors::Error::Fatal)?;
BridgeHandle { id, control }
};
match self.start_dhcpv4_client(bridge.id).await {
Ok(()) => {}
Err(errors::Error::NonFatal(e)) => {
error!("failed to start DHCPv4 client on bridge: {}", e)
}
Err(errors::Error::Fatal(e)) => return Err(errors::Error::Fatal(e)),
}
let did_enable = bridge
.control
.enable()
.await
.context("call enable")
.map_err(errors::Error::Fatal)?
.map_err(|e| anyhow!("failed to enable interface: {:?}", e))
.map_err(errors::Error::Fatal)?;
assert!(
did_enable,
"the bridge should have been disabled on creation and then enabled by Control.Enable",
);
debug!("enabled bridge interface {}", bridge.id);
Ok(bridge)
}
async fn destroy_bridge(&self, handle: BridgeHandle) -> Result<(), errors::Error> {
let BridgeHandle { id: _, control } = handle;
control
.remove()
.await
.context("calling remove bridge")
.map_err(errors::Error::Fatal)?
.map_err(|err: fnet_interfaces_admin::ControlRemoveError| {
errors::Error::Fatal(anyhow::anyhow!("failed to remove bridge: {:?}", err))
})?;
let _: fnet_interfaces_ext::admin::TerminalError<_> = control.wait_termination().await;
Ok(())
}
}
#[cfg(test)]
mod tests {
use futures::channel::mpsc;
use futures::SinkExt as _;
use test_case::test_case;
use super::*;
#[derive(Copy, Clone, Debug, PartialEq)]
enum Guest {
A,
B,
}
impl Guest {
fn id(&self) -> u64 {
match self {
Self::A => 1,
Self::B => 2,
}
}
}
#[derive(Copy, Clone, Debug, PartialEq)]
enum Upstream {
A,
B,
}
impl Upstream {
fn id(&self) -> u64 {
match self {
Self::A => 11,
Self::B => 12,
}
}
}
#[derive(Debug, PartialEq)]
enum BridgeEvent {
Destroyed,
Created { interfaces: HashSet<u64>, upstream_interface: u64 },
}
impl BridgeEvent {
fn created(interfaces: Vec<Guest>, upstream_interface: Upstream) -> Self {
Self::Created {
interfaces: interfaces.iter().map(Guest::id).collect(),
upstream_interface: upstream_interface.id(),
}
}
fn destroyed() -> Self {
Self::Destroyed
}
}
struct BridgeServer {
id: u64,
_request_stream: fnet_interfaces_admin::ControlRequestStream,
}
impl std::fmt::Debug for BridgeServer {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("BridgeServer")
.field("id", &self.id)
.field("request_stream", &"_")
.finish()
}
}
struct BridgeHandlerTestImplInner {
bridge: Option<BridgeServer>,
events: mpsc::Sender<BridgeEvent>,
}
struct BridgeHandlerTestImpl {
inner: std::cell::RefCell<BridgeHandlerTestImplInner>,
}
impl BridgeHandlerTestImpl {
fn new(events: mpsc::Sender<BridgeEvent>) -> Self {
BridgeHandlerTestImpl {
inner: std::cell::RefCell::new(BridgeHandlerTestImplInner { bridge: None, events }),
}
}
}
#[async_trait(?Send)]
impl BridgeHandler for BridgeHandlerTestImpl {
async fn destroy_bridge(
&self,
BridgeHandle { id, control: _ }: BridgeHandle,
) -> Result<(), errors::Error> {
let BridgeHandlerTestImplInner { bridge, events } = &mut *self.inner.borrow_mut();
let bridge = bridge.take();
assert_eq!(
bridge.map(|BridgeServer { id, _request_stream: _ }| id),
Some(id),
"cannot destroy a non-existent bridge"
);
events.send(BridgeEvent::Destroyed).await.expect("send event");
Ok(())
}
async fn build_bridge(
&self,
interfaces: impl Iterator<Item = u64> + 'async_trait,
upstream_interface: u64,
) -> Result<BridgeHandle, errors::Error> {
let BridgeHandlerTestImplInner { bridge, events } = &mut *self.inner.borrow_mut();
assert_matches::assert_matches!(
*bridge,
None,
"cannot create a bridge since there is already an existing bridge",
);
const BRIDGE_IF: u64 = 99;
let (control, server) =
fnet_interfaces_ext::admin::Control::create_endpoints().expect("create endpoints");
*bridge = Some(BridgeServer { id: BRIDGE_IF, _request_stream: server.into_stream() });
events
.send(BridgeEvent::Created { interfaces: interfaces.collect(), upstream_interface })
.await
.expect("send event");
Ok(BridgeHandle { id: BRIDGE_IF, control })
}
}
enum Action {
AddGuest(Guest),
RemoveGuest(Guest),
UpstreamOnline(Upstream),
UpstreamOffline(Upstream),
RemoveUpstream(Upstream),
}
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
];
"wait for guest"
)]
#[test_case(
[
(Action::AddGuest(Guest::A), vec![]),
(
Action::UpstreamOnline(Upstream::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
];
"wait for upstream"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
(Action::RemoveUpstream(Upstream::A), vec![BridgeEvent::destroyed()]),
];
"destroy bridge when no upstream"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
(
Action::AddGuest(Guest::B),
vec![
BridgeEvent::destroyed(),
BridgeEvent::created([Guest::A, Guest::B].into(), Upstream::A),
],
),
(
Action::RemoveGuest(Guest::B),
vec![
BridgeEvent::destroyed(),
BridgeEvent::created([Guest::A].into(), Upstream::A),
],
),
(
Action::RemoveGuest(Guest::A),
vec![BridgeEvent::destroyed()],
),
];
"multiple interfaces"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
(Action::RemoveGuest(Guest::A), vec![BridgeEvent::destroyed()]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
];
"remember upstream"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
(Action::RemoveUpstream(Upstream::A), vec![BridgeEvent::destroyed()]),
(Action::AddGuest(Guest::B), vec![]),
(Action::RemoveGuest(Guest::A), vec![]),
(
Action::UpstreamOnline(Upstream::A),
vec![BridgeEvent::created([Guest::B].into(), Upstream::A)],
),
];
"remember guest interfaces"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
(Action::RemoveUpstream(Upstream::A), vec![BridgeEvent::destroyed()]),
];
"remove upstream"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
(Action::UpstreamOffline(Upstream::A), vec![]),
];
"upstream offline not removed"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(Action::UpstreamOffline(Upstream::A), vec![]),
(Action::AddGuest(Guest::A), vec![]),
(
Action::UpstreamOnline(Upstream::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
];
"do not bridge with offline upstream"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::A)],
),
(Action::UpstreamOnline(Upstream::B), vec![]),
(
Action::RemoveUpstream(Upstream::A),
vec![
BridgeEvent::destroyed(),
BridgeEvent::created([Guest::A].into(), Upstream::B),
],
),
];
"replace upstream"
)]
#[test_case(
[
(Action::UpstreamOnline(Upstream::A), vec![]),
(Action::UpstreamOnline(Upstream::B), vec![]),
(Action::UpstreamOffline(Upstream::A), vec![]),
(
Action::AddGuest(Guest::A),
vec![BridgeEvent::created([Guest::A].into(), Upstream::B)],
),
];
"replace upstream with no guests"
)]
#[fuchsia::test]
async fn bridge(steps: impl IntoIterator<Item = (Action, Vec<BridgeEvent>)>) {
let (events_tx, mut events_rx) = mpsc::channel(2);
let mut bridge = Bridge {
allowed_bridge_upstream_device_classes: Default::default(),
bridge_handler: BridgeHandlerTestImpl::new(events_tx),
bridge_state: Default::default(),
};
for (action, expected_events) in steps {
match action {
Action::AddGuest(guest) => {
bridge.add_guest_to_bridge(guest.id()).await.expect("add guest to bridge");
}
Action::RemoveGuest(guest) => {
bridge
.remove_guest_from_bridge(guest.id())
.await
.expect("remove guest from bridge");
}
Action::UpstreamOnline(upstream) => {
bridge
.handle_interface_online(upstream.id(), true)
.await
.expect("upstream interface online");
}
Action::UpstreamOffline(upstream) => {
bridge
.handle_interface_offline(upstream.id(), true)
.await
.expect("upstream interface offline");
}
Action::RemoveUpstream(upstream) => {
bridge
.handle_interface_removed(upstream.id())
.await
.expect("upstream interface removed");
}
}
for event in expected_events {
assert_eq!(events_rx.next().await.expect("receive event"), event);
}
let _: mpsc::TryRecvError =
events_rx.try_next().expect_err("got unexpected bridge event");
}
}
}