You've already forked AstralRinth
forked from xxxOFFxxx/AstralRinth
* Update Rust version * Update async-compression 0.4.25 -> 0.4.27 * Update async-tungstenite 0.29.1 -> 0.30.0 * Update bytemuck 1.23.0 -> 1.23.1 * Update clap 4.5.40 -> 4.5.43 * Update deadpool-redis 0.21.1 -> 0.22.0 and redis 0.31.0 -> 0.32.4 * Update enumset 1.1.6 -> 1.1.7 * Update hyper-util 0.1.14 -> 0.1.16 * Update indexmap 2.9.0 -> 2.10.0 * Update indicatif 0.17.11 -> 0.18.0 * Update jemalloc_pprof 0.7.0 -> 0.8.1 * Update lettre 0.11.17 -> 0.11.18 * Update meilisearch-sdk 0.28.0 -> 0.29.1 * Update notify 8.0.0 -> 8.2.0 and notify-debouncer-mini 0.6.0 -> 0.7.0 * Update quick-xml 0.37.5 -> 0.38.1 * Fix theseus lint * Update reqwest 0.12.20 -> 0.12.22 * Cargo fmt in theseus * Update rgb 0.8.50 -> 0.8.52 * Update sentry 0.41.0 -> 0.42.0 and sentry-actix 0.41.0 -> 0.42.0 * Update serde_json 1.0.140 -> 1.0.142 * Update serde_with 3.13.0 -> 3.14.0 * Update spdx 0.10.8 -> 0.10.9 * Update sysinfo 0.35.2 -> 0.36.1 * Update tauri suite * Fix build by updating mappings * Update tokio 1.45.1 -> 1.47.1 and tokio-util 0.7.15 -> 0.7.16 * Update tracing-actix-web 0.7.18 -> 0.7.19 * Update zip 4.2.0 -> 4.3.0 * Misc Cargo.lock updates * Update Dockerfiles
163 lines
4.4 KiB
Rust
163 lines
4.4 KiB
Rust
use std::{fmt::Write, sync::LazyLock};
|
|
|
|
use itertools::Itertools;
|
|
use regex::Regex;
|
|
use validator::{ValidationErrors, ValidationErrorsKind};
|
|
|
|
use crate::models::pats::Scopes;
|
|
|
|
pub static RE_URL_SAFE: LazyLock<Regex> =
|
|
LazyLock::new(|| Regex::new(r#"^[a-zA-Z0-9!@$()`.+,_"-]*$"#).unwrap());
|
|
pub static RE_USERNAME: LazyLock<Regex> =
|
|
LazyLock::new(|| Regex::new(r#"^[a-zA-Z0-9_-]*$"#).unwrap());
|
|
|
|
//TODO: In order to ensure readability, only the first error is printed, this may need to be expanded on in the future!
|
|
pub fn validation_errors_to_string(
|
|
errors: ValidationErrors,
|
|
adder: Option<String>,
|
|
) -> String {
|
|
let mut output = String::new();
|
|
|
|
let map = errors.into_errors();
|
|
|
|
let key_option = map.keys().next();
|
|
|
|
if let Some(field) = key_option
|
|
&& let Some(error) = map.get(field)
|
|
{
|
|
return match error {
|
|
ValidationErrorsKind::Struct(errors) => {
|
|
validation_errors_to_string(
|
|
*errors.clone(),
|
|
Some(format!("of item {field}")),
|
|
)
|
|
}
|
|
ValidationErrorsKind::List(list) => {
|
|
if let Some((index, errors)) = list.iter().next() {
|
|
output.push_str(&validation_errors_to_string(
|
|
*errors.clone(),
|
|
Some(format!("of list {field} with index {index}")),
|
|
));
|
|
}
|
|
|
|
output
|
|
}
|
|
ValidationErrorsKind::Field(errors) => {
|
|
if let Some(error) = errors.first() {
|
|
if let Some(adder) = adder {
|
|
write!(
|
|
&mut output,
|
|
"Field {field} {adder} failed validation with error: {}",
|
|
error.code
|
|
).unwrap();
|
|
} else {
|
|
write!(
|
|
&mut output,
|
|
"Field {field} failed validation with error: {}",
|
|
error.code
|
|
)
|
|
.unwrap();
|
|
}
|
|
}
|
|
|
|
output
|
|
}
|
|
};
|
|
}
|
|
|
|
String::new()
|
|
}
|
|
|
|
pub fn validate_deps(
|
|
values: &[crate::models::projects::Dependency],
|
|
) -> Result<(), validator::ValidationError> {
|
|
if values
|
|
.iter()
|
|
.duplicates_by(|x| {
|
|
format!(
|
|
"{}-{}-{}",
|
|
x.version_id.unwrap_or(crate::models::ids::VersionId(0)),
|
|
x.project_id.unwrap_or(crate::models::ids::ProjectId(0)),
|
|
x.file_name.as_deref().unwrap_or_default()
|
|
)
|
|
})
|
|
.next()
|
|
.is_some()
|
|
{
|
|
return Err(validator::ValidationError::new("duplicate dependency"));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn validate_url(value: &str) -> Result<(), validator::ValidationError> {
|
|
let url = url::Url::parse(value)
|
|
.ok()
|
|
.ok_or_else(|| validator::ValidationError::new("invalid URL"))?;
|
|
|
|
if url.scheme() != "https" {
|
|
return Err(validator::ValidationError::new("URL must be https"));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn validate_url_hashmap_optional_values(
|
|
values: &std::collections::HashMap<String, Option<String>>,
|
|
) -> Result<(), validator::ValidationError> {
|
|
for value in values.values().flatten() {
|
|
validate_url(value)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn validate_url_hashmap_values(
|
|
values: &std::collections::HashMap<String, String>,
|
|
) -> Result<(), validator::ValidationError> {
|
|
for value in values.values() {
|
|
validate_url(value)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn validate_no_restricted_scopes(
|
|
value: &Scopes,
|
|
) -> Result<(), validator::ValidationError> {
|
|
if value.is_restricted() {
|
|
return Err(validator::ValidationError::new(
|
|
"Restricted scopes not allowed",
|
|
));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn validate_name(value: &str) -> Result<(), validator::ValidationError> {
|
|
if value.trim().is_empty() {
|
|
return Err(validator::ValidationError::new(
|
|
"Name cannot contain only whitespace.",
|
|
));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn validate_name_with_valid_input() {
|
|
let result = validate_name("My Test mod");
|
|
assert!(result.is_ok());
|
|
}
|
|
|
|
#[test]
|
|
fn validate_name_with_invalid_input_returns_error() {
|
|
let result = validate_name(" ");
|
|
assert!(result.is_err());
|
|
}
|
|
}
|