config rework

This commit is contained in:
Reid 2023-10-17 17:19:50 -07:00
parent 6e7550927f
commit 7024a1fa28
Signed by: reidlab
GPG key ID: 6C9EAA3364F962C8
8 changed files with 49 additions and 70 deletions

View file

@ -13,8 +13,6 @@
append_path = "/" append_path = "/"
# where can your server be accessible? # where can your server be accessible?
port = 8000 port = 8000
# your realip header, if you're behind a reverse proxy
realip_header = "X-Real-IP"
[accounts] [accounts]
# allow new accounts to be created # allow new accounts to be created

View file

@ -33,8 +33,9 @@ i've run out of ideas.
## todo ## todo
- User icons in account management pages - User icons in account management pages
- Flesh out account management page
- Account settings page - Account settings page
- Better web design - Better web design (make formatting more consistant)
- Use chrono for dates in database, add recent - Use chrono for dates in database, add recent
- 2.2's friends only unlisted - 2.2's friends only unlisted
- Dailies, weeklies, events(?) - Dailies, weeklies, events(?)
@ -43,4 +44,8 @@ i've run out of ideas.
- Cache authentication - Cache authentication
- Panic less - Panic less
- Make a proper rank system (reuploading, uploading music, rating, etc.) - Make a proper rank system (reuploading, uploading music, rating, etc.)
- Swap to a better web framework - Use serde to make the forms whateverCaseThisIs rather than breaking our lint convention
- Swap to `sqlx` im gonna be honest `diesel` is pretty shit.
- Swap to `sqlite` from `postgres`. Postgres feels too clunky and it just solos honestly
- Add back `realip` header support
- Add configurable form limits

View file

@ -1,52 +1,27 @@
use serde::Deserialize;
use std::fs; use std::fs;
use std::sync::LazyLock; use std::sync::LazyLock;
#[derive(Deserialize)] use toml::Table;
pub struct Config {
pub general: ConfigGeneral,
pub accounts: ConfigAccounts,
pub db: ConfigDB,
pub levels: ConfigLevels
}
#[derive(Deserialize)] pub static CONFIG: LazyLock<Table> = LazyLock::new(|| {
pub struct ConfigGeneral { let toml_str = fs::read_to_string("config.toml").expect("error finding toml config");
pub append_path: String, let config: Table = toml::from_str(toml_str.as_str()).expect("error parsing toml config");
pub port: u16,
pub realip_header: String
}
#[derive(Deserialize)]
pub struct ConfigAccounts {
pub allow_registration: bool
}
#[derive(Deserialize)]
pub struct ConfigDB {
pub data_folder: String
}
#[derive(Deserialize)]
pub struct ConfigLevels {
pub max_objects: i32,
pub blocklist: Vec<i32>,
pub reupload: bool
}
impl Config {
pub fn load_from_file(file_path: &str) -> Self {
let toml_str = fs::read_to_string(file_path).expect("Error finding toml config:");
let config: Config = toml::from_str(toml_str.as_str()).expect("Error parsing toml config:");
return config;
}
}
pub static CONFIG: LazyLock<Config> = LazyLock::new(|| {
let config = Config::load_from_file("config.toml");
return config; return config;
}); });
pub fn config_get(key: &str) -> Option<&toml::Value> {
let this = &CONFIG;
let mut current = this.get(key)?;
for val in key.split(".").skip(1) {
current = current.as_table()?.get(val)?;
}
Some(current)
}
pub fn config_get_with_default<T: serde::Deserialize<'static>>(key: &str, default: T) -> T {
config_get(key)
.and_then(|v| v.clone().try_into().ok())
.unwrap_or(default)
}

View file

@ -7,7 +7,7 @@ use diesel::result::Error;
use password_auth::generate_hash; use password_auth::generate_hash;
use crate::CONFIG; use crate::config;
use crate::helpers; use crate::helpers;
use crate::db; use crate::db;
@ -22,7 +22,7 @@ pub struct FormRegisterAccount {
pub fn register_account(input: Form<FormRegisterAccount>) -> status::Custom<&'static str> { pub fn register_account(input: Form<FormRegisterAccount>) -> status::Custom<&'static str> {
let connection = &mut db::establish_connection_pg(); let connection = &mut db::establish_connection_pg();
if CONFIG.accounts.allow_registration == false { if config::config_get_with_default("accounts.allow_registration", true) == false {
return status::Custom(Status::Ok, "-1") return status::Custom(Status::Ok, "-1")
} }
@ -55,7 +55,7 @@ pub fn register_account(input: Form<FormRegisterAccount>) -> status::Custom<&'st
use crate::schema::accounts::dsl::*; use crate::schema::accounts::dsl::*;
let account_name_usage = accounts.filter(username.eq(input.userName.clone())).count().get_result::<i64>(connection) as Result<i64, Error>; let account_name_usage = accounts.filter(username.eq(input.userName.clone())).count().get_result::<i64>(connection) as Result<i64, Error>;
let account_name_used = account_name_usage.expect("Fatal database name query error") != 0; let account_name_used = account_name_usage.expect("database name query error") != 0;
if account_name_used { if account_name_used {
return status::Custom(Status::Ok, "-2") return status::Custom(Status::Ok, "-2")
} }
@ -70,7 +70,7 @@ pub fn register_account(input: Form<FormRegisterAccount>) -> status::Custom<&'st
inserted_account = diesel::insert_into(accounts) inserted_account = diesel::insert_into(accounts)
.values(&new_account) .values(&new_account)
.get_result::<Account, >(connection) .get_result::<Account, >(connection)
.expect("Fatal error saving the new account"); .expect("error saving the new account");
} }
// user management // user management
@ -88,7 +88,7 @@ pub fn register_account(input: Form<FormRegisterAccount>) -> status::Custom<&'st
diesel::insert_into(users) diesel::insert_into(users)
.values(&new_user) .values(&new_user)
.get_result::<User, >(connection) .get_result::<User, >(connection)
.expect("Fatal error saving the new user"); .expect("error saving the new user");
} }
return status::Custom(Status::Ok, "1") return status::Custom(Status::Ok, "1")

View file

@ -13,6 +13,7 @@ use std::fs;
use std::io::prelude::*; use std::io::prelude::*;
use crate::helpers; use crate::helpers;
use crate::config;
use crate::db; use crate::db;
#[derive(FromForm)] #[derive(FromForm)]
@ -98,7 +99,7 @@ pub fn download_level(input: Form<FormDownloadLevel>) -> status::Custom<&'static
xor_pass = level.password.clone().unwrap_or(String::from("0")); xor_pass = level.password.clone().unwrap_or(String::from("0"));
} }
let compressed_level_data = fs::read(format!("{}/{}/{}.lvl", crate::CONFIG.db.data_folder, "levels", level.id)).expect("couldnt read level file"); let compressed_level_data = fs::read(format!("{}/{}/{}.lvl", config::config_get_with_default("db.data_folder", "data"), "levels", level.id)).expect("couldnt read level file");
let uncompressed_level_data = String::from_utf8(if compressed_level_data.starts_with(&[0x1F, 0x8B]) { let uncompressed_level_data = String::from_utf8(if compressed_level_data.starts_with(&[0x1F, 0x8B]) {
// gzip!! // gzip!!

View file

@ -8,7 +8,7 @@ use base64::{Engine as _, engine::general_purpose};
use std::fs; use std::fs;
use crate::config::CONFIG; use crate::config;
use crate::helpers; use crate::helpers;
use crate::db; use crate::db;
@ -98,12 +98,12 @@ pub fn upload_level(input: Form<FormUploadLevel>) -> status::Custom<&'static str
} }
// too many objects // too many objects
if objects_val > CONFIG.levels.max_objects as usize { if config::config_get_with_default("levels.max_objects", 0) != 0 && objects_val > config::config_get_with_default("levels.max_objects", 0) {
return status::Custom(Status::Ok, "-1") return status::Custom(Status::Ok, "-1")
} }
// forbidden object checking // forbidden object checking
if let Some(_forbidden_object) = level_objects.iter().find(|obj| crate::CONFIG.levels.blocklist.contains(&obj.id())) { if let Some(_forbidden_object) = level_objects.iter().find(|obj| config::config_get_with_default("levels.blocklist", Vec::new() as Vec<i32>).contains(&obj.id())) {
return status::Custom(Status::Ok, "-1") return status::Custom(Status::Ok, "-1")
} }
@ -158,7 +158,7 @@ pub fn upload_level(input: Form<FormUploadLevel>) -> status::Custom<&'static str
.get_result::<Level, >(connection) .get_result::<Level, >(connection)
.expect("failed to update level"); .expect("failed to update level");
fs::write(format!("{}/levels/{}.lvl", crate::CONFIG.db.data_folder, updated_level.id), general_purpose::URL_SAFE.decode(input.levelString.clone()).expect("user provided invalid level string")).expect("couldnt write level to file"); fs::write(format!("{}/levels/{}.lvl", config::config_get_with_default("db.data_folder", "data"), updated_level.id), general_purpose::URL_SAFE.decode(input.levelString.clone()).expect("user provided invalid level string")).expect("couldnt write level to file");
return status::Custom(Status::Ok, Box::leak(input.levelID.to_string().into_boxed_str())) return status::Custom(Status::Ok, Box::leak(input.levelID.to_string().into_boxed_str()))
} else { } else {
@ -194,7 +194,7 @@ pub fn upload_level(input: Form<FormUploadLevel>) -> status::Custom<&'static str
.get_result::<Level, >(connection) .get_result::<Level, >(connection)
.expect("failed to insert level"); .expect("failed to insert level");
fs::write(format!("{}/levels/{}.lvl", crate::CONFIG.db.data_folder, inserted_level.id), general_purpose::URL_SAFE.decode(input.levelString.clone()).expect("user provided invalid level string")).expect("couldnt write level to file"); fs::write(format!("{}/levels/{}.lvl", config::config_get_with_default("db.data_folder", "data"), inserted_level.id), general_purpose::URL_SAFE.decode(input.levelString.clone()).expect("user provided invalid level string")).expect("couldnt write level to file");
return status::Custom(Status::Ok, "1") return status::Custom(Status::Ok, "1")
} }

View file

@ -38,15 +38,14 @@ fn rocket() -> _ {
crate::helpers::reupload::init(); crate::helpers::reupload::init();
// data directories // data directories
// this is a bit scuffed // unhardcore this maybe?
fs::create_dir_all(&CONFIG.db.data_folder).expect("failed to create data directory!"); fs::create_dir_all(config::config_get_with_default("db.data_folder", "data")).expect("failed to create data directory!");
fs::create_dir_all(format!("{}/levels", &CONFIG.db.data_folder)).expect("failed to create data directory for levels"); fs::create_dir_all(format!("{}/levels", config::config_get_with_default("db.data_folder", "data"))).expect("failed to create data directory for levels");
rocket::build() rocket::build()
// conf // conf
.configure(rocket::Config::figment() .configure(rocket::Config::figment()
.merge(("port", CONFIG.general.port)) .merge(("port", config::config_get_with_default("general.port", 8000)))
.merge(("ip_header", CONFIG.general.realip_header.as_str()))
.merge(("limits", Limits::new().limit("forms", 10.megabytes())))) .merge(("limits", Limits::new().limit("forms", 10.megabytes()))))
// actual website // actual website
.mount("/", routes![ .mount("/", routes![
@ -67,7 +66,7 @@ fn rocket() -> _ {
files files
]) ])
// https://www.youtube.com/watch?v=_pLrtsf5yfE // https://www.youtube.com/watch?v=_pLrtsf5yfE
.mount(CONFIG.general.append_path.as_str(), routes![ .mount(config::config_get_with_default("general.append_path", "/"), routes![
endpoints::accounts::login_account::login_account, endpoints::accounts::login_account::login_account,
endpoints::accounts::register_account::register_account, endpoints::accounts::register_account::register_account,
endpoints::accounts::update_account_settings::update_account_settings, endpoints::accounts::update_account_settings::update_account_settings,

View file

@ -16,6 +16,7 @@ use diesel::prelude::*;
use crate::helpers; use crate::helpers;
use crate::db; use crate::db;
use crate::config;
#[derive(Deserialize)] #[derive(Deserialize)]
struct LevelResults { struct LevelResults {
@ -39,7 +40,7 @@ pub struct FormReupload {
pub async fn post_reupload(input: Form<FormReupload>) -> Template { pub async fn post_reupload(input: Form<FormReupload>) -> Template {
let connection = &mut db::establish_connection_pg(); let connection = &mut db::establish_connection_pg();
let disabled = !crate::CONFIG.levels.reupload; let disabled = !config::config_get_with_default("levels.reupload", true);
if !disabled { if !disabled {
let remote_level_id = input.level_id; let remote_level_id = input.level_id;
@ -105,7 +106,7 @@ pub async fn post_reupload(input: Form<FormReupload>) -> Template {
.get_result::<Level, >(connection) .get_result::<Level, >(connection)
.expect("failed to insert level"); .expect("failed to insert level");
fs::write(format!("{}/levels/{}.lvl", crate::CONFIG.db.data_folder, inserted_level.id), general_purpose::URL_SAFE.decode(level_data.get("k4").expect("no level data?!").as_bytes()).expect("user provided invalid level string")).expect("couldnt write level to file"); fs::write(format!("{}/levels/{}.lvl", config::config_get_with_default("db.data_folder", "data"), inserted_level.id), general_purpose::URL_SAFE.decode(level_data.get("k4").expect("no level data?!").as_bytes()).expect("user provided invalid level string")).expect("couldnt write level to file");
return Template::render("reupload", context! { return Template::render("reupload", context! {
level_id: inserted_level.id level_id: inserted_level.id
@ -119,7 +120,7 @@ pub async fn post_reupload(input: Form<FormReupload>) -> Template {
#[get("/tools/reupload")] #[get("/tools/reupload")]
pub fn get_reupload() -> Template { pub fn get_reupload() -> Template {
let disabled = !crate::CONFIG.levels.reupload; let disabled = !config::config_get_with_default("levels.reupload", true);
Template::render("reupload", context! { Template::render("reupload", context! {
disabled: disabled disabled: disabled