tmdb_api/collection/
details.rs1use std::borrow::Cow;
2
3use crate::common::MediaType;
4
5#[derive(Clone, Debug, Default)]
25pub struct CollectionDetails {
26 pub collection_id: u64,
28 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}