musicus/crates/musicus_client/src/lib.rs

177 lines
4.9 KiB
Rust
Raw Normal View History

2021-02-06 17:20:41 +01:00
use isahc::{AsyncBody, Request, Response};
2020-11-14 22:32:21 +01:00
use isahc::http::StatusCode;
use isahc::prelude::*;
2020-11-17 15:52:47 +01:00
use serde::Serialize;
2020-11-28 22:01:59 +01:00
use std::time::Duration;
2021-02-04 21:47:22 +01:00
use std::cell::RefCell;
2020-11-17 15:52:47 +01:00
2020-11-28 22:23:35 +01:00
pub mod ensembles;
pub use ensembles::*;
2021-02-04 21:47:22 +01:00
pub mod error;
pub use error::*;
2020-11-28 23:07:31 +01:00
pub mod instruments;
pub use instruments::*;
2021-01-30 16:06:04 +01:00
pub mod mediums;
pub use mediums::*;
2020-11-28 23:07:31 +01:00
pub mod persons;
pub use persons::*;
2020-11-29 01:00:19 +01:00
pub mod recordings;
pub use recordings::*;
2021-01-30 23:16:44 +01:00
pub mod register;
pub use register::*;
2020-11-29 00:12:23 +01:00
pub mod works;
pub use works::*;
2020-11-17 15:52:47 +01:00
/// Credentials used for login.
#[derive(Serialize, Debug, Clone)]
#[serde(rename_all = "camelCase")]
pub struct LoginData {
pub username: String,
pub password: String,
}
2020-11-14 22:32:21 +01:00
2021-02-04 21:47:22 +01:00
/// A client for accessing the Wolfgang API.
pub struct Client {
server_url: RefCell<Option<String>>,
login_data: RefCell<Option<LoginData>>,
token: RefCell<Option<String>>,
}
2020-11-17 15:52:47 +01:00
2021-02-04 21:47:22 +01:00
impl Client {
/// Create a new client.
pub fn new() -> Self {
Self {
server_url: RefCell::new(None),
login_data: RefCell::new(None),
token: RefCell::new(None),
2020-11-17 15:52:47 +01:00
}
}
/// Set the URL of the Musicus server to connect to.
2021-02-04 21:47:22 +01:00
pub fn set_server_url(&self, url: &str) {
self.server_url.replace(Some(url.to_owned()));
2020-11-17 15:52:47 +01:00
}
/// Get the currently set server URL.
pub fn get_server_url(&self) -> Option<String> {
self.server_url.borrow().clone()
}
2021-02-04 21:47:22 +01:00
/// Set the user credentials to use.
pub fn set_login_data(&self, data: LoginData) {
self.login_data.replace(Some(data));
self.token.replace(None);
2021-02-04 16:31:37 +01:00
}
2020-11-17 15:52:47 +01:00
/// Get the currently stored login credentials.
pub fn get_login_data(&self) -> Option<LoginData> {
self.login_data.borrow().clone()
}
2020-11-14 22:32:21 +01:00
/// Try to login a user with the provided credentials and return, wether the login suceeded.
pub async fn login(&self) -> Result<bool> {
2021-02-04 16:31:37 +01:00
let server_url = self.server_url()?;
let data = self.login_data()?;
2020-11-14 22:32:21 +01:00
let request = Request::post(format!("{}/login", server_url))
2020-11-29 11:30:43 +01:00
.timeout(Duration::from_secs(10))
2020-11-14 22:32:21 +01:00
.header("Content-Type", "application/json")
.body(serde_json::to_string(&data)?)?;
let mut response = isahc::send_async(request).await?;
let success = match response.status() {
StatusCode::OK => {
2021-02-06 17:20:41 +01:00
let token = response.text().await?;
2021-02-04 21:47:22 +01:00
self.token.replace(Some(token));
2020-11-14 22:32:21 +01:00
true
}
StatusCode::UNAUTHORIZED => false,
2021-02-04 16:31:37 +01:00
status_code => Err(Error::UnexpectedResponse(status_code))?,
2020-11-14 22:32:21 +01:00
};
Ok(success)
}
2020-11-28 22:01:59 +01:00
/// Make an unauthenticated get request to the server.
async fn get(&self, url: &str) -> Result<String> {
2021-02-04 16:31:37 +01:00
let server_url = self.server_url()?;
2020-11-28 22:01:59 +01:00
let mut response = Request::get(format!("{}/{}", server_url, url))
.timeout(Duration::from_secs(10))
.body(())?
.send_async()
.await?;
2021-02-06 17:20:41 +01:00
let body = response.text().await?;
2020-11-28 22:01:59 +01:00
Ok(body)
}
/// Make an authenticated post request to the server.
async fn post(&self, url: &str, body: String) -> Result<String> {
2021-02-04 21:47:22 +01:00
let body = if self.token.borrow().is_some() {
let mut response = self.post_priv(url, body.clone()).await?;
2020-11-28 22:01:59 +01:00
2021-02-04 21:47:22 +01:00
// Try one more time (maybe the token was expired)
if response.status() == StatusCode::UNAUTHORIZED {
if self.login().await? {
response = self.post_priv(url, body).await?;
2020-11-28 22:01:59 +01:00
} else {
2021-02-04 21:47:22 +01:00
Err(Error::LoginFailed)?;
}
2020-11-28 22:01:59 +01:00
}
2021-02-04 21:47:22 +01:00
2021-02-06 17:20:41 +01:00
response.text().await?
2021-02-04 21:47:22 +01:00
} else {
let mut response = if self.login().await? {
self.post_priv(url, body).await?
} else {
Err(Error::LoginFailed)?
};
2021-02-06 17:20:41 +01:00
response.text().await?
2020-11-28 22:01:59 +01:00
};
Ok(body)
}
/// Post something to the server assuming there is a valid login token.
2021-02-06 17:20:41 +01:00
async fn post_priv(&self, url: &str, body: String) -> Result<Response<AsyncBody>> {
2021-02-04 16:31:37 +01:00
let server_url = self.server_url()?;
2021-02-04 21:47:22 +01:00
let token = self.token()?;
2020-11-28 22:01:59 +01:00
let response = Request::post(format!("{}/{}", server_url, url))
2020-11-29 11:30:43 +01:00
.timeout(Duration::from_secs(10))
2020-11-28 22:01:59 +01:00
.header("Authorization", format!("Bearer {}", token))
.header("Content-Type", "application/json")
.body(body)?
.send_async()
.await?;
Ok(response)
}
2021-02-04 21:47:22 +01:00
/// Require the server URL to be set.
fn server_url(&self) -> Result<String> {
self.get_server_url().ok_or(Error::Other("The server URL is not available!"))
}
/// Require the login data to be set.
fn login_data(&self) -> Result<LoginData> {
self.get_login_data().ok_or(Error::Other("The login data is unset!"))
}
/// Require a login token to be set.
fn token(&self) -> Result<String> {
self.token.borrow().clone().ok_or(Error::Other("No login token found!"))
}
2020-11-14 22:32:21 +01:00
}