veloren/server/src/auth_provider.rs

84 lines
2.9 KiB
Rust
Raw Normal View History

2020-01-01 20:17:43 +00:00
use authc::{AuthClient, AuthToken, Uuid};
2019-12-21 17:02:39 +00:00
use common::msg::ServerError;
2019-08-11 19:48:02 +00:00
use hashbrown::HashMap;
2019-12-21 17:02:39 +00:00
use std::str::FromStr;
2020-01-01 20:17:43 +00:00
fn derive_uuid(username: &str) -> Uuid {
let mut state: [u8; 16] = [
52, 17, 19, 239, 52, 17, 19, 239, 52, 17, 19, 239, 52, 17, 19, 239,
];
for mix_byte_1 in username.as_bytes() {
for i in 0..16 {
let mix_byte_step: u8 = mix_byte_1
.wrapping_pow(239)
.wrapping_mul((i as u8).wrapping_pow(43));
let mix_byte_2 = state[i + mix_byte_step as usize % 16];
let rot_step: u8 = mix_byte_1
.wrapping_pow(29)
.wrapping_mul((i as u8).wrapping_pow(163));
state[i] = (state[i] ^ mix_byte_1)
.wrapping_mul(mix_byte_2)
.rotate_left(rot_step as u32);
}
}
Uuid::from_slice(&state).unwrap()
}
pub struct AuthProvider {
accounts: HashMap<Uuid, String>,
2019-12-21 17:02:39 +00:00
auth_server: Option<AuthClient>,
}
impl AuthProvider {
2019-12-21 17:02:39 +00:00
pub fn new(auth_addr: Option<String>) -> Self {
let auth_server = match auth_addr {
Some(addr) => Some(AuthClient::new(addr)),
None => None,
};
AuthProvider {
accounts: HashMap::new(),
2019-12-21 17:02:39 +00:00
auth_server,
}
}
2019-12-21 17:02:39 +00:00
pub fn query(&mut self, username_or_token: String) -> Result<bool, ServerError> {
// Based on whether auth server is provided or not we expect an username or
// token
match &self.auth_server {
// Token from auth server expected
Some(srv) => {
log::info!("Validating '{}' token.", &username_or_token);
2020-01-01 20:17:43 +00:00
if let Ok(token) = AuthToken::from_str(&username_or_token) {
match srv.validate(token) {
Ok(uuid) => {
if self.accounts.contains_key(&uuid) {
2020-01-01 20:17:43 +00:00
return Err(ServerError::AlreadyLoggedIn);
}
let username = srv.uuid_to_username(uuid.clone())?;
self.accounts.insert(uuid, username);
2020-01-01 20:17:43 +00:00
Ok(true)
},
Err(e) => Err(ServerError::from(e)),
2019-12-21 17:02:39 +00:00
}
2020-01-01 20:17:43 +00:00
} else {
Ok(false)
2019-12-21 17:02:39 +00:00
}
},
// Username is expected
None => {
// Assume username was provided
let username = username_or_token;
let uuid = derive_uuid(&username);
if !self.accounts.contains_key(&uuid) {
log::info!("New User '{}'", username);
self.accounts.insert(uuid, username);
2019-12-21 17:02:39 +00:00
Ok(true)
} else {
Err(ServerError::AlreadyLoggedIn)
}
},
}
}
}