2022-04-30 20:32:08 +00:00
|
|
|
use std::fs;
|
|
|
|
|
2022-04-23 19:21:04 +00:00
|
|
|
use axum::{extract::rejection::JsonRejection, Extension, Json};
|
2022-04-23 18:44:02 +00:00
|
|
|
use homedisk_database::{Database, User};
|
2022-04-19 11:05:47 +00:00
|
|
|
use homedisk_types::{
|
|
|
|
auth::login::{Request, Response},
|
2022-06-08 19:16:12 +00:00
|
|
|
config::Config,
|
2022-07-13 09:20:30 +00:00
|
|
|
errors::{AuthError, FsError, ServerError},
|
2022-04-19 13:14:17 +00:00
|
|
|
};
|
2022-04-23 19:21:04 +00:00
|
|
|
|
|
|
|
use crate::middleware::{create_token, validate_json};
|
2022-04-19 11:05:47 +00:00
|
|
|
|
|
|
|
pub async fn handle(
|
2022-05-01 18:34:28 +00:00
|
|
|
Extension(db): Extension<Database>,
|
|
|
|
Extension(config): Extension<Config>,
|
2022-04-23 19:21:04 +00:00
|
|
|
request: Result<Json<Request>, JsonRejection>,
|
2022-04-19 11:05:47 +00:00
|
|
|
) -> Result<Json<Response>, ServerError> {
|
2022-06-07 20:36:26 +00:00
|
|
|
// validate json request
|
2022-07-25 21:02:25 +00:00
|
|
|
let request = validate_json(request)?;
|
2022-04-23 19:21:04 +00:00
|
|
|
|
2022-04-30 19:56:06 +00:00
|
|
|
// username must contain at least 4 characters
|
|
|
|
if request.username.len() < 4 {
|
|
|
|
return Err(ServerError::AuthError(AuthError::UsernameTooShort));
|
|
|
|
}
|
|
|
|
|
2022-04-30 20:38:58 +00:00
|
|
|
// username must be less than 25 characters
|
|
|
|
if request.username.len() > 25 {
|
2022-04-30 19:56:06 +00:00
|
|
|
return Err(ServerError::AuthError(AuthError::UsernameTooLong));
|
|
|
|
}
|
|
|
|
|
|
|
|
// password must contain at least 8 characters
|
|
|
|
if request.password.len() < 8 {
|
|
|
|
return Err(ServerError::AuthError(AuthError::PasswordTooShort));
|
|
|
|
}
|
|
|
|
|
2022-07-25 21:02:25 +00:00
|
|
|
// create `User` type and hash password
|
2022-04-19 13:14:17 +00:00
|
|
|
let user = User::new(&request.username, &request.password);
|
|
|
|
|
2022-07-25 21:02:25 +00:00
|
|
|
// create user in the database
|
2022-04-19 13:14:17 +00:00
|
|
|
let response = match db.create_user(&user).await {
|
2022-07-25 21:02:25 +00:00
|
|
|
Ok(_result) => {
|
2022-04-30 20:32:08 +00:00
|
|
|
let token = create_token(&user, config.jwt.secret.as_bytes(), config.jwt.expires)?;
|
2022-04-19 13:14:17 +00:00
|
|
|
|
2022-08-30 11:33:15 +00:00
|
|
|
Response {
|
2022-04-23 19:21:04 +00:00
|
|
|
access_token: token,
|
2022-04-19 13:14:17 +00:00
|
|
|
}
|
2022-07-12 19:59:11 +00:00
|
|
|
},
|
2022-04-19 11:05:47 +00:00
|
|
|
|
2022-06-11 08:19:47 +00:00
|
|
|
// error while searching for a user
|
|
|
|
Err(err) => {
|
|
|
|
// user already exists
|
|
|
|
if err.to_string().contains("UNIQUE constraint failed") {
|
2022-04-19 11:05:47 +00:00
|
|
|
return Err(ServerError::AuthError(AuthError::UserAlreadyExists));
|
|
|
|
}
|
|
|
|
|
2022-06-11 08:19:47 +00:00
|
|
|
// other error
|
|
|
|
return Err(ServerError::AuthError(AuthError::Other(err.to_string())));
|
2022-07-12 19:59:11 +00:00
|
|
|
},
|
2022-04-19 11:05:47 +00:00
|
|
|
};
|
|
|
|
|
2022-07-25 21:02:25 +00:00
|
|
|
// create directory for the user files
|
|
|
|
fs::create_dir_all(&format!("{}/{}", config.storage.path, user.username,))
|
2022-07-13 09:20:30 +00:00
|
|
|
.map_err(|e| ServerError::FsError(FsError::CreateDirectory(e.to_string())))?;
|
2022-04-30 20:32:08 +00:00
|
|
|
|
2022-07-25 21:02:25 +00:00
|
|
|
// send response
|
2022-04-19 11:05:47 +00:00
|
|
|
Ok(Json(response))
|
|
|
|
}
|