use crate::{
api::{Endpoint, prelude::*},
model::{ContextType, Offset, TrackId},
};
#[derive(Debug, Default, Clone)]
pub struct StartPlayback {
pub device_id: Option<String>,
pub context_uri: Option<ContextType>,
pub uris: Option<Vec<TrackId>>,
pub offset: Option<Offset>,
pub position_ms: Option<u32>,
}
impl StartPlayback {
pub fn device_id(mut self, device_id: impl Into<String>) -> Self {
self.device_id = Some(device_id.into());
self
}
pub fn context_uri(mut self, context_uri: ContextType) -> Self {
self.context_uri = Some(context_uri);
self
}
pub fn uris(mut self, uris: Vec<TrackId>) -> Self {
self.uris = Some(uris);
self
}
pub fn uri(mut self, uri: TrackId) -> Self {
self.uris.get_or_insert_with(Vec::new).push(uri);
self
}
pub fn offset(mut self, offset: impl Into<Offset>) -> Self {
self.offset = Some(offset.into());
self
}
pub fn position_ms(mut self, position_ms: u32) -> Self {
self.position_ms = Some(position_ms);
self
}
}
impl<T: Into<String>> From<T> for StartPlayback {
fn from(device_id: T) -> Self {
Self {
device_id: Some(device_id.into()),
context_uri: None,
uris: None,
offset: None,
position_ms: None,
}
}
}
impl Endpoint for StartPlayback {
fn method(&self) -> Method {
Method::PUT
}
fn endpoint(&self) -> Cow<'static, str> {
"me/player/play".into()
}
fn parameters(&self) -> QueryParams<'_> {
let mut params = QueryParams::default();
params.push_opt("device_id", self.device_id.as_ref());
params
}
fn body(&self) -> Result<Option<(&'static str, Vec<u8>)>, BodyError> {
let mut body = serde_json::json!({});
if let Some(context_uri) = self.context_uri.as_ref() {
body["context_uri"] = context_uri.uri().into();
}
if let Some(uris) = &self.uris {
body["uris"] = uris.iter().map(|uri| uri.uri()).collect::<Vec<_>>().into();
}
if let Some(offset) = self.offset.as_ref() {
body["offset"] = match offset {
Offset::Position(pos) => {
serde_json::json!({ "position": pos })
}
Offset::Uri(context) => serde_json::json!({ "uri": context.uri() }),
}
}
if let Some(position_ms) = self.position_ms {
body["position_ms"] = position_ms.into();
}
if body == serde_json::json!({}) {
return Ok(None);
}
JsonParams::into_body(&body)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{
api::{self, Query as _},
model::AlbumId,
test::client::{ExpectedUrl, SingleTestClient},
};
#[test]
fn test_start_playback_endpoint() {
let endpoint = ExpectedUrl::builder()
.method(Method::PUT)
.endpoint("me/player/play")
.add_query_params(&[("device_id", "xxxxxxxxxxxxxxxxxxxxxx")])
.build();
let client = SingleTestClient::new_raw(endpoint, "");
let endpoint = StartPlayback::from("xxxxxxxxxxxxxxxxxxxxxx");
api::ignore(endpoint).query(&client).unwrap();
}
#[test]
fn test_start_playback_endpoint_with_context_and_offset() {
let endpoint = ExpectedUrl::builder()
.method(Method::PUT)
.content_type("application/json")
.endpoint("me/player/play")
.add_query_params(&[("device_id", "xxxxxxxxxxxxxxxxxxxxxx")])
.body_str(r#"{"context_uri":"spotify:album:5ht7ItJgpBH7W6vJ5BqpPr","offset":{"position":5},"position_ms":0}"#)
.build();
let client = SingleTestClient::new_raw(endpoint, "");
let endpoint = StartPlayback::default()
.context_uri(ContextType::Album(
AlbumId::from_id("5ht7ItJgpBH7W6vJ5BqpPr").unwrap(),
))
.offset(Offset::Position(5))
.position_ms(0)
.device_id("xxxxxxxxxxxxxxxxxxxxxx");
api::ignore(endpoint).query(&client).unwrap();
}
}