use crate::input_handler::{InputHandlerStatus, UnhandledInputHandler};
use crate::{consumer_controls_binding, input_device, metrics};
use anyhow::{Context, Error};
use async_trait::async_trait;
use fidl::endpoints::Proxy;
use fuchsia_inspect::health::Reporter;
use futures::channel::mpsc;
use futures::{StreamExt, TryStreamExt};
use metrics_registry::*;
use std::cell::RefCell;
use std::collections::HashMap;
use std::rc::Rc;
use zx::AsHandleRef;
use {
fidl_fuchsia_input_interaction_observation as interaction_observation,
fidl_fuchsia_input_report as fidl_input_report, fidl_fuchsia_ui_input as fidl_ui_input,
fidl_fuchsia_ui_policy as fidl_ui_policy, fuchsia_async as fasync,
};
pub struct MediaButtonsHandler {
inner: RefCell<MediaButtonsHandlerInner>,
pub inspect_status: InputHandlerStatus,
metrics_logger: metrics::MetricsLogger,
aggregator_proxy: Option<interaction_observation::AggregatorProxy>,
}
#[derive(Debug)]
struct MediaButtonsHandlerInner {
pub listeners: HashMap<u32, fidl_ui_policy::MediaButtonsListenerProxy>,
pub last_event: Option<fidl_ui_input::MediaButtonsEvent>,
pub send_event_task_tracker: LocalTaskTracker,
}
#[async_trait(?Send)]
impl UnhandledInputHandler for MediaButtonsHandler {
async fn handle_unhandled_input_event(
self: Rc<Self>,
unhandled_input_event: input_device::UnhandledInputEvent,
) -> Vec<input_device::InputEvent> {
match unhandled_input_event {
input_device::UnhandledInputEvent {
device_event:
input_device::InputDeviceEvent::ConsumerControls(ref media_buttons_event),
device_descriptor:
input_device::InputDeviceDescriptor::ConsumerControls(ref device_descriptor),
event_time,
trace_id: _,
} => {
self.inspect_status.count_received_event(input_device::InputEvent::from(
unhandled_input_event.clone(),
));
let media_buttons_event = Self::create_media_buttons_event(
media_buttons_event,
device_descriptor.device_id,
);
self.send_event_to_listeners(&media_buttons_event).await;
self.inner.borrow_mut().last_event = Some(media_buttons_event);
if let Err(e) = self.report_media_buttons_activity(event_time).await {
log::error!("report_media_buttons_activity failed: {}", e);
}
self.inspect_status.count_handled_event();
vec![input_device::InputEvent::from(unhandled_input_event).into_handled()]
}
_ => vec![input_device::InputEvent::from(unhandled_input_event)],
}
}
fn set_handler_healthy(self: std::rc::Rc<Self>) {
self.inspect_status.health_node.borrow_mut().set_ok();
}
fn set_handler_unhealthy(self: std::rc::Rc<Self>, msg: &str) {
self.inspect_status.health_node.borrow_mut().set_unhealthy(msg);
}
}
impl MediaButtonsHandler {
pub fn new(
input_handlers_node: &fuchsia_inspect::Node,
metrics_logger: metrics::MetricsLogger,
) -> Rc<Self> {
let inspect_status =
InputHandlerStatus::new(input_handlers_node, "media_buttons_handler", false);
let aggregator_proxy = match fuchsia_component::client::connect_to_protocol::<
interaction_observation::AggregatorMarker,
>() {
Ok(proxy) => Some(proxy),
Err(e) => {
log::error!("MedaButtonsHandler failed to connect to fuchsia.input.interaction.observation.Aggregator: {}", e);
None
}
};
Self::new_internal(inspect_status, metrics_logger, aggregator_proxy)
}
fn new_internal(
inspect_status: InputHandlerStatus,
metrics_logger: metrics::MetricsLogger,
aggregator_proxy: Option<interaction_observation::AggregatorProxy>,
) -> Rc<Self> {
let media_buttons_handler = Self {
aggregator_proxy,
inner: RefCell::new(MediaButtonsHandlerInner {
listeners: HashMap::new(),
last_event: None,
send_event_task_tracker: LocalTaskTracker::new(),
}),
inspect_status,
metrics_logger,
};
Rc::new(media_buttons_handler)
}
pub async fn handle_device_listener_registry_request_stream(
self: &Rc<Self>,
mut stream: fidl_ui_policy::DeviceListenerRegistryRequestStream,
) -> Result<(), Error> {
while let Some(request) = stream
.try_next()
.await
.context("Error handling device listener registry request stream")?
{
match request {
fidl_ui_policy::DeviceListenerRegistryRequest::RegisterListener {
listener,
responder,
} => {
let proxy = listener.into_proxy();
self.inner
.borrow_mut()
.listeners
.insert(proxy.as_channel().raw_handle(), proxy.clone());
let proxy_clone = proxy.clone();
if let Some(event) = &self.inner.borrow().last_event {
let event_to_send = event.clone();
let fut = async move {
match proxy_clone.on_event(&event_to_send).await {
Ok(_) => {}
Err(e) => {
log::info!(
"Failed to send media buttons event to listener {:?}",
e
)
}
}
};
let metrics_logger_clone = self.metrics_logger.clone();
self.inner
.borrow()
.send_event_task_tracker
.track(metrics_logger_clone, fasync::Task::local(fut));
}
let _ = responder.send();
}
_ => {}
}
}
Ok(())
}
fn create_media_buttons_event(
event: &consumer_controls_binding::ConsumerControlsEvent,
device_id: u32,
) -> fidl_ui_input::MediaButtonsEvent {
let mut new_event = fidl_ui_input::MediaButtonsEvent {
volume: Some(0),
mic_mute: Some(false),
pause: Some(false),
camera_disable: Some(false),
power: Some(false),
function: Some(false),
device_id: Some(device_id),
..Default::default()
};
for button in &event.pressed_buttons {
match button {
fidl_input_report::ConsumerControlButton::VolumeUp => {
new_event.volume = Some(new_event.volume.unwrap().saturating_add(1));
}
fidl_input_report::ConsumerControlButton::VolumeDown => {
new_event.volume = Some(new_event.volume.unwrap().saturating_sub(1));
}
fidl_input_report::ConsumerControlButton::MicMute => {
new_event.mic_mute = Some(true);
}
fidl_input_report::ConsumerControlButton::Pause => {
new_event.pause = Some(true);
}
fidl_input_report::ConsumerControlButton::CameraDisable => {
new_event.camera_disable = Some(true);
}
fidl_input_report::ConsumerControlButton::Function => {
new_event.function = Some(true);
}
fidl_input_report::ConsumerControlButton::Power => {
new_event.power = Some(true);
}
_ => {}
}
}
new_event
}
async fn send_event_to_listeners(self: &Rc<Self>, event: &fidl_ui_input::MediaButtonsEvent) {
let tracker = &self.inner.borrow().send_event_task_tracker;
for (handle, listener) in &self.inner.borrow().listeners {
let weak_handler = Rc::downgrade(&self);
let listener_clone = listener.clone();
let handle_clone = handle.clone();
let event_to_send = event.clone();
let fut = async move {
match listener_clone.on_event(&event_to_send).await {
Ok(_) => {}
Err(e) => {
if let Some(handler) = weak_handler.upgrade() {
handler.inner.borrow_mut().listeners.remove(&handle_clone);
log::info!(
"Unregistering listener; unable to send MediaButtonsEvent: {:?}",
e
)
}
}
}
};
let metrics_logger_clone = self.metrics_logger.clone();
tracker.track(metrics_logger_clone, fasync::Task::local(fut));
}
}
async fn report_media_buttons_activity(
&self,
event_time: zx::MonotonicInstant,
) -> Result<(), fidl::Error> {
if let Some(proxy) = self.aggregator_proxy.clone() {
return proxy.report_discrete_activity(event_time.into_nanos()).await;
}
Ok(())
}
}
#[derive(Debug)]
pub struct LocalTaskTracker {
sender: mpsc::UnboundedSender<fasync::Task<()>>,
_receiver_task: fasync::Task<()>,
}
impl LocalTaskTracker {
pub fn new() -> Self {
let (sender, receiver) = mpsc::unbounded();
let receiver_task = fasync::Task::local(async move {
receiver.for_each_concurrent(None, |task: fasync::Task<()>| task).await
});
Self { sender, _receiver_task: receiver_task }
}
pub fn track(&self, metrics_logger: metrics::MetricsLogger, task: fasync::Task<()>) {
match self.sender.unbounded_send(task) {
Ok(_) => {}
Err(e) => {
metrics_logger.log_error(
InputPipelineErrorMetricDimensionEvent::MediaButtonErrorWhilePushingTask,
std::format!("Unexpected {e:?} while pushing task"),
);
}
};
}
}
#[cfg(test)]
mod tests {
use crate::input_handler::InputHandler;
use super::*;
use crate::testing_utilities;
use assert_matches::assert_matches;
use fidl::endpoints::create_proxy_and_stream;
use futures::channel::oneshot;
use pretty_assertions::assert_eq;
use std::task::Poll;
use {fidl_fuchsia_input_report as fidl_input_report, fuchsia_async as fasync};
fn spawn_device_listener_registry_server(
handler: Rc<MediaButtonsHandler>,
) -> fidl_ui_policy::DeviceListenerRegistryProxy {
let (device_listener_proxy, device_listener_stream) =
create_proxy_and_stream::<fidl_ui_policy::DeviceListenerRegistryMarker>();
fasync::Task::local(async move {
let _ = handler
.handle_device_listener_registry_request_stream(device_listener_stream)
.await;
})
.detach();
device_listener_proxy
}
fn create_ui_input_media_buttons_event(
volume: Option<i8>,
mic_mute: Option<bool>,
pause: Option<bool>,
camera_disable: Option<bool>,
power: Option<bool>,
function: Option<bool>,
) -> fidl_ui_input::MediaButtonsEvent {
fidl_ui_input::MediaButtonsEvent {
volume,
mic_mute,
pause,
camera_disable,
power,
function,
device_id: Some(0),
..Default::default()
}
}
fn spawn_aggregator_request_server(
expected_time: i64,
) -> Option<interaction_observation::AggregatorProxy> {
let (proxy, mut stream) =
fidl::endpoints::create_proxy_and_stream::<interaction_observation::AggregatorMarker>();
fasync::Task::local(async move {
if let Some(request) = stream.next().await {
match request {
Ok(interaction_observation::AggregatorRequest::ReportDiscreteActivity {
event_time,
responder,
}) => {
assert_eq!(event_time, expected_time);
responder.send().expect("failed to respond");
}
other => panic!("expected aggregator report request, but got {:?}", other),
};
} else {
panic!("AggregatorRequestStream failed.");
}
})
.detach();
Some(proxy)
}
fn make_signalable_task<T: Default + 'static>(
) -> (oneshot::Sender<T>, fasync::Task<()>, Rc<RefCell<T>>) {
let (sender, receiver) = oneshot::channel();
let task_completed = Rc::new(RefCell::new(<T as Default>::default()));
let task_completed_ = task_completed.clone();
let task = fasync::Task::local(async move {
if let Ok(value) = receiver.await {
*task_completed_.borrow_mut() = value;
}
});
(sender, task, task_completed)
}
#[fasync::run_singlethreaded(test)]
async fn register_media_buttons_listener() {
let inspector = fuchsia_inspect::Inspector::default();
let test_node = inspector.root().create_child("test_node");
let inspect_status = InputHandlerStatus::new(
&test_node,
"media_buttons_handler",
false,
);
let (aggregator_proxy, _) =
fidl::endpoints::create_proxy_and_stream::<interaction_observation::AggregatorMarker>();
let media_buttons_handler = Rc::new(MediaButtonsHandler {
aggregator_proxy: Some(aggregator_proxy),
inner: RefCell::new(MediaButtonsHandlerInner {
listeners: HashMap::new(),
last_event: Some(create_ui_input_media_buttons_event(
Some(1),
None,
None,
None,
None,
None,
)),
send_event_task_tracker: LocalTaskTracker::new(),
}),
inspect_status,
metrics_logger: metrics::MetricsLogger::default(),
});
let device_listener_proxy =
spawn_device_listener_registry_server(media_buttons_handler.clone());
let (listener, mut listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>();
let register_listener_fut = async {
let res = device_listener_proxy.register_listener(listener).await;
assert!(res.is_ok());
};
let expected_event =
create_ui_input_media_buttons_event(Some(1), None, None, None, None, None);
let assert_fut = async {
match listener_stream.next().await {
Some(Ok(fidl_ui_policy::MediaButtonsListenerRequest::OnEvent {
event,
responder,
})) => {
assert_eq!(event, expected_event);
responder.send().expect("responder failed.");
}
_ => assert!(false),
}
};
futures::join!(register_listener_fut, assert_fut);
assert_eq!(media_buttons_handler.inner.borrow().listeners.len(), 1);
}
#[fasync::run_singlethreaded(test)]
async fn listener_receives_all_buttons() {
let event_time = zx::MonotonicInstant::get();
let inspector = fuchsia_inspect::Inspector::default();
let test_node = inspector.root().create_child("test_node");
let inspect_status = InputHandlerStatus::new(
&test_node,
"media_buttons_handler",
false,
);
let aggregator_proxy = spawn_aggregator_request_server(event_time.into_nanos());
let media_buttons_handler = MediaButtonsHandler::new_internal(
inspect_status,
metrics::MetricsLogger::default(),
aggregator_proxy,
);
let device_listener_proxy =
spawn_device_listener_registry_server(media_buttons_handler.clone());
let (listener, listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>();
let _ = device_listener_proxy.register_listener(listener).await;
let descriptor = testing_utilities::consumer_controls_device_descriptor();
let input_events = vec![testing_utilities::create_consumer_controls_event(
vec![
fidl_input_report::ConsumerControlButton::VolumeUp,
fidl_input_report::ConsumerControlButton::VolumeDown,
fidl_input_report::ConsumerControlButton::Pause,
fidl_input_report::ConsumerControlButton::MicMute,
fidl_input_report::ConsumerControlButton::CameraDisable,
fidl_input_report::ConsumerControlButton::Function,
fidl_input_report::ConsumerControlButton::Power,
],
event_time,
&descriptor,
)];
let expected_events = vec![create_ui_input_media_buttons_event(
Some(0),
Some(true),
Some(true),
Some(true),
Some(true),
Some(true),
)];
use crate::input_handler::InputHandler as _; assert_input_event_sequence_generates_media_buttons_events!(
input_handler: media_buttons_handler,
input_events: input_events,
expected_events: expected_events,
media_buttons_listener_request_stream: vec![listener_stream],
);
}
#[fasync::run_singlethreaded(test)]
async fn multiple_listeners_receive_event() {
let event_time = zx::MonotonicInstant::get();
let inspector = fuchsia_inspect::Inspector::default();
let test_node = inspector.root().create_child("test_node");
let inspect_status = InputHandlerStatus::new(
&test_node,
"media_buttons_handler",
false,
);
let aggregator_proxy = spawn_aggregator_request_server(event_time.into_nanos());
let media_buttons_handler = MediaButtonsHandler::new_internal(
inspect_status,
metrics::MetricsLogger::default(),
aggregator_proxy,
);
let device_listener_proxy =
spawn_device_listener_registry_server(media_buttons_handler.clone());
let (first_listener, first_listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>();
let (second_listener, second_listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>();
let _ = device_listener_proxy.register_listener(first_listener).await;
let _ = device_listener_proxy.register_listener(second_listener).await;
let descriptor = testing_utilities::consumer_controls_device_descriptor();
let input_events = vec![testing_utilities::create_consumer_controls_event(
vec![fidl_input_report::ConsumerControlButton::VolumeUp],
event_time,
&descriptor,
)];
let expected_events = vec![create_ui_input_media_buttons_event(
Some(1),
Some(false),
Some(false),
Some(false),
Some(false),
Some(false),
)];
use crate::input_handler::InputHandler as _; assert_input_event_sequence_generates_media_buttons_events!(
input_handler: media_buttons_handler,
input_events: input_events,
expected_events: expected_events,
media_buttons_listener_request_stream:
vec![first_listener_stream, second_listener_stream],
);
}
#[fuchsia::test]
fn unregister_listener_if_channel_closed() {
let mut exec = fasync::TestExecutor::new();
let event_time = zx::MonotonicInstant::get();
let inspector = fuchsia_inspect::Inspector::default();
let test_node = inspector.root().create_child("test_node");
let inspect_status = InputHandlerStatus::new(
&test_node,
"media_buttons_handler",
false,
);
let aggregator_proxy = spawn_aggregator_request_server(event_time.into_nanos());
let media_buttons_handler = MediaButtonsHandler::new_internal(
inspect_status,
metrics::MetricsLogger::default(),
aggregator_proxy,
);
let media_buttons_handler_clone = media_buttons_handler.clone();
let mut task = fasync::Task::local(async move {
let device_listener_proxy =
spawn_device_listener_registry_server(media_buttons_handler.clone());
let (first_listener, mut first_listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>(
);
let (second_listener, mut second_listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>(
);
let (third_listener, third_listener_stream) = fidl::endpoints::create_request_stream::<
fidl_ui_policy::MediaButtonsListenerMarker,
>();
let _ = device_listener_proxy.register_listener(first_listener).await;
let _ = device_listener_proxy.register_listener(second_listener).await;
let _ = device_listener_proxy.register_listener(third_listener).await;
assert_eq!(media_buttons_handler.inner.borrow().listeners.len(), 3);
let descriptor = testing_utilities::consumer_controls_device_descriptor();
let input_event = testing_utilities::create_consumer_controls_event(
vec![fidl_input_report::ConsumerControlButton::VolumeUp],
event_time,
&descriptor,
);
let expected_media_buttons_event = create_ui_input_media_buttons_event(
Some(1),
Some(false),
Some(false),
Some(false),
Some(false),
Some(false),
);
std::mem::drop(third_listener_stream);
let _ = media_buttons_handler.clone().handle_input_event(input_event).await;
if let Some(request) = first_listener_stream.next().await {
match request {
Ok(fidl_ui_policy::MediaButtonsListenerRequest::OnEvent {
event,
responder: _,
}) => {
pretty_assertions::assert_eq!(event, expected_media_buttons_event);
}
_ => assert!(false),
}
} else {
assert!(false);
}
if let Some(request) = second_listener_stream.next().await {
match request {
Ok(fidl_ui_policy::MediaButtonsListenerRequest::OnEvent {
event,
responder,
}) => {
pretty_assertions::assert_eq!(event, expected_media_buttons_event);
let _ = responder.send();
}
_ => assert!(false),
}
} else {
assert!(false);
}
});
let _ = exec.run_until_stalled(&mut task);
let _ = exec.run_singlethreaded(async {
assert_eq!(media_buttons_handler_clone.inner.borrow().listeners.len(), 2);
});
}
#[fasync::run_singlethreaded(test)]
async fn stuck_reader_wont_block_input_pipeline() {
let event_time = zx::MonotonicInstant::get();
let inspector = fuchsia_inspect::Inspector::default();
let test_node = inspector.root().create_child("test_node");
let inspect_status = InputHandlerStatus::new(
&test_node,
"media_buttons_handler",
false,
);
let aggregator_proxy = spawn_aggregator_request_server(event_time.into_nanos());
let media_buttons_handler = MediaButtonsHandler::new_internal(
inspect_status,
metrics::MetricsLogger::default(),
aggregator_proxy,
);
let device_listener_proxy =
spawn_device_listener_registry_server(media_buttons_handler.clone());
let (first_listener, mut first_listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>();
let (second_listener, mut second_listener_stream) =
fidl::endpoints::create_request_stream::<fidl_ui_policy::MediaButtonsListenerMarker>();
let _ = device_listener_proxy.register_listener(first_listener).await;
let _ = device_listener_proxy.register_listener(second_listener).await;
let descriptor = testing_utilities::consumer_controls_device_descriptor();
let first_unhandled_input_event = input_device::UnhandledInputEvent {
device_event: input_device::InputDeviceEvent::ConsumerControls(
consumer_controls_binding::ConsumerControlsEvent::new(vec![
fidl_input_report::ConsumerControlButton::VolumeUp,
]),
),
device_descriptor: descriptor.clone(),
event_time,
trace_id: None,
};
let first_expected_media_buttons_event = create_ui_input_media_buttons_event(
Some(1),
Some(false),
Some(false),
Some(false),
Some(false),
Some(false),
);
assert_matches!(
media_buttons_handler
.clone()
.handle_unhandled_input_event(first_unhandled_input_event)
.await
.as_slice(),
[input_device::InputEvent { handled: input_device::Handled::Yes, .. }]
);
let mut save_responder = None;
if let Some(request) = first_listener_stream.next().await {
match request {
Ok(fidl_ui_policy::MediaButtonsListenerRequest::OnEvent { event, responder }) => {
pretty_assertions::assert_eq!(event, first_expected_media_buttons_event);
save_responder = Some(responder);
}
_ => assert!(false),
}
} else {
assert!(false)
}
if let Some(request) = second_listener_stream.next().await {
match request {
Ok(fidl_ui_policy::MediaButtonsListenerRequest::OnEvent { event, responder }) => {
pretty_assertions::assert_eq!(event, first_expected_media_buttons_event);
let _ = responder.send();
}
_ => assert!(false),
}
} else {
assert!(false)
}
let second_unhandled_input_event = input_device::UnhandledInputEvent {
device_event: input_device::InputDeviceEvent::ConsumerControls(
consumer_controls_binding::ConsumerControlsEvent::new(vec![
fidl_input_report::ConsumerControlButton::MicMute,
]),
),
device_descriptor: descriptor.clone(),
event_time,
trace_id: None,
};
let second_expected_media_buttons_event = create_ui_input_media_buttons_event(
Some(0),
Some(true),
Some(false),
Some(false),
Some(false),
Some(false),
);
assert_matches!(
media_buttons_handler
.clone()
.handle_unhandled_input_event(second_unhandled_input_event)
.await
.as_slice(),
[input_device::InputEvent { handled: input_device::Handled::Yes, .. }]
);
if let Some(request) = second_listener_stream.next().await {
match request {
Ok(fidl_ui_policy::MediaButtonsListenerRequest::OnEvent { event, responder }) => {
pretty_assertions::assert_eq!(event, second_expected_media_buttons_event);
let _ = responder.send();
}
_ => assert!(false),
}
} else {
assert!(false)
}
match save_responder {
Some(save_responder) => {
let _ = save_responder.send();
if let Some(request) = first_listener_stream.next().await {
match request {
Ok(fidl_ui_policy::MediaButtonsListenerRequest::OnEvent {
event,
responder: _,
}) => {
pretty_assertions::assert_eq!(
event,
second_expected_media_buttons_event
);
}
_ => assert!(false),
}
} else {
assert!(false)
}
}
None => {
assert!(false)
}
}
}
#[fuchsia::test]
fn local_task_tracker_test() -> Result<(), Error> {
let mut exec = fasync::TestExecutor::new();
let (mut sender_1, task_1, completed_1) = make_signalable_task::<bool>();
let (sender_2, task_2, completed_2) = make_signalable_task::<bool>();
let mut tracker = LocalTaskTracker::new();
tracker.track(metrics::MetricsLogger::default(), task_1);
tracker.track(metrics::MetricsLogger::default(), task_2);
assert_matches!(exec.run_until_stalled(&mut tracker._receiver_task), Poll::Pending);
assert_eq!(Rc::strong_count(&completed_1), 2);
assert_eq!(Rc::strong_count(&completed_2), 2);
assert!(!sender_1.is_canceled());
assert!(!sender_2.is_canceled());
assert!(sender_2.send(true).is_ok());
assert_matches!(exec.run_until_stalled(&mut tracker._receiver_task), Poll::Pending);
assert_eq!(Rc::strong_count(&completed_1), 2);
assert_eq!(Rc::strong_count(&completed_2), 1);
assert_eq!(*completed_1.borrow(), false);
assert_eq!(*completed_2.borrow(), true);
assert!(!sender_1.is_canceled());
drop(tracker);
let mut sender_1_cancellation = sender_1.cancellation();
assert_matches!(exec.run_until_stalled(&mut sender_1_cancellation), Poll::Ready(()));
assert_eq!(Rc::strong_count(&completed_1), 1);
assert!(sender_1.is_canceled());
Ok(())
}
#[fasync::run_singlethreaded(test)]
async fn media_buttons_handler_initialized_with_inspect_node() {
let inspector = fuchsia_inspect::Inspector::default();
let fake_handlers_node = inspector.root().create_child("input_handlers_node");
let _handler =
MediaButtonsHandler::new(&fake_handlers_node, metrics::MetricsLogger::default());
diagnostics_assertions::assert_data_tree!(inspector, root: {
input_handlers_node: {
media_buttons_handler: {
events_received_count: 0u64,
events_handled_count: 0u64,
last_received_timestamp_ns: 0u64,
"fuchsia.inspect.Health": {
status: "STARTING_UP",
start_timestamp_nanos: diagnostics_assertions::AnyProperty
},
}
}
});
}
#[fasync::run_singlethreaded(test)]
async fn media_buttons_handler_inspect_counts_events() {
let inspector = fuchsia_inspect::Inspector::default();
let fake_handlers_node = inspector.root().create_child("input_handlers_node");
let media_buttons_handler =
MediaButtonsHandler::new(&fake_handlers_node, metrics::MetricsLogger::default());
let descriptor = testing_utilities::consumer_controls_device_descriptor();
let events = vec![
input_device::InputEvent {
device_event: input_device::InputDeviceEvent::ConsumerControls(
consumer_controls_binding::ConsumerControlsEvent::new(vec![
fidl_input_report::ConsumerControlButton::VolumeUp,
]),
),
device_descriptor: descriptor.clone(),
event_time: zx::MonotonicInstant::get(),
handled: input_device::Handled::No,
trace_id: None,
},
input_device::InputEvent {
device_event: input_device::InputDeviceEvent::ConsumerControls(
consumer_controls_binding::ConsumerControlsEvent::new(vec![
fidl_input_report::ConsumerControlButton::VolumeUp,
]),
),
device_descriptor: descriptor.clone(),
event_time: zx::MonotonicInstant::get(),
handled: input_device::Handled::Yes,
trace_id: None,
},
input_device::InputEvent {
device_event: input_device::InputDeviceEvent::ConsumerControls(
consumer_controls_binding::ConsumerControlsEvent::new(vec![
fidl_input_report::ConsumerControlButton::VolumeDown,
]),
),
device_descriptor: descriptor.clone(),
event_time: zx::MonotonicInstant::get(),
handled: input_device::Handled::No,
trace_id: None,
},
];
let last_event_timestamp: u64 =
events[2].clone().event_time.into_nanos().try_into().unwrap();
for event in events {
media_buttons_handler.clone().handle_input_event(event).await;
}
diagnostics_assertions::assert_data_tree!(inspector, root: {
input_handlers_node: {
media_buttons_handler: {
events_received_count: 2u64,
events_handled_count: 2u64,
last_received_timestamp_ns: last_event_timestamp,
"fuchsia.inspect.Health": {
status: "STARTING_UP",
start_timestamp_nanos: diagnostics_assertions::AnyProperty
},
}
}
});
}
}