Can sign out
This commit is contained in:
parent
d54f9e4503
commit
7fbdf629d1
@ -1,6 +1,7 @@
|
|||||||
use crate::constants::StaticConstraints;
|
use crate::constants::StaticConstraints;
|
||||||
use crate::controllers::HttpResult;
|
use crate::controllers::HttpResult;
|
||||||
use crate::models::{User, UserID};
|
use crate::models::{User, UserID};
|
||||||
|
use crate::services::login_token_service::LoginTokenValue;
|
||||||
use crate::services::rate_limiter_service::RatedAction;
|
use crate::services::rate_limiter_service::RatedAction;
|
||||||
use crate::services::{login_token_service, openid_service, rate_limiter_service, users_service};
|
use crate::services::{login_token_service, openid_service, rate_limiter_service, users_service};
|
||||||
use actix_remote_ip::RemoteIP;
|
use actix_remote_ip::RemoteIP;
|
||||||
@ -304,3 +305,10 @@ pub async fn finish_openid_login(
|
|||||||
|
|
||||||
finish_login(&user).await
|
finish_login(&user).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Logout user
|
||||||
|
pub async fn logout(token: LoginTokenValue) -> HttpResult {
|
||||||
|
login_token_service::delete_token(&token).await?;
|
||||||
|
|
||||||
|
Ok(HttpResponse::NoContent().finish())
|
||||||
|
}
|
||||||
|
@ -51,6 +51,7 @@ async fn main() -> std::io::Result<()> {
|
|||||||
"/auth/finish_openid_login",
|
"/auth/finish_openid_login",
|
||||||
web::post().to(auth_controller::finish_openid_login),
|
web::post().to(auth_controller::finish_openid_login),
|
||||||
)
|
)
|
||||||
|
.route("/auth/logout", web::get().to(auth_controller::logout))
|
||||||
// User controller
|
// User controller
|
||||||
.route("/user/info", web::get().to(user_controller::auth_info))
|
.route("/user/info", web::get().to(user_controller::auth_info))
|
||||||
})
|
})
|
||||||
|
@ -6,12 +6,19 @@ use crate::utils::string_utils;
|
|||||||
use crate::utils::time_utils::time;
|
use crate::utils::time_utils::time;
|
||||||
use actix_web::dev::Payload;
|
use actix_web::dev::Payload;
|
||||||
use actix_web::{FromRequest, HttpRequest};
|
use actix_web::{FromRequest, HttpRequest};
|
||||||
|
use std::future::{ready, Ready};
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct LoginTokenValue(String);
|
||||||
|
|
||||||
#[derive(serde::Serialize, serde::Deserialize, Debug, Clone)]
|
#[derive(serde::Serialize, serde::Deserialize, Debug, Clone)]
|
||||||
pub struct LoginToken {
|
pub struct LoginToken {
|
||||||
|
#[serde(rename = "e")]
|
||||||
expire: u64,
|
expire: u64,
|
||||||
|
#[serde(rename = "h")]
|
||||||
hb: u64,
|
hb: u64,
|
||||||
|
#[serde(rename = "u")]
|
||||||
pub user_id: UserID,
|
pub user_id: UserID,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -64,6 +71,12 @@ pub async fn gen_new_token(user: &User) -> anyhow::Result<String> {
|
|||||||
Ok(key)
|
Ok(key)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Delete a login token
|
||||||
|
pub async fn delete_token(token: &LoginTokenValue) -> anyhow::Result<()> {
|
||||||
|
redis_connection::remove_value(&redis_key(&token.0)).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
/// Get a user information from its token
|
/// Get a user information from its token
|
||||||
async fn get_token(key: &str) -> anyhow::Result<Option<LoginToken>> {
|
async fn get_token(key: &str) -> anyhow::Result<Option<LoginToken>> {
|
||||||
let token = match redis_connection::get_value::<LoginToken>(&redis_key(key)).await? {
|
let token = match redis_connection::get_value::<LoginToken>(&redis_key(key)).await? {
|
||||||
@ -87,6 +100,20 @@ async fn get_token(key: &str) -> anyhow::Result<Option<LoginToken>> {
|
|||||||
Ok(Some(token))
|
Ok(Some(token))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl FromRequest for LoginTokenValue {
|
||||||
|
type Error = actix_web::Error;
|
||||||
|
type Future = Ready<Result<Self, Self::Error>>;
|
||||||
|
|
||||||
|
fn from_request(req: &HttpRequest, _payload: &mut Payload) -> Self::Future {
|
||||||
|
ready(match req.headers().get("X-Auth-Token") {
|
||||||
|
Some(v) => Ok(LoginTokenValue(v.to_str().unwrap_or("").to_string())),
|
||||||
|
None => Err(actix_web::error::ErrorBadRequest(
|
||||||
|
"Missing auth token header!",
|
||||||
|
)),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl FromRequest for LoginToken {
|
impl FromRequest for LoginToken {
|
||||||
type Error = actix_web::Error;
|
type Error = actix_web::Error;
|
||||||
type Future = futures_util::future::LocalBoxFuture<'static, Result<Self, Self::Error>>;
|
type Future = futures_util::future::LocalBoxFuture<'static, Result<Self, Self::Error>>;
|
||||||
@ -94,16 +121,9 @@ impl FromRequest for LoginToken {
|
|||||||
fn from_request(req: &HttpRequest, _payload: &mut Payload) -> Self::Future {
|
fn from_request(req: &HttpRequest, _payload: &mut Payload) -> Self::Future {
|
||||||
let req = req.clone();
|
let req = req.clone();
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
let token = match req.headers().get("X-Auth-Token") {
|
let token = LoginTokenValue::extract(&req).await?;
|
||||||
Some(v) => v.to_str().unwrap_or(""),
|
|
||||||
None => {
|
|
||||||
return Err(actix_web::error::ErrorBadRequest(
|
|
||||||
"Missing auth token header!",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let token = match get_token(token).await {
|
let token = match get_token(&token.0).await {
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::error!("Failed to load auth token! {}", e);
|
log::error!("Failed to load auth token! {}", e);
|
||||||
return Err(actix_web::error::ErrorInternalServerError(
|
return Err(actix_web::error::ErrorInternalServerError(
|
||||||
|
Loading…
Reference in New Issue
Block a user