use crate::constants;
use crate::diagnostics::{GlobalConnectionStats, TRACE_CATEGORY};
use crate::identity::ComponentIdentity;
use crate::inspect::collector::{self as collector, InspectData};
use crate::pipeline::ComponentAllowlist;
use diagnostics_data::{self as schema, InspectHandleName};
use diagnostics_hierarchy::DiagnosticsHierarchy;
use fidl::endpoints::Proxy;
use flyweights::FlyStr;
use fuchsia_async::{self as fasync, DurationExt, TimeoutExt};
use fuchsia_inspect::reader::snapshot::{Snapshot, SnapshotTree};
use futures::channel::oneshot;
use futures::{FutureExt, Stream};
use log::warn;
use selectors::SelectorExt;
use std::collections::HashMap;
use std::future::Future;
use std::sync::{Arc, Weak};
use std::time::Duration;
use zx::{self as zx, AsHandleRef};
use {
fidl_fuchsia_diagnostics as fdiagnostics, fidl_fuchsia_inspect as finspect,
fidl_fuchsia_io as fio, fuchsia_trace as ftrace, inspect_fidl_load as deprecated_inspect,
};
#[derive(Debug)]
pub enum InspectHandle {
Tree {
proxy: finspect::TreeProxy,
name: Option<FlyStr>,
},
Directory {
proxy: fio::DirectoryProxy,
},
Escrow {
vmo: Arc<zx::Vmo>,
name: Option<FlyStr>,
token: finspect::EscrowToken,
related_koid: zx::Koid,
},
}
impl InspectHandle {
fn is_closed(&self) -> bool {
match self {
Self::Directory { proxy } => proxy.as_channel().is_closed(),
Self::Tree { proxy, .. } => proxy.as_channel().is_closed(),
Self::Escrow { .. } => false,
}
}
async fn on_closed(&self) -> Result<zx::Signals, zx::Status> {
match self {
Self::Tree { proxy, .. } => proxy.on_closed().await,
Self::Directory { proxy, .. } => proxy.on_closed().await,
Self::Escrow { token, .. } => {
fasync::OnSignals::new(&token.token, fidl::Signals::OBJECT_PEER_CLOSED).await
}
}
}
pub fn koid(&self) -> zx::Koid {
match self {
Self::Directory { proxy } => {
proxy.as_channel().as_handle_ref().get_koid().expect("DirectoryProxy has koid")
}
Self::Tree { proxy, .. } => {
proxy.as_channel().as_handle_ref().get_koid().expect("TreeProxy has koid")
}
Self::Escrow { related_koid, .. } => *related_koid,
}
}
pub fn tree<T: Into<FlyStr>>(proxy: finspect::TreeProxy, name: Option<T>) -> Self {
InspectHandle::Tree { proxy, name: name.map(|n| n.into()) }
}
pub fn escrow<T: Into<FlyStr>>(
vmo: zx::Vmo,
token: finspect::EscrowToken,
name: Option<T>,
) -> Self {
let related_koid = token.token.basic_info().unwrap().related_koid;
InspectHandle::Escrow {
vmo: Arc::new(vmo),
name: name.map(|n| n.into()),
token,
related_koid,
}
}
pub fn directory(proxy: fio::DirectoryProxy) -> Self {
InspectHandle::Directory { proxy }
}
}
struct StoredInspectHandle {
handle: Arc<InspectHandle>,
_control: oneshot::Sender<()>,
}
#[derive(Default)]
pub struct InspectArtifactsContainer {
inspect_handles: HashMap<zx::Koid, StoredInspectHandle>,
}
impl InspectArtifactsContainer {
pub fn remove_handle(&mut self, koid: zx::Koid) -> (Option<Arc<InspectHandle>>, usize) {
let stored = self.inspect_handles.remove(&koid);
(stored.map(|stored| stored.handle), self.inspect_handles.len())
}
pub fn push_handle(
&mut self,
handle: InspectHandle,
on_closed: impl FnOnce(zx::Koid),
) -> Option<impl Future<Output = ()>> {
if !self.inspect_handles.is_empty() && matches!(handle, InspectHandle::Directory { .. }) {
return None;
}
let (control_snd, control_rcv) = oneshot::channel();
let handle = Arc::new(handle);
let stored = StoredInspectHandle { _control: control_snd, handle: Arc::clone(&handle) };
let koid = handle.koid();
self.inspect_handles.insert(koid, stored);
Some(async move {
if !handle.is_closed() {
let closed_fut = handle.on_closed();
futures::pin_mut!(closed_fut);
let _ = futures::future::select(closed_fut, control_rcv).await;
}
on_closed(koid);
})
}
pub fn create_unpopulated(
&self,
identity: &Arc<ComponentIdentity>,
allowlist: ComponentAllowlist,
dynamic_selectors: &Option<impl AsRef<[fdiagnostics::Selector]>>,
) -> Option<UnpopulatedInspectDataContainer> {
if self.inspect_handles.is_empty() {
return None;
}
if allowlist.all_filtered_out() {
return None;
}
let mut dynamic_selectors = dynamic_selectors.as_ref().map(|all_components| {
all_components
.as_ref()
.iter()
.filter(|s| matches!(identity.moniker.matches_selector(s), Ok(true)))
.peekable()
});
if let Some(None) = dynamic_selectors.as_mut().map(|s| s.peek()) {
return None;
}
Some(UnpopulatedInspectDataContainer {
identity: Arc::clone(identity),
component_allowlist: allowlist,
inspect_handles: self
.inspect_handles
.values()
.filter(|s| {
Self::name_filters_satisfied(s.handle.as_ref(), dynamic_selectors.clone())
})
.map(|s| Arc::downgrade(&s.handle))
.collect::<Vec<_>>(),
})
}
fn name_filters_satisfied<'a>(
handle: &InspectHandle,
mut dynamic_selectors: Option<impl Iterator<Item = &'a fdiagnostics::Selector>>,
) -> bool {
let Some(ref mut selectors) = dynamic_selectors else {
return true;
};
match handle {
InspectHandle::Tree { name, .. } | InspectHandle::Escrow { name, .. } => {
selectors.any(|s| {
let name = match name.as_ref() {
Some(fly) => fly.as_str(),
None => finspect::DEFAULT_TREE_NAME,
};
selectors::match_tree_name_against_selector(name, s)
})
}
InspectHandle::Directory { .. } => selectors.any(|s| {
matches!(s.tree_names.as_ref(), None | Some(fdiagnostics::TreeNames::All(_)))
}),
}
}
}
#[cfg(test)]
impl InspectArtifactsContainer {
pub(crate) fn handles(&self) -> impl ExactSizeIterator<Item = &InspectHandle> {
self.inspect_handles.values().map(|stored| stored.handle.as_ref())
}
}
static TIMEOUT_MESSAGE: &str = "Exceeded per-component time limit for fetching diagnostics data";
#[derive(Debug)]
pub enum ReadSnapshot {
Single(Snapshot),
Tree(SnapshotTree),
Finished(DiagnosticsHierarchy),
}
#[derive(Debug)]
pub struct SnapshotData {
pub name: Option<InspectHandleName>,
pub timestamp: zx::BootInstant,
pub errors: Vec<schema::InspectError>,
pub snapshot: Option<ReadSnapshot>,
pub escrowed: bool,
}
impl SnapshotData {
async fn new(
name: Option<InspectHandleName>,
data: InspectData,
lazy_child_timeout: zx::MonotonicDuration,
identity: &ComponentIdentity,
parent_trace_id: ftrace::Id,
) -> SnapshotData {
let trace_id = ftrace::Id::random();
let _trace_guard = ftrace::async_enter!(
trace_id,
TRACE_CATEGORY,
c"SnapshotData::new",
"parent_trace_id" => u64::from(parent_trace_id),
"trace_id" => u64::from(trace_id),
"moniker" => identity.to_string().as_ref(),
"filename" => name
.as_ref()
.and_then(InspectHandleName::as_filename)
.unwrap_or(""),
"name" => name
.as_ref()
.and_then(InspectHandleName::as_name)
.unwrap_or("")
);
match data {
InspectData::Tree(tree) => {
let lazy_child_timeout =
Duration::from_nanos(lazy_child_timeout.into_nanos() as u64);
match SnapshotTree::try_from_with_timeout(&tree, lazy_child_timeout).await {
Ok(snapshot_tree) => {
SnapshotData::successful(ReadSnapshot::Tree(snapshot_tree), name, false)
}
Err(e) => SnapshotData::failed(
schema::InspectError { message: format!("{e:?}") },
name,
false,
),
}
}
InspectData::DeprecatedFidl(inspect_proxy) => {
match deprecated_inspect::load_hierarchy(inspect_proxy).await {
Ok(hierarchy) => {
SnapshotData::successful(ReadSnapshot::Finished(hierarchy), name, false)
}
Err(e) => SnapshotData::failed(
schema::InspectError { message: format!("{e:?}") },
name,
false,
),
}
}
InspectData::Vmo { data: vmo, escrowed } => match Snapshot::try_from(vmo.as_ref()) {
Ok(snapshot) => {
SnapshotData::successful(ReadSnapshot::Single(snapshot), name, escrowed)
}
Err(e) => SnapshotData::failed(
schema::InspectError { message: format!("{e:?}") },
name,
escrowed,
),
},
InspectData::File(contents) => match Snapshot::try_from(contents) {
Ok(snapshot) => {
SnapshotData::successful(ReadSnapshot::Single(snapshot), name, false)
}
Err(e) => SnapshotData::failed(
schema::InspectError { message: format!("{e:?}") },
name,
false,
),
},
}
}
fn successful(
snapshot: ReadSnapshot,
name: Option<InspectHandleName>,
escrowed: bool,
) -> SnapshotData {
SnapshotData {
name,
timestamp: zx::BootInstant::get(),
errors: Vec::new(),
snapshot: Some(snapshot),
escrowed,
}
}
fn failed(
error: schema::InspectError,
name: Option<InspectHandleName>,
escrowed: bool,
) -> SnapshotData {
SnapshotData {
name,
timestamp: zx::BootInstant::get(),
errors: vec![error],
snapshot: None,
escrowed,
}
}
}
pub struct PopulatedInspectDataContainer {
pub identity: Arc<ComponentIdentity>,
pub snapshot: SnapshotData,
pub component_allowlist: ComponentAllowlist,
}
enum Status {
Begin,
Pending(collector::InspectHandleDeque),
}
struct State {
status: Status,
unpopulated: Arc<UnpopulatedInspectDataContainer>,
batch_timeout: zx::MonotonicDuration,
elapsed_time: zx::MonotonicDuration,
global_stats: Arc<GlobalConnectionStats>,
trace_guard: Arc<Option<ftrace::AsyncScope>>,
trace_id: ftrace::Id,
}
impl State {
fn into_pending(
self,
pending: collector::InspectHandleDeque,
start_time: zx::MonotonicInstant,
) -> Self {
Self {
unpopulated: self.unpopulated,
status: Status::Pending(pending),
batch_timeout: self.batch_timeout,
global_stats: self.global_stats,
elapsed_time: self.elapsed_time + (zx::MonotonicInstant::get() - start_time),
trace_guard: self.trace_guard,
trace_id: self.trace_id,
}
}
fn add_elapsed_time(&mut self, start_time: zx::MonotonicInstant) {
self.elapsed_time += zx::MonotonicInstant::get() - start_time
}
async fn iterate(
mut self,
start_time: zx::MonotonicInstant,
) -> Option<(PopulatedInspectDataContainer, State)> {
loop {
match &mut self.status {
Status::Begin => {
let data_map =
collector::populate_data_map(&self.unpopulated.inspect_handles).await;
self = self.into_pending(data_map, start_time);
}
Status::Pending(ref mut pending) => match pending.pop_front() {
None => {
self.global_stats.record_component_duration(
self.unpopulated.identity.moniker.to_string(),
self.elapsed_time + (zx::MonotonicInstant::get() - start_time),
);
return None;
}
Some((name, data)) => {
let snapshot = SnapshotData::new(
name,
data,
self.batch_timeout / constants::LAZY_NODE_TIMEOUT_PROPORTION,
&self.unpopulated.identity,
self.trace_id,
)
.await;
let result = PopulatedInspectDataContainer {
identity: Arc::clone(&self.unpopulated.identity),
snapshot,
component_allowlist: self.unpopulated.component_allowlist.clone(),
};
self.add_elapsed_time(start_time);
return Some((result, self));
}
},
}
}
}
}
#[derive(Debug)]
pub struct UnpopulatedInspectDataContainer {
pub identity: Arc<ComponentIdentity>,
pub inspect_handles: Vec<Weak<InspectHandle>>,
pub component_allowlist: ComponentAllowlist,
}
impl UnpopulatedInspectDataContainer {
pub fn populate(
self,
timeout: i64,
global_stats: Arc<GlobalConnectionStats>,
parent_trace_id: ftrace::Id,
) -> impl Stream<Item = PopulatedInspectDataContainer> {
let trace_id = ftrace::Id::random();
let trace_guard = ftrace::async_enter!(
trace_id,
TRACE_CATEGORY,
c"ReaderServer::stream.populate",
"parent_trace_id" => u64::from(parent_trace_id),
"trace_id" => u64::from(trace_id),
"moniker" => self.identity.to_string().as_ref()
);
let this = Arc::new(self);
let state = State {
status: Status::Begin,
unpopulated: this,
batch_timeout: zx::MonotonicDuration::from_seconds(timeout),
global_stats,
elapsed_time: zx::MonotonicDuration::ZERO,
trace_guard: Arc::new(trace_guard),
trace_id,
};
futures::stream::unfold(state, |state| {
let unpopulated_for_timeout = Arc::clone(&state.unpopulated);
let timeout = state.batch_timeout;
let elapsed_time = state.elapsed_time;
let global_stats = Arc::clone(&state.global_stats);
let start_time = zx::MonotonicInstant::get();
let trace_guard = Arc::clone(&state.trace_guard);
let trace_id = state.trace_id;
state
.iterate(start_time)
.on_timeout((timeout - elapsed_time).after_now(), move || {
warn!(identity:? = unpopulated_for_timeout.identity.moniker; "{TIMEOUT_MESSAGE}");
global_stats.add_timeout();
let result = PopulatedInspectDataContainer {
identity: Arc::clone(&unpopulated_for_timeout.identity),
component_allowlist: unpopulated_for_timeout.component_allowlist.clone(),
snapshot: SnapshotData::failed(
schema::InspectError { message: TIMEOUT_MESSAGE.to_string() },
None,
false,
),
};
Some((
result,
State {
status: Status::Pending(collector::InspectHandleDeque::new()),
unpopulated: unpopulated_for_timeout,
batch_timeout: timeout,
global_stats,
elapsed_time: elapsed_time + (zx::MonotonicInstant::get() - start_time),
trace_guard,
trace_id,
},
))
})
.boxed()
})
}
}
#[cfg(test)]
mod test {
use super::*;
use fuchsia_inspect::Node;
use futures::StreamExt;
use std::sync::LazyLock;
static EMPTY_IDENTITY: LazyLock<Arc<ComponentIdentity>> =
LazyLock::new(|| Arc::new(ComponentIdentity::unknown()));
static O_K_IDENTITY: LazyLock<Arc<ComponentIdentity>> =
LazyLock::new(|| Arc::new(ComponentIdentity::from(vec!["o", "k"])));
#[fuchsia::test]
async fn population_times_out() {
let (directory, mut stream) =
fidl::endpoints::create_proxy_and_stream::<fio::DirectoryMarker>();
fasync::Task::spawn(async move {
while stream.next().await.is_some() {
fasync::Timer::new(fasync::MonotonicInstant::after(
zx::MonotonicDuration::from_seconds(100000),
))
.await;
}
})
.detach();
let handle = Arc::new(InspectHandle::directory(directory));
let container = UnpopulatedInspectDataContainer {
identity: EMPTY_IDENTITY.clone(),
inspect_handles: vec![Arc::downgrade(&handle)],
component_allowlist: ComponentAllowlist::new_disabled(),
};
let mut stream = container.populate(
0,
Arc::new(GlobalConnectionStats::new(Node::default())),
ftrace::Id::random(),
);
let res = stream.next().await.unwrap();
assert_eq!(res.snapshot.name, None);
assert_eq!(
res.snapshot.errors,
vec![schema::InspectError { message: TIMEOUT_MESSAGE.to_string() }]
);
}
#[fuchsia::test]
async fn no_inspect_files_do_not_give_an_error_response() {
let directory = Arc::new(InspectHandle::directory(
fuchsia_fs::directory::open_in_namespace("/tmp", fio::PERM_READABLE).unwrap(),
));
let container = UnpopulatedInspectDataContainer {
identity: EMPTY_IDENTITY.clone(),
inspect_handles: vec![Arc::downgrade(&directory)],
component_allowlist: ComponentAllowlist::new_disabled(),
};
let mut stream = container.populate(
1000000,
Arc::new(GlobalConnectionStats::new(Node::default())),
ftrace::Id::random(),
);
assert!(stream.next().await.is_none());
}
#[fuchsia::test]
fn only_one_directory_proxy_is_populated() {
let _executor = fuchsia_async::LocalExecutor::new();
let (directory, _) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
let mut container = InspectArtifactsContainer::default();
let _rx = container.push_handle(InspectHandle::directory(directory), |_| {});
let (directory2, _) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
assert!(container.push_handle(InspectHandle::directory(directory2), |_| {}).is_none());
}
fn inspect_artifacts_container_with_one_dir() -> InspectArtifactsContainer {
let (directory, _) = fidl::endpoints::create_proxy_and_stream::<fio::DirectoryMarker>();
let handle = InspectHandle::directory(directory);
let mut container = InspectArtifactsContainer::default();
container.push_handle(handle, |_| {});
container
}
#[fuchsia::test]
fn prefilter_on_names_for_directories() {
let _executor = fuchsia_async::LocalExecutor::new();
let all_selector = fdiagnostics::Selector {
tree_names: Some(fdiagnostics::TreeNames::All(fdiagnostics::All {})),
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![all_selector]);
let iac = inspect_artifacts_container_with_one_dir();
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(1, container.inspect_handles.len());
let selector_with_wrong_tree_name = fdiagnostics::Selector {
tree_names: Some(fdiagnostics::TreeNames::Some(vec!["a".to_string()])),
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![selector_with_wrong_tree_name]);
let iac = inspect_artifacts_container_with_one_dir();
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(0, container.inspect_handles.len());
let selector_with_no_tree_names = fdiagnostics::Selector {
tree_names: None,
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![selector_with_no_tree_names]);
let iac = inspect_artifacts_container_with_one_dir();
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(1, container.inspect_handles.len());
}
fn inspect_artifacts_container_with_n_trees<'a>(
tree_names: impl IntoIterator<Item = &'a str>,
) -> InspectArtifactsContainer {
let mut container = InspectArtifactsContainer::default();
for name in tree_names {
let (proxy, _) = fidl::endpoints::create_proxy::<finspect::TreeMarker>();
let handle = InspectHandle::tree(proxy, Some(name));
container.push_handle(handle, |_| {});
}
container
}
#[fuchsia::test]
fn name_filter_with_no_matches() {
let _executor = fuchsia_async::LocalExecutor::new();
let b_only = fdiagnostics::Selector {
tree_names: Some(fdiagnostics::TreeNames::Some(vec!["b".to_string()])),
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![b_only]);
let iac = inspect_artifacts_container_with_n_trees(["a"]);
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(0, container.inspect_handles.len());
}
#[fuchsia::test]
fn all_name_filter_matches_everything() {
let _executor = fuchsia_async::LocalExecutor::new();
let iac = inspect_artifacts_container_with_n_trees(["a", "b"]);
let container = iac
.create_unpopulated(
&O_K_IDENTITY,
ComponentAllowlist::new_disabled(),
&None::<&[fdiagnostics::Selector]>,
)
.unwrap();
assert_eq!(2, container.inspect_handles.len());
let all_selector = fdiagnostics::Selector {
tree_names: Some(fdiagnostics::TreeNames::All(fdiagnostics::All {})),
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![all_selector]);
let iac = inspect_artifacts_container_with_n_trees(["a", "b"]);
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(2, container.inspect_handles.len());
}
#[fuchsia::test]
fn none_name_filter_matches_root_only() {
let _executor = fuchsia_async::LocalExecutor::new();
let lists_each_name = fdiagnostics::Selector {
tree_names: None,
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![lists_each_name]);
let iac = inspect_artifacts_container_with_n_trees(["a", "b", "root"]);
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(1, container.inspect_handles.len());
}
#[fuchsia::test]
fn select_subset_of_names() {
let _executor = fuchsia_async::LocalExecutor::new();
let a_only = fdiagnostics::Selector {
tree_names: Some(fdiagnostics::TreeNames::Some(vec!["a".to_string()])),
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![a_only]);
let iac = inspect_artifacts_container_with_n_trees(["a", "b"]);
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(1, container.inspect_handles.len());
match container.inspect_handles[0].upgrade().unwrap().as_ref() {
InspectHandle::Tree { name: Some(n), .. } => assert_eq!(n.as_str(), "a"),
bad_handle => {
panic!("filtering failed, got {bad_handle:?}");
}
};
}
#[fuchsia::test]
fn names_are_case_sensitive() {
let _executor = fuchsia_async::LocalExecutor::new();
let a_but_wrong_case = fdiagnostics::Selector {
tree_names: Some(fdiagnostics::TreeNames::Some(vec!["A".to_string()])),
component_selector: Some(fdiagnostics::ComponentSelector {
moniker_segments: Some(vec![
fdiagnostics::StringSelector::ExactMatch("o".to_string()),
fdiagnostics::StringSelector::ExactMatch("k".to_string()),
]),
..Default::default()
}),
tree_selector: Some(fdiagnostics::TreeSelector::SubtreeSelector(
fdiagnostics::SubtreeSelector {
node_path: vec![fdiagnostics::StringSelector::ExactMatch("root".to_string())],
},
)),
..Default::default()
};
let selectors = Some(vec![a_but_wrong_case]);
let iac = inspect_artifacts_container_with_n_trees(["a", "b"]);
let container = iac
.create_unpopulated(&O_K_IDENTITY, ComponentAllowlist::new_disabled(), &selectors)
.unwrap();
assert_eq!(0, container.inspect_handles.len());
}
}