1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
// Copyright 2023 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 crate::common_utils::common::macros::{fx_err_and_bail, with_line};
use crate::media_session::types::PlayerStatusWrapper;
use anyhow::Error;
use fidl::endpoints::create_request_stream;
use fidl_fuchsia_media_sessions2::{
    self as sessions2, ActiveSessionMarker, ActiveSessionProxy, PlayerRegistration, PublisherMarker,
};
use fuchsia_async as fasync;
use fuchsia_component::client::connect_to_protocol;
use fuchsia_sync::RwLock;
use futures::StreamExt;
use std::sync::Arc;
use tracing::info;

#[derive(Debug)]
pub struct MediaSessionFacade {
    active_session_proxy: Option<ActiveSessionProxy>,
    player_task: RwLock<Option<fasync::Task<()>>>,
    requests_received: Arc<RwLock<Vec<String>>>,
}

impl MediaSessionFacade {
    pub fn new() -> Self {
        Self {
            active_session_proxy: None,
            player_task: RwLock::new(None),
            requests_received: Arc::new(RwLock::new(Vec::new())),
        }
    }

    /// Returns the active session proxy.
    fn get_active_session_proxy(&self) -> Result<ActiveSessionProxy, Error> {
        if let Some(proxy) = &self.active_session_proxy {
            Ok(proxy.clone())
        } else {
            match connect_to_protocol::<ActiveSessionMarker>() {
                Ok(proxy) => Ok(proxy),
                Err(e) => fx_err_and_bail!(
                    &with_line!("MediaSessionFacade::get_active_session_proxy"),
                    format_err!("Failed to create proxy: {:?}", e)
                ),
            }
        }
    }

    /// Returns the active media session's player status.
    /// If there's no active session, it will return a None type value.
    pub async fn watch_active_session_status(&self) -> Result<Option<PlayerStatusWrapper>, Error> {
        let active_session_proxy = &self
            .get_active_session_proxy()
            .map_err(|e| format_err!("Failed to get active session proxy: {}", e))?;
        let session = active_session_proxy
            .watch_active_session()
            .await
            .map_err(|e| format_err!("Failed to watch active session: {}", e))?;
        if let Some(session_control_marker) = session {
            let info_delta = session_control_marker
                .into_proxy()
                .map_err(|e| format_err!("Failed to get session control proxy: {}", e))?
                .watch_status()
                .await
                .map_err(|e| format_err!("Failed to watch session status: {}", e))?;
            Ok(Some(info_delta.player_status.into()))
        } else {
            Ok(None)
        }
    }

    /// Publishes a mock Media Player via the fuchsia.media.sessions2 Publisher protocol.
    /// Clients can watch Media Sessions via the fuchsia.media.sessions2 Discovery protocol.
    /// This mock player does not route requests; it just records the requests it receives.
    /// The list of requests can be accessed through this facade's ListReceivedRequests command.
    /// The Bluetooth AVRCP component forwards commands to the most recently activated player.
    pub async fn publish_mock_player(&self) -> Result<(), Error> {
        let requests_received = self.requests_received.clone();
        let fut = async move {
            let publisher = fuchsia_component::client::connect_to_protocol::<PublisherMarker>()
                .expect("Failed to connect to Publisher");

            let (player_client, mut player_request_stream) =
                create_request_stream().expect("Failed to create request stream");

            publisher
                .publish(
                    player_client,
                    &PlayerRegistration {
                        domain: Some("domain://sl4f".to_string()),
                        ..Default::default()
                    },
                )
                .await
                .expect("Failed to publish Player");

            let mut _player_info_change_responder = None;
            let mut sent_response = false;

            loop {
                match player_request_stream.next().await {
                    Some(Ok(request)) => match request {
                        sessions2::PlayerRequest::WatchInfoChange { responder } => {
                            if !sent_response {
                                let _ = responder.send(&sessions2::PlayerInfoDelta::default());
                                sent_response = true;
                            } else {
                                // Save responder so request stream is not dropped
                                _player_info_change_responder = Some(responder);
                            }
                        }
                        request => {
                            requests_received.write().push(request.method_name().to_string());
                        }
                    },
                    Some(Err(e)) => {
                        info!(?e, "Mock player request stream error");
                        break;
                    }
                    None => {
                        info!("Mock player request stream terminated");
                        break;
                    }
                }
            }
        };

        if let Some(_task) = self.player_task.write().take() {
            info!("Dropping old mock player");
        }
        *self.player_task.write() = Some(fasync::Task::spawn(fut));
        info!("Publishing mock player");

        Ok(())
    }

    /// Unpublishes a mock Media Player and resets the list of received requests.
    /// Returns true if a mock Media Player was running and is now stopped.
    pub async fn stop_mock_player(&self) -> Result<bool, Error> {
        if let Some(_task) = self.player_task.write().take() {
            info!("Stopping mock player");
            self.requests_received.write().clear();
            return Ok(true);
        }
        info!("Mock player stoppage requested, but no session exists");
        Ok(false)
    }

    /// Returns a list of the player requests received by mock Media Players spawned by this
    /// facade's PublishMockPlayer command.
    pub async fn list_received_requests(&self) -> Result<Vec<String>, Error> {
        Ok(self.requests_received.read().clone())
    }
}