// Copyright 2020 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
use anyhow::{Context as _, Error};
use fidl_fuchsia_bluetooth_le::{
ConnectionProxy, PeripheralEvent, PeripheralMarker, PeripheralProxy,
};
use fidl_fuchsia_hardware_bluetooth::EmulatorProxy;
use fuchsia_bluetooth::expectation::asynchronous::{
expectable, Expectable, ExpectableExt, ExpectableState,
};
use fuchsia_bluetooth::types::le::Peer;
use futures::future::{self, BoxFuture};
use futures::{FutureExt, TryFutureExt, TryStreamExt};
use std::ops::{Deref, DerefMut};
use std::sync::Arc;
use test_harness::{SharedState, TestHarness, SHARED_STATE_TEST_COMPONENT_INDEX};
use crate::core_realm::{CoreRealm, SHARED_STATE_INDEX};
use crate::emulator::{watch_advertising_states, EmulatorState};
use crate::host_watcher::ActivatedFakeHost;
/// A snapshot of the current LE peripheral procedure states of the controller.
#[derive(Clone, Debug, Default)]
pub struct PeripheralState {
emulator_state: EmulatorState,
/// Observed peer connections.
pub connections: Vec<(Peer, ConnectionProxy)>,
}
impl PeripheralState {
/// Resets to the default state.
pub fn reset(&mut self) {
self.emulator_state = EmulatorState::default();
self.connections.clear();
}
}
impl AsMut<EmulatorState> for PeripheralState {
fn as_mut(&mut self) -> &mut EmulatorState {
&mut self.emulator_state
}
}
impl AsRef<EmulatorState> for PeripheralState {
fn as_ref(&self) -> &EmulatorState {
&self.emulator_state
}
}
#[derive(Clone)]
pub struct PeripheralHarness(Expectable<PeripheralState, Aux>);
impl Deref for PeripheralHarness {
type Target = Expectable<PeripheralState, Aux>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl DerefMut for PeripheralHarness {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.0
}
}
/// Auxilliary data for the PeripheralHarness
pub struct Aux {
pub peripheral: PeripheralProxy,
emulator: EmulatorProxy,
}
impl AsRef<EmulatorProxy> for Aux {
fn as_ref(&self) -> &EmulatorProxy {
&self.emulator
}
}
impl TestHarness for PeripheralHarness {
type Env = (ActivatedFakeHost, Arc<CoreRealm>);
type Runner = BoxFuture<'static, Result<(), Error>>;
fn init(
shared_state: &Arc<SharedState>,
) -> BoxFuture<'static, Result<(Self, Self::Env, Self::Runner), Error>> {
let shared_state = shared_state.clone();
async move {
let test_component: Arc<String> = shared_state
.get(SHARED_STATE_TEST_COMPONENT_INDEX)
.expect("SharedState must have TEST-COMPONENT")?;
let inserter = move || CoreRealm::create(test_component.to_string());
let realm = shared_state.get_or_insert_with(SHARED_STATE_INDEX, inserter).await?;
let host = ActivatedFakeHost::new(realm.clone()).await?;
let peripheral = realm
.instance()
.connect_to_protocol_at_exposed_dir::<PeripheralMarker>()
.context("Failed to connect to BLE Peripheral service")?;
let harness = PeripheralHarness(expectable(
Default::default(),
Aux { peripheral, emulator: host.emulator().clone() },
));
// Create a task to process the state update watcher
let watch_adv = watch_advertising_states(harness.deref().clone());
let watch_conn = watch_connections(harness.clone());
let run_peripheral =
future::try_join(watch_adv, watch_conn).map_ok(|((), ())| ()).boxed();
Ok((harness, (host, realm), run_peripheral))
}
.boxed()
}
fn terminate((emulator, realm): Self::Env) -> BoxFuture<'static, Result<(), Error>> {
// The realm must be kept alive in order for emulator.release() to work properly.
async move {
let _realm = realm;
emulator.release().await
}
.boxed()
}
}
async fn watch_connections(harness: PeripheralHarness) -> Result<(), Error> {
let mut events = harness.aux().peripheral.take_event_stream();
while let Some(e) = events.try_next().await? {
match e {
PeripheralEvent::OnPeerConnected { peer, connection } => {
harness.write_state().connections.push((peer.try_into()?, connection.into_proxy()));
}
}
harness.notify_state_changed();
}
Ok(())
}