mirror of
https://github.com/danbulant/Portfolio
synced 2026-06-08 17:20:52 +00:00
Merge pull request #171 from EETagent/school_order_selection_starvy
(backend) data structure stabilization
This commit is contained in:
commit
13c4210f06
20 changed files with 309 additions and 101 deletions
79
Cargo.lock
generated
79
Cargo.lock
generated
|
|
@ -122,6 +122,15 @@ dependencies = [
|
||||||
"version_check",
|
"version_check",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "aho-corasick"
|
||||||
|
version = "0.7.20"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "cc936419f96fa211c1b9166887b38e5e40b19958e5b895be7c1f93adec7071ac"
|
||||||
|
dependencies = [
|
||||||
|
"memchr",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "aliasable"
|
name = "aliasable"
|
||||||
version = "0.1.3"
|
version = "0.1.3"
|
||||||
|
|
@ -1489,6 +1498,17 @@ dependencies = [
|
||||||
"cxx-build",
|
"cxx-build",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "idna"
|
||||||
|
version = "0.2.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "418a0a6fab821475f634efe3ccc45c013f742efe03d853e8d3355d5cb850ecf8"
|
||||||
|
dependencies = [
|
||||||
|
"matches",
|
||||||
|
"unicode-bidi",
|
||||||
|
"unicode-normalization",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "idna"
|
name = "idna"
|
||||||
version = "0.3.0"
|
version = "0.3.0"
|
||||||
|
|
@ -1499,6 +1519,12 @@ dependencies = [
|
||||||
"unicode-normalization",
|
"unicode-normalization",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "if_chain"
|
||||||
|
version = "1.0.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "cb56e1aa765b4b4f3aadfab769793b7087bb03a4ea4920644a6d238e2df5b9ed"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "indexmap"
|
name = "indexmap"
|
||||||
version = "1.9.2"
|
version = "1.9.2"
|
||||||
|
|
@ -1701,6 +1727,12 @@ dependencies = [
|
||||||
"regex-automata",
|
"regex-automata",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "matches"
|
||||||
|
version = "0.1.9"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "a3e378b66a060d48947b590737b30a1be76706c8dd7b8ba0f2fe3989c68a853f"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "md-5"
|
name = "md-5"
|
||||||
version = "0.10.5"
|
version = "0.10.5"
|
||||||
|
|
@ -2170,6 +2202,7 @@ dependencies = [
|
||||||
"serial_test",
|
"serial_test",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
"validator",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|
@ -2399,6 +2432,8 @@ version = "1.7.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "48aaa5748ba571fb95cd2c85c09f629215d3a6ece942baa100950af03a34f733"
|
checksum = "48aaa5748ba571fb95cd2c85c09f629215d3a6ece942baa100950af03a34f733"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"aho-corasick",
|
||||||
|
"memchr",
|
||||||
"regex-syntax",
|
"regex-syntax",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
@ -3595,7 +3630,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "0d68c799ae75762b8c3fe375feb6600ef5602c883c5d21eb51c09f22b83c4643"
|
checksum = "0d68c799ae75762b8c3fe375feb6600ef5602c883c5d21eb51c09f22b83c4643"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"form_urlencoded",
|
"form_urlencoded",
|
||||||
"idna",
|
"idna 0.3.0",
|
||||||
"percent-encoding",
|
"percent-encoding",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
@ -3609,6 +3644,48 @@ dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "validator"
|
||||||
|
version = "0.15.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f07b0a1390e01c0fc35ebb26b28ced33c9a3808f7f9fbe94d3cc01e233bfeed5"
|
||||||
|
dependencies = [
|
||||||
|
"idna 0.2.3",
|
||||||
|
"lazy_static",
|
||||||
|
"regex",
|
||||||
|
"serde",
|
||||||
|
"serde_derive",
|
||||||
|
"serde_json",
|
||||||
|
"url",
|
||||||
|
"validator_derive",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "validator_derive"
|
||||||
|
version = "0.15.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "ea7ed5e8cf2b6bdd64a6c4ce851da25388a89327b17b88424ceced6bd5017923"
|
||||||
|
dependencies = [
|
||||||
|
"if_chain",
|
||||||
|
"lazy_static",
|
||||||
|
"proc-macro-error",
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"regex",
|
||||||
|
"syn",
|
||||||
|
"validator_types",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "validator_types"
|
||||||
|
version = "0.15.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d2ddf34293296847abfc1493b15c6e2f5d3cd19f57ad7d22673bf4c6278da329"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"syn",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "valuable"
|
name = "valuable"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
|
|
|
||||||
|
|
@ -256,7 +256,6 @@ pub mod tests {
|
||||||
))
|
))
|
||||||
.dispatch();
|
.dispatch();
|
||||||
|
|
||||||
println!("{:?}", response);
|
|
||||||
(
|
(
|
||||||
response.cookies().get("id").unwrap().to_owned(),
|
response.cookies().get("id").unwrap().to_owned(),
|
||||||
response.cookies().get("key").unwrap().to_owned(),
|
response.cookies().get("key").unwrap().to_owned(),
|
||||||
|
|
|
||||||
|
|
@ -102,6 +102,7 @@ pub async fn post_details(
|
||||||
) -> Result<Json<ApplicationDetails>, Custom<String>> {
|
) -> Result<Json<ApplicationDetails>, Custom<String>> {
|
||||||
let db = conn.into_inner();
|
let db = conn.into_inner();
|
||||||
let form = details.into_inner();
|
let form = details.into_inner();
|
||||||
|
form.candidate.validate_self().map_err(to_custom_error)?;
|
||||||
let application: application::Model = session.into();
|
let application: application::Model = session.into();
|
||||||
let candidate = ApplicationService::find_related_candidate(&db, &application).await.map_err(to_custom_error)?; // TODO
|
let candidate = ApplicationService::find_related_candidate(&db, &application).await.map_err(to_custom_error)?; // TODO
|
||||||
|
|
||||||
|
|
@ -308,9 +309,11 @@ mod tests {
|
||||||
\"candidate\": {
|
\"candidate\": {
|
||||||
\"name\": \"idk\",
|
\"name\": \"idk\",
|
||||||
\"surname\": \"idk\",
|
\"surname\": \"idk\",
|
||||||
|
\"birthSurname\": \"surname\",
|
||||||
\"birthplace\": \"Praha 1\",
|
\"birthplace\": \"Praha 1\",
|
||||||
\"birthdate\": \"2015-09-18\",
|
\"birthdate\": \"2015-09-18\",
|
||||||
\"address\": \"Stefanikova jidelna\",
|
\"address\": \"Stefanikova jidelna\",
|
||||||
|
\"letterAddress\": \"Stefanikova jidelna\",
|
||||||
\"telephone\": \"000111222333\",
|
\"telephone\": \"000111222333\",
|
||||||
\"citizenship\": \"Czech Republic\",
|
\"citizenship\": \"Czech Republic\",
|
||||||
\"email\": \"magor@magor.cz\",
|
\"email\": \"magor@magor.cz\",
|
||||||
|
|
@ -319,6 +322,8 @@ mod tests {
|
||||||
\"schoolName\": \"29988383\",
|
\"schoolName\": \"29988383\",
|
||||||
\"healthInsurance\": \"000\",
|
\"healthInsurance\": \"000\",
|
||||||
\"grades\": [],
|
\"grades\": [],
|
||||||
|
\"firstSchool\": {\"name\": \"SSPŠ\", \"field\": \"KB\"},
|
||||||
|
\"secondSchool\": {\"name\": \"SSPŠ\", \"field\": \"IT\"},
|
||||||
\"testLanguage\": \"CZ\"
|
\"testLanguage\": \"CZ\"
|
||||||
},
|
},
|
||||||
\"parents\": [
|
\"parents\": [
|
||||||
|
|
|
||||||
|
|
@ -13,6 +13,8 @@ portfolio-entity = { path = "../entity" }
|
||||||
serde = { version = "^1.0", features = ["derive"] }
|
serde = { version = "^1.0", features = ["derive"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
|
|
||||||
|
validator = { version = "^0.15", features = ["derive"] }
|
||||||
|
|
||||||
# csv
|
# csv
|
||||||
csv = "^1.1"
|
csv = "^1.1"
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -34,10 +34,10 @@ pub fn random_12_char_string() -> String {
|
||||||
|
|
||||||
/// Exclude O and 0, lowercase letters
|
/// Exclude O and 0, lowercase letters
|
||||||
fn is_usable_char(c: &char) -> bool {
|
fn is_usable_char(c: &char) -> bool {
|
||||||
('1'..='9').contains(&c) ||
|
('1'..='9').contains(c) ||
|
||||||
('A'..='N').contains(&c) ||
|
('A'..='N').contains(c) ||
|
||||||
('P'..'Z').contains(&c) ||
|
('P'..'Z').contains(c) ||
|
||||||
['@', '#', '$', '%'].contains(&c)
|
['@', '#', '$', '%'].contains(c)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn hash_password(password_plain_text: String) -> Result<String, ServiceError> {
|
pub async fn hash_password(password_plain_text: String) -> Result<String, ServiceError> {
|
||||||
|
|
|
||||||
|
|
@ -44,9 +44,11 @@ impl Mutation {
|
||||||
|
|
||||||
candidate.name = Set(enc_candidate.name.map(|e| e.into()));
|
candidate.name = Set(enc_candidate.name.map(|e| e.into()));
|
||||||
candidate.surname = Set(enc_candidate.surname.map(|e| e.into()));
|
candidate.surname = Set(enc_candidate.surname.map(|e| e.into()));
|
||||||
|
candidate.birth_surname = Set(enc_candidate.birth_surname.map(|e| e.into()));
|
||||||
candidate.birthplace = Set(enc_candidate.birthplace.map(|e| e.into()));
|
candidate.birthplace = Set(enc_candidate.birthplace.map(|e| e.into()));
|
||||||
candidate.birthdate = Set(enc_candidate.birthdate.map(|e| e.into()));
|
candidate.birthdate = Set(enc_candidate.birthdate.map(|e| e.into()));
|
||||||
candidate.address = Set(enc_candidate.address.map(|e| e.into()));
|
candidate.address = Set(enc_candidate.address.map(|e| e.into()));
|
||||||
|
candidate.letter_address = Set(enc_candidate.letter_address.map(|e| e.into()));
|
||||||
candidate.telephone = Set(enc_candidate.telephone.map(|e| e.into()));
|
candidate.telephone = Set(enc_candidate.telephone.map(|e| e.into()));
|
||||||
candidate.citizenship = Set(enc_candidate.citizenship.map(|e| e.into()));
|
candidate.citizenship = Set(enc_candidate.citizenship.map(|e| e.into()));
|
||||||
candidate.email = Set(enc_candidate.email.map(|e| e.into()));
|
candidate.email = Set(enc_candidate.email.map(|e| e.into()));
|
||||||
|
|
@ -54,6 +56,8 @@ impl Mutation {
|
||||||
candidate.school_name = Set(enc_candidate.school_name.map(|e| e.into()));
|
candidate.school_name = Set(enc_candidate.school_name.map(|e| e.into()));
|
||||||
candidate.health_insurance = Set(enc_candidate.health_insurance.map(|e| e.into()));
|
candidate.health_insurance = Set(enc_candidate.health_insurance.map(|e| e.into()));
|
||||||
candidate.grades_json = Set(enc_candidate.grades_json.map(|e| e.into()));
|
candidate.grades_json = Set(enc_candidate.grades_json.map(|e| e.into()));
|
||||||
|
candidate.first_school = Set(enc_candidate.first_school.map(|e| e.into()));
|
||||||
|
candidate.second_school = Set(enc_candidate.second_school.map(|e| e.into()));
|
||||||
candidate.test_language = Set(enc_candidate.test_language.map(|s| s));
|
candidate.test_language = Set(enc_candidate.test_language.map(|s| s));
|
||||||
candidate.encrypted_by_id = Set(Some(encrypted_by_id));
|
candidate.encrypted_by_id = Set(Some(encrypted_by_id));
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -14,6 +14,10 @@ pub enum ServiceError {
|
||||||
Forbidden,
|
Forbidden,
|
||||||
#[error("Session expired, please login again")]
|
#[error("Session expired, please login again")]
|
||||||
ExpiredSession,
|
ExpiredSession,
|
||||||
|
#[error("Missing details")]
|
||||||
|
MissingDetails,
|
||||||
|
#[error("Validation error: {0}")]
|
||||||
|
ValidationError(#[from] validator::ValidationErrors),
|
||||||
#[error("User already exists")]
|
#[error("User already exists")]
|
||||||
UserAlreadyExists,
|
UserAlreadyExists,
|
||||||
#[error("Candidate not found")]
|
#[error("Candidate not found")]
|
||||||
|
|
@ -82,6 +86,8 @@ impl ServiceError {
|
||||||
// 4XX
|
// 4XX
|
||||||
ServiceError::InvalidApplicationId => 400,
|
ServiceError::InvalidApplicationId => 400,
|
||||||
ServiceError::ParentOverflow => 400,
|
ServiceError::ParentOverflow => 400,
|
||||||
|
ServiceError::MissingDetails => 400,
|
||||||
|
ServiceError::ValidationError(_) => 400,
|
||||||
ServiceError::Unauthorized => 401,
|
ServiceError::Unauthorized => 401,
|
||||||
ServiceError::InvalidCredentials => 401,
|
ServiceError::InvalidCredentials => 401,
|
||||||
ServiceError::ExpiredSession => 401,
|
ServiceError::ExpiredSession => 401,
|
||||||
|
|
|
||||||
|
|
@ -47,9 +47,11 @@ pub struct ApplicationRow {
|
||||||
pub application: i32,
|
pub application: i32,
|
||||||
pub name: Option<String>,
|
pub name: Option<String>,
|
||||||
pub surname: Option<String>,
|
pub surname: Option<String>,
|
||||||
|
pub birth_surname: Option<String>,
|
||||||
pub birthplace: Option<String>,
|
pub birthplace: Option<String>,
|
||||||
pub birthdate: Option<String>,
|
pub birthdate: Option<String>,
|
||||||
pub address: Option<String>,
|
pub address: Option<String>,
|
||||||
|
pub letter_address: Option<String>,
|
||||||
pub telephone: Option<String>,
|
pub telephone: Option<String>,
|
||||||
pub citizenship: Option<String>,
|
pub citizenship: Option<String>,
|
||||||
pub email: Option<String>,
|
pub email: Option<String>,
|
||||||
|
|
@ -58,6 +60,15 @@ pub struct ApplicationRow {
|
||||||
pub school_name: Option<String>,
|
pub school_name: Option<String>,
|
||||||
pub health_insurance: Option<String>,
|
pub health_insurance: Option<String>,
|
||||||
|
|
||||||
|
pub diploma_1_8: String,
|
||||||
|
pub diploma_2_8: String,
|
||||||
|
pub diploma_1_9: String,
|
||||||
|
|
||||||
|
pub first_school_name: Option<String>,
|
||||||
|
pub first_school_field: Option<String>,
|
||||||
|
pub second_school_name: Option<String>,
|
||||||
|
pub second_school_field: Option<String>,
|
||||||
|
|
||||||
pub parent_name: Option<String>,
|
pub parent_name: Option<String>,
|
||||||
pub parent_surname: Option<String>,
|
pub parent_surname: Option<String>,
|
||||||
pub parent_telephone: Option<String>,
|
pub parent_telephone: Option<String>,
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,13 @@
|
||||||
use chrono::NaiveDate;
|
use chrono::NaiveDate;
|
||||||
use entity::{application, candidate};
|
use entity::{application, candidate};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
use validator::Validate;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
error::ServiceError,
|
error::ServiceError,
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{candidate_details::{EncryptedString, EncryptedCandidateDetails}, grade::GradeList};
|
use super::{candidate_details::{EncryptedString, EncryptedCandidateDetails}, grade::GradeList, school::School};
|
||||||
|
|
||||||
pub enum FieldOfStudy {
|
pub enum FieldOfStudy {
|
||||||
G,
|
G,
|
||||||
|
|
@ -57,24 +58,47 @@ pub struct CreateCandidateResponse {
|
||||||
pub password: String,
|
pub password: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq)]
|
#[derive(Debug, Serialize, Deserialize, Validate, Clone, PartialEq, Eq)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub struct CandidateDetails {
|
pub struct CandidateDetails {
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub name: String,
|
pub name: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub surname: String,
|
pub surname: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
|
pub birth_surname: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub birthplace: String,
|
pub birthplace: String,
|
||||||
pub birthdate: NaiveDate,
|
pub birthdate: NaiveDate,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub address: String,
|
pub address: String,
|
||||||
|
pub letter_address: String,
|
||||||
|
#[validate(length(min = 1, max = 31))]
|
||||||
pub telephone: String,
|
pub telephone: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub citizenship: String,
|
pub citizenship: String,
|
||||||
|
#[validate(email)]
|
||||||
pub email: String,
|
pub email: String,
|
||||||
pub sex: String,
|
pub sex: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub personal_id_number: String,
|
pub personal_id_number: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub school_name: String,
|
pub school_name: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
pub health_insurance: String,
|
pub health_insurance: String,
|
||||||
pub grades: GradeList,
|
pub grades: GradeList,
|
||||||
|
pub first_school: School,
|
||||||
|
pub second_school: School,
|
||||||
pub test_language: String,
|
pub test_language: String,
|
||||||
}
|
}
|
||||||
|
impl CandidateDetails {
|
||||||
|
pub fn validate_self(&self) -> Result<(), ServiceError> {
|
||||||
|
self.first_school.validate()?;
|
||||||
|
self.second_school.validate()?;
|
||||||
|
self.validate()
|
||||||
|
.map_err(ServiceError::ValidationError)
|
||||||
|
}
|
||||||
|
}
|
||||||
#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq)]
|
#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub struct ParentDetails {
|
pub struct ParentDetails {
|
||||||
|
|
|
||||||
|
|
@ -5,7 +5,7 @@ use futures::future;
|
||||||
|
|
||||||
use crate::{crypto, models::candidate::{ApplicationDetails}, error::ServiceError, utils::date::parse_naive_date_from_opt_str};
|
use crate::{crypto, models::candidate::{ApplicationDetails}, error::ServiceError, utils::date::parse_naive_date_from_opt_str};
|
||||||
|
|
||||||
use super::{candidate::{CandidateDetails, ParentDetails}, application::ApplicationRow, grade::GradeList};
|
use super::{candidate::{CandidateDetails, ParentDetails}, grade::GradeList, school::School};
|
||||||
|
|
||||||
pub const NAIVE_DATE_FMT: &str = "%Y-%m-%d";
|
pub const NAIVE_DATE_FMT: &str = "%Y-%m-%d";
|
||||||
|
|
||||||
|
|
@ -16,9 +16,11 @@ pub struct EncryptedString(String);
|
||||||
pub struct EncryptedCandidateDetails {
|
pub struct EncryptedCandidateDetails {
|
||||||
pub name: Option<EncryptedString>,
|
pub name: Option<EncryptedString>,
|
||||||
pub surname: Option<EncryptedString>,
|
pub surname: Option<EncryptedString>,
|
||||||
|
pub birth_surname: Option<EncryptedString>,
|
||||||
pub birthplace: Option<EncryptedString>,
|
pub birthplace: Option<EncryptedString>,
|
||||||
pub birthdate: Option<EncryptedString>,
|
pub birthdate: Option<EncryptedString>,
|
||||||
pub address: Option<EncryptedString>,
|
pub address: Option<EncryptedString>,
|
||||||
|
pub letter_address: Option<EncryptedString>,
|
||||||
pub telephone: Option<EncryptedString>,
|
pub telephone: Option<EncryptedString>,
|
||||||
pub citizenship: Option<EncryptedString>,
|
pub citizenship: Option<EncryptedString>,
|
||||||
pub email: Option<EncryptedString>,
|
pub email: Option<EncryptedString>,
|
||||||
|
|
@ -27,6 +29,8 @@ pub struct EncryptedCandidateDetails {
|
||||||
pub school_name: Option<EncryptedString>,
|
pub school_name: Option<EncryptedString>,
|
||||||
pub health_insurance: Option<EncryptedString>,
|
pub health_insurance: Option<EncryptedString>,
|
||||||
pub grades_json: Option<EncryptedString>,
|
pub grades_json: Option<EncryptedString>,
|
||||||
|
pub first_school: Option<EncryptedString>,
|
||||||
|
pub second_school: Option<EncryptedString>,
|
||||||
pub test_language: Option<String>,
|
pub test_language: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -121,12 +125,16 @@ impl EncryptedCandidateDetails {
|
||||||
) -> Result<EncryptedCandidateDetails, ServiceError> {
|
) -> Result<EncryptedCandidateDetails, ServiceError> {
|
||||||
let birthdate_str = form.birthdate.format(NAIVE_DATE_FMT).to_string();
|
let birthdate_str = form.birthdate.format(NAIVE_DATE_FMT).to_string();
|
||||||
let grades_str = form.grades.to_string();
|
let grades_str = form.grades.to_string();
|
||||||
|
let (first_school_str, second_school_str) =
|
||||||
|
(form.first_school.to_string(), form.second_school.to_string());
|
||||||
let d = tokio::try_join!(
|
let d = tokio::try_join!(
|
||||||
EncryptedString::new_option(&form.name, recipients),
|
EncryptedString::new_option(&form.name, recipients),
|
||||||
EncryptedString::new_option(&form.surname, recipients),
|
EncryptedString::new_option(&form.surname, recipients),
|
||||||
|
EncryptedString::new_option(&form.birth_surname, recipients),
|
||||||
EncryptedString::new_option(&form.birthplace, recipients),
|
EncryptedString::new_option(&form.birthplace, recipients),
|
||||||
EncryptedString::new_option(&birthdate_str, recipients),
|
EncryptedString::new_option(&birthdate_str, recipients),
|
||||||
EncryptedString::new_option(&form.address, recipients),
|
EncryptedString::new_option(&form.address, recipients),
|
||||||
|
EncryptedString::new_option(&form.letter_address, recipients),
|
||||||
EncryptedString::new_option(&form.telephone, recipients),
|
EncryptedString::new_option(&form.telephone, recipients),
|
||||||
EncryptedString::new_option(&form.citizenship, recipients),
|
EncryptedString::new_option(&form.citizenship, recipients),
|
||||||
EncryptedString::new_option(&form.email, recipients),
|
EncryptedString::new_option(&form.email, recipients),
|
||||||
|
|
@ -135,23 +143,29 @@ impl EncryptedCandidateDetails {
|
||||||
EncryptedString::new_option(&form.school_name, recipients),
|
EncryptedString::new_option(&form.school_name, recipients),
|
||||||
EncryptedString::new_option(&form.health_insurance, recipients),
|
EncryptedString::new_option(&form.health_insurance, recipients),
|
||||||
EncryptedString::new_option(&grades_str, recipients),
|
EncryptedString::new_option(&grades_str, recipients),
|
||||||
|
EncryptedString::new_option(&first_school_str, recipients),
|
||||||
|
EncryptedString::new_option(&second_school_str, recipients),
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
Ok(
|
Ok(
|
||||||
EncryptedCandidateDetails {
|
EncryptedCandidateDetails {
|
||||||
name: d.0,
|
name: d.0,
|
||||||
surname: d.1,
|
surname: d.1,
|
||||||
birthplace: d.2,
|
birth_surname: d.2,
|
||||||
birthdate: d.3,
|
birthplace: d.3,
|
||||||
address: d.4,
|
birthdate: d.4,
|
||||||
telephone: d.5,
|
address: d.5,
|
||||||
citizenship: d.6,
|
letter_address: d.6,
|
||||||
email: d.7,
|
telephone: d.7,
|
||||||
sex: d.8,
|
citizenship: d.8,
|
||||||
personal_id_number: d.9,
|
email: d.9,
|
||||||
school_name: d.10,
|
sex: d.10,
|
||||||
health_insurance: d.11,
|
personal_id_number: d.11,
|
||||||
grades_json: d.12,
|
school_name: d.12,
|
||||||
|
health_insurance: d.13,
|
||||||
|
grades_json: d.14,
|
||||||
|
first_school: d.15,
|
||||||
|
second_school: d.16,
|
||||||
test_language: Some(form.test_language.to_owned()),
|
test_language: Some(form.test_language.to_owned()),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
@ -161,33 +175,41 @@ impl EncryptedCandidateDetails {
|
||||||
let d = tokio::try_join!(
|
let d = tokio::try_join!(
|
||||||
EncryptedString::decrypt_option(&self.name, priv_key), // 0
|
EncryptedString::decrypt_option(&self.name, priv_key), // 0
|
||||||
EncryptedString::decrypt_option(&self.surname, priv_key), // 1
|
EncryptedString::decrypt_option(&self.surname, priv_key), // 1
|
||||||
EncryptedString::decrypt_option(&self.birthplace, priv_key), // 2
|
EncryptedString::decrypt_option(&self.birth_surname, priv_key), // 2
|
||||||
EncryptedString::decrypt_option(&self.birthdate, priv_key), // 3
|
EncryptedString::decrypt_option(&self.birthplace, priv_key), // 3
|
||||||
EncryptedString::decrypt_option(&self.address, priv_key), // 4
|
EncryptedString::decrypt_option(&self.birthdate, priv_key), // 4
|
||||||
EncryptedString::decrypt_option(&self.telephone, priv_key), // 5
|
EncryptedString::decrypt_option(&self.address, priv_key), // 5
|
||||||
EncryptedString::decrypt_option(&self.citizenship, priv_key), // 6
|
EncryptedString::decrypt_option(&self.letter_address, priv_key), // 6
|
||||||
EncryptedString::decrypt_option(&self.email, priv_key), // 7
|
EncryptedString::decrypt_option(&self.telephone, priv_key), // 7
|
||||||
EncryptedString::decrypt_option(&self.sex, priv_key), // 8
|
EncryptedString::decrypt_option(&self.citizenship, priv_key), // 8
|
||||||
EncryptedString::decrypt_option(&self.personal_id_number, priv_key),// 9
|
EncryptedString::decrypt_option(&self.email, priv_key), // 9
|
||||||
EncryptedString::decrypt_option(&self.school_name, priv_key), // 10
|
EncryptedString::decrypt_option(&self.sex, priv_key), // 10
|
||||||
EncryptedString::decrypt_option(&self.health_insurance, priv_key), // 11
|
EncryptedString::decrypt_option(&self.personal_id_number, priv_key),// 11
|
||||||
EncryptedString::decrypt_option(&self.grades_json, priv_key), // 12
|
EncryptedString::decrypt_option(&self.school_name, priv_key), // 12
|
||||||
|
EncryptedString::decrypt_option(&self.health_insurance, priv_key), // 13
|
||||||
|
EncryptedString::decrypt_option(&self.grades_json, priv_key), // 14
|
||||||
|
EncryptedString::decrypt_option(&self.first_school, priv_key), // 15
|
||||||
|
EncryptedString::decrypt_option(&self.second_school, priv_key), // 16
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
Ok(CandidateDetails {
|
Ok(CandidateDetails {
|
||||||
name: d.0.unwrap_or_default(),
|
name: d.0.unwrap_or_default(),
|
||||||
surname: d.1.unwrap_or_default(),
|
surname: d.1.unwrap_or_default(),
|
||||||
birthplace: d.2.unwrap_or_default(),
|
birth_surname: d.2.unwrap_or_default(),
|
||||||
birthdate: parse_naive_date_from_opt_str(d.3, NAIVE_DATE_FMT)?,
|
birthplace: d.3.unwrap_or_default(),
|
||||||
address: d.4.unwrap_or_default(),
|
birthdate: parse_naive_date_from_opt_str(d.4, NAIVE_DATE_FMT)?,
|
||||||
telephone: d.5.unwrap_or_default(),
|
address: d.5.unwrap_or_default(),
|
||||||
citizenship: d.6.unwrap_or_default(),
|
letter_address: d.6.unwrap_or_default(),
|
||||||
email: d.7.unwrap_or_default(),
|
telephone: d.7.unwrap_or_default(),
|
||||||
sex: d.8.unwrap_or_default(),
|
citizenship: d.8.unwrap_or_default(),
|
||||||
personal_id_number: d.9.unwrap_or_default(),
|
email: d.9.unwrap_or_default(),
|
||||||
school_name: d.10.unwrap_or_default(),
|
sex: d.10.unwrap_or_default(),
|
||||||
health_insurance: d.11.unwrap_or_default(),
|
personal_id_number: d.11.unwrap_or_default(),
|
||||||
grades: GradeList::from_opt_str(d.12).unwrap_or_default(),
|
school_name: d.12.unwrap_or_default(),
|
||||||
|
health_insurance: d.13.unwrap_or_default(),
|
||||||
|
grades: GradeList::from_opt_str(d.14).unwrap_or_default(),
|
||||||
|
first_school: School::from_opt_str(d.15).unwrap_or_default(),
|
||||||
|
second_school: School::from_opt_str(d.16).unwrap_or_default(),
|
||||||
test_language: self.test_language.to_owned().unwrap_or_default().to_string(),
|
test_language: self.test_language.to_owned().unwrap_or_default().to_string(),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
@ -202,8 +224,12 @@ impl EncryptedCandidateDetails {
|
||||||
self.telephone.is_some() &&
|
self.telephone.is_some() &&
|
||||||
self.citizenship.is_some() &&
|
self.citizenship.is_some() &&
|
||||||
self.email.is_some() &&
|
self.email.is_some() &&
|
||||||
// self.sex.is_some() &&
|
self.personal_id_number.is_some() &&
|
||||||
self.personal_id_number.is_some()
|
self.school_name.is_some() &&
|
||||||
|
self.health_insurance.is_some() &&
|
||||||
|
self.first_school.is_some() &&
|
||||||
|
self.second_school.is_some()
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
impl From<&candidate::Model> for EncryptedCandidateDetails {
|
impl From<&candidate::Model> for EncryptedCandidateDetails {
|
||||||
|
|
@ -213,9 +239,11 @@ impl From<&candidate::Model> for EncryptedCandidateDetails {
|
||||||
EncryptedCandidateDetails {
|
EncryptedCandidateDetails {
|
||||||
name: EncryptedString::try_from(&candidate.name).ok(),
|
name: EncryptedString::try_from(&candidate.name).ok(),
|
||||||
surname: EncryptedString::try_from(&candidate.surname).ok(),
|
surname: EncryptedString::try_from(&candidate.surname).ok(),
|
||||||
|
birth_surname: EncryptedString::try_from(&candidate.birth_surname).ok(),
|
||||||
birthplace: EncryptedString::try_from(&candidate.birthplace).ok(),
|
birthplace: EncryptedString::try_from(&candidate.birthplace).ok(),
|
||||||
birthdate: EncryptedString::try_from(&candidate.birthdate).ok(),
|
birthdate: EncryptedString::try_from(&candidate.birthdate).ok(),
|
||||||
address: EncryptedString::try_from(&candidate.address).ok(),
|
address: EncryptedString::try_from(&candidate.address).ok(),
|
||||||
|
letter_address: EncryptedString::try_from(&candidate.letter_address).ok(),
|
||||||
telephone: EncryptedString::try_from(&candidate.telephone).ok(),
|
telephone: EncryptedString::try_from(&candidate.telephone).ok(),
|
||||||
citizenship: EncryptedString::try_from(&candidate.citizenship).ok(),
|
citizenship: EncryptedString::try_from(&candidate.citizenship).ok(),
|
||||||
email: EncryptedString::try_from(&candidate.email).ok(),
|
email: EncryptedString::try_from(&candidate.email).ok(),
|
||||||
|
|
@ -224,6 +252,8 @@ impl From<&candidate::Model> for EncryptedCandidateDetails {
|
||||||
school_name: EncryptedString::try_from(&candidate.school_name).ok(),
|
school_name: EncryptedString::try_from(&candidate.school_name).ok(),
|
||||||
health_insurance: EncryptedString::try_from(&candidate.health_insurance).ok(),
|
health_insurance: EncryptedString::try_from(&candidate.health_insurance).ok(),
|
||||||
grades_json: EncryptedString::try_from(&candidate.grades_json).ok(),
|
grades_json: EncryptedString::try_from(&candidate.grades_json).ok(),
|
||||||
|
first_school: EncryptedString::try_from(&candidate.first_school).ok(),
|
||||||
|
second_school: EncryptedString::try_from(&candidate.second_school).ok(),
|
||||||
test_language: candidate.test_language.to_owned(),
|
test_language: candidate.test_language.to_owned(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -327,9 +357,9 @@ impl EncryptedApplicationDetails {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<(&candidate::Model, Vec<parent::Model>)> for EncryptedApplicationDetails {
|
impl From<(&candidate::Model, &Vec<parent::Model>)> for EncryptedApplicationDetails {
|
||||||
fn from(
|
fn from(
|
||||||
(candidate, parents): (&candidate::Model, Vec<parent::Model>),
|
(candidate, parents): (&candidate::Model, &Vec<parent::Model>),
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let enc_parents = parents.iter()
|
let enc_parents = parents.iter()
|
||||||
.map(|m| EncryptedParentDetails::from(m))
|
.map(|m| EncryptedParentDetails::from(m))
|
||||||
|
|
@ -342,50 +372,6 @@ impl From<(&candidate::Model, Vec<parent::Model>)> for EncryptedApplicationDetai
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TryFrom<ApplicationRow> for EncryptedApplicationDetails {
|
|
||||||
type Error = ServiceError;
|
|
||||||
|
|
||||||
fn try_from(
|
|
||||||
cp: ApplicationRow,
|
|
||||||
) -> Result<Self, Self::Error> {
|
|
||||||
Ok(EncryptedApplicationDetails {
|
|
||||||
candidate: EncryptedCandidateDetails {
|
|
||||||
name: EncryptedString::try_from(&cp.name).ok(),
|
|
||||||
surname: EncryptedString::try_from(&cp.surname).ok(),
|
|
||||||
birthplace: EncryptedString::try_from(&cp.birthplace).ok(),
|
|
||||||
birthdate: EncryptedString::try_from(&cp.birthdate).ok(),
|
|
||||||
address: EncryptedString::try_from(&cp.address).ok(),
|
|
||||||
telephone: EncryptedString::try_from(&cp.telephone).ok(),
|
|
||||||
citizenship: EncryptedString::try_from(&cp.citizenship).ok(),
|
|
||||||
email: EncryptedString::try_from(&cp.email).ok(),
|
|
||||||
sex: EncryptedString::try_from(&cp.sex).ok(),
|
|
||||||
personal_id_number: EncryptedString::try_from(&cp.personal_identification_number).ok(),
|
|
||||||
school_name: EncryptedString::try_from(&cp.school_name).ok(),
|
|
||||||
health_insurance: EncryptedString::try_from(&cp.health_insurance).ok(),
|
|
||||||
grades_json: None, // TODO
|
|
||||||
test_language: None // TODO
|
|
||||||
},
|
|
||||||
parents: vec![EncryptedParentDetails {
|
|
||||||
name: EncryptedString::try_from(&cp.parent_name).ok(),
|
|
||||||
surname: EncryptedString::try_from(&cp.parent_surname).ok(),
|
|
||||||
telephone: EncryptedString::try_from(&cp.parent_telephone).ok(),
|
|
||||||
email: EncryptedString::try_from(&cp.parent_email).ok(),
|
|
||||||
}]
|
|
||||||
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* pub 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("")),
|
|
||||||
}
|
|
||||||
} */
|
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
pub mod tests {
|
pub mod tests {
|
||||||
use std::sync::Mutex;
|
use std::sync::Mutex;
|
||||||
|
|
@ -395,7 +381,7 @@ pub mod tests {
|
||||||
use once_cell::sync::Lazy;
|
use once_cell::sync::Lazy;
|
||||||
use sea_orm::{DbConn, Set, ActiveModelTrait};
|
use sea_orm::{DbConn, Set, ActiveModelTrait};
|
||||||
|
|
||||||
use crate::{crypto, models::{candidate::{CandidateDetails, ParentDetails}, grade::GradeList}, utils::db::get_memory_sqlite_connection, services::candidate_service::tests::put_user_data};
|
use crate::{crypto, models::{candidate::{CandidateDetails, ParentDetails}, grade::GradeList, school::School}, utils::db::get_memory_sqlite_connection, services::candidate_service::tests::put_user_data};
|
||||||
|
|
||||||
use super::{ApplicationDetails, EncryptedApplicationDetails, EncryptedString};
|
use super::{ApplicationDetails, EncryptedApplicationDetails, EncryptedString};
|
||||||
|
|
||||||
|
|
@ -407,9 +393,11 @@ pub mod tests {
|
||||||
candidate: CandidateDetails {
|
candidate: CandidateDetails {
|
||||||
name: "name".to_string(),
|
name: "name".to_string(),
|
||||||
surname: "surname".to_string(),
|
surname: "surname".to_string(),
|
||||||
|
birth_surname: "birth_surname".to_string(),
|
||||||
birthplace: "birthplace".to_string(),
|
birthplace: "birthplace".to_string(),
|
||||||
birthdate: chrono::NaiveDate::from_ymd_opt(2000, 1, 1).unwrap(),
|
birthdate: chrono::NaiveDate::from_ymd_opt(2000, 1, 1).unwrap(),
|
||||||
address: "address".to_string(),
|
address: "address".to_string(),
|
||||||
|
letter_address: "letter_address".to_string(),
|
||||||
telephone: "telephone".to_string(),
|
telephone: "telephone".to_string(),
|
||||||
citizenship: "citizenship".to_string(),
|
citizenship: "citizenship".to_string(),
|
||||||
email: "email".to_string(),
|
email: "email".to_string(),
|
||||||
|
|
@ -418,6 +406,8 @@ pub mod tests {
|
||||||
school_name: "school_name".to_string(),
|
school_name: "school_name".to_string(),
|
||||||
health_insurance: "health_insurance".to_string(),
|
health_insurance: "health_insurance".to_string(),
|
||||||
grades: GradeList::from(vec![]),
|
grades: GradeList::from(vec![]),
|
||||||
|
first_school: School::from_opt_str(Some("{\"name\": \"SSPS\", \"field\": \"KB\"}".to_string())).unwrap(),
|
||||||
|
second_school: School::from_opt_str(Some("{\"name\": \"SSPS\", \"field\": \"IT\"}".to_string())).unwrap(),
|
||||||
test_language: "test_language".to_string(),
|
test_language: "test_language".to_string(),
|
||||||
},
|
},
|
||||||
parents: vec![ParentDetails {
|
parents: vec![ParentDetails {
|
||||||
|
|
@ -518,7 +508,7 @@ pub mod tests {
|
||||||
|
|
||||||
let (_, candidate, parents) = put_user_data(&db).await;
|
let (_, candidate, parents) = put_user_data(&db).await;
|
||||||
|
|
||||||
let encrypted_details = EncryptedApplicationDetails::try_from((&candidate, parents)).unwrap();
|
let encrypted_details = EncryptedApplicationDetails::try_from((&candidate, &parents)).unwrap();
|
||||||
|
|
||||||
let application_details = encrypted_details
|
let application_details = encrypted_details
|
||||||
.decrypt(PRIVATE_KEY.to_string()) // decrypt with admin's private key
|
.decrypt(PRIVATE_KEY.to_string()) // decrypt with admin's private key
|
||||||
|
|
|
||||||
|
|
@ -16,6 +16,23 @@ impl GradeList {
|
||||||
|grades| serde_json::from_str(&grades).unwrap() // TODO: handle error
|
|grades| serde_json::from_str(&grades).unwrap() // TODO: handle error
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn group_by_semester(&self) -> (GradeList, GradeList, GradeList) {
|
||||||
|
let mut first_semester = GradeList::default();
|
||||||
|
let mut second_semester = GradeList::default();
|
||||||
|
let mut third_semester = GradeList::default();
|
||||||
|
|
||||||
|
for grade in &self.0 {
|
||||||
|
match grade.semester.as_str() {
|
||||||
|
"1/8" => first_semester.0.push(grade.clone()),
|
||||||
|
"2/8" => second_semester.0.push(grade.clone()),
|
||||||
|
"1/9" => third_semester.0.push(grade.clone()),
|
||||||
|
_ => panic!("Invalid semester"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
(first_semester, second_semester, third_semester)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for GradeList {
|
impl Default for GradeList {
|
||||||
|
|
|
||||||
|
|
@ -3,3 +3,4 @@ pub mod candidate;
|
||||||
pub mod auth;
|
pub mod auth;
|
||||||
pub mod application;
|
pub mod application;
|
||||||
pub mod grade;
|
pub mod grade;
|
||||||
|
pub mod school;
|
||||||
48
core/src/models/school.rs
Normal file
48
core/src/models/school.rs
Normal file
|
|
@ -0,0 +1,48 @@
|
||||||
|
use serde::{Serialize, Deserialize};
|
||||||
|
use validator::Validate;
|
||||||
|
|
||||||
|
use crate::error::ServiceError;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Serialize, Deserialize, Validate, PartialEq, Eq)]
|
||||||
|
pub struct School {
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
|
name: String,
|
||||||
|
#[validate(length(min = 1, max = 255))]
|
||||||
|
field: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl School {
|
||||||
|
pub fn from_opt_str(school: Option<String>) -> Option<Self> {
|
||||||
|
school.map(
|
||||||
|
|school| serde_json::from_str(&school).unwrap() // TODO: handle error
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn validate_self(&self) -> Result<(), ServiceError> {
|
||||||
|
self.validate()
|
||||||
|
.map_err(ServiceError::ValidationError)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn name(&self) -> &str {
|
||||||
|
&self.name
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn field(&self) -> &str {
|
||||||
|
&self.field
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToString for School {
|
||||||
|
fn to_string(&self) -> String {
|
||||||
|
serde_json::to_string(&self).unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for School {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
name: String::default(),
|
||||||
|
field: String::default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -237,7 +237,7 @@ impl ApplicationService {
|
||||||
let candidate = ApplicationService::find_related_candidate(db, application).await?;
|
let candidate = ApplicationService::find_related_candidate(db, application).await?;
|
||||||
|
|
||||||
let parents = Query::find_candidate_parents(db, &candidate).await?;
|
let parents = Query::find_candidate_parents(db, &candidate).await?;
|
||||||
let enc_details = EncryptedApplicationDetails::from((&candidate, parents));
|
let enc_details = EncryptedApplicationDetails::from((&candidate, &parents));
|
||||||
|
|
||||||
if enc_details.is_filled() {
|
if enc_details.is_filled() {
|
||||||
enc_details.decrypt(private_key).await
|
enc_details.decrypt(private_key).await
|
||||||
|
|
@ -330,7 +330,7 @@ impl ApplicationService {
|
||||||
recipients.append(&mut admin_public_keys);
|
recipients.append(&mut admin_public_keys);
|
||||||
recipients.append(&mut applications.iter().map(|a| a.public_key.to_owned()).collect());
|
recipients.append(&mut applications.iter().map(|a| a.public_key.to_owned()).collect());
|
||||||
|
|
||||||
let dec_details = EncryptedApplicationDetails::from((&candidate, parents.clone()))
|
let dec_details = EncryptedApplicationDetails::from((&candidate, &parents))
|
||||||
.decrypt(admin_private_key).await?;
|
.decrypt(admin_private_key).await?;
|
||||||
|
|
||||||
let enc_details = EncryptedApplicationDetails::new(&dec_details, recipients).await?;
|
let enc_details = EncryptedApplicationDetails::new(&dec_details, recipients).await?;
|
||||||
|
|
|
||||||
|
|
@ -134,7 +134,7 @@ pub mod tests {
|
||||||
let dec_priv_key = crypto::decrypt_password(application.private_key.clone(), password)
|
let dec_priv_key = crypto::decrypt_password(application.private_key.clone(), password)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let enc_details = EncryptedApplicationDetails::try_from((&enc_candidate, enc_parent))
|
let enc_details = EncryptedApplicationDetails::try_from((&enc_candidate, &enc_parent))
|
||||||
.ok()
|
.ok()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let dec_details = enc_details.decrypt(dec_priv_key).await.ok().unwrap();
|
let dec_details = enc_details.decrypt(dec_priv_key).await.ok().unwrap();
|
||||||
|
|
|
||||||
|
|
@ -54,16 +54,18 @@ mod tests {
|
||||||
|
|
||||||
use once_cell::sync::Lazy;
|
use once_cell::sync::Lazy;
|
||||||
|
|
||||||
use crate::{utils::db::get_memory_sqlite_connection, models::{candidate::{ParentDetails, ApplicationDetails, CandidateDetails}, candidate_details::EncryptedApplicationDetails, grade::GradeList}, services::{candidate_service::{CandidateService, tests::put_user_data}, application_service::ApplicationService, parent_service::ParentService}, crypto};
|
use crate::{utils::db::get_memory_sqlite_connection, models::{candidate::{ParentDetails, ApplicationDetails, CandidateDetails}, candidate_details::EncryptedApplicationDetails, grade::GradeList, school::School}, services::{candidate_service::{CandidateService, tests::put_user_data}, application_service::ApplicationService, parent_service::ParentService}, crypto};
|
||||||
|
|
||||||
pub static APPLICATION_DETAILS_TWO_PARENTS: Lazy<Mutex<ApplicationDetails>> = Lazy::new(||
|
pub static APPLICATION_DETAILS_TWO_PARENTS: Lazy<Mutex<ApplicationDetails>> = Lazy::new(||
|
||||||
Mutex::new(ApplicationDetails {
|
Mutex::new(ApplicationDetails {
|
||||||
candidate: CandidateDetails {
|
candidate: CandidateDetails {
|
||||||
name: "name".to_string(),
|
name: "name".to_string(),
|
||||||
surname: "surname".to_string(),
|
surname: "surname".to_string(),
|
||||||
|
birth_surname: "birth_surname".to_string(),
|
||||||
birthplace: "birthplace".to_string(),
|
birthplace: "birthplace".to_string(),
|
||||||
birthdate: chrono::NaiveDate::from_ymd_opt(2000, 1, 1).unwrap(),
|
birthdate: chrono::NaiveDate::from_ymd_opt(2000, 1, 1).unwrap(),
|
||||||
address: "address".to_string(),
|
address: "address".to_string(),
|
||||||
|
letter_address: "letter_address".to_string(),
|
||||||
telephone: "telephone".to_string(),
|
telephone: "telephone".to_string(),
|
||||||
citizenship: "citizenship".to_string(),
|
citizenship: "citizenship".to_string(),
|
||||||
email: "email".to_string(),
|
email: "email".to_string(),
|
||||||
|
|
@ -72,6 +74,8 @@ mod tests {
|
||||||
school_name: "school_name".to_string(),
|
school_name: "school_name".to_string(),
|
||||||
health_insurance: "health_insurance".to_string(),
|
health_insurance: "health_insurance".to_string(),
|
||||||
grades: GradeList::from(vec![]),
|
grades: GradeList::from(vec![]),
|
||||||
|
first_school: School::from_opt_str(Some("{\"name\": \"SSPS\", \"field\": \"KB\"}".to_string())).unwrap(),
|
||||||
|
second_school: School::from_opt_str(Some("{\"name\": \"SSPS\", \"field\": \"IT\"}".to_string())).unwrap(),
|
||||||
test_language: "test_language".to_string(),
|
test_language: "test_language".to_string(),
|
||||||
},
|
},
|
||||||
parents: vec![ParentDetails {
|
parents: vec![ParentDetails {
|
||||||
|
|
@ -113,7 +117,7 @@ mod tests {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
let priv_key = crypto::decrypt_password(application.private_key.clone(), plain_text_password).await.unwrap();
|
let priv_key = crypto::decrypt_password(application.private_key.clone(), plain_text_password).await.unwrap();
|
||||||
let dec_details = EncryptedApplicationDetails::try_from((&candidate, parents))
|
let dec_details = EncryptedApplicationDetails::try_from((&candidate, &parents))
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.decrypt(priv_key)
|
.decrypt(priv_key)
|
||||||
.await
|
.await
|
||||||
|
|
|
||||||
|
|
@ -77,7 +77,6 @@ mod tests {
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
// println!("{}", session.err().unwrap().1);
|
|
||||||
assert!(
|
assert!(
|
||||||
ApplicationService::auth(db, Uuid::parse_str(&session).unwrap())
|
ApplicationService::auth(db, Uuid::parse_str(&session).unwrap())
|
||||||
.await
|
.await
|
||||||
|
|
|
||||||
|
|
@ -9,20 +9,32 @@ use sea_orm::DbConn;
|
||||||
impl From<(i32, ApplicationDetails)> for ApplicationRow {
|
impl From<(i32, ApplicationDetails)> for ApplicationRow {
|
||||||
fn from((application, d): (i32, ApplicationDetails)) -> Self {
|
fn from((application, d): (i32, ApplicationDetails)) -> Self {
|
||||||
let c = d.candidate;
|
let c = d.candidate;
|
||||||
|
let (diploma_1_8, diploma_2_8, diploma_1_9) = c.grades.group_by_semester();
|
||||||
Self {
|
Self {
|
||||||
application,
|
application,
|
||||||
name: Some(c.name),
|
name: Some(c.name),
|
||||||
surname: Some(c.surname),
|
surname: Some(c.surname),
|
||||||
|
birth_surname: Some(c.birth_surname),
|
||||||
birthplace: Some(c.birthplace),
|
birthplace: Some(c.birthplace),
|
||||||
birthdate: Some(c.birthdate.to_string()),
|
birthdate: Some(c.birthdate.to_string()),
|
||||||
address: Some(c.address),
|
address: Some(c.address),
|
||||||
|
letter_address: Some(c.letter_address),
|
||||||
telephone: Some(c.telephone),
|
telephone: Some(c.telephone),
|
||||||
citizenship: Some(c.citizenship),
|
citizenship: Some(c.citizenship),
|
||||||
email: Some(c.email),
|
email: Some(c.email),
|
||||||
sex: Some(c.sex),
|
sex: Some(c.sex),
|
||||||
|
personal_identification_number: Some(c.personal_id_number),
|
||||||
health_insurance: Some(c.health_insurance),
|
health_insurance: Some(c.health_insurance),
|
||||||
school_name: Some(c.school_name),
|
school_name: Some(c.school_name),
|
||||||
personal_identification_number: Some(c.personal_id_number),
|
|
||||||
|
diploma_1_8: diploma_1_8.to_string(),
|
||||||
|
diploma_2_8: diploma_2_8.to_string(),
|
||||||
|
diploma_1_9: diploma_1_9.to_string(),
|
||||||
|
|
||||||
|
first_school_name: Some(c.first_school.name().to_owned()),
|
||||||
|
first_school_field: Some(c.first_school.field().to_owned()),
|
||||||
|
second_school_name: Some(c.second_school.name().to_owned()),
|
||||||
|
second_school_field: Some(c.second_school.field().to_owned()),
|
||||||
|
|
||||||
parent_name: d.parents.get(0).map(|p| p.name.clone()),
|
parent_name: d.parents.get(0).map(|p| p.name.clone()),
|
||||||
parent_surname: d.parents.get(0).map(|p| p.surname.clone()),
|
parent_surname: d.parents.get(0).map(|p| p.surname.clone()),
|
||||||
|
|
@ -45,7 +57,7 @@ pub async fn export(db: &DbConn, private_key: String) -> Result<Vec<u8>, Service
|
||||||
let candidate = ApplicationService::find_related_candidate(db, &application).await?;
|
let candidate = ApplicationService::find_related_candidate(db, &application).await?;
|
||||||
let parents = Query::find_candidate_parents(db, &candidate).await?;
|
let parents = Query::find_candidate_parents(db, &candidate).await?;
|
||||||
|
|
||||||
let row: ApplicationRow = match EncryptedApplicationDetails::try_from((&candidate, parents))
|
let row: ApplicationRow = match EncryptedApplicationDetails::try_from((&candidate, &parents))
|
||||||
{
|
{
|
||||||
Ok(d) => ApplicationRow::from(
|
Ok(d) => ApplicationRow::from(
|
||||||
d.decrypt(private_key.to_string())
|
d.decrypt(private_key.to_string())
|
||||||
|
|
|
||||||
|
|
@ -13,6 +13,7 @@ pub struct Model {
|
||||||
pub birthplace: Option<String>,
|
pub birthplace: Option<String>,
|
||||||
pub birthdate: Option<String>,
|
pub birthdate: Option<String>,
|
||||||
pub address: Option<String>,
|
pub address: Option<String>,
|
||||||
|
pub letter_address: Option<String>,
|
||||||
pub telephone: Option<String>,
|
pub telephone: Option<String>,
|
||||||
pub citizenship: Option<String>,
|
pub citizenship: Option<String>,
|
||||||
pub email: Option<String>,
|
pub email: Option<String>,
|
||||||
|
|
@ -21,6 +22,8 @@ pub struct Model {
|
||||||
pub school_name: Option<String>,
|
pub school_name: Option<String>,
|
||||||
pub health_insurance: Option<String>,
|
pub health_insurance: Option<String>,
|
||||||
pub grades_json: Option<String>,
|
pub grades_json: Option<String>,
|
||||||
|
pub first_school: Option<String>,
|
||||||
|
pub second_school: Option<String>,
|
||||||
pub test_language: Option<String>,
|
pub test_language: Option<String>,
|
||||||
pub encrypted_by_id: Option<i32>,
|
pub encrypted_by_id: Option<i32>,
|
||||||
pub created_at: DateTime,
|
pub created_at: DateTime,
|
||||||
|
|
|
||||||
|
|
@ -24,6 +24,7 @@ impl MigrationTrait for Migration {
|
||||||
.col(ColumnDef::new(Candidate::Birthplace).string())
|
.col(ColumnDef::new(Candidate::Birthplace).string())
|
||||||
.col(ColumnDef::new(Candidate::Birthdate).string())
|
.col(ColumnDef::new(Candidate::Birthdate).string())
|
||||||
.col(ColumnDef::new(Candidate::Address).string())
|
.col(ColumnDef::new(Candidate::Address).string())
|
||||||
|
.col(ColumnDef::new(Candidate::LetterAddress).string())
|
||||||
.col(ColumnDef::new(Candidate::Telephone).string())
|
.col(ColumnDef::new(Candidate::Telephone).string())
|
||||||
.col(ColumnDef::new(Candidate::Citizenship).string())
|
.col(ColumnDef::new(Candidate::Citizenship).string())
|
||||||
.col(ColumnDef::new(Candidate::Email).string())
|
.col(ColumnDef::new(Candidate::Email).string())
|
||||||
|
|
@ -32,6 +33,8 @@ impl MigrationTrait for Migration {
|
||||||
.col(ColumnDef::new(Candidate::SchoolName).string())
|
.col(ColumnDef::new(Candidate::SchoolName).string())
|
||||||
.col(ColumnDef::new(Candidate::HealthInsurance).string())
|
.col(ColumnDef::new(Candidate::HealthInsurance).string())
|
||||||
.col(ColumnDef::new(Candidate::GradesJson).string())
|
.col(ColumnDef::new(Candidate::GradesJson).string())
|
||||||
|
.col(ColumnDef::new(Candidate::FirstSchool).string())
|
||||||
|
.col(ColumnDef::new(Candidate::SecondSchool).string())
|
||||||
.col(ColumnDef::new(Candidate::TestLanguage).string())
|
.col(ColumnDef::new(Candidate::TestLanguage).string())
|
||||||
.col(ColumnDef::new(Candidate::EncryptedById).integer())
|
.col(ColumnDef::new(Candidate::EncryptedById).integer())
|
||||||
.col(ColumnDef::new(Candidate::CreatedAt).date_time().not_null())
|
.col(ColumnDef::new(Candidate::CreatedAt).date_time().not_null())
|
||||||
|
|
@ -58,6 +61,7 @@ pub enum Candidate {
|
||||||
Birthplace,
|
Birthplace,
|
||||||
Birthdate,
|
Birthdate,
|
||||||
Address,
|
Address,
|
||||||
|
LetterAddress,
|
||||||
Telephone,
|
Telephone,
|
||||||
Citizenship,
|
Citizenship,
|
||||||
Email,
|
Email,
|
||||||
|
|
@ -66,6 +70,8 @@ pub enum Candidate {
|
||||||
SchoolName,
|
SchoolName,
|
||||||
HealthInsurance,
|
HealthInsurance,
|
||||||
GradesJson,
|
GradesJson,
|
||||||
|
FirstSchool,
|
||||||
|
SecondSchool,
|
||||||
TestLanguage,
|
TestLanguage,
|
||||||
EncryptedById,
|
EncryptedById,
|
||||||
CreatedAt,
|
CreatedAt,
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue