tmdb_api/collection/
details.rs

1use std::borrow::Cow;
2
3use crate::common::MediaType;
4
5/// Command to get the details of a collection
6///
7/// ```rust
8/// use tmdb_api::prelude::Command;
9/// use tmdb_api::client::Client;
10/// use tmdb_api::client::reqwest::ReqwestExecutor;
11/// use tmdb_api::collection::details::CollectionDetails;
12///
13/// #[tokio::main]
14/// async fn main() {
15///     let client = Client::<ReqwestExecutor>::new("this-is-my-secret-token".into());
16///     let cmd = CollectionDetails::new(1);
17///     let result = cmd.execute(&client).await;
18///     match result {
19///         Ok(res) => println!("found: {:#?}", res),
20///         Err(err) => eprintln!("error: {:?}", err),
21///     };
22/// }
23/// ```
24#[derive(Clone, Debug, Default)]
25pub struct CollectionDetails {
26    /// ID of the collection
27    pub collection_id: u64,
28    /// ISO 639-1 value to display translated data for the fields that support it.
29    pub language: Option<String>,
30}
31
32#[derive(Debug, Deserialize, Serialize)]
33pub struct CollectionDetailsResult {
34    #[serde(flatten)]
35    pub inner: super::CollectionBase,
36    pub parts: Vec<Media>,
37}
38
39#[derive(Debug, Deserialize, Serialize)]
40pub struct Media {
41    pub id: u64,
42    pub media_type: MediaType,
43    pub title: String,
44    pub original_language: String,
45    pub original_title: String,
46    pub overview: String,
47    pub poster_path: Option<String>,
48    pub backdrop_path: Option<String>,
49    #[serde(default)]
50    pub genre_ids: Vec<u64>,
51    #[serde(default)]
52    pub popularity: f64,
53    #[serde(default)]
54    pub adult: bool,
55    #[serde(default)]
56    pub video: bool,
57    #[serde(default)]
58    pub vote_average: f64,
59    #[serde(default)]
60    pub vote_count: u64,
61    #[serde(default, deserialize_with = "crate::util::empty_string::deserialize")]
62    pub release_date: Option<chrono::NaiveDate>,
63}
64
65impl CollectionDetails {
66    pub fn new(collection_id: u64) -> Self {
67        Self {
68            collection_id,
69            language: None,
70        }
71    }
72
73    pub fn with_language(mut self, value: Option<String>) -> Self {
74        self.language = value;
75        self
76    }
77}
78
79impl crate::prelude::Command for CollectionDetails {
80    type Output = CollectionDetailsResult;
81
82    fn path(&self) -> Cow<'static, str> {
83        Cow::Owned(format!("/collection/{}", self.collection_id))
84    }
85
86    fn params(&self) -> Vec<(&'static str, Cow<'_, str>)> {
87        if let Some(ref language) = self.language {
88            vec![("language", Cow::Borrowed(language))]
89        } else {
90            Vec::new()
91        }
92    }
93}
94
95#[cfg(test)]
96mod tests {
97    use mockito::Matcher;
98
99    use crate::client::Client;
100    use crate::client::reqwest::ReqwestExecutor;
101    use crate::prelude::Command;
102
103    use super::CollectionDetails;
104
105    #[tokio::test]
106    async fn it_works() {
107        let mut server = mockito::Server::new_async().await;
108        let client = Client::<ReqwestExecutor>::builder()
109            .with_api_key("secret".into())
110            .with_base_url(server.url())
111            .build()
112            .unwrap();
113
114        let _m = server
115            .mock("GET", "/collection/10")
116            .match_query(Matcher::UrlEncoded("api_key".into(), "secret".into()))
117            .with_status(200)
118            .with_header("content-type", "application/json")
119            .with_body(include_str!("../../assets/collection-details.json"))
120            .create_async()
121            .await;
122
123        let result = CollectionDetails::new(10).execute(&client).await.unwrap();
124        assert_eq!(result.inner.id, 10);
125    }
126
127    #[tokio::test]
128    async fn invalid_api_key() {
129        let mut server = mockito::Server::new_async().await;
130        let client = Client::<ReqwestExecutor>::builder()
131            .with_api_key("secret".into())
132            .with_base_url(server.url())
133            .build()
134            .unwrap();
135
136        let _m = server
137            .mock("GET", "/collection/0")
138            .match_query(Matcher::UrlEncoded("api_key".into(), "secret".into()))
139            .with_status(401)
140            .with_header("content-type", "application/json")
141            .with_body(include_str!("../../assets/invalid-api-key.json"))
142            .create_async()
143            .await;
144
145        let err = CollectionDetails::new(0)
146            .execute(&client)
147            .await
148            .unwrap_err();
149        let server_err = err.as_server_error().unwrap();
150        assert_eq!(server_err.status_code, 7);
151    }
152
153    #[tokio::test]
154    async fn resource_not_found() {
155        let mut server = mockito::Server::new_async().await;
156        let client = Client::<ReqwestExecutor>::builder()
157            .with_api_key("secret".into())
158            .with_base_url(server.url())
159            .build()
160            .unwrap();
161
162        let _m = server
163            .mock("GET", "/collection/0")
164            .match_query(Matcher::UrlEncoded("api_key".into(), "secret".into()))
165            .with_status(404)
166            .with_header("content-type", "application/json")
167            .with_body(include_str!("../../assets/resource-not-found.json"))
168            .create_async()
169            .await;
170
171        let err = CollectionDetails::new(0)
172            .execute(&client)
173            .await
174            .unwrap_err();
175        let server_err = err.as_server_error().unwrap();
176        assert_eq!(server_err.status_code, 34);
177    }
178}
179
180#[cfg(all(test, feature = "integration"))]
181mod integration_tests {
182    use crate::client::Client;
183    use crate::client::reqwest::ReqwestExecutor;
184    use crate::prelude::Command;
185
186    use super::CollectionDetails;
187
188    #[tokio::test]
189    async fn execute() {
190        let secret = std::env::var("TMDB_TOKEN_V3").unwrap();
191        let client = Client::<ReqwestExecutor>::new(secret);
192
193        for i in [10, 1196769] {
194            let result = CollectionDetails::new(i).execute(&client).await.unwrap();
195            assert_eq!(result.inner.id, i);
196        }
197    }
198}