#![warn(missing_docs)]
pub mod constants;
pub mod devices;
pub mod dhcpv4;
pub mod interfaces;
pub mod ndp;
pub mod nud;
pub mod packets;
pub mod ping;
#[macro_use]
pub mod realms;
use anyhow::Context as _;
use component_events::events::EventStream;
use diagnostics_hierarchy::{filter_hierarchy, DiagnosticsHierarchy, HierarchyMatcher};
use fidl::endpoints::DiscoverableProtocolMarker;
use fidl_fuchsia_diagnostics::Selector;
use fidl_fuchsia_inspect_deprecated::InspectMarker;
use fuchsia_async::{self as fasync, DurationExt as _};
use fuchsia_component::client;
use futures::future::FutureExt as _;
use futures::stream::{Stream, StreamExt as _, TryStreamExt as _};
use futures::{select, Future};
use std::pin::pin;
use {fidl_fuchsia_io as fio, fidl_fuchsia_netemul as fnetemul};
use crate::realms::TestSandboxExt as _;
pub type Result<T = ()> = std::result::Result<T, anyhow::Error>;
pub const ASYNC_EVENT_POSITIVE_CHECK_TIMEOUT: zx::MonotonicDuration =
zx::MonotonicDuration::from_seconds(120);
pub const ASYNC_EVENT_NEGATIVE_CHECK_TIMEOUT: zx::MonotonicDuration =
zx::MonotonicDuration::from_seconds(5);
pub const ASYNC_EVENT_CHECK_INTERVAL: zx::MonotonicDuration =
zx::MonotonicDuration::from_seconds(1);
pub async fn try_any<S: Stream<Item = Result<bool>>>(stream: S) -> Result<bool> {
let stream = pin!(stream);
stream.try_filter(|v| futures::future::ready(*v)).next().await.unwrap_or(Ok(false))
}
pub async fn try_all<S: Stream<Item = Result<bool>>>(stream: S) -> Result<bool> {
let stream = pin!(stream);
stream.try_filter(|v| futures::future::ready(!*v)).next().await.unwrap_or(Ok(true))
}
pub async fn sleep(secs: i64) {
fasync::Timer::new(zx::MonotonicDuration::from_seconds(secs).after_now()).await;
}
pub async fn get_component_stopped_event_stream() -> Result<component_events::events::EventStream> {
EventStream::open_at_path("/events/stopped")
.await
.context("failed to subscribe to `Stopped` events")
}
pub async fn wait_for_component_stopped_with_stream(
event_stream: &mut component_events::events::EventStream,
realm: &netemul::TestRealm<'_>,
component_moniker: &str,
status_matcher: Option<component_events::matcher::ExitStatusMatcher>,
) -> Result<component_events::events::Stopped> {
let matcher = get_child_component_event_matcher(realm, component_moniker)
.await
.context("get child component matcher")?;
matcher.stop(status_matcher).wait::<component_events::events::Stopped>(event_stream).await
}
pub async fn wait_for_component_stopped(
realm: &netemul::TestRealm<'_>,
component_moniker: &str,
status_matcher: Option<component_events::matcher::ExitStatusMatcher>,
) -> Result<component_events::events::Stopped> {
let mut stream = get_component_stopped_event_stream().await?;
wait_for_component_stopped_with_stream(&mut stream, realm, component_moniker, status_matcher)
.await
}
pub async fn get_child_component_event_matcher(
realm: &netemul::TestRealm<'_>,
component_moniker: &str,
) -> Result<component_events::matcher::EventMatcher> {
let realm_moniker = &realm.get_moniker().await.context("calling get moniker")?;
let moniker_for_match =
format!("./{}/{}/{}", NETEMUL_SANDBOX_MONIKER, realm_moniker, component_moniker);
Ok(component_events::matcher::EventMatcher::ok().moniker(moniker_for_match))
}
const NETEMUL_SANDBOX_MONIKER: &str = "sandbox";
pub async fn get_component_moniker<'a>(
realm: &netemul::TestRealm<'a>,
component: &str,
) -> Result<String> {
let realm_moniker = realm.get_moniker().await.context("calling get moniker")?;
Ok([NETEMUL_SANDBOX_MONIKER, &realm_moniker, component].join("/"))
}
pub async fn get_inspect_data(
realm: &netemul::TestRealm<'_>,
component_moniker: impl Into<String>,
tree_selector: impl Into<String>,
) -> Result<diagnostics_hierarchy::DiagnosticsHierarchy> {
let moniker = realm.get_moniker().await.context("calling get moniker")?;
let realm_moniker = selectors::sanitize_string_for_selectors(&moniker);
let mut data = diagnostics_reader::ArchiveReader::new()
.retry(diagnostics_reader::RetryConfig::MinSchemaCount(1))
.add_selector(
diagnostics_reader::ComponentSelector::new(vec![
NETEMUL_SANDBOX_MONIKER.into(),
realm_moniker.into_owned(),
component_moniker.into(),
])
.with_tree_selector(tree_selector.into()),
)
.snapshot::<diagnostics_reader::Inspect>()
.await
.context("snapshot did not return any inspect data")?
.into_iter()
.map(|inspect_data| {
inspect_data.payload.ok_or_else(|| {
anyhow::anyhow!(
"empty inspect payload, metadata errors: {:?}",
inspect_data.metadata.errors
)
})
});
let Some(datum) = data.next() else {
unreachable!("archive reader RetryConfig specifies non-empty")
};
let data: Vec<_> = data.collect();
assert!(
data.is_empty(),
"expected a single inspect entry; got {:?} and also {:?}",
datum,
data
);
datum
}
pub async fn get_inspect_property(
realm: &netemul::TestRealm<'_>,
component_moniker: impl Into<String>,
property_selector: impl Into<String>,
) -> Result<diagnostics_hierarchy::Property> {
let property_selector = property_selector.into();
let hierarchy = get_inspect_data(&realm, component_moniker, property_selector.clone())
.await
.context("getting hierarchy")?;
let property_selector = property_selector.split(&['/', ':']).skip(1).collect::<Vec<_>>();
let property = hierarchy
.get_property_by_path(&property_selector)
.ok_or_else(|| anyhow::anyhow!("property not found in hierarchy: {hierarchy:?}"))?;
Ok(property.clone())
}
pub async fn get_deprecated_netstack2_inspect_data(
diagnostics_dir: &fio::DirectoryProxy,
subdir: &str,
selectors: impl IntoIterator<Item = Selector>,
) -> DiagnosticsHierarchy {
let matcher = HierarchyMatcher::new(selectors.into_iter()).expect("invalid selectors");
loop {
let proxy = client::connect_to_named_protocol_at_dir_root::<InspectMarker>(
diagnostics_dir,
&format!("{subdir}/{}", InspectMarker::PROTOCOL_NAME),
)
.unwrap();
match inspect_fidl_load::load_hierarchy(proxy).await {
Ok(hierarchy) => return filter_hierarchy(hierarchy, &matcher).unwrap(),
Err(err) => {
println!("Failed to load hierarchy, retrying. Error: {err:?}")
}
}
fasync::Timer::new(fasync::MonotonicDuration::from_millis(100)).await;
}
}
pub async fn setup_network<'a, N: realms::Netstack>(
sandbox: &'a netemul::TestSandbox,
name: &'a str,
metric: Option<u32>,
) -> Result<(
netemul::TestNetwork<'a>,
netemul::TestRealm<'a>,
netemul::TestInterface<'a>,
netemul::TestFakeEndpoint<'a>,
)> {
setup_network_with::<N, _>(sandbox, name, metric, std::iter::empty::<fnetemul::ChildDef>())
.await
}
pub async fn setup_network_with<'a, N: realms::Netstack, I>(
sandbox: &'a netemul::TestSandbox,
name: &'a str,
metric: Option<u32>,
children: I,
) -> Result<(
netemul::TestNetwork<'a>,
netemul::TestRealm<'a>,
netemul::TestInterface<'a>,
netemul::TestFakeEndpoint<'a>,
)>
where
I: IntoIterator,
I::Item: Into<fnetemul::ChildDef>,
{
let network = sandbox.create_network(name).await.context("failed to create network")?;
let realm = sandbox
.create_netstack_realm_with::<N, _, _>(name, children)
.context("failed to create netstack realm")?;
let fake_ep = network.create_fake_endpoint()?;
let iface = realm
.join_network_with_if_config(
&network,
name,
netemul::InterfaceConfig { name: Some(name.into()), metric, ..Default::default() },
)
.await
.context("failed to configure networking")?;
Ok((network, realm, iface, fake_ep))
}
pub async fn pause_fake_clock(realm: &netemul::TestRealm<'_>) -> Result<()> {
let fake_clock_control = realm
.connect_to_protocol::<fidl_fuchsia_testing::FakeClockControlMarker>()
.context("failed to connect to FakeClockControl")?;
let () = fake_clock_control.pause().await.context("failed to pause time")?;
Ok(())
}
#[track_caller]
pub fn annotate<'a, 'b: 'a, T>(
fut: impl Future<Output = T> + 'a,
interval: std::time::Duration,
event_name: &'b str,
) -> impl Future<Output = T> + 'a {
let caller = std::panic::Location::caller();
async move {
let mut fut = pin!(fut.fuse());
let event_name = event_name.to_string();
let mut print_fut = pin!(futures::stream::repeat(())
.for_each(|()| async {
fasync::Timer::new(interval).await;
eprintln!("waiting for {} at {}", event_name, caller);
})
.fuse());
let result = select! {
result = fut => result,
() = print_fut => unreachable!("should repeat printing forever"),
};
eprintln!("completed {} at {}", event_name, caller);
result
}
}