use crate::error::Result;
use crate::v2::client::gloo::check::check_res;
use crate::v2::interface::oauth::IOauth;
use crate::v2::model::oauth::enums::scope::Scope;
use crate::v2::model::oauth::structs::o_token::OToken;
use gloo_net::http::Request;
use std::sync::{Arc, Mutex};
use wasm_bindgen::JsValue;
use web_sys::console;
#[derive(Clone, Default)]
pub struct GlooOauth {
pub o_token: Arc<Mutex<OToken>>,
pub proxy_url: Arc<Mutex<String>>,
}
impl IOauth for GlooOauth {
async fn get_token_with_code(
&self,
client_id: u64,
client_secret: &str,
code: &str,
redirect_uri: &str,
) -> Result<OToken> {
console::log_1(&JsValue::from_str("GlooOauth get_token_with_code"));
let proxy_url = {
let url = self.proxy_url.lock().unwrap();
url.clone()
};
let form_data = format!(
"client_id={client_id}&client_secret={client_secret}&code={code}&grant_type=authorization_code&redirect_uri={redirect_uri}"
);
let url = format!("{proxy_url}https://osu.ppy.sh/oauth/token");
let res = Request::post(&url)
.header("Accept", "application/json")
.header("Content-Type", "application/x-www-form-urlencoded")
.body(form_data)?
.send()
.await?;
let response = check_res(res)?;
let token: OToken = response.json().await?;
{
let mut stored_token = self.o_token.lock().unwrap();
*stored_token = token.clone();
}
Ok(token)
}
async fn get_token_without_code(&self, client_id: u64, client_secret: &str) -> Result<OToken> {
console::log_1(&JsValue::from_str("GlooOauth get_token_without_code"));
let proxy_url = {
let url = self.proxy_url.lock().unwrap();
url.clone()
};
let form_data = format!(
"client_id={client_id}&client_secret={client_secret}&grant_type=client_credentials&scope=public"
);
let url = format!("{proxy_url}https://osu.ppy.sh/oauth/token");
let res = Request::post(&url)
.header("Accept", "application/json")
.header("Content-Type", "application/x-www-form-urlencoded")
.body(form_data)?
.send()
.await?;
let response = check_res(res)?;
let token: OToken = response.json().await?;
{
let mut stored_token = self.o_token.lock().unwrap();
*stored_token = token.clone();
}
Ok(token)
}
async fn refresh_token(
&self,
client_id: u64,
client_secret: &str,
scope: Option<Vec<Scope>>,
) -> Result<OToken> {
console::log_1(&JsValue::from_str("GlooOauth refresh_token"));
let proxy_url = {
let url = self.proxy_url.lock().unwrap();
url.clone()
};
let refresh_token = {
let token = self.o_token.lock().unwrap();
token.refresh_token.clone().unwrap_or_default()
};
let scope_str = scope
.map(|scopes| {
scopes
.iter()
.map(|s| s.to_string())
.collect::<Vec<_>>()
.join(" ")
})
.unwrap_or_else(|| "public".to_string());
let form_data = format!(
"client_id={client_id}&client_secret={client_secret}&grant_type=refresh_token&refresh_token={refresh_token}&scope={scope_str}"
);
let url = format!("{proxy_url}https://osu.ppy.sh/oauth/token");
let res = Request::post(&url)
.header("Accept", "application/json")
.header("Content-Type", "application/x-www-form-urlencoded")
.body(form_data)?
.send()
.await?;
let response = check_res(res)?;
let token: OToken = response.json().await?;
{
let mut stored_token = self.o_token.lock().unwrap();
*stored_token = token.clone();
}
Ok(token)
}
async fn revoke_current_token(&self) -> Result<()> {
console::log_1(&JsValue::from_str("GlooOauth revoke_current_token"));
let access_token = {
let token = self.o_token.lock().unwrap();
token.access_token.clone()
};
let proxy_url = {
let url = self.proxy_url.lock().unwrap();
url.clone()
};
let url = format!("{proxy_url}https://osu.ppy.sh/api/v2/oauth/tokens/current");
let res = Request::delete(&url)
.header("Accept", "application/json")
.header("Authorization", &format!("Bearer {access_token}"))
.send()
.await?;
check_res(res)?;
Ok(())
}
}