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
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
//! Gets video information by video ID (one or more), user ID (one only), or game ID (one only).
//! [`get-videos`](https://dev.twitch.tv/docs/api/reference#get-videos)
//!
//! # Accessing the endpoint
//!
//! ## Request: [GetVideosRequest]
//!
//! To use this endpoint, construct a [`GetVideosRequest`] with the [`GetVideosRequest::builder()`] method.
//!
//! ```rust
//! use twitch_api2::helix::videos::get_videos;
//! let request = get_videos::GetVideosRequest::builder()
//!     .user_id(Some("1234".into()))
//!     .build();
//! ```
//!
//! ## Response: [Video]
//!
//! Send the request to receive the response with [`HelixClient::req_get()`](helix::HelixClient::req_get).
//!
//! ```rust, no_run
//! use twitch_api2::helix::{self, videos::get_videos};
//! # use twitch_api2::client;
//! # #[tokio::main]
//! # async fn main() -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> {
//! # let client: helix::HelixClient<'static, client::DummyHttpClient> = helix::HelixClient::default();
//! # let token = twitch_oauth2::AccessToken::new("validtoken".to_string());
//! # let token = twitch_oauth2::UserToken::from_existing(&client, token, None, None).await?;
//! let request = get_videos::GetVideosRequest::builder()
//!     .user_id(Some("1234".into()))
//!     .build();
//! let response: Vec<get_videos::Video> = client.req_get(request, &token).await?.data;
//! # Ok(())
//! # }
//! ```
//!
//! You can also get the [`http::Request`] with [`request.create_request(&token, &client_id)`](helix::RequestGet::create_request)
//! and parse the [`http::Response`] with [`GetVideosRequest::parse_response(None, &request.get_uri(), response)`](GetVideosRequest::parse_response)

use super::*;
use helix::RequestGet;

// FIXME: One of id, user_id or game_id needs to be specified. typed_builder should have enums. id can not be used with other params
/// Query Parameters for [Get Videos](super::get_videos)
///
/// [`get-videos`](https://dev.twitch.tv/docs/api/reference#get-videos)
#[derive(PartialEq, typed_builder::TypedBuilder, Deserialize, Serialize, Clone, Debug)]
#[non_exhaustive]
pub struct GetVideosRequest {
    /// ID of the video being queried. Limit: 100. If this is specified, you cannot use any of the optional query parameters below.
    #[builder(default)]
    pub id: Vec<types::VideoId>,
    /// ID of the user who owns the video.
    #[builder(default, setter(into))]
    pub user_id: Option<types::UserId>,
    /// ID of the game the video is of.
    #[builder(default, setter(into))]
    pub game_id: Option<types::CategoryId>,
    /// Cursor for forward pagination: tells the server where to start fetching the next set of results, in a multi-page response. The cursor value specified here is from the pagination response field of a prior query.
    #[builder(default)]
    pub after: Option<helix::Cursor>,
    /// Cursor for backward pagination: tells the server where to start fetching the next set of results, in a multi-page response. The cursor value specified here is from the pagination response field of a prior query.
    #[builder(default)]
    pub before: Option<helix::Cursor>,
    /// Number of values to be returned when getting videos by user or game ID. Limit: 100. Default: 20.
    #[builder(default, setter(into))]
    pub first: Option<usize>,
    /// Language of the video being queried. Limit: 1.
    #[builder(default, setter(into))]
    pub language: Option<String>,
    /// Period during which the video was created. Valid values: "all", "day", "week", "month". Default: "all".
    #[builder(default, setter(into))]
    pub period: Option<VideoPeriod>,
    /// Sort order of the videos. Valid values: "time", "trending", "views". Default: "time".
    #[builder(default, setter(into))]
    pub sort: Option<Sort>,
    /// Type of video. Valid values: "all", "upload", "archive", "highlight". Default: "all".
    #[serde(rename = "type")]
    #[builder(default, setter(into))]
    pub type_: Option<VideoTypeFilter>,
}

/// Return Values for [Get Videos](super::get_videos)
///
/// [`get-videos`](https://dev.twitch.tv/docs/api/reference#get-videos)
#[derive(PartialEq, Deserialize, Serialize, Debug, Clone)]
#[cfg_attr(feature = "deny_unknown_fields", serde(deny_unknown_fields))]
#[non_exhaustive]
pub struct Video {
    /// Date when the video was created.
    pub created_at: types::Timestamp,
    /// Description of the video.
    pub description: String,
    /// Length of the video.
    pub duration: String,
    /// ID of the video.
    pub id: types::VideoId,
    /// Language of the video.
    pub language: String,
    /// Muted segments in the video.
    #[serde(deserialize_with = "helix::deserialize_default_from_null")]
    pub muted_segments: Vec<MutedSegment>,
    /// Date when the video was published.
    pub published_at: types::Timestamp,
    /// ID of the stream that the video originated from if the type is "archive". Otherwise set to null.
    pub stream_id: Option<types::StreamId>,
    /// Template URL for the thumbnail of the video.
    pub thumbnail_url: String,
    /// Title of the video.
    pub title: String,
    /// Type of video. Valid values: "upload", "archive", "highlight".
    #[serde(rename = "type")]
    pub type_: types::VideoType,
    /// URL of the video.
    pub url: String,
    /// ID of the user who owns the video.
    pub user_id: types::UserId,
    /// Display name corresponding to user_id.
    pub user_name: types::DisplayName,
    /// Login of the user who owns the video.
    pub user_login: types::UserName,
    /// Number of times the video has been viewed.
    pub view_count: i64,
    /// Indicates whether the video is publicly viewable. Valid values: "public", "private".
    pub viewable: types::VideoPrivacy,
}

/// muted segment in a video.
#[derive(PartialEq, Deserialize, Serialize, Debug, Clone)]
#[cfg_attr(feature = "deny_unknown_fields", serde(deny_unknown_fields))]
#[non_exhaustive]
pub struct MutedSegment {
    /// Duration of the muted segment.
    pub duration: i64,
    /// Offset in the video at which the muted segment begins.
    pub offset: i64,
}

impl Request for GetVideosRequest {
    type Response = Vec<Video>;

    const PATH: &'static str = "videos";
    #[cfg(feature = "twitch_oauth2")]
    const SCOPE: &'static [twitch_oauth2::Scope] = &[];
}

impl RequestGet for GetVideosRequest {}

impl helix::Paginated for GetVideosRequest {
    fn set_pagination(&mut self, cursor: Option<helix::Cursor>) { self.after = cursor }
}

#[cfg(test)]
#[test]
fn test_request() {
    use helix::*;
    let req = GetVideosRequest::builder()
        .id(vec!["234482848".into()])
        .build();

    // From twitch docs
    let data = br#"
{
  "data": [
    {
      "id": "335921245",
      "stream_id": null,
      "user_id": "141981764",
      "user_login": "twitchdev",
      "user_name": "TwitchDev",
      "title": "Twitch Developers 101",
      "description": "Welcome to Twitch development! Here is a quick overview of our products and information to help you get started.",
      "created_at": "2018-11-14T21:30:18Z",
      "published_at": "2018-11-14T22:04:30Z",
      "url": "https://www.twitch.tv/videos/335921245",
      "thumbnail_url": "https://static-cdn.jtvnw.net/cf_vods/d2nvs31859zcd8/twitchdev/335921245/ce0f3a7f-57a3-4152-bc06-0c6610189fb3/thumb/index-0000000000-%{width}x%{height}.jpg",
      "viewable": "public",
      "view_count": 1863062,
      "language": "en",
      "type": "upload",
      "duration": "3m21s",
      "muted_segments": [
        {
          "duration": 30,
          "offset": 120
        }
      ]
    }
  ],
  "pagination": {}
}
"#
        .to_vec();

    let http_response = http::Response::builder().body(data).unwrap();

    let uri = req.get_uri().unwrap();
    assert_eq!(
        uri.to_string(),
        "https://api.twitch.tv/helix/videos?id=234482848"
    );

    dbg!(GetVideosRequest::parse_response(Some(req), &uri, http_response).unwrap());
}