use serde::{Deserialize, Serialize};
use crate::token::*;
use futures_util::future::BoxFuture;
use lazy_static::*;
use slack_morphism_models::{SlackClientId, SlackClientSecret};
use url::Url;
#[derive(Debug)]
pub struct SlackClient<SCHC>
where
SCHC: SlackClientHttpConnector + Send,
{
pub http_api: SlackClientHttpApi<SCHC>,
}
#[derive(Debug)]
pub struct SlackClientHttpApi<SCHC>
where
SCHC: SlackClientHttpConnector + Send,
{
pub connector: SCHC,
}
#[derive(Debug)]
pub struct SlackClientSession<'a, SCHC>
where
SCHC: SlackClientHttpConnector + Send,
{
pub http_api: SlackClientHttpSessionApi<'a, SCHC>,
client: &'a SlackClient<SCHC>,
token: &'a SlackApiToken,
}
#[derive(Debug)]
pub struct SlackClientHttpSessionApi<'a, SCHC>
where
SCHC: SlackClientHttpConnector + Send,
{
client: &'a SlackClient<SCHC>,
token: &'a SlackApiToken,
}
pub trait SlackClientHttpConnector {
fn http_get_uri<'a, RS>(
&'a self,
full_uri: Url,
token: Option<&'a SlackApiToken>,
) -> BoxFuture<'a, ClientResult<RS>>
where
RS: for<'de> serde::de::Deserialize<'de> + Send + 'a + 'a + Send;
fn http_get_with_client_secret<'a, RS>(
&'a self,
full_uri: Url,
client_id: &'a SlackClientId,
client_secret: &'a SlackClientSecret,
) -> BoxFuture<'a, ClientResult<RS>>
where
RS: for<'de> serde::de::Deserialize<'de> + Send + 'a + 'a + Send;
fn http_get_token<'a, 'p, RS, PT, TS>(
&'a self,
method_relative_uri: &str,
params: &'p PT,
token: Option<&'a SlackApiToken>,
) -> BoxFuture<'a, ClientResult<RS>>
where
RS: for<'de> serde::de::Deserialize<'de> + Send + 'a,
PT: std::iter::IntoIterator<Item = (&'p str, Option<&'p TS>)> + Clone,
TS: std::string::ToString + 'p + 'a + Send,
{
let full_uri = SlackClientHttpApiUri::create_url_with_params(
&SlackClientHttpApiUri::create_method_uri_path(&method_relative_uri),
params,
);
self.http_get_uri(full_uri, token)
}
fn http_get<'a, 'p, RS, PT, TS>(
&'a self,
method_relative_uri: &str,
params: &'p PT,
) -> BoxFuture<'a, ClientResult<RS>>
where
RS: for<'de> serde::de::Deserialize<'de> + Send + 'a,
PT: std::iter::IntoIterator<Item = (&'p str, Option<&'p TS>)> + Clone,
TS: std::string::ToString + 'p + 'a + Send,
{
self.http_get_token(&method_relative_uri, params, None)
}
fn http_post_uri<'a, RQ, RS>(
&'a self,
full_uri: Url,
request_body: &'a RQ,
token: Option<&'a SlackApiToken>,
) -> BoxFuture<'a, ClientResult<RS>>
where
RQ: serde::ser::Serialize + Send + Sync,
RS: for<'de> serde::de::Deserialize<'de> + Send + 'a + Send + 'a;
fn http_post_token<'a, RQ, RS>(
&'a self,
method_relative_uri: &str,
request: &'a RQ,
token: Option<&'a SlackApiToken>,
) -> BoxFuture<'a, ClientResult<RS>>
where
RQ: serde::ser::Serialize + Send + Sync,
RS: for<'de> serde::de::Deserialize<'de> + Send + 'a,
{
let full_uri = SlackClientHttpApiUri::create_url(
&SlackClientHttpApiUri::create_method_uri_path(&method_relative_uri),
);
self.http_post_uri(full_uri, request, token)
}
fn http_post<'a, RQ, RS>(
&'a self,
method_relative_uri: &str,
request: &'a RQ,
) -> BoxFuture<'a, ClientResult<RS>>
where
RQ: serde::ser::Serialize + Send + Sync,
RS: for<'de> serde::de::Deserialize<'de> + Send + 'a,
{
self.http_post_token(method_relative_uri, request, None)
}
}
pub type ClientResult<T> = std::result::Result<T, Box<dyn std::error::Error + Send + Sync>>;
#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)]
pub struct SlackEnvelopeMessage {
pub ok: bool,
pub error: Option<String>,
pub warnings: Option<Vec<String>>,
}
lazy_static! {
pub static ref SLACK_HTTP_EMPTY_GET_PARAMS: Vec<(&'static str, Option<&'static String>)> =
vec![];
}
impl<SCHC> SlackClientHttpApi<SCHC>
where
SCHC: SlackClientHttpConnector + Send + Sync,
{
fn new(http_connector: SCHC) -> Self {
Self {
connector: http_connector,
}
}
}
pub struct SlackClientHttpApiUri;
impl SlackClientHttpApiUri {
const SLACK_API_URI_STR: &'static str = "https://slack.com/api";
pub fn create_method_uri_path(method_relative_uri: &str) -> String {
format!("{}/{}", Self::SLACK_API_URI_STR, method_relative_uri)
}
pub(crate) fn create_url(url_str: &str) -> Url {
url_str.parse().unwrap()
}
pub fn create_url_with_params<'p, PT, TS>(url_str: &str, params: &'p PT) -> Url
where
PT: std::iter::IntoIterator<Item = (&'p str, Option<&'p TS>)> + Clone,
TS: std::string::ToString + 'p,
{
let url_query_params: Vec<(String, String)> = params
.clone()
.into_iter()
.map(|(k, vo)| vo.map(|v| (k.to_string(), v.to_string())))
.flatten()
.collect();
Url::parse_with_params(url_str, url_query_params)
.unwrap()
.as_str()
.parse()
.unwrap()
}
}
impl<SCHC> SlackClient<SCHC>
where
SCHC: SlackClientHttpConnector + Send + Sync,
{
pub fn new(http_connector: SCHC) -> Self {
Self {
http_api: SlackClientHttpApi::new(http_connector),
}
}
pub fn open_session<'a>(&'a self, token: &'a SlackApiToken) -> SlackClientSession<'a, SCHC> {
let http_session_api = SlackClientHttpSessionApi {
client: self,
token,
};
SlackClientSession {
client: self,
token,
http_api: http_session_api,
}
}
}
impl<'a, SCHC> SlackClientHttpSessionApi<'a, SCHC>
where
SCHC: SlackClientHttpConnector + Send,
{
pub async fn http_get_uri<RS, PT, TS>(&self, full_uri: Url) -> ClientResult<RS>
where
RS: for<'de> serde::de::Deserialize<'de> + Send,
{
self.client
.http_api
.connector
.http_get_uri(full_uri, Some(&self.token))
.await
}
pub async fn http_get<'p, RS, PT, TS>(
&self,
method_relative_uri: &str,
params: &'p PT,
) -> ClientResult<RS>
where
RS: for<'de> serde::de::Deserialize<'de> + Send,
PT: std::iter::IntoIterator<Item = (&'p str, Option<&'p TS>)> + Clone,
TS: std::string::ToString + 'p + Send,
{
self.client
.http_api
.connector
.http_get_token(&method_relative_uri, params, Some(&self.token))
.await
}
pub async fn http_post<RQ, RS>(
&self,
method_relative_uri: &str,
request: &RQ,
) -> ClientResult<RS>
where
RQ: serde::ser::Serialize + Send + Sync,
RS: for<'de> serde::de::Deserialize<'de> + Send,
{
self.client
.http_api
.connector
.http_post_token(&method_relative_uri, &request, Some(&self.token))
.await
}
pub async fn http_post_uri<RQ, RS>(&self, full_uri: Url, request: &RQ) -> ClientResult<RS>
where
RQ: serde::ser::Serialize + Send + Sync,
RS: for<'de> serde::de::Deserialize<'de> + Send,
{
self.client
.http_api
.connector
.http_post_uri(full_uri, &request, Some(&self.token))
.await
}
}