use std::{
borrow::Cow,
io::{Read, Write},
str::FromStr,
string::ToString,
};
use crate::{
cast::proxies::{self, media::Item},
errors::Error,
message_manager::{CastMessage, CastMessagePayload, MessageManager},
Lrc,
};
const CHANNEL_NAMESPACE: &str = "urn:x-cast:com.google.cast.media";
const MESSAGE_TYPE_GET_STATUS: &str = "GET_STATUS";
const MESSAGE_TYPE_LOAD: &str = "LOAD";
const MESSAGE_TYPE_PLAY: &str = "PLAY";
const MESSAGE_TYPE_PAUSE: &str = "PAUSE";
const MESSAGE_TYPE_STOP: &str = "STOP";
const MESSAGE_TYPE_SEEK: &str = "SEEK";
const MESSAGE_TYPE_QUEUE_REMOVE: &str = "QUEUE_REMOVE";
const MESSAGE_TYPE_QUEUE_INSERT: &str = "QUEUE_INSERT";
const MESSAGE_TYPE_QUEUE_LOAD: &str = "QUEUE_LOAD";
const MESSAGE_TYPE_QUEUE_GET_ITEMS: &str = "QUEUE_GET_ITEMS";
const MESSAGE_TYPE_QUEUE_PREV: &str = "QUEUE_PREV";
const MESSAGE_TYPE_QUEUE_NEXT: &str = "QUEUE_NEXT";
const MESSAGE_TYPE_MEDIA_STATUS: &str = "MEDIA_STATUS";
const MESSAGE_TYPE_LOAD_CANCELLED: &str = "LOAD_CANCELLED";
const MESSAGE_TYPE_LOAD_FAILED: &str = "LOAD_FAILED";
const MESSAGE_TYPE_INVALID_PLAYER_STATE: &str = "INVALID_PLAYER_STATE";
const MESSAGE_TYPE_INVALID_REQUEST: &str = "INVALID_REQUEST";
pub enum RepeatMode {
Off,
All,
Single,
AllAndShuffle,
}
impl ToString for RepeatMode {
fn to_string(&self) -> String {
match *self {
RepeatMode::Off => "REPEAT_OFF",
RepeatMode::All => "REPEAT_ALL",
RepeatMode::Single => "REPEAT_SINGLE",
RepeatMode::AllAndShuffle => "REPEAT_ALL_AND_SHUFFLE",
}
.to_string()
}
}
#[derive(Copy, Clone, Debug)]
pub enum StreamType {
None,
Buffered,
Live,
}
impl FromStr for StreamType {
type Err = Error;
fn from_str(s: &str) -> Result<StreamType, Error> {
match s {
"BUFFERED" | "buffered" => Ok(StreamType::Buffered),
"LIVE" | "live" => Ok(StreamType::Live),
_ => Ok(StreamType::None),
}
}
}
impl ToString for StreamType {
fn to_string(&self) -> String {
let stream_type = match *self {
StreamType::None => "NONE",
StreamType::Buffered => "BUFFERED",
StreamType::Live => "LIVE",
};
stream_type.to_string()
}
}
#[derive(Clone, Debug)]
pub enum Metadata {
Generic(GenericMediaMetadata),
Movie(MovieMediaMetadata),
TvShow(TvShowMediaMetadata),
MusicTrack(MusicTrackMediaMetadata),
Photo(PhotoMediaMetadata),
}
#[derive(Clone, Debug)]
pub struct GenericMediaMetadata {
pub title: Option<String>,
pub subtitle: Option<String>,
pub images: Vec<Image>,
pub release_date: Option<String>,
}
#[derive(Clone, Debug)]
pub struct MovieMediaMetadata {
pub title: Option<String>,
pub subtitle: Option<String>,
pub studio: Option<String>,
pub images: Vec<Image>,
pub release_date: Option<String>,
}
#[derive(Clone, Debug)]
pub struct TvShowMediaMetadata {
pub series_title: Option<String>,
pub episode_title: Option<String>,
pub season: Option<u32>,
pub episode: Option<u32>,
pub images: Vec<Image>,
pub original_air_date: Option<String>,
}
#[derive(Clone, Debug)]
pub struct MusicTrackMediaMetadata {
pub album_name: Option<String>,
pub title: Option<String>,
pub album_artist: Option<String>,
pub artist: Option<String>,
pub composer: Option<String>,
pub track_number: Option<u32>,
pub disc_number: Option<u32>,
pub images: Vec<Image>,
pub release_date: Option<String>,
}
#[derive(Clone, Debug)]
pub struct PhotoMediaMetadata {
pub title: Option<String>,
pub artist: Option<String>,
pub location: Option<String>,
pub latitude_longitude: Option<(f64, f64)>,
pub dimensions: Option<(u32, u32)>,
pub creation_date_time: Option<String>,
}
#[derive(Clone, Debug)]
pub struct Image {
pub url: String,
pub dimensions: Option<(u32, u32)>,
}
impl Image {
pub fn new(url: String) -> Image {
Image {
url,
dimensions: None,
}
}
fn encode(&self) -> proxies::media::Image {
proxies::media::Image {
url: self.url.clone(),
width: self.dimensions.map(|d| d.0),
height: self.dimensions.map(|d| d.1),
}
}
}
#[derive(Copy, Clone, Debug)]
pub enum PlayerState {
Idle,
Playing,
Buffering,
Paused,
}
impl FromStr for PlayerState {
type Err = Error;
fn from_str(s: &str) -> Result<PlayerState, Error> {
match s {
"IDLE" => Ok(PlayerState::Idle),
"PLAYING" => Ok(PlayerState::Playing),
"BUFFERING" => Ok(PlayerState::Buffering),
"PAUSED" => Ok(PlayerState::Paused),
_ => Err(Error::Internal(format!("Unknown player state {}", s))),
}
}
}
impl ToString for PlayerState {
fn to_string(&self) -> String {
let player_state = match *self {
PlayerState::Idle => "IDLE",
PlayerState::Playing => "PLAYING",
PlayerState::Buffering => "BUFFERING",
PlayerState::Paused => "PAUSED",
};
player_state.to_string()
}
}
#[derive(Copy, Clone, Debug)]
pub enum IdleReason {
Cancelled,
Interrupted,
Finished,
Error,
}
impl FromStr for IdleReason {
type Err = Error;
fn from_str(s: &str) -> Result<IdleReason, Error> {
match s {
"CANCELLED" => Ok(IdleReason::Cancelled),
"INTERRUPTED" => Ok(IdleReason::Interrupted),
"FINISHED" => Ok(IdleReason::Finished),
"ERROR" => Ok(IdleReason::Error),
_ => Err(Error::Internal(format!("Unknown idle reason {}", s))),
}
}
}
#[derive(Copy, Clone, Debug)]
pub enum ResumeState {
PlaybackStart,
PlaybackPause,
}
impl FromStr for ResumeState {
type Err = Error;
fn from_str(s: &str) -> Result<ResumeState, Error> {
match s {
"PLAYBACK_START" | "start" => Ok(ResumeState::PlaybackStart),
"PLAYBACK_PAUSE" | "pause" => Ok(ResumeState::PlaybackPause),
_ => Err(Error::Internal(format!("Unknown resume state {}", s))),
}
}
}
impl ToString for ResumeState {
fn to_string(&self) -> String {
let resume_state = match *self {
ResumeState::PlaybackStart => "PLAYBACK_START",
ResumeState::PlaybackPause => "PLAYBACK_PAUSE",
};
resume_state.to_string()
}
}
#[derive(Clone, Debug)]
pub struct Media {
pub content_id: String,
pub stream_type: StreamType,
pub content_type: String,
pub metadata: Option<Metadata>,
pub duration: Option<f32>,
}
#[derive(Clone, Debug)]
pub struct Status {
pub request_id: i32,
pub entries: Vec<StatusEntry>,
}
#[derive(Clone, Debug)]
pub struct StatusEntry {
pub media_session_id: i32,
pub media: Option<Media>,
pub playback_rate: f32,
pub player_state: PlayerState,
pub idle_reason: Option<IdleReason>,
pub current_time: Option<f32>,
pub supported_media_commands: u32,
pub items: Option<Vec<Item>>,
pub current_item_id: Option<i32>,
}
#[derive(Copy, Clone, Debug)]
pub struct LoadCancelled {
pub request_id: i32,
}
#[derive(Copy, Clone, Debug)]
pub struct LoadFailed {
pub request_id: i32,
}
#[derive(Copy, Clone, Debug)]
pub struct InvalidPlayerState {
pub request_id: i32,
}
#[derive(Clone, Debug)]
pub struct InvalidRequest {
pub request_id: i32,
pub reason: Option<String>,
}
#[derive(Clone, Debug)]
pub enum MediaResponse {
Status(Status),
LoadCancelled(LoadCancelled),
LoadFailed(LoadFailed),
InvalidPlayerState(InvalidPlayerState),
InvalidRequest(InvalidRequest),
NotImplemented(String, serde_json::Value),
}
pub struct MediaChannel<'a, W>
where
W: Read + Write,
{
sender: Cow<'a, str>,
message_manager: Lrc<MessageManager<W>>,
}
impl<'a, W> MediaChannel<'a, W>
where
W: Read + Write,
{
pub fn new<S>(sender: S, message_manager: Lrc<MessageManager<W>>) -> MediaChannel<'a, W>
where
S: Into<Cow<'a, str>>,
{
MediaChannel {
sender: sender.into(),
message_manager,
}
}
pub fn get_status<S>(
&self,
destination: S,
media_session_id: Option<i32>,
) -> Result<Status, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::GetStatusRequest {
typ: MESSAGE_TYPE_GET_STATUS.to_string(),
request_id,
media_session_id,
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.message_manager.receive_find_map(|message| {
if !self.can_handle(message) {
return Ok(None);
}
match self.parse(message)? {
MediaResponse::Status(status) => {
if status.request_id == request_id {
return Ok(Some(status));
}
}
MediaResponse::InvalidRequest(error) => {
if error.request_id == request_id {
return Err(Error::Internal(format!(
"Invalid request ({}).",
error.reason.unwrap_or_else(|| "Unknown".to_string())
)));
}
}
_ => {}
}
Ok(None)
})
}
pub fn load<S>(&self, destination: S, session_id: S, media: &Media) -> Result<Status, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let metadata = media.metadata.as_ref().map(|m| match *m {
Metadata::Generic(ref x) => proxies::media::Metadata {
title: x.title.clone(),
subtitle: x.subtitle.clone(),
images: x.images.iter().map(|i| i.encode()).collect(),
release_date: x.release_date.clone(),
..proxies::media::Metadata::new(0)
},
Metadata::Movie(ref x) => proxies::media::Metadata {
title: x.title.clone(),
subtitle: x.subtitle.clone(),
studio: x.studio.clone(),
images: x.images.iter().map(|i| i.encode()).collect(),
release_date: x.release_date.clone(),
..proxies::media::Metadata::new(1)
},
Metadata::TvShow(ref x) => proxies::media::Metadata {
series_title: x.series_title.clone(),
subtitle: x.episode_title.clone(),
season: x.season,
episode: x.episode,
images: x.images.iter().map(|i| i.encode()).collect(),
original_air_date: x.original_air_date.clone(),
..proxies::media::Metadata::new(2)
},
Metadata::MusicTrack(ref x) => proxies::media::Metadata {
album_name: x.album_name.clone(),
title: x.title.clone(),
album_artist: x.album_artist.clone(),
artist: x.artist.clone(),
composer: x.composer.clone(),
track_number: x.track_number,
disc_number: x.disc_number,
images: x.images.iter().map(|i| i.encode()).collect(),
release_date: x.release_date.clone(),
..proxies::media::Metadata::new(3)
},
Metadata::Photo(ref x) => proxies::media::Metadata {
title: x.title.clone(),
artist: x.artist.clone(),
location: x.location.clone(),
latitude: x.latitude_longitude.map(|coord| coord.0),
longitude: x.latitude_longitude.map(|coord| coord.1),
width: x.dimensions.map(|dims| dims.0),
height: x.dimensions.map(|dims| dims.1),
creation_date_time: x.creation_date_time.clone(),
..proxies::media::Metadata::new(4)
},
});
let payload = serde_json::to_string(&proxies::media::MediaRequest {
request_id,
session_id: session_id.into().to_string(),
typ: MESSAGE_TYPE_LOAD.to_string(),
media: proxies::media::Media {
content_id: media.content_id.clone(),
stream_type: media.stream_type.to_string(),
content_type: media.content_type.clone(),
metadata,
duration: media.duration,
},
current_time: 0_f64,
autoplay: true,
preload_time: 20_f64,
custom_data: proxies::media::CustomData { queue: None },
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.message_manager.receive_find_map(|message| {
if !self.can_handle(message) {
return Ok(None);
}
match self.parse(message)? {
MediaResponse::Status(status) => {
if status.request_id == request_id {
return Ok(Some(status));
}
let has_media = {
status.entries.iter().any(|entry| {
if let Some(ref loaded_media) = entry.media {
return loaded_media.content_id == media.content_id;
}
false
})
};
if has_media {
return Ok(Some(status));
}
}
MediaResponse::LoadFailed(error) => {
if error.request_id == request_id {
return Err(Error::Internal("Failed to load media.".to_string()));
}
}
MediaResponse::LoadCancelled(error) => {
if error.request_id == request_id {
return Err(Error::Internal(
"Load cancelled by another request.".to_string(),
));
}
}
MediaResponse::InvalidPlayerState(error) => {
if error.request_id == request_id {
return Err(Error::Internal(
"Load failed because of invalid player state.".to_string(),
));
}
}
MediaResponse::InvalidRequest(error) => {
if error.request_id == request_id {
return Err(Error::Internal(format!(
"Load failed because of invalid media request (reason: {}).",
error.reason.unwrap_or_else(|| "UNKNOWN".to_string())
)));
}
}
_ => {}
}
Ok(None)
})
}
pub fn pause<S>(&self, destination: S, media_session_id: i32) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackGenericRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_PAUSE.to_string(),
custom_data: proxies::media::CustomData::new(),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn play<S>(&self, destination: S, media_session_id: i32) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackGenericRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_PLAY.to_string(),
custom_data: proxies::media::CustomData::new(),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn stop<S>(&self, destination: S, media_session_id: i32) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackGenericRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_STOP.to_string(),
custom_data: proxies::media::CustomData::new(),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn seek<S>(
&self,
destination: S,
media_session_id: i32,
current_time: Option<f32>,
resume_state: Option<ResumeState>,
) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackSeekRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_SEEK.to_string(),
current_time,
resume_state: resume_state.map(|s| s.to_string()),
custom_data: proxies::media::CustomData::new(),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn queue_remove<S>(
&self,
destination: S,
media_session_id: i32,
item_id: i32,
) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackQueueRemoveRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_QUEUE_REMOVE.to_string(),
item_ids: vec![item_id],
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn queue_insert<S>(
&self,
destination: S,
media_session_id: i32,
medias: Vec<Media>,
insert_before: Option<i32>,
) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let medias = medias
.into_iter()
.map(|media| proxies::media::Media {
content_id: media.content_id.clone(),
stream_type: media.stream_type.to_string(),
content_type: media.content_type.clone(),
metadata: get_metadata(&media),
duration: media.duration,
})
.collect::<Vec<proxies::media::Media>>();
let items = medias
.into_iter()
.map(|media| proxies::media::QueueItem {
media,
auto_play: true,
start_time: 0.0,
custom_data: proxies::media::CustomData { queue: Some(true) },
})
.collect();
let payload = serde_json::to_string(&proxies::media::PlaybackQueueInsertRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_QUEUE_INSERT.to_string(),
items,
current_item_index: None,
insert_before,
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn queue_load<S>(
&self,
destination: S,
medias: Vec<Media>,
start_index: Option<i32>,
repeat_mode: Option<RepeatMode>,
) -> Result<(), Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let medias = medias
.into_iter()
.map(|media| proxies::media::Media {
content_id: media.content_id.clone(),
stream_type: media.stream_type.to_string(),
content_type: media.content_type.clone(),
metadata: get_metadata(&media),
duration: media.duration,
})
.collect::<Vec<proxies::media::Media>>();
let items = medias
.into_iter()
.map(|media| proxies::media::QueueItem {
media,
auto_play: true,
start_time: 0.0,
custom_data: proxies::media::CustomData { queue: Some(true) },
})
.collect();
let payload = serde_json::to_string(&proxies::media::PlaybackQueueLoadRequest {
request_id,
typ: MESSAGE_TYPE_QUEUE_LOAD.to_string(),
items,
start_index,
repeat_mode: repeat_mode.map(|m| m.to_string()),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
Ok(())
}
pub fn get_queue_items<S>(
&self,
destination: S,
media_session_id: i32,
) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackGenericRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_QUEUE_GET_ITEMS.to_string(),
custom_data: proxies::media::CustomData::new(),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn previous<S>(&self, destination: S, media_session_id: i32) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackGenericRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_QUEUE_PREV.to_string(),
custom_data: proxies::media::CustomData::new(),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn next<S>(&self, destination: S, media_session_id: i32) -> Result<StatusEntry, Error>
where
S: Into<Cow<'a, str>>,
{
let request_id = self.message_manager.generate_request_id();
let payload = serde_json::to_string(&proxies::media::PlaybackGenericRequest {
request_id,
media_session_id,
typ: MESSAGE_TYPE_QUEUE_NEXT.to_string(),
custom_data: proxies::media::CustomData::new(),
})?;
self.message_manager.send(CastMessage {
namespace: CHANNEL_NAMESPACE.to_string(),
source: self.sender.to_string(),
destination: destination.into().to_string(),
payload: CastMessagePayload::String(payload),
})?;
self.receive_status_entry(request_id, media_session_id)
}
pub fn can_handle(&self, message: &CastMessage) -> bool {
message.namespace == CHANNEL_NAMESPACE
}
pub fn parse(&self, message: &CastMessage) -> Result<MediaResponse, Error> {
let reply = match message.payload {
CastMessagePayload::String(ref payload) => {
serde_json::from_str::<serde_json::Value>(payload)?
}
_ => {
return Err(Error::Internal(
"Binary payload is not supported!".to_string(),
))
}
};
let message_type = reply
.as_object()
.and_then(|object| object.get("type"))
.and_then(|property| property.as_str())
.unwrap_or("")
.to_string();
let response = match message_type.as_ref() {
MESSAGE_TYPE_MEDIA_STATUS => {
let reply: proxies::media::StatusReply = serde_json::value::from_value(reply)?;
let statuses_entries = reply.status.iter().map(|x| StatusEntry {
media_session_id: x.media_session_id,
media: x.media.as_ref().map(|m| {
let metadata = match m.metadata {
Some(ref metadata) => {
Some(Metadata::MusicTrack(MusicTrackMediaMetadata {
album_name: metadata.album_name.clone(),
album_artist: metadata.album_artist.clone(),
artist: metadata.artist.clone(),
title: metadata.title.clone(),
track_number: metadata.track_number.clone(),
composer: metadata.composer.clone(),
disc_number: metadata.disc_number.clone(),
images: metadata
.images
.clone()
.into_iter()
.map(|i| Image {
url: i.url,
dimensions: None,
})
.collect(),
release_date: metadata.release_date.clone(),
}))
}
None => None,
};
Media {
content_id: m.content_id.to_string(),
stream_type: StreamType::from_str(m.stream_type.as_ref()).unwrap(),
content_type: m.content_type.to_string(),
metadata,
duration: m.duration,
}
}),
playback_rate: x.playback_rate,
player_state: PlayerState::from_str(x.player_state.as_ref()).unwrap(),
idle_reason: x
.idle_reason
.as_ref()
.map(|reason| IdleReason::from_str(reason).unwrap()),
current_time: x.current_time,
current_item_id: x.current_item_id,
supported_media_commands: x.supported_media_commands,
items: x.items.as_ref().map(|items| {
items
.iter()
.map(|item| Item {
item_id: item.item_id,
media: item.clone().media,
auto_play: item.auto_play,
custom_data: item.clone().custom_data,
})
.collect::<Vec<Item>>()
}),
});
MediaResponse::Status(Status {
request_id: reply.request_id,
entries: statuses_entries.collect::<Vec<StatusEntry>>(),
})
}
MESSAGE_TYPE_LOAD_CANCELLED => {
let reply: proxies::media::LoadCancelledReply =
serde_json::value::from_value(reply)?;
MediaResponse::LoadCancelled(LoadCancelled {
request_id: reply.request_id,
})
}
MESSAGE_TYPE_LOAD_FAILED => {
let reply: proxies::media::LoadFailedReply = serde_json::value::from_value(reply)?;
MediaResponse::LoadFailed(LoadFailed {
request_id: reply.request_id,
})
}
MESSAGE_TYPE_INVALID_PLAYER_STATE => {
let reply: proxies::media::InvalidPlayerStateReply =
serde_json::value::from_value(reply)?;
MediaResponse::InvalidPlayerState(InvalidPlayerState {
request_id: reply.request_id,
})
}
MESSAGE_TYPE_INVALID_REQUEST => {
let reply: proxies::media::InvalidRequestReply =
serde_json::value::from_value(reply)?;
MediaResponse::InvalidRequest(InvalidRequest {
request_id: reply.request_id,
reason: reply.reason,
})
}
_ => MediaResponse::NotImplemented(message_type.to_string(), reply),
};
Ok(response)
}
fn receive_status_entry(
&self,
request_id: i32,
media_session_id: i32,
) -> Result<StatusEntry, Error> {
self.message_manager.receive_find_map(|message| {
if !self.can_handle(message) {
return Ok(None);
}
match self.parse(message)? {
MediaResponse::Status(mut status) => {
if status.request_id == request_id {
let position = status
.entries
.iter()
.position(|e| e.media_session_id == media_session_id);
return Ok(position.map(|position| status.entries.remove(position)));
}
}
MediaResponse::InvalidPlayerState(error) => {
if error.request_id == request_id {
return Err(Error::Internal(
"Request failed because of invalid player state.".to_string(),
));
}
}
MediaResponse::InvalidRequest(error) => {
if error.request_id == request_id {
return Err(Error::Internal(format!(
"Invalid request ({}).",
error.reason.unwrap_or_else(|| "Unknown".to_string())
)));
}
}
_ => {}
}
Ok(None)
})
}
}
fn get_metadata(media: &Media) -> Option<proxies::media::Metadata> {
media.metadata.as_ref().map(|m| match *m {
Metadata::Generic(ref x) => proxies::media::Metadata {
title: x.title.clone(),
subtitle: x.subtitle.clone(),
images: x.images.iter().map(|i| i.encode()).collect(),
release_date: x.release_date.clone(),
..proxies::media::Metadata::new(0)
},
Metadata::Movie(ref x) => proxies::media::Metadata {
title: x.title.clone(),
subtitle: x.subtitle.clone(),
studio: x.studio.clone(),
images: x.images.iter().map(|i| i.encode()).collect(),
release_date: x.release_date.clone(),
..proxies::media::Metadata::new(1)
},
Metadata::TvShow(ref x) => proxies::media::Metadata {
series_title: x.series_title.clone(),
subtitle: x.episode_title.clone(),
season: x.season,
episode: x.episode,
images: x.images.iter().map(|i| i.encode()).collect(),
original_air_date: x.original_air_date.clone(),
..proxies::media::Metadata::new(2)
},
Metadata::MusicTrack(ref x) => proxies::media::Metadata {
album_name: x.album_name.clone(),
title: x.title.clone(),
album_artist: x.album_artist.clone(),
artist: x.artist.clone(),
composer: x.composer.clone(),
track_number: x.track_number,
disc_number: x.disc_number,
images: x.images.iter().map(|i| i.encode()).collect(),
release_date: x.release_date.clone(),
..proxies::media::Metadata::new(3)
},
Metadata::Photo(ref x) => proxies::media::Metadata {
title: x.title.clone(),
artist: x.artist.clone(),
location: x.location.clone(),
latitude: x.latitude_longitude.map(|coord| coord.0),
longitude: x.latitude_longitude.map(|coord| coord.1),
width: x.dimensions.map(|dims| dims.0),
height: x.dimensions.map(|dims| dims.1),
creation_date_time: x.creation_date_time.clone(),
..proxies::media::Metadata::new(4)
},
})
}