//! This module provides the functionality to parse the lua config and convert the config options //! into rust readable form. use super::parser_models::Style; use rlua::Lua; use std::{format, fs, path::Path}; /// A named struct which stores the parsed config file options. /// /// # Fields // /// * `port` - It stores the parsed port number option on which the server should launch. /// * `binding_ip_addr` - It stores the parsed ip address option on which the server should launch /// * `style` - It stores the theming options for the website. /// * `redis_connection_url` - It stores the redis connection url address on which the redis /// client should connect. #[derive(Clone)] pub struct Config { pub port: u16, pub binding_ip_addr: String, pub style: Style, pub redis_connection_url: String, } impl Config { /// A function which parses the config.lua file and puts all the parsed options in the newly /// contructed Config struct and returns it. /// /// # Error /// /// Returns a lua parse error if parsing of the config.lua file fails or has a syntax error /// or io error if the config.lua file doesn't exists otherwise it returns a newly contructed /// Config struct with all the parsed config options from the parsed config file. pub fn parse() -> Result> { Lua::new().context(|context| -> Result> { let globals = context.globals(); context .load(&fs::read_to_string( Config::handle_different_config_file_path()?, )?) .exec()?; Ok(Config { port: globals.get::<_, u16>("port")?, binding_ip_addr: globals.get::<_, String>("binding_ip_addr")?, style: Style::new( globals.get::<_, String>("theme")?, globals.get::<_, String>("colorscheme")?, ), redis_connection_url: globals.get::<_, String>("redis_connection_url")?, }) }) } /// A helper function which returns an appropriate config file path checking if the config /// file exists on that path. /// /// # Error /// /// Returns a `config file not found!!` error if the config file is not present under following /// paths which are: /// 1. `~/.config/websurfx/` if it not present here then it fallbacks to the next one (2) /// 2. `/etc/xdg/websurfx/config.lua` if it is not present here then it fallbacks to the next /// one (3). /// 3. `websurfx/` (under project folder ( or codebase in other words)) if it is not present /// here then it returns an error as mentioned above. fn handle_different_config_file_path() -> Result> { if Path::new( format!( "{}/.config/websurfx/config.lua", std::env::var("HOME").unwrap() ) .as_str(), ) .exists() { Ok(format!( "{}/.config/websurfx/config.lua", std::env::var("HOME").unwrap() )) } else if Path::new("/etc/xdg/websurfx/config.lua").exists() { Ok("/etc/xdg/websurfx/config.lua".to_string()) } else if Path::new("./websurfx/config.lua").exists() { Ok("./websurfx/config.lua".to_string()) } else { Err(format!("Config file not found!!").into()) } } }