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
#![deny(intra_doc_link_resolution_failure)]
#![warn(missing_docs)]
#![doc(html_root_url = "https://docs.rs/twitch_api2/0.2.1")]
//! Rust library for talking with the new Twitch API aka. "Helix" and TMI.
//!
//! [![github]](https://github.com/emilgardis/twitch_api2) [![crates-io]](https://crates.io/crates/twitch_api2) [![docs-rs]](https://docs.rs/twitch_api2/0.2.1/twitch_api2)
//!
//! [github]: https://img.shields.io/badge/github-emilgardis/twitch__utils-8da0cb?style=for-the-badge&labelColor=555555&logo=github"
//! [crates-io]: https://img.shields.io/crates/v/twitch_api2.svg?style=for-the-badge&color=fc8d62&logo=rust"
//! [docs-rs]: https://img.shields.io/badge/docs.rs-twitch__api2-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo="
//!
//! <br>
//!
//! <h5>Use Twitch endpoints fearlessly</h5>
//!
//! ```rust,no_run
//! use twitch_api2::{Client, helix::channel::GetChannelRequest};
//! use twitch_oauth2::{AppAccessToken, Scope, TokenError, TwitchToken};
//!
//! # #[tokio::main]
//! # async fn run() -> Result<(), Box<dyn std::error::Error + 'static>> {
//! # let client_id = twitch_oauth2::ClientId::new("validclientid".to_string());
//! # let client_secret = twitch_oauth2::ClientSecret::new("validclientsecret".to_string());
//! let client =
//!     match AppAccessToken::get_app_access_token(client_id, client_secret, Scope::all()).await {
//!         Ok(t) => Client::with_token(t).unwrap(),
//!         Err(TokenError::RequestError(e)) => panic!("got error: {:?}", e),
//!         Err(e) => panic!(e),
//!     };
//!
//! let req = GetChannelRequest {
//!     broadcaster_id: client.helix.token().await.as_ref().validate_token().await?.user_id,
//! };
//!
//! println!("{:?}", &client.helix.req_get(req).await?.data[0].title);
//! # Ok(())
//! # }
//! # fn main() {run().unwrap();}
//! ```
//!
//!
#[doc(no_inline)]
pub use crate::helix::HelixClient;
#[doc(no_inline)]
pub use crate::tmi::TMIClient;

#[doc(no_inline)]
pub use twitch_oauth2;

pub mod helix;
pub mod tmi;

static TWITCH_HELIX_URL: &str = "https://api.twitch.tv/helix/";
static TWITCH_TMI_URL: &str = "https://tmi.twitch.tv/";

/// Client for Twitch APIs.
#[derive(Clone)]
pub struct Client {
    /// Helix endpoint. See [helix]
    pub helix: HelixClient,
    /// TMI endpoint. See [tmi]
    pub tmi: TMIClient,
}

impl Client {
    /// Create a new [Client] with an [twitch_oauth2::AppAccessToken]
    pub async fn new(
        client_id: twitch_oauth2::ClientId,
        client_secret: twitch_oauth2::ClientSecret,
        scopes: Vec<twitch_oauth2::Scope>,
    ) -> Result<Client, Error>
    {
        let token =
            twitch_oauth2::AppAccessToken::get_app_access_token(client_id, client_secret, scopes)
                .await?;
        Client::with_token(token)
    }

    /// Create a new [Client] with a generic [twitch_oauth2::TwitchToken]
    pub fn with_token<T>(token: T) -> Result<Client, Error>
    where T: twitch_oauth2::TwitchToken + Sized + Send + Sync + 'static {
        let helix = HelixClient::new(token.into());
        Ok(Client {
            tmi: TMIClient::new_with_client(helix.clone_client()),
            helix,
        })
    }
}

#[allow(missing_docs)]
#[derive(Debug, thiserror::Error)]
pub enum Error {
    #[error("could not make token")]
    TokenError(#[from] twitch_oauth2::TokenError),
}