bfile/filed/src/env.rs

110 lines
3.9 KiB
Rust
Raw Normal View History

2023-09-29 12:15:07 +02:00
/*
env.rs - The environment loader. It loads all the .env config to a convenient `Env` struct.
This file provides the `loadenv` function that will do just that.
*/
use std::{env::var, net::SocketAddr, path::Path, fs};
2023-09-29 12:15:07 +02:00
2023-10-20 17:31:08 +02:00
pub const DEFAULT_CONFIG: &'static str = include_str!("../config/filed.toml.example");
2023-10-01 02:14:35 +02:00
#[derive(Debug, Clone)]
pub struct Redis {
pub pass: String,
pub host: String,
pub port: u16,
pub prefix: String
}
#[derive(Debug, Clone)]
2023-09-29 12:15:07 +02:00
pub struct Env {
2023-09-29 12:57:53 +02:00
pub logging: bool,
2023-10-01 02:14:35 +02:00
pub listen: SocketAddr,
pub redis: Redis,
pub filedir: String,
2023-10-07 11:29:11 +02:00
pub instanceurl: String,
2023-10-20 17:12:16 +02:00
pub uploadspath: String,
pub confpath: String
2023-09-29 12:15:07 +02:00
}
fn get_var<T: Into<String>, O: From<String>>(name: T) -> Result<O, String> {
let name: String = name.into();
let v = var(name.clone());
if v.is_err() {
return Err(format!("Variable {name} does not exist!"));
}
return Ok(v.unwrap().into())
}
pub fn loadenv() -> Result<Env, Box<dyn std::error::Error>> {
Ok(
Env {
logging: get_var::<&str, String>("APP_LOGGING")?.to_lowercase() == "true",
2023-10-01 02:14:35 +02:00
listen: get_var::<&str, String>("APP_HOST")?.parse::<SocketAddr>().unwrap(),
redis: Redis {
pass: get_var("REDIS_PASS")?,
host: get_var("REDIS_HOST")?,
port: get_var::<&str, String>("REDIS_PORT")?.parse().unwrap(),
prefix: get_var("REDIS_PREFIX")?
2023-10-01 06:17:58 +02:00
},
filedir: {
let spath: String = get_var("USERCONTENT_DIR")?;
let path = Path::new(&spath);
if ! path.exists() {
fs::create_dir_all(path).map_err(|err| format!("Could not create usercontent directory: {err}"))?;
}
if ! path.is_dir() {
return Err(format!("USERCONTENT_DIR is set to \"{}\", which exists but is not a directory!", &spath).into())
2023-10-01 06:17:58 +02:00
}
spath
},
2023-10-07 11:29:11 +02:00
instanceurl: get_var("INSTANCE_URL")?,
2023-10-20 17:12:16 +02:00
uploadspath: get_var("UPLOADS_PATH")?,
confpath: {
2023-10-20 17:12:52 +02:00
let spath: String = get_var("CONF_FILE").unwrap_or("/etc/filed/filed.toml".into());
2023-10-20 17:12:16 +02:00
let path = Path::new(&spath);
let mut dirpath = path.components();
2023-10-20 17:31:08 +02:00
dirpath.next_back();
let dirpath = dirpath.as_path();
2023-10-20 17:12:16 +02:00
if ! path.is_file() {
2023-10-20 17:35:13 +02:00
log::error!("Config file is not a file");
log::info!("Trying to recover from error");
if ! dirpath.is_dir() {
log::info!("The config file directory does not exist. Trying to create it");
let created = fs::create_dir_all(dirpath);
if created.is_err() {
log::warn!("Could not create the config directory: {:?}", created.unwrap_err());
} else {
log::info!("Created the config directory");
}
}
2023-10-20 17:31:08 +02:00
if dirpath.is_dir() {
log::info!("Config file does not exist, trying to write the example");
2023-10-20 17:31:08 +02:00
let wrote = fs::write(path, DEFAULT_CONFIG);
2023-10-20 17:31:08 +02:00
if wrote.is_err() {
log::warn!("Could not write example because of the following error: {:?}", wrote.unwrap_err());
2023-10-20 17:31:08 +02:00
} else {
log::info!("Wrote example to {}", spath);
}
}
2023-10-20 17:35:13 +02:00
log::info!("Giving up");
2023-10-20 17:12:16 +02:00
return Err(format!("CONF_FILE is {}, which is not a file!", spath).into())
}
spath
}
2023-09-29 12:15:07 +02:00
}
)
2023-10-01 06:17:58 +02:00
}
impl Env {
pub fn usercontent_dir(self: &Self) -> Box<&Path> {
Box::new(Path::new(&self.filedir))
}
2023-09-29 12:15:07 +02:00
}