commit
36ae371beb
@ -0,0 +1,2 @@ |
|||||||
|
target |
||||||
|
Cargo.lock |
@ -0,0 +1,20 @@ |
|||||||
|
[package] |
||||||
|
name = "mammut" |
||||||
|
version = "0.1.0" |
||||||
|
description = "A wrapper around the Mastodon API." |
||||||
|
authors = ["Aaron Power <theaaronepower@gmail.com>"] |
||||||
|
license = "MIT/Apache-2.0" |
||||||
|
readme = "README.md" |
||||||
|
repository = "https://github.com/Aaronepower/mastodon.rs.git" |
||||||
|
keywords = ["api", "web", "social", "mastodon", "wrapper"] |
||||||
|
categories = ["web-programming", "http-client"] |
||||||
|
|
||||||
|
[dependencies] |
||||||
|
reqwest = "0.5" |
||||||
|
serde = "0.9" |
||||||
|
serde_json = "0.9" |
||||||
|
serde_derive = "0.9" |
||||||
|
|
||||||
|
[dependencies.chrono] |
||||||
|
version = "0.3" |
||||||
|
features = ["serde"] |
@ -0,0 +1,13 @@ |
|||||||
|
Copyright 2016 Aaron Power |
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
you may not use this file except in compliance with the License. |
||||||
|
You may obtain a copy of the License at |
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software |
||||||
|
distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
See the License for the specific language governing permissions and |
||||||
|
limitations under the License. |
@ -0,0 +1,21 @@ |
|||||||
|
MIT License (MIT) |
||||||
|
|
||||||
|
Copyright (c) 2016 Aaron Power |
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy |
||||||
|
of this software and associated documentation files (the "Software"), to deal |
||||||
|
in the Software without restriction, including without limitation the rights |
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
||||||
|
copies of the Software, and to permit persons to whom the Software is |
||||||
|
furnished to do so, subject to the following conditions: |
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in |
||||||
|
all copies or substantial portions of the Software. |
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
||||||
|
THE SOFTWARE. |
@ -0,0 +1,5 @@ |
|||||||
|
# Mammut. A API Wrapper for the Mastodon API. |
||||||
|
|
||||||
|
### [Documentation](https://docs.rs/mastodon) |
||||||
|
|
||||||
|
A wrapper around the [API](https://github.com/tootsuite/mastodon/blob/master/docs/Using-the-API/API.md#tag) for [Mastodon](https://mastodon.social/) |
@ -0,0 +1,32 @@ |
|||||||
|
#[derive(Debug, Default, Serialize)] |
||||||
|
pub struct AppBuilder<'a> { |
||||||
|
pub client_name: &'a str, |
||||||
|
pub redirect_uris: &'a str, |
||||||
|
pub scopes: Scope, |
||||||
|
#[serde(skip_serializing_if="Option::is_none")] |
||||||
|
pub website: Option<&'a str>, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy, Serialize)] |
||||||
|
pub enum Scope { |
||||||
|
#[serde(rename = "read write follow")] |
||||||
|
All, |
||||||
|
#[serde(rename = "follow")] |
||||||
|
Follow, |
||||||
|
#[serde(rename = "read")] |
||||||
|
Read, |
||||||
|
#[serde(rename = "read follow")] |
||||||
|
ReadFollow, |
||||||
|
#[serde(rename = "read write")] |
||||||
|
ReadWrite, |
||||||
|
#[serde(rename = "write")] |
||||||
|
Write, |
||||||
|
#[serde(rename = "write follow")] |
||||||
|
WriteFollow, |
||||||
|
} |
||||||
|
|
||||||
|
impl Default for Scope { |
||||||
|
fn default() -> Self { |
||||||
|
Scope::Read |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,17 @@ |
|||||||
|
use chrono::prelude::*; |
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Account { |
||||||
|
pub id: u64, |
||||||
|
pub username: String, |
||||||
|
pub acct: String, |
||||||
|
pub display_name: String, |
||||||
|
pub note: String, |
||||||
|
pub url: String, |
||||||
|
pub avatar: String, |
||||||
|
pub header: String, |
||||||
|
pub locked: bool, |
||||||
|
pub created_at: DateTime<UTC>, |
||||||
|
pub followers_count: u64, |
||||||
|
pub following_count: u64, |
||||||
|
pub statuses_count: u64, |
||||||
|
} |
@ -0,0 +1,20 @@ |
|||||||
|
#[derive(Debug, Clone, Deserialize)] |
||||||
|
pub struct Attachment { |
||||||
|
pub id: u64, |
||||||
|
#[serde(rename="type")] |
||||||
|
pub media_type: MediaType, |
||||||
|
pub url: String, |
||||||
|
pub remote_url: String, |
||||||
|
pub preview_url: String, |
||||||
|
pub text_url: String, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Debug, Deserialize, Clone, Copy)] |
||||||
|
pub enum MediaType { |
||||||
|
#[serde(rename = "image")] |
||||||
|
Image, |
||||||
|
#[serde(rename = "video")] |
||||||
|
Video, |
||||||
|
#[serde(rename = "gifv")] |
||||||
|
Gifv, |
||||||
|
} |
@ -0,0 +1,7 @@ |
|||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Card { |
||||||
|
pub url: String, |
||||||
|
pub title: String, |
||||||
|
pub description: String, |
||||||
|
pub image: String, |
||||||
|
} |
@ -0,0 +1,7 @@ |
|||||||
|
use super::status::Status; |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Context { |
||||||
|
pub ancestors: Vec<Status>, |
||||||
|
pub descendants: Vec<Status>, |
||||||
|
} |
@ -0,0 +1,7 @@ |
|||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Instance { |
||||||
|
pub uri: String, |
||||||
|
pub title: String, |
||||||
|
pub description: String, |
||||||
|
pub email: String, |
||||||
|
} |
@ -0,0 +1,28 @@ |
|||||||
|
pub mod account; |
||||||
|
pub mod attachment; |
||||||
|
pub mod card; |
||||||
|
pub mod context; |
||||||
|
pub mod instance; |
||||||
|
pub mod notification; |
||||||
|
pub mod relationship; |
||||||
|
pub mod report; |
||||||
|
pub mod search_result; |
||||||
|
pub mod status; |
||||||
|
|
||||||
|
/// An empty JSON object.
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Empty {} |
||||||
|
|
||||||
|
pub mod prelude { |
||||||
|
pub use super::Empty; |
||||||
|
pub use super::account::Account; |
||||||
|
pub use super::attachment::{Attachment, MediaType}; |
||||||
|
pub use super::card::Card; |
||||||
|
pub use super::context::Context; |
||||||
|
pub use super::instance::Instance; |
||||||
|
pub use super::notification::Notification; |
||||||
|
pub use super::relationship::Relationship; |
||||||
|
pub use super::report::Report; |
||||||
|
pub use super::search_result::SearchResult; |
||||||
|
pub use super::status::{Status, Application}; |
||||||
|
} |
@ -0,0 +1,20 @@ |
|||||||
|
use chrono::prelude::*; |
||||||
|
use super::account::Account; |
||||||
|
use super::status::Status; |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Notification { |
||||||
|
pub id: u64, |
||||||
|
pub notification_type: NotificationType, |
||||||
|
pub created_at: DateTime<UTC>, |
||||||
|
pub account: Account, |
||||||
|
pub status: Option<Status>, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub enum NotificationType { |
||||||
|
Mention, |
||||||
|
Reblog, |
||||||
|
Favourite, |
||||||
|
Follow, |
||||||
|
} |
@ -0,0 +1,8 @@ |
|||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Relationship { |
||||||
|
pub following: bool, |
||||||
|
pub followed_by: bool, |
||||||
|
pub blocking: bool, |
||||||
|
pub muting: bool, |
||||||
|
pub requested: bool, |
||||||
|
} |
@ -0,0 +1,5 @@ |
|||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Report { |
||||||
|
pub id: u64, |
||||||
|
pub action_taken: String, |
||||||
|
} |
@ -0,0 +1,8 @@ |
|||||||
|
use super::prelude::{Account, Status}; |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct SearchResult { |
||||||
|
pub accounts: Vec<Account>, |
||||||
|
pub statuses: Vec<Status>, |
||||||
|
pub hashtags: Vec<String>, |
||||||
|
} |
@ -0,0 +1,49 @@ |
|||||||
|
use chrono::prelude::*; |
||||||
|
use super::prelude::*; |
||||||
|
use status_builder::Visibility; |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Status { |
||||||
|
pub id: i64, |
||||||
|
pub uri: String, |
||||||
|
pub url: String, |
||||||
|
pub account: Account, |
||||||
|
pub in_reply_to_id: Option<u64>, |
||||||
|
pub in_reply_to_account_id: Option<u64>, |
||||||
|
pub reblog: Option<Box<Status>>, |
||||||
|
pub content: String, |
||||||
|
pub created_at: DateTime<UTC>, |
||||||
|
pub reblogs_count: u64, |
||||||
|
pub favourites_count: u64, |
||||||
|
pub reblogged: bool, |
||||||
|
pub favourited: bool, |
||||||
|
pub sensitive: bool, |
||||||
|
pub spoiler_text: String, |
||||||
|
pub visibility: Visibility, |
||||||
|
pub media_attachments: Vec<Attachment>, |
||||||
|
pub mentions: Vec<Mention>, |
||||||
|
pub tags: Vec<Tag>, |
||||||
|
pub application: Application, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Mention { |
||||||
|
pub url: String, |
||||||
|
pub username: String, |
||||||
|
pub acct: String, |
||||||
|
pub id: u64, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Tag { |
||||||
|
pub name: String, |
||||||
|
pub url: String, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
pub struct Application { |
||||||
|
pub name: String, |
||||||
|
pub website: String, |
||||||
|
} |
||||||
|
|
||||||
|
|
@ -0,0 +1,374 @@ |
|||||||
|
//! # Mammut: API Wrapper around the Mastodon API.
|
||||||
|
//!
|
||||||
|
//! Most of the api is documented on [Mastodon's
|
||||||
|
//! github](https://github.com/tootsuite/mastodon/blob/master/docs/Using-the-API/API.md#tag)
|
||||||
|
#![deny(unused_must_use)] |
||||||
|
|
||||||
|
#[cfg_attr(test, deny(warnings))] |
||||||
|
|
||||||
|
#[macro_use] extern crate serde_derive; |
||||||
|
#[macro_use] extern crate serde_json as json; |
||||||
|
extern crate chrono; |
||||||
|
extern crate reqwest; |
||||||
|
extern crate serde; |
||||||
|
|
||||||
|
pub mod apps; |
||||||
|
pub mod status_builder; |
||||||
|
pub mod entities; |
||||||
|
|
||||||
|
use json::Error as SerdeError; |
||||||
|
use reqwest::Error as HttpError; |
||||||
|
use reqwest::Client; |
||||||
|
use reqwest::header::{Authorization, Bearer, Headers}; |
||||||
|
|
||||||
|
use entities::prelude::*; |
||||||
|
use status_builder::StatusBuilder; |
||||||
|
|
||||||
|
pub type Result<T> = std::result::Result<T, Error>; |
||||||
|
|
||||||
|
macro_rules! methods { |
||||||
|
($($method:ident,)+) => { |
||||||
|
$( |
||||||
|
fn $method<T: serde::Deserialize>(&self, url: String) |
||||||
|
-> Result<T> |
||||||
|
{ |
||||||
|
Ok(self.client.$method(&url) |
||||||
|
.headers(self.access_token.clone().unwrap()) |
||||||
|
.send()? |
||||||
|
.json()?) |
||||||
|
} |
||||||
|
)+ |
||||||
|
}; |
||||||
|
} |
||||||
|
|
||||||
|
macro_rules! route { |
||||||
|
|
||||||
|
((post ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { |
||||||
|
/// Requires `access_token` or will return error.
|
||||||
|
pub fn $name(&self, $($param: $typ,)*) -> Result<$ret> { |
||||||
|
self.has_access_token()?; |
||||||
|
|
||||||
|
let form_data = json!({ |
||||||
|
$( |
||||||
|
stringify!($param): $param, |
||||||
|
)* |
||||||
|
}); |
||||||
|
|
||||||
|
Ok(self.client.post(&self.route(concat!("/api/v1/", $url))) |
||||||
|
.headers(self.access_token.clone().unwrap()) |
||||||
|
.form(&form_data) |
||||||
|
.send()? |
||||||
|
.json()?) |
||||||
|
} |
||||||
|
route!{$($rest)*} |
||||||
|
}; |
||||||
|
|
||||||
|
(($method:ident) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { |
||||||
|
/// Requires `access_token` or will return error.
|
||||||
|
pub fn $name(&self) -> Result<$ret> { |
||||||
|
self.has_access_token()?; |
||||||
|
|
||||||
|
self.$method(self.route(concat!("/api/v1/", $url))) |
||||||
|
} |
||||||
|
|
||||||
|
route!{$($rest)*} |
||||||
|
}; |
||||||
|
|
||||||
|
() => {} |
||||||
|
} |
||||||
|
|
||||||
|
macro_rules! route_id { |
||||||
|
|
||||||
|
($(($method:ident) $name:ident: $url:expr => $ret:ty,)*) => { |
||||||
|
$( |
||||||
|
/// Requires `access_token` or will return error.
|
||||||
|
pub fn $name(&self, id: u64) -> Result<$ret> { |
||||||
|
self.has_access_token()?; |
||||||
|
|
||||||
|
|
||||||
|
self.$method(self.route(&format!(concat!("/api/v1/", $url), id))) |
||||||
|
} |
||||||
|
)* |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Clone, Debug)] |
||||||
|
pub struct Mastodon { |
||||||
|
base_url: String, |
||||||
|
client: Client, |
||||||
|
client_id: Option<String>, |
||||||
|
client_secret: Option<String>, |
||||||
|
redirect_uri: Option<String>, |
||||||
|
access_token: Option<Headers>, |
||||||
|
id: Option<u64>, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Deserialize)] |
||||||
|
struct OAuth { |
||||||
|
client_id: String, |
||||||
|
client_secret: String, |
||||||
|
id: u64, |
||||||
|
redirect_uri: String, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Debug)] |
||||||
|
pub enum Error { |
||||||
|
Serde(SerdeError), |
||||||
|
Http(HttpError), |
||||||
|
ClientIdRequired, |
||||||
|
ClientSecretRequired, |
||||||
|
AccessTokenRequired, |
||||||
|
} |
||||||
|
|
||||||
|
impl Mastodon { |
||||||
|
/// Inits new Mastodon object. `base_url` is expected in the following
|
||||||
|
/// format `https://mastodon.social` with no leading forward slash.
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use mammut::Mastodon;
|
||||||
|
///
|
||||||
|
/// let mastodon = Mastodon::new("https://mastodon.social").unwrap();
|
||||||
|
/// ```
|
||||||
|
pub fn new<I: Into<String>>(base_url: I) -> Result<Self> { |
||||||
|
Ok(Mastodon { |
||||||
|
base_url: base_url.into(), |
||||||
|
client: Client::new()?, |
||||||
|
client_id: None, |
||||||
|
client_secret: None, |
||||||
|
redirect_uri: None, |
||||||
|
access_token: None, |
||||||
|
id: None, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
/// Register the application with the server from `base_url`.
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// # extern crate mammut;
|
||||||
|
/// # fn main() {
|
||||||
|
/// # try().unwrap();
|
||||||
|
/// # }
|
||||||
|
///
|
||||||
|
/// # fn try() -> mammut::Result<()> {
|
||||||
|
/// use mammut::Mastodon;
|
||||||
|
/// use mammut::apps::{AppBuilder, Scope};
|
||||||
|
///
|
||||||
|
/// let app = AppBuilder {
|
||||||
|
/// client_name: "mammut_test",
|
||||||
|
/// redirect_uris: "urn:ietf:wg:oauth:2.0:oob",
|
||||||
|
/// scopes: Scope::Read,
|
||||||
|
/// website: None,
|
||||||
|
/// };
|
||||||
|
///
|
||||||
|
/// let mut mastodon = Mastodon::new("https://mastodon.social")?;
|
||||||
|
/// mastodon.register(app)?;
|
||||||
|
/// # Ok(())
|
||||||
|
/// # }
|
||||||
|
/// ```
|
||||||
|
pub fn register(&mut self, app_builder: apps::AppBuilder) -> Result<()> { |
||||||
|
let url = self.route("/api/v1/apps"); |
||||||
|
|
||||||
|
let app: OAuth = self.client.post(&url).form(&app_builder).send()?.json()?; |
||||||
|
|
||||||
|
self.id = Some(app.id); |
||||||
|
self.client_id = Some(app.client_id); |
||||||
|
self.client_secret = Some(app.client_secret); |
||||||
|
self.redirect_uri = Some(app.redirect_uri); |
||||||
|
|
||||||
|
Ok(()) |
||||||
|
} |
||||||
|
|
||||||
|
/// Returns the full url needed for authorisation. This needs to be opened
|
||||||
|
/// in a browser.
|
||||||
|
pub fn authorise(&mut self) -> Result<String> { |
||||||
|
self.is_registered()?; |
||||||
|
|
||||||
|
let url = format!( |
||||||
|
"{}/oauth/authorize?client_id={}&redirect_uri={}&response_type=code", |
||||||
|
self.base_url, |
||||||
|
self.client_id.clone().unwrap(), |
||||||
|
self.redirect_uri.clone().unwrap(), |
||||||
|
); |
||||||
|
|
||||||
|
Ok(url) |
||||||
|
} |
||||||
|
|
||||||
|
/// Set `access_token` required to use any method about the user.
|
||||||
|
pub fn set_access_token(&mut self, access_token: String) { |
||||||
|
let mut headers = Headers::new(); |
||||||
|
|
||||||
|
headers.set(Authorization(Bearer { token: access_token })); |
||||||
|
|
||||||
|
self.access_token = Some(headers); |
||||||
|
} |
||||||
|
|
||||||
|
route! { |
||||||
|
(get) verify: "accounts/verify_credentials" => Account, |
||||||
|
(get) blocks: "blocks" => Vec<Account>, |
||||||
|
(get) follow_requests: "follow_requests" => Vec<Account>, |
||||||
|
(get) mutes: "mutes" => Vec<Account>, |
||||||
|
(get) notifications: "notifications" => Vec<Notification>, |
||||||
|
(get) reports: "reports" => Vec<Report>, |
||||||
|
(get) get_home_timeline: "timelines/home" => Vec<Status>, |
||||||
|
(post (id: u64,)) allow_follow_request: "accounts/follow_requests/authorize" => Empty, |
||||||
|
(post (id: u64,)) reject_follow_request: "accounts/follow_requests/reject" => Empty, |
||||||
|
(post (uri: String,)) follows: "follows" => Account, |
||||||
|
(post) clear_notifications: "notifications/clear" => Empty, |
||||||
|
(post (file: Vec<u8>,)) media: "media" => Attachment, |
||||||
|
(post (account_id: u64, status_ids: Vec<u64>, comment: String,)) report: |
||||||
|
"reports" => Report, |
||||||
|
(post (q: String, resolve: bool,)) search: "search" => SearchResult, |
||||||
|
(post (status: StatusBuilder,)) new_status: "statuses" => Status, |
||||||
|
} |
||||||
|
|
||||||
|
route_id! { |
||||||
|
(get) get_account: "accounts/{}" => Account, |
||||||
|
(get) followers: "accounts/{}/followers" => Vec<Account>, |
||||||
|
(get) following: "accounts/{}/following" => Vec<Account>, |
||||||
|
(get) follow: "accounts/{}/follow" => Account, |
||||||
|
(get) unfollow: "accounts/{}/unfollow" => Account, |
||||||
|
(get) block: "accounts/{}/block" => Account, |
||||||
|
(get) unblock: "accounts/{}/unblock" => Account, |
||||||
|
(get) mute: "accounts/{}/mute" => Account, |
||||||
|
(get) unmute: "accounts/{}/unmute" => Account, |
||||||
|
(get) get_notification: "notifications/{}" => Notification, |
||||||
|
(get) get_status: "statuses/{}" => Status, |
||||||
|
(get) get_context: "statuses/{}/context" => Context, |
||||||
|
(get) get_card: "statuses/{}/card" => Card, |
||||||
|
(get) reblogged_by: "statuses/{}/reblogged_by" => Vec<Account>, |
||||||
|
(get) favourited_by: "statuses/{}/favourited_by" => Vec<Account>, |
||||||
|
(post) reblog: "statuses/{}/reblog" => Status, |
||||||
|
(post) unreblog: "statuses/{}/unreblog" => Status, |
||||||
|
(post) favourite: "statuses/{}/favourite" => Status, |
||||||
|
(post) unfavourite: "statuses/{}/unfavourite" => Status, |
||||||
|
(delete) delete_status: "statuses/{}" => Empty, |
||||||
|
} |
||||||
|
|
||||||
|
pub fn get_public_timeline(&self, local: bool) -> Result<Vec<Status>> { |
||||||
|
self.has_access_token()?; |
||||||
|
|
||||||
|
let mut url = self.route("/api/v1/timelines/public"); |
||||||
|
|
||||||
|
if local { |
||||||
|
url += "?local=1"; |
||||||
|
} |
||||||
|
|
||||||
|
self.get(url) |
||||||
|
} |
||||||
|
|
||||||
|
pub fn get_tagged_timeline(&self, hashtag: String, local: bool) -> Result<Vec<Status>> { |
||||||
|
self.has_access_token()?; |
||||||
|
|
||||||
|
let mut url = self.route("/api/v1/timelines/tag/"); |
||||||
|
url += &hashtag; |
||||||
|
|
||||||
|
if local { |
||||||
|
url += "?local=1"; |
||||||
|
} |
||||||
|
|
||||||
|
self.get(url) |
||||||
|
} |
||||||
|
|
||||||
|
pub fn statuses(&self, id: u64, only_media: bool, exclude_replies: bool) |
||||||
|
-> Result<Vec<Status>> |
||||||
|
{ |
||||||
|
self.has_access_token()?; |
||||||
|
let mut url = format!("{}/api/v1/accounts/{}/statuses", self.base_url, id); |
||||||
|
|
||||||
|
if only_media { |
||||||
|
url += "?only_media=1"; |
||||||
|
} |
||||||
|
|
||||||
|
if exclude_replies { |
||||||
|
url += if only_media { |
||||||
|
"&" |
||||||
|
} else { |
||||||
|
"?" |
||||||
|
}; |
||||||
|
|
||||||
|
url += "exclude_replies=1"; |
||||||
|
} |
||||||
|
|
||||||
|
self.get(url) |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
pub fn relationships(&self, ids: &[u64]) -> Result<Vec<Relationship>> { |
||||||
|
self.has_access_token()?; |
||||||
|
|
||||||
|
let mut url = self.route("/api/v1/accounts/relationships?"); |
||||||
|
|
||||||
|
if ids.len() == 1 { |
||||||
|
url += "id="; |
||||||
|
url += &ids[0].to_string(); |
||||||
|
} else { |
||||||
|
for id in ids { |
||||||
|
url += "id[]="; |
||||||
|
url += &id.to_string(); |
||||||
|
url += "&"; |
||||||
|
} |
||||||
|
url.pop(); |
||||||
|
} |
||||||
|
|
||||||
|
self.get(url) |
||||||
|
} |
||||||
|
|
||||||
|
// TODO: Add a limit fn
|
||||||
|
pub fn search_accounts(&self, query: &str) -> Result<Vec<Account>> { |
||||||
|
self.has_access_token()?; |
||||||
|
self.get(format!("{}/api/v1/accounts/search?q={}", self.base_url, query)) |
||||||
|
} |
||||||
|
|
||||||
|
pub fn instance(&self) -> Result<Instance> { |
||||||
|
self.is_registered()?; |
||||||
|
|
||||||
|
self.get(self.route("/api/v1/instance")) |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
fn has_access_token(&self) -> Result<()> { |
||||||
|
if self.access_token.is_none() { |
||||||
|
Err(Error::AccessTokenRequired) |
||||||
|
} else { |
||||||
|
Ok(()) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
fn is_registered(&self) -> Result<()> { |
||||||
|
if self.client_id.is_none() { |
||||||
|
Err(Error::ClientIdRequired) |
||||||
|
} else if self.client_secret.is_none() { |
||||||
|
Err(Error::ClientSecretRequired) |
||||||
|
} else { |
||||||
|
Ok(()) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
methods![get, post, delete,]; |
||||||
|
|
||||||
|
fn route(&self, url: &str) -> String { |
||||||
|
let mut s = self.base_url.clone(); |
||||||
|
s += url; |
||||||
|
s |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
macro_rules! from { |
||||||
|
($($typ:ident, $variant:ident,)*) => { |
||||||
|
$( |
||||||
|
impl From<$typ> for Error { |
||||||
|
fn from(from: $typ) -> Self { |
||||||
|
use Error::*; |
||||||
|
$variant(from) |
||||||
|
} |
||||||
|
} |
||||||
|
)* |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
from! { |
||||||
|
SerdeError, Serde, |
||||||
|
HttpError, Http, |
||||||
|
} |
@ -0,0 +1,42 @@ |
|||||||
|
#[derive(Debug, Default, Clone, Serialize)] |
||||||
|
pub struct StatusBuilder { |
||||||
|
status: String, |
||||||
|
#[serde(skip_serializing_if="Option::is_none")] |
||||||
|
in_reply_to_id: Option<u64>, |
||||||
|
#[serde(skip_serializing_if="Option::is_none")] |
||||||
|
media_ids: Option<Vec<u64>>, |
||||||
|
#[serde(skip_serializing_if="Option::is_none")] |
||||||
|
sensitive: Option<bool>, |
||||||
|
#[serde(skip_serializing_if="Option::is_none")] |
||||||
|
spoiler_text: Option<String>, |
||||||
|
#[serde(skip_serializing_if="Option::is_none")] |
||||||
|
visibility: Option<Visibility>, |
||||||
|
} |
||||||
|
|
||||||
|
#[derive(Clone, Copy, Debug, Deserialize, Serialize)] |
||||||
|
pub enum Visibility { |
||||||
|
#[serde(rename = "direct")] |
||||||
|
Direct, |
||||||
|
#[serde(rename = "private")] |
||||||
|
Private, |
||||||
|
#[serde(rename = "unlisted")] |
||||||
|
Unlisted, |
||||||
|
#[serde(rename = "public")] |
||||||
|
Public, |
||||||
|
} |
||||||
|
|
||||||
|
impl StatusBuilder { |
||||||
|
|
||||||
|
pub fn new(status: String) -> Self { |
||||||
|
StatusBuilder { |
||||||
|
status: status, |
||||||
|
..Self::default() |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
impl Default for Visibility { |
||||||
|
fn default() -> Self { |
||||||
|
Visibility::Public |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue