use byteorder::{LittleEndian, WriteBytesExt};
use fidl::endpoints::{create_endpoints, create_proxy, ControlHandle, RequestStream, ServerEnd};
use fidl_fuchsia_component_decl::{
Capability, Component, Dictionary, Expose, ExposeDictionary, ExposeProtocol, ParentRef,
Protocol, Ref, SelfRef,
};
use fidl_fuchsia_io::{self as fio, DirectoryMarker};
use fidl_fuchsia_sys2 as fsys2;
use futures::{StreamExt, TryStreamExt};
use moniker::Moniker;
use std::collections::HashMap;
use std::io::Write;
use std::path::Path;
use std::rc::Rc;
use std::str::FromStr;
use std::sync::atomic::{AtomicBool, Ordering};
use zx_status::Status;
#[derive(Default)]
pub struct MockRealmQueryBuilder {
mapping: HashMap<String, Box<MockRealmQueryBuilderInner>>,
}
pub struct MockRealmQueryBuilderInner {
when: Moniker,
moniker: Moniker,
exposes: Vec<Expose>,
parent: Option<Box<MockRealmQueryBuilder>>,
}
impl MockRealmQueryBuilderInner {
pub fn moniker(mut self, moniker: &str) -> Self {
self.moniker = moniker.try_into().unwrap();
self
}
pub fn exposes(mut self, exposes: Vec<Expose>) -> Self {
self.exposes = exposes;
self
}
pub fn add(mut self) -> MockRealmQueryBuilder {
let mut parent = *self.parent.unwrap();
self.parent = None;
parent.mapping.insert(self.when.to_string(), Box::new(self));
parent
}
pub fn serve_exposed_dir(&self, server_end: ServerEnd<DirectoryMarker>, path: &str) {
let mut mock_dir_top = MockDir::new("expose".to_owned());
let mut mock_accessors = MockDir::new("diagnostics-accessors".to_owned());
for expose in &self.exposes {
let Expose::Protocol(ExposeProtocol {
source_name: Some(name), source_dictionary, ..
}) = expose
else {
continue;
};
if matches!(source_dictionary, Some(d) if d == "diagnostics-accessors") {
mock_accessors = mock_accessors.add_entry(MockFile::new_arc(name.to_owned()));
}
}
match path {
"diagnostics-accessors" => {
fuchsia_async::Task::local(async move {
Rc::new(mock_accessors).serve(server_end).await
})
.detach();
}
_ => {
mock_dir_top = mock_dir_top.add_entry(Rc::new(mock_accessors));
fuchsia_async::Task::local(
async move { Rc::new(mock_dir_top).serve(server_end).await },
)
.detach();
}
}
}
fn to_instance(&self) -> fsys2::Instance {
fsys2::Instance {
moniker: Some(self.moniker.to_string()),
url: Some("".to_owned()),
instance_id: None,
resolved_info: Some(fsys2::ResolvedInfo {
resolved_url: Some("".to_owned()),
..Default::default()
}),
..Default::default()
}
}
fn make_manifest(&self) -> Component {
let capabilities = self
.exposes
.iter()
.map(|expose| match expose {
Expose::Protocol(ExposeProtocol {
source_name: Some(name),
source: Some(Ref::Self_(SelfRef)),
..
}) => Capability::Protocol(Protocol {
name: Some(name.clone()),
source_path: Some(format!("/svc/{}", name)),
..Protocol::default()
}),
Expose::Dictionary(ExposeDictionary {
source_name: Some(name),
source: Some(Ref::Self_(SelfRef)),
..
}) => Capability::Dictionary(Dictionary {
name: Some(name.clone()),
source: Some(Ref::Self_(SelfRef)),
..Dictionary::default()
}),
_ => unreachable!("we just add protocols for the test purposes"),
})
.collect::<Vec<_>>();
Component {
capabilities: Some(capabilities),
exposes: Some(self.exposes.clone()),
..Default::default()
}
}
}
impl MockRealmQueryBuilder {
pub fn new() -> Self {
Self::default()
}
pub fn when(self, at: &str) -> MockRealmQueryBuilderInner {
MockRealmQueryBuilderInner {
when: at.try_into().unwrap(),
moniker: Moniker::root(),
exposes: vec![],
parent: Some(Box::new(self)),
}
}
pub fn build(self) -> MockRealmQuery {
MockRealmQuery { mapping: self.mapping }
}
pub fn prefilled() -> Self {
Self::new()
.when("example/component")
.moniker("./example/component")
.exposes(vec![
Expose::Protocol(ExposeProtocol {
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
source_name: Some("fuchsia.diagnostics.ArchiveAccessor".to_owned()),
target_name: Some("fuchsia.diagnostics.ArchiveAccessor".to_owned()),
source_dictionary: Some("diagnostics-accessors".to_owned()),
..Default::default()
}),
Expose::Dictionary(ExposeDictionary {
source_name: Some("diagnostics-accessors".into()),
target_name: Some("diagnostics-accessors".into()),
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
..Default::default()
}),
])
.add()
.when("other/component")
.moniker("./other/component")
.exposes(vec![Expose::Protocol(ExposeProtocol {
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
source_name: Some("src".to_owned()),
target_name: Some("fuchsia.io.SomeOtherThing".to_owned()),
..Default::default()
})])
.add()
.when("other/component")
.moniker("./other/component")
.exposes(vec![Expose::Protocol(ExposeProtocol {
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
source_name: Some("src".to_owned()),
target_name: Some("fuchsia.io.MagicStuff".to_owned()),
..Default::default()
})])
.add()
.when("foo/component")
.moniker("./foo/component")
.exposes(vec![
Expose::Protocol(ExposeProtocol {
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
source_name: Some("fuchsia.diagnostics.ArchiveAccessor.feedback".to_owned()),
target_name: Some("fuchsia.diagnostics.ArchiveAccessor.feedback".to_owned()),
source_dictionary: Some("diagnostics-accessors".to_owned()),
..Default::default()
}),
Expose::Dictionary(ExposeDictionary {
source_name: Some("diagnostics-accessors".into()),
target_name: Some("diagnostics-accessors".into()),
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
..Default::default()
}),
])
.add()
.when("foo/bar/thing:instance")
.moniker("./foo/bar/thing:instance")
.exposes(vec![
Expose::Protocol(ExposeProtocol {
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
source_name: Some("fuchsia.diagnostics.ArchiveAccessor.feedback".to_owned()),
target_name: Some("fuchsia.diagnostics.ArchiveAccessor.feedback".to_owned()),
source_dictionary: Some("diagnostics-accessors".to_owned()),
..Default::default()
}),
Expose::Dictionary(ExposeDictionary {
source_name: Some("diagnostics-accessors".into()),
target_name: Some("diagnostics-accessors".into()),
source: Some(Ref::Self_(SelfRef)),
target: Some(Ref::Parent(ParentRef)),
..Default::default()
}),
])
.add()
}
}
pub struct MockRealmQuery {
mapping: HashMap<String, Box<MockRealmQueryBuilderInner>>,
}
impl Default for MockRealmQuery {
fn default() -> Self {
MockRealmQueryBuilder::prefilled().build()
}
}
impl MockRealmQuery {
pub async fn serve(self: Rc<Self>, object: ServerEnd<fsys2::RealmQueryMarker>) {
let mut stream = object.into_stream();
while let Ok(Some(request)) = stream.try_next().await {
match request {
fsys2::RealmQueryRequest::GetInstance { moniker, responder } => {
let query_moniker = Moniker::from_str(moniker.as_str()).unwrap();
let res = self.mapping.get(&query_moniker.to_string()).unwrap();
responder.send(Ok(&res.to_instance())).unwrap();
}
fsys2::RealmQueryRequest::DeprecatedOpen {
moniker,
dir_type,
object,
responder,
path,
..
} => {
let query_moniker = Moniker::from_str(moniker.as_str()).unwrap();
if let Some(res) = self.mapping.get(&query_moniker.to_string()) {
if dir_type == fsys2::OpenDirType::ExposedDir {
res.serve_exposed_dir(object.into_channel().into(), &path);
}
responder.send(Ok(())).unwrap();
} else {
responder.send(Err(fsys2::OpenError::InstanceNotFound)).unwrap();
}
}
fsys2::RealmQueryRequest::OpenDirectory {
moniker,
dir_type,
object,
responder,
..
} => {
let query_moniker = Moniker::from_str(moniker.as_str()).unwrap();
if let Some(res) = self.mapping.get(&query_moniker.to_string()) {
if dir_type == fsys2::OpenDirType::OutgoingDir {
res.serve_exposed_dir(object, "");
}
responder.send(Ok(())).unwrap();
} else {
responder.send(Err(fsys2::OpenError::InstanceNotFound)).unwrap();
}
}
fsys2::RealmQueryRequest::GetManifest { moniker, responder, .. } => {
let query_moniker = Moniker::from_str(moniker.as_str()).unwrap();
let res = self.mapping.get(&query_moniker.to_string()).unwrap();
let manifest = res.make_manifest();
let manifest = fidl::persist(&manifest).unwrap();
let (client_end, server_end) =
create_endpoints::<fsys2::ManifestBytesIteratorMarker>();
fuchsia_async::Task::spawn(async move {
let mut stream = server_end.into_stream();
let fsys2::ManifestBytesIteratorRequest::Next { responder } =
stream.next().await.unwrap().unwrap();
responder.send(manifest.as_slice()).unwrap();
let fsys2::ManifestBytesIteratorRequest::Next { responder } =
stream.next().await.unwrap().unwrap();
responder.send(&[]).unwrap();
})
.detach();
responder.send(Ok(client_end)).unwrap();
}
fsys2::RealmQueryRequest::GetResolvedDeclaration { moniker, responder, .. } => {
let query_moniker = Moniker::from_str(moniker.as_str()).unwrap();
let res = self.mapping.get(&query_moniker.to_string()).unwrap();
let manifest = res.make_manifest();
let manifest = fidl::persist(&manifest).unwrap();
let (client_end, server_end) =
create_endpoints::<fsys2::ManifestBytesIteratorMarker>();
fuchsia_async::Task::spawn(async move {
let mut stream = server_end.into_stream();
let fsys2::ManifestBytesIteratorRequest::Next { responder } =
stream.next().await.unwrap().unwrap();
responder.send(manifest.as_slice()).unwrap();
let fsys2::ManifestBytesIteratorRequest::Next { responder } =
stream.next().await.unwrap().unwrap();
responder.send(&[]).unwrap();
})
.detach();
responder.send(Ok(client_end)).unwrap();
}
fsys2::RealmQueryRequest::GetAllInstances { responder } => {
let instances: Vec<fsys2::Instance> =
self.mapping.values().map(|m| m.to_instance()).collect();
let (client_end, server_end) =
create_endpoints::<fsys2::InstanceIteratorMarker>();
fuchsia_async::Task::spawn(async move {
let mut stream = server_end.into_stream();
let fsys2::InstanceIteratorRequest::Next { responder } =
stream.next().await.unwrap().unwrap();
responder.send(&instances).unwrap();
let fsys2::InstanceIteratorRequest::Next { responder } =
stream.next().await.unwrap().unwrap();
responder.send(&[]).unwrap();
})
.detach();
responder.send(Ok(client_end)).unwrap();
}
_ => unreachable!("request {:?}", request),
}
}
}
pub async fn get_proxy(self: Rc<Self>) -> fsys2::RealmQueryProxy {
let (proxy, server_end) = create_proxy::<fsys2::RealmQueryMarker>();
fuchsia_async::Task::local(async move { self.serve(server_end).await }).detach();
proxy
}
}
pub trait Entry {
fn open(self: Rc<Self>, flags: fio::OpenFlags, path: &str, object: ServerEnd<fio::NodeMarker>);
fn encode(&self, buf: &mut Vec<u8>);
fn name(&self) -> String;
}
pub struct MockDir {
subdirs: HashMap<String, Rc<dyn Entry>>,
name: String,
at_end: AtomicBool,
}
impl MockDir {
pub fn new(name: String) -> Self {
MockDir { name, subdirs: HashMap::new(), at_end: AtomicBool::new(false) }
}
pub fn new_arc(name: String) -> Rc<Self> {
Rc::new(Self::new(name))
}
pub fn add_entry(mut self, entry: Rc<dyn Entry>) -> Self {
self.subdirs.insert(entry.name(), entry);
self
}
async fn serve(self: Rc<Self>, object: ServerEnd<fio::DirectoryMarker>) {
let mut stream = object.into_stream();
let _ = stream.control_handle().send_on_open_(
Status::OK.into_raw(),
Some(fio::NodeInfoDeprecated::Directory(fio::DirectoryObject {})),
);
while let Ok(Some(request)) = stream.try_next().await {
match request {
fio::DirectoryRequest::Open { flags, mode: _, path, object, .. } => {
self.clone().open(flags, &path, object);
}
fio::DirectoryRequest::Rewind { responder, .. } => {
self.at_end.store(false, Ordering::Relaxed);
responder.send(Status::OK.into_raw()).unwrap();
}
fio::DirectoryRequest::ReadDirents { max_bytes: _, responder, .. } => {
let entries = match self.at_end.compare_exchange(
false,
true,
Ordering::Relaxed,
Ordering::Relaxed,
) {
Ok(false) => encode_entries(&self.subdirs),
Err(true) => Vec::new(),
_ => unreachable!(),
};
responder.send(Status::OK.into_raw(), &entries).unwrap();
}
x => panic!("unsupported request: {:?}", x),
}
}
}
}
fn encode_entries(subdirs: &HashMap<String, Rc<dyn Entry>>) -> Vec<u8> {
let mut buf = Vec::new();
let mut data = subdirs.iter().collect::<Vec<(_, _)>>();
data.sort_by(|a, b| a.0.cmp(b.0));
for (_, entry) in data.iter() {
entry.encode(&mut buf);
}
buf
}
impl Entry for MockDir {
fn open(self: Rc<Self>, flags: fio::OpenFlags, path: &str, object: ServerEnd<fio::NodeMarker>) {
let path = Path::new(path);
let mut path_iter = path.iter();
let segment = if let Some(segment) = path_iter.next() {
if let Some(segment) = segment.to_str() {
segment
} else {
send_error(object, Status::NOT_FOUND);
return;
}
} else {
"."
};
if segment == "." {
fuchsia_async::Task::local(self.serve(ServerEnd::new(object.into_channel()))).detach();
return;
}
if let Some(entry) = self.subdirs.get(segment) {
entry.clone().open(flags, path_iter.as_path().to_str().unwrap(), object);
} else {
send_error(object, Status::NOT_FOUND);
}
}
fn encode(&self, buf: &mut Vec<u8>) {
buf.write_u64::<LittleEndian>(fio::INO_UNKNOWN).expect("writing mockdir ino to work");
buf.write_u8(self.name.len() as u8).expect("writing mockdir size to work");
buf.write_u8(fio::DirentType::Directory.into_primitive())
.expect("writing mockdir type to work");
buf.write_all(self.name.as_ref()).expect("writing mockdir name to work");
}
fn name(&self) -> String {
self.name.clone()
}
}
impl Entry for fio::DirectoryProxy {
fn open(self: Rc<Self>, flags: fio::OpenFlags, path: &str, object: ServerEnd<fio::NodeMarker>) {
let _ = fio::DirectoryProxy::open(&self, flags, fio::ModeType::empty(), path, object);
}
fn encode(&self, _buf: &mut Vec<u8>) {
unimplemented!();
}
fn name(&self) -> String {
unimplemented!();
}
}
struct MockFile {
name: String,
}
impl MockFile {
pub fn new(name: String) -> Self {
MockFile { name }
}
pub fn new_arc(name: String) -> Rc<Self> {
Rc::new(Self::new(name))
}
}
impl Entry for MockFile {
fn open(
self: Rc<Self>,
_flags: fio::OpenFlags,
_path: &str,
_object: ServerEnd<fio::NodeMarker>,
) {
unimplemented!();
}
fn encode(&self, buf: &mut Vec<u8>) {
buf.write_u64::<LittleEndian>(fio::INO_UNKNOWN).expect("writing mockdir ino to work");
buf.write_u8(self.name.len() as u8).expect("writing mockdir size to work");
buf.write_u8(fio::DirentType::Service.into_primitive())
.expect("writing mockdir type to work");
buf.write_all(self.name.as_ref()).expect("writing mockdir name to work");
}
fn name(&self) -> String {
self.name.clone()
}
}
fn send_error(object: ServerEnd<fio::NodeMarker>, status: Status) {
let stream = object.into_stream();
let control_handle = stream.control_handle();
let _ = control_handle.send_on_open_(status.into_raw(), None);
control_handle.shutdown_with_epitaph(status);
}