use crate::reader::ReaderError;
use crate::Inspector;
use async_trait::async_trait;
#[cfg(target_os = "fuchsia")]
pub type SnapshotSource = fuchsia_zircon::Vmo;
#[cfg(not(target_os = "fuchsia"))]
pub type SnapshotSource = Vec<u8>;
#[async_trait]
pub trait ReadableTree: Sized {
async fn tree_names(&self) -> Result<Vec<String>, ReaderError>;
async fn vmo(&self) -> Result<SnapshotSource, ReaderError>;
async fn read_tree(&self, name: &str) -> Result<Self, ReaderError>;
}
#[async_trait]
impl ReadableTree for Inspector {
async fn vmo(&self) -> Result<SnapshotSource, ReaderError> {
self.duplicate_vmo().ok_or(ReaderError::DuplicateVmo)
}
async fn tree_names(&self) -> Result<Vec<String>, ReaderError> {
match self.state() {
None => Ok(vec![]),
Some(state) => {
let state = state.try_lock().map_err(ReaderError::FailedToLockState)?;
let names =
state.callbacks().keys().map(|k| k.to_string()).collect::<Vec<String>>();
Ok(names)
}
}
}
async fn read_tree(&self, name: &str) -> Result<Self, ReaderError> {
let result = self.state().and_then(|state| match state.try_lock() {
Err(_) => None,
Ok(state) => state.callbacks().get(name).map(|cb| cb()),
});
match result {
Some(cb_result) => cb_result.await.map_err(ReaderError::LazyCallback),
None => return Err(ReaderError::FailedToLoadTree(name.to_string())),
}
}
}
#[cfg(target_os = "fuchsia")]
#[async_trait]
impl ReadableTree for fidl_fuchsia_inspect::TreeProxy {
async fn vmo(&self) -> Result<fuchsia_zircon::Vmo, ReaderError> {
let tree_content = self.get_content().await.map_err(|e| ReaderError::Fidl(e.into()))?;
tree_content.buffer.map(|b| b.vmo).ok_or(ReaderError::FetchVmo)
}
async fn tree_names(&self) -> Result<Vec<String>, ReaderError> {
let (name_iterator, server_end) =
fidl::endpoints::create_proxy::<fidl_fuchsia_inspect::TreeNameIteratorMarker>()
.map_err(|e| ReaderError::Fidl(e.into()))?;
self.list_child_names(server_end).map_err(|e| ReaderError::Fidl(e.into()))?;
let mut names = vec![];
loop {
let subset_names =
name_iterator.get_next().await.map_err(|e| ReaderError::Fidl(e.into()))?;
if subset_names.is_empty() {
return Ok(names);
}
names.extend(subset_names.into_iter());
}
}
async fn read_tree(&self, name: &str) -> Result<Self, ReaderError> {
let (child_tree, server_end) =
fidl::endpoints::create_proxy::<fidl_fuchsia_inspect::TreeMarker>()
.map_err(|e| ReaderError::Fidl(e.into()))?;
self.open_child(name, server_end).map_err(|e| ReaderError::Fidl(e.into()))?;
Ok(child_tree)
}
}