macro_rules! methods { ($($method:ident and $method_with_call_id:ident,)+) => { $( doc_comment! { concat!("Make a ", stringify!($method), " API request, and deserialize the result into T"), async fn $method serde::Deserialize<'de> + serde::Serialize>(&self, url: impl AsRef) -> Result { let call_id = uuid::Uuid::new_v4(); self.$method_with_call_id(url, call_id).await } } doc_comment! { concat!( "Make a ", stringify!($method), " API request, and deserialize the result into T.\n\n", "Logging will use the provided UUID, rather than generating one before making the request.", ), async fn $method_with_call_id serde::Deserialize<'de> + serde::Serialize>(&self, url: impl AsRef, call_id: Uuid) -> Result { use log::{debug, as_debug}; let url = url.as_ref(); debug!(url = url, method = stringify!($method), call_id = as_debug!(call_id); "making API request"); let response = self.authenticated(self.client.$method(url)).header("Accept", "application/json").send().await?; read_response(response).await } } )+ }; } macro_rules! paged_routes { (($method:ident) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `", stringify!($method), " /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set.", "\n", "```no_run", "use mastodon_async::prelude::*;\n", "let data = Data::default();\n", "let client = Mastodon::from(data);\n", "client.", stringify!($name), "();\n", "```" ), pub async fn $name(&self) -> Result> { use log::{debug, as_debug}; let url = self.route(concat!("/api/v1/", $url)); let call_id = uuid::Uuid::new_v4(); debug!(url = url, method = stringify!($method), call_id = as_debug!(call_id); "making API request"); let response = self.authenticated(self.client.$method(&url)).header("Accept", "application/json").send().await?; Page::new(self.clone(), response, call_id).await } } paged_routes!{$($rest)*} }; ((get ($($(#[$m:meta])* $param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `get /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set." ), pub async fn $name<'a>(&self, $($param: $typ,)*) -> Result> { use serde_urlencoded; use log::{debug, as_debug}; let call_id = uuid::Uuid::new_v4(); #[derive(Serialize)] struct Data<'a> { $( $( #[$m] )* $param: $typ, )* #[serde(skip)] _marker: ::std::marker::PhantomData<&'a ()>, } #[allow(clippy::redundant_field_names)] let qs_data = Data { $( $param: $param, )* _marker: ::std::marker::PhantomData, }; let qs = serde_urlencoded::to_string(&qs_data)?; let url = format!(concat!("/api/v1/", $url, "?{}"), &qs); debug!(url = url, method = "get", call_id = as_debug!(call_id); "making API request"); let response = self.authenticated(self.client.get(&url)).header("Accept", "application/json").send().await?; Page::new(self.clone(), response, call_id).await } } paged_routes!{$($rest)*} }; () => {} } macro_rules! route_v2 { ((get ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `get /api/v2/", $url, "`\n# Errors\nIf `access_token` is not set." ), pub async fn $name<'a>(&self, $($param: $typ,)*) -> Result<$ret> { use serde_urlencoded; use log::{debug, as_serde}; use uuid::Uuid; let call_id = Uuid::new_v4(); #[derive(Serialize)] struct Data<'a> { $( $param: $typ, )* #[serde(skip)] _marker: ::std::marker::PhantomData<&'a ()>, } #[allow(clippy::redundant_field_names)] let qs_data = Data { $( $param: $param, )* _marker: ::std::marker::PhantomData, }; let qs = serde_urlencoded::to_string(&qs_data)?; debug!(query_string_data = as_serde!(qs_data); "URL-encoded data to be sent in API request"); let url = format!(concat!("/api/v2/", $url, "?{}"), &qs); self.get_with_call_id(self.route(&url), call_id).await } } route_v2!{$($rest)*} }; ((post multipart with description ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `post /api/v2/", $url, "`, with a description/alt-text.", "\n# Errors\nIf `access_token` is not set."), pub async fn $name(&self $(, $param: $typ)*, description: Option) -> Result<$ret> { use reqwest::multipart::Form; use log::{debug, as_debug}; use uuid::Uuid; let call_id = Uuid::new_v4(); let form_data = Form::new() $( .part(stringify!($param), Self::get_form_part($param)?) )*; let form_data = if let Some(description) = description { form_data.text("description", description) } else { form_data }; let url = &self.route(concat!("/api/v2/", $url)); debug!( url = url, method = stringify!($method), multipart_form_data = as_debug!(form_data), call_id = as_debug!(call_id); "making API request" ); let response = self.authenticated(self.client.post(url)) .multipart(form_data) .header("Accept", "application/json") .send() .await?; read_response(response).await } } route_v2! { $($rest)* } }; ((post multipart ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `post /api/v2/", $url, "`\n# Errors\nIf `access_token` is not set."), pub async fn $name(&self, $($param: $typ,)*) -> Result<$ret> { use reqwest::multipart::Form; use log::{debug, as_debug}; use uuid::Uuid; let call_id = Uuid::new_v4(); let form_data = Form::new() $( .part(stringify!($param), Self::get_form_part($param)?) )*; let url = &self.route(concat!("/api/v2/", $url)); debug!( url = url, method = stringify!($method), multipart_form_data = as_debug!(form_data), call_id = as_debug!(call_id); "making API request" ); let response = self.authenticated(self.client.post(url)) .multipart(form_data) .header("Accept", "application/json") .send() .await?; read_response(response).await } } route_v2!{$($rest)*} }; () => {} } macro_rules! route { ((post multipart ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `post /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set."), pub async fn $name(&self, $($param: $typ,)*) -> Result<$ret> { use reqwest::multipart::Form; use log::{debug, as_debug}; use uuid::Uuid; let call_id = Uuid::new_v4(); let form_data = Form::new() $( .part(stringify!($param), Self::get_form_part($param)?) )*; let url = &self.route(concat!("/api/v1/", $url)); debug!( url = url, method = stringify!($method), multipart_form_data = as_debug!(form_data), call_id = as_debug!(call_id); "making API request" ); let response = self.authenticated(self.client.post(url)) .multipart(form_data) .header("Accept", "application/json") .send() .await?; read_response(response).await } } route!{$($rest)*} }; ((post multipart with description ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `post /api/v1/", $url, "`, with a description/alt-text.", "\n# Errors\nIf `access_token` is not set."), pub async fn $name(&self $(, $param: $typ)*, description: Option) -> Result<$ret> { use reqwest::multipart::Form; use log::{debug, as_debug}; use uuid::Uuid; let call_id = Uuid::new_v4(); let form_data = Form::new() $( .part(stringify!($param), Self::get_form_part($param)?) )*; let form_data = if let Some(description) = description { form_data.text("description", description) } else { form_data }; let url = &self.route(concat!("/api/v1/", $url)); debug!( url = url, method = stringify!($method), multipart_form_data = as_debug!(form_data), call_id = as_debug!(call_id); "making API request" ); let response = self.authenticated(self.client.post(url)) .multipart(form_data) .header("Accept", "application/json") .send() .await?; read_response(response).await } } route! { $($rest)* } }; ((get ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `get /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set." ), pub async fn $name<'a>(&self, $($param: $typ,)*) -> Result<$ret> { use serde_urlencoded; use log::{debug, as_serde}; use uuid::Uuid; let call_id = Uuid::new_v4(); #[derive(Serialize)] struct Data<'a> { $( $param: $typ, )* #[serde(skip)] _marker: ::std::marker::PhantomData<&'a ()>, } #[allow(clippy::redundant_field_names)] let qs_data = Data { $( $param: $param, )* _marker: ::std::marker::PhantomData, }; let qs = serde_urlencoded::to_string(&qs_data)?; debug!(query_string_data = as_serde!(qs_data); "URL-encoded data to be sent in API request"); let url = format!(concat!("/api/v1/", $url, "?{}"), &qs); self.get_with_call_id(self.route(&url), call_id).await } } route!{$($rest)*} }; (($method:ident ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `", stringify!($method), " /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set.", ), pub async fn $name(&self, $($param: $typ,)*) -> Result<$ret> { use log::{debug, as_debug, as_serde}; use uuid::Uuid; let call_id = Uuid::new_v4(); let form_data = json!({ $( stringify!($param): $param, )* }); let url = &self.route(concat!("/api/v1/", $url)); debug!( url = url.as_str(), method = stringify!($method), call_id = as_debug!(call_id), form_data = as_serde!(&form_data); "making API request" ); let response = self.authenticated(self.client.$method(url)) .json(&form_data) .header("Accept", "application/json") .send() .await?; read_response(response).await } } route!{$($rest)*} }; (($method:ident) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `", stringify!($method), " /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set.", "\n", "```no_run", "use mastodon_async::prelude::*;\n", "let data = Data::default();\n", "let client = Mastodon::from(data);\n", "client.", stringify!($name), "();\n", "```" ), pub async fn $name(&self) -> Result<$ret> { self.$method(self.route(concat!("/api/v1/", $url))).await } } route!{$($rest)*} }; () => {} } macro_rules! route_id { ($(($method:ident) $name:ident[$id_type:ty]: $url:expr => $ret:ty,)*) => { $( doc_comment! { concat!( "Equivalent to `", stringify!($method), " /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set.", "\n", "```no_run", "use mastodon_async::prelude::*;\n", "let data = Data::default();\n", "let client = Mastodon::from(data);\n", "client.", stringify!($name), "(\"42\");\n", "# Ok(())\n", "# }\n", "```" ), pub async fn $name(&self, id: &$id_type) -> Result<$ret> { self.$method(self.route(&format!(concat!("/api/v1/", $url), id))).await } } )* } } macro_rules! paged_routes_with_id { (($method:ident) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { doc_comment! { concat!( "Equivalent to `", stringify!($method), " /api/v1/", $url, "`\n# Errors\nIf `access_token` is not set.", "\n", "```no_run", "use mastodon_async::prelude::*;\n", "let data = Data::default();", "let client = Mastodon::from(data);\n", "client.", stringify!($name), "(\"some-id\");\n", "```" ), pub async fn $name(&self, id: impl AsRef) -> Result> { use log::{debug, as_debug}; use uuid::Uuid; let call_id = Uuid::new_v4(); let url = self.route(&format!(concat!("/api/v1/", $url), id.as_ref())); debug!(url = url, method = stringify!($method), call_id = as_debug!(call_id); "making API request"); let response = self.authenticated(self.client.$method(&url)).header("Accept", "application/json").send().await?; Page::new(self.clone(), response, call_id).await } } paged_routes_with_id!{$($rest)*} }; () => {} } macro_rules! streaming { ($desc:tt $fn_name:ident@$stream:literal, $($rest:tt)*) => { doc_comment! { concat!( $desc, "\n\nExample:\n\n", " use mastodon_async::prelude::*; use mastodon_async::entities::event::Event; use futures_util::{pin_mut, StreamExt, TryStreamExt}; tokio_test::block_on(async { let data = Data::default(); let client = Mastodon::from(data); let stream = client.", stringify!($fn_name), "().await.unwrap(); stream.try_for_each(|event| async move { match event { Event::Update(ref status) => { /* .. */ }, Event::Notification(ref notification) => { /* .. */ }, Event::Delete(ref id) => { /* .. */ }, Event::FiltersChanged => { /* .. */ }, } Ok(()) }).await.unwrap(); });" ), pub async fn $fn_name(&self) -> Result> { let url = self.route(&format!("/api/v1/streaming/{}", $stream)); let response = self.authenticated(self.client.get(&url)).header("Accept", "application/json").send().await?; debug!( status = log_serde!(response Status), url = &url, headers = log_serde!(response Headers); "received API response" ); let status = response.status(); if status.is_success() { Ok(event_stream(response, url)) } else { let response = response.json().await?; Err(Error::Api{ status, response }) } } } streaming! { $($rest)* } }; ($desc:tt $fn_name:ident($param:ident: $param_type:ty, like $param_doc_val:literal)@$stream:literal, $($rest:tt)*) => { doc_comment! { concat!( $desc, "\n\nExample:\n\n", " use mastodon_async::prelude::*; use mastodon_async::entities::event::Event; use futures_util::{pin_mut, StreamExt, TryStreamExt}; tokio_test::block_on(async { let data = Data::default(); let client = Mastodon::from(data); let stream = client.", stringify!($fn_name), "(", $param_doc_val, ").await.unwrap(); stream.try_for_each(|event| async move { match event { Event::Update(ref status) => { /* .. */ }, Event::Notification(ref notification) => { /* .. */ }, Event::Delete(ref id) => { /* .. */ }, Event::FiltersChanged => { /* .. */ }, } Ok(()) }).await.unwrap(); });" ), pub async fn $fn_name(&self, $param: $param_type) -> Result> { let mut url: Url = self.route(concat!("/api/v1/streaming/", stringify!($stream))).parse()?; url.query_pairs_mut().append_pair(stringify!($param), $param.as_ref()); let url = url.to_string(); let response = self.authenticated(self.client.get(url.as_str())).header("Accept", "application/json").send().await?; debug!( status = log_serde!(response Status), url = as_debug!(url), headers = log_serde!(response Headers); "received API response" ); let status = response.status(); if status.is_success() { Ok(event_stream(response, url)) } else { let response = response.json().await?; Err(Error::Api{ status, response }) } } } streaming! { $($rest)* } }; () => {} }