mirror of
https://github.com/danbulant/Portfolio
synced 2026-06-24 17:11:49 +00:00
feat: list candidates with id, name, surname, study
This commit is contained in:
parent
b6e4af6ef7
commit
4375b9d932
9 changed files with 145 additions and 37 deletions
|
|
@ -70,6 +70,11 @@ async fn start() -> Result<(), rocket::Error> {
|
||||||
routes::admin::create_candidate,
|
routes::admin::create_candidate,
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
.mount(
|
||||||
|
"/admin/list",
|
||||||
|
routes![
|
||||||
|
routes::admin::list_candidates,
|
||||||
|
])
|
||||||
.register("/", catchers![])
|
.register("/", catchers![])
|
||||||
.launch()
|
.launch()
|
||||||
.await
|
.await
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,7 @@ use std::net::SocketAddr;
|
||||||
|
|
||||||
use portfolio_core::{
|
use portfolio_core::{
|
||||||
crypto::random_8_char_string,
|
crypto::random_8_char_string,
|
||||||
services::{admin_service::AdminService, candidate_service::CandidateService, application_service::ApplicationService},
|
services::{admin_service::AdminService, candidate_service::CandidateService, application_service::ApplicationService}, responses::CandidateResponse,
|
||||||
};
|
};
|
||||||
use requests::{AdminLoginRequest, RegisterRequest};
|
use requests::{AdminLoginRequest, RegisterRequest};
|
||||||
use rocket::http::{Cookie, Status, CookieJar};
|
use rocket::http::{Cookie, Status, CookieJar};
|
||||||
|
|
@ -86,3 +86,19 @@ pub async fn create_candidate(
|
||||||
|
|
||||||
Ok(plain_text_password)
|
Ok(plain_text_password)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[get("/candidates?<field>")]
|
||||||
|
pub async fn list_candidates(
|
||||||
|
conn: Connection<'_, Db>,
|
||||||
|
session: AdminAuth,
|
||||||
|
field: Option<String>,
|
||||||
|
) -> Result<Json<Vec<CandidateResponse>>, Custom<String>> {
|
||||||
|
let db = conn.into_inner();
|
||||||
|
let private_key = session.get_private_key();
|
||||||
|
|
||||||
|
let candidates = CandidateService::list_candidates(private_key, db, field)
|
||||||
|
.await
|
||||||
|
.map_err(|e| Custom(Status::InternalServerError, e.to_string()))?;
|
||||||
|
|
||||||
|
Ok(Json(candidates))
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -185,6 +185,33 @@ impl TryFrom<(candidate::Model, parent::Model)> for EncryptedApplicationDetails
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: use this for ApplicationResult???
|
||||||
|
/* impl TryFrom<ApplicationResult> for EncryptedApplicationDetails {
|
||||||
|
type Error = ServiceError;
|
||||||
|
|
||||||
|
fn try_from(value: ApplicationResult) -> Result<Self, Self::Error> {
|
||||||
|
Ok(EncryptedApplicationDetails {
|
||||||
|
name: EncryptedString::try_from(value.name)?,
|
||||||
|
surname: EncryptedString::try_from(value.surname)?,
|
||||||
|
birthplace: todo!(),
|
||||||
|
birthdate: todo!(),
|
||||||
|
address: todo!(),
|
||||||
|
telephone: todo!(),
|
||||||
|
citizenship: todo!(),
|
||||||
|
email: todo!(),
|
||||||
|
sex: todo!(),
|
||||||
|
study: todo!(),
|
||||||
|
parent_name: todo!(),
|
||||||
|
parent_surname: todo!(),
|
||||||
|
parent_telephone: todo!(),
|
||||||
|
parent_email: todo!(),
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
} */
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub struct ApplicationDetails {
|
pub struct ApplicationDetails {
|
||||||
// Candidate
|
// Candidate
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,23 @@
|
||||||
use crate::Query;
|
use crate::{Query};
|
||||||
|
|
||||||
use ::entity::{candidate, candidate::Entity as Candidate};
|
use ::entity::{candidate, candidate::Entity as Candidate, parent};
|
||||||
use sea_orm::*;
|
use sea_orm::*;
|
||||||
|
use serde::Serialize;
|
||||||
|
|
||||||
|
|
||||||
|
pub const PAGE_SIZE: u64 = 20;
|
||||||
|
|
||||||
|
#[derive(FromQueryResult, Serialize)]
|
||||||
|
pub struct ApplicationResult {
|
||||||
|
pub application: i32,
|
||||||
|
pub name: Option<String>,
|
||||||
|
pub surname: Option<String>,
|
||||||
|
pub study: Option<String>,
|
||||||
|
pub citizenship: Option<String>,
|
||||||
|
|
||||||
|
pub parent_name: Option<String>,
|
||||||
|
pub parent_surname: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
impl Query {
|
impl Query {
|
||||||
pub async fn find_candidate_by_id(
|
pub async fn find_candidate_by_id(
|
||||||
|
|
@ -10,6 +26,21 @@ impl Query {
|
||||||
) -> Result<Option<candidate::Model>, DbErr> {
|
) -> Result<Option<candidate::Model>, DbErr> {
|
||||||
Candidate::find_by_id(id).one(db).await
|
Candidate::find_by_id(id).one(db).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn list_candidates(
|
||||||
|
db: &DbConn,
|
||||||
|
field_of_study: Option<String>,
|
||||||
|
) -> Result<Vec<ApplicationResult>, DbErr> {
|
||||||
|
Candidate::find()
|
||||||
|
.join(JoinType::InnerJoin, candidate::Relation::Parent.def())
|
||||||
|
.column_as(parent::Column::Name, "parent_name")
|
||||||
|
.column_as(parent::Column::Surname, "parent_surname")
|
||||||
|
.into_model::<ApplicationResult>()
|
||||||
|
.paginate(db, PAGE_SIZE)
|
||||||
|
.fetch()
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|
|
||||||
|
|
@ -5,6 +5,7 @@ pub mod services;
|
||||||
pub mod error;
|
pub mod error;
|
||||||
pub mod candidate_details;
|
pub mod candidate_details;
|
||||||
pub mod util;
|
pub mod util;
|
||||||
|
pub mod responses;
|
||||||
|
|
||||||
pub use database::mutation::*;
|
pub use database::mutation::*;
|
||||||
pub use database::query::*;
|
pub use database::query::*;
|
||||||
|
|
|
||||||
47
core/src/responses.rs
Normal file
47
core/src/responses.rs
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
use serde::Serialize;
|
||||||
|
|
||||||
|
use crate::{candidate_details::EncryptedString, error::ServiceError};
|
||||||
|
|
||||||
|
#[derive(Debug, Serialize)]
|
||||||
|
pub struct CandidateResponse {
|
||||||
|
pub application_id: i32,
|
||||||
|
pub name: String,
|
||||||
|
pub surname: String,
|
||||||
|
pub study: String,
|
||||||
|
pub submitted: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CandidateResponse {
|
||||||
|
pub async fn from_encrypted(
|
||||||
|
private_key: &String,
|
||||||
|
application_id: i32,
|
||||||
|
name_opt: Option<String>,
|
||||||
|
surname_opt: Option<String>,
|
||||||
|
study_opt: Option<String>,
|
||||||
|
submitted: bool,
|
||||||
|
) -> Result<Self, ServiceError> {
|
||||||
|
let name = decrypt_if_exists(private_key, name_opt).await?;
|
||||||
|
let surname = decrypt_if_exists(private_key, surname_opt).await?;
|
||||||
|
let study = decrypt_if_exists(private_key, study_opt).await?;
|
||||||
|
Ok(
|
||||||
|
Self {
|
||||||
|
application_id,
|
||||||
|
name,
|
||||||
|
surname,
|
||||||
|
study,
|
||||||
|
submitted,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn decrypt_if_exists(
|
||||||
|
private_key: &String,
|
||||||
|
encrypted_string: Option<String>,
|
||||||
|
) -> Result<String, ServiceError> {
|
||||||
|
match EncryptedString::try_from(encrypted_string) {
|
||||||
|
Ok(encrypted_string) => Ok(encrypted_string.decrypt(private_key).await?),
|
||||||
|
Err(_) => Ok(String::from("")),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -423,6 +423,19 @@ mod tests {
|
||||||
assert!(!CandidateService::is_application_id_valid(101));
|
assert!(!CandidateService::is_application_id_valid(101));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO
|
||||||
|
/* #[tokio::test]
|
||||||
|
async fn test_list_candidates() {
|
||||||
|
let db = get_memory_sqlite_connection().await;
|
||||||
|
let candidates = CandidateService::list_candidates(&db, None).await.unwrap();
|
||||||
|
assert_eq!(candidates.len(), 0);
|
||||||
|
|
||||||
|
put_user_data(&db).await;
|
||||||
|
|
||||||
|
let candidates = CandidateService::list_candidates(&db, None).await.unwrap();
|
||||||
|
assert_eq!(candidates.len(), 1);
|
||||||
|
} */
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
async fn test_encrypt_decrypt_private_key_with_passphrase() {
|
async fn test_encrypt_decrypt_private_key_with_passphrase() {
|
||||||
let db = get_memory_sqlite_connection().await;
|
let db = get_memory_sqlite_connection().await;
|
||||||
|
|
@ -479,7 +492,7 @@ mod tests {
|
||||||
citizenship: "test".to_string(),
|
citizenship: "test".to_string(),
|
||||||
email: "test".to_string(),
|
email: "test".to_string(),
|
||||||
sex: "test".to_string(),
|
sex: "test".to_string(),
|
||||||
study: "test".to_string(),
|
study: "KB".to_string(),
|
||||||
parent_name: "test".to_string(),
|
parent_name: "test".to_string(),
|
||||||
parent_surname: "test".to_string(),
|
parent_surname: "test".to_string(),
|
||||||
parent_telephone: "test".to_string(),
|
parent_telephone: "test".to_string(),
|
||||||
|
|
|
||||||
|
|
@ -3,4 +3,3 @@ pub mod candidate_service;
|
||||||
pub mod admin_service;
|
pub mod admin_service;
|
||||||
pub mod parent_service;
|
pub mod parent_service;
|
||||||
pub mod application_service;
|
pub mod application_service;
|
||||||
pub mod portfolio_service;
|
|
||||||
|
|
@ -1,31 +0,0 @@
|
||||||
use std::path::Path;
|
|
||||||
|
|
||||||
use tokio::io::AsyncWriteExt;
|
|
||||||
|
|
||||||
use crate::error::ServiceError;
|
|
||||||
|
|
||||||
pub struct PortfolioService;
|
|
||||||
|
|
||||||
impl PortfolioService {
|
|
||||||
pub async fn write_portfolio_file(
|
|
||||||
candidate_id: i32,
|
|
||||||
data: Vec<u8>,
|
|
||||||
filename: &str,
|
|
||||||
) -> Result<(), ServiceError> {
|
|
||||||
let cache_path = Path::new(&candidate_id.to_string()).join("cache");
|
|
||||||
|
|
||||||
let mut file = tokio::fs::File::create(cache_path.join(filename)).await?;
|
|
||||||
|
|
||||||
file.write_all(&data).await?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn is_portfolio_complete(candidate_id: i32) -> bool {
|
|
||||||
let cache_path = Path::new(&candidate_id.to_string()).join("cache");
|
|
||||||
|
|
||||||
tokio::fs::metadata(cache_path.join("MOTIVACNI_DOPIS.pdf")).await.is_ok()
|
|
||||||
&& tokio::fs::metadata(cache_path.join("PORTFOLIO.pdf")).await.is_ok()
|
|
||||||
&& tokio::fs::metadata(cache_path.join("PORTFOLIO.zip")).await.is_ok()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Loading…
Reference in a new issue