use anyhow::Error;
use fidl_fuchsia_hardware_power_statecontrol::{
AdminProxy, AdminRebootResult, AdminRequest, AdminRequestStream,
};
use fuchsia_async as fasync;
use futures::{TryFutureExt, TryStreamExt};
use std::sync::Arc;
pub use fidl_fuchsia_hardware_power_statecontrol::RebootReason;
pub struct MockRebootService {
call_hook: Box<dyn Fn(RebootReason) -> AdminRebootResult + Send + Sync>,
}
impl MockRebootService {
pub fn new(call_hook: Box<dyn Fn(RebootReason) -> AdminRebootResult + Send + Sync>) -> Self {
Self { call_hook }
}
pub async fn run_reboot_service(
self: Arc<Self>,
mut stream: AdminRequestStream,
) -> Result<(), Error> {
while let Some(event) = stream.try_next().await.expect("received request") {
match event {
AdminRequest::Reboot { reason, responder } => {
let result = (self.call_hook)(reason);
responder.send(result)?;
}
_ => {
panic!("unhandled RebootService method {event:?}");
}
}
}
Ok(())
}
pub fn spawn_reboot_service(self: Arc<Self>) -> AdminProxy {
let (proxy, stream) = fidl::endpoints::create_proxy_and_stream::<
fidl_fuchsia_hardware_power_statecontrol::AdminMarker,
>();
fasync::Task::spawn(
self.run_reboot_service(stream)
.unwrap_or_else(|e| panic!("error running reboot service: {e:?}")),
)
.detach();
proxy
}
}
#[cfg(test)]
mod tests {
use super::*;
use fuchsia_async as fasync;
use std::sync::atomic::{AtomicU32, Ordering};
#[fasync::run_singlethreaded(test)]
async fn test_mock_reboot() {
let reboot_service = Arc::new(MockRebootService::new(Box::new(|_| Ok(()))));
let reboot_service_clone = Arc::clone(&reboot_service);
let proxy = reboot_service_clone.spawn_reboot_service();
proxy
.reboot(RebootReason::SystemUpdate)
.await
.expect("made reboot call")
.expect("reboot call succeeded");
}
#[fasync::run_singlethreaded(test)]
async fn test_mock_reboot_fails() {
let reboot_service =
Arc::new(MockRebootService::new(Box::new(|_| Err(zx::Status::INTERNAL.into_raw()))));
let reboot_service_clone = Arc::clone(&reboot_service);
let proxy = reboot_service_clone.spawn_reboot_service();
let reboot_result =
proxy.reboot(RebootReason::SystemUpdate).await.expect("made reboot call");
assert_eq!(reboot_result, Err(zx::Status::INTERNAL.into_raw()));
}
#[fasync::run_singlethreaded(test)]
async fn test_mock_reboot_call_hook() {
let reboot_service = Arc::new(MockRebootService::new(Box::new(|reason| match reason {
RebootReason::UserRequest => Ok(()),
_ => Err(zx::Status::NOT_SUPPORTED.into_raw()),
})));
let reboot_service_clone = Arc::clone(&reboot_service);
let proxy = reboot_service_clone.spawn_reboot_service();
let () = proxy
.reboot(RebootReason::UserRequest)
.await
.expect("made reboot call")
.expect("reboot call succeeded");
let error_reboot_result =
proxy.reboot(RebootReason::SystemUpdate).await.expect("made reboot call");
assert_eq!(error_reboot_result, Err(zx::Status::NOT_SUPPORTED.into_raw()));
}
#[fasync::run_singlethreaded(test)]
async fn test_mock_reboot_with_external_state() {
let called = Arc::new(AtomicU32::new(0));
let called_clone = Arc::clone(&called);
let reboot_service = Arc::new(MockRebootService::new(Box::new(move |_| {
called_clone.fetch_add(1, Ordering::SeqCst);
Ok(())
})));
let reboot_service_clone = Arc::clone(&reboot_service);
let proxy = reboot_service_clone.spawn_reboot_service();
proxy
.reboot(RebootReason::SystemUpdate)
.await
.expect("made reboot call")
.expect("reboot call succeeded");
assert_eq!(called.load(Ordering::SeqCst), 1);
}
}