diff --git a/docs/content/configuration/config-file/index.md b/docs/content/configuration/config-file/index.md index 8ccf31f4f..433d4f3ca 100644 --- a/docs/content/configuration/config-file/index.md +++ b/docs/content/configuration/config-file/index.md @@ -12,8 +12,8 @@ If no config file argument is given, it will automatically look for a config fil | Linux | `~/.config/bottom/bottom.toml`
`$XDG_CONFIG_HOME/bottom/bottom.toml` | | Windows | `C:\Users\\AppData\Roaming\bottom\bottom.toml` | -Like if a path is passed with `-C`/`--config`, if a file doesn't exist at the path, bottom will automatically create a -new, default config file at that location. +If the config file doesn't exist at the path, bottom will automatically try to create a new config file at the location +with default values. ## JSON Schema diff --git a/src/main.rs b/src/main.rs index b6e703abf..c7c979ce3 100644 --- a/src/main.rs +++ b/src/main.rs @@ -27,7 +27,6 @@ use std::{ boxed::Box, io::{stderr, stdout, Write}, panic::{self, PanicInfo}, - path::Path, sync::{ mpsc::{self, Receiver, Sender}, Arc, Condvar, Mutex, @@ -36,7 +35,6 @@ use std::{ time::{Duration, Instant}, }; -use anyhow::Context; use app::{layout_manager::UsedWidgets, App, AppConfigFields, DataFilters}; use crossterm::{ event::{ @@ -49,7 +47,7 @@ use crossterm::{ }; use data_conversion::*; use event::{handle_key_event_or_break, handle_mouse_event, BottomEvent, CollectionThreadEvent}; -use options::{args, get_config_path, get_or_create_config, init_app}; +use options::{args, get_or_create_config, init_app}; use tui::{backend::CrosstermBackend, Terminal}; #[allow(unused_imports)] use utils::logging::*; @@ -330,18 +328,8 @@ fn main() -> anyhow::Result<()> { } // Read from config file. - let config = { - let config_path = get_config_path(args.general.config_location.as_deref()); - get_or_create_config(config_path.as_deref()).with_context(|| { - format!( - "Unable to parse or create the config file at: {}", - config_path - .as_deref() - .unwrap_or_else(|| Path::new("")) - .display() - ) - })? - }; + + let config = get_or_create_config(args.general.config_location.as_deref())?; // Create the "app" and initialize a bunch of stuff. let (mut app, widget_layout, styling) = init_app(args, config)?; diff --git a/src/options.rs b/src/options.rs index b15003292..b770b3474 100644 --- a/src/options.rs +++ b/src/options.rs @@ -68,7 +68,7 @@ macro_rules! is_flag_enabled { /// /// For more details on this, see [dirs](https://docs.rs/dirs/latest/dirs/fn.config_dir.html)' /// documentation. -pub fn get_config_path(override_config_path: Option<&Path>) -> Option { +fn get_config_path(override_config_path: Option<&Path>) -> Option { const DEFAULT_CONFIG_FILE_PATH: &str = "bottom/bottom.toml"; if let Some(conf_loc) = override_config_path { @@ -92,29 +92,73 @@ pub fn get_config_path(override_config_path: Option<&Path>) -> Option { }) } +fn create_config_at_path(path: &Path) -> anyhow::Result { + if let Some(parent_path) = path.parent() { + fs::create_dir_all(parent_path)?; + } + + let mut file = fs::File::create(path)?; + file.write_all(CONFIG_TEXT.as_bytes())?; + + Ok(Config::default()) +} + /// Get the config at `config_path`. If there is no config file at the specified /// path, it will try to create a new file with the default settings, and return -/// the default config. If bottom fails to write a new config, it will silently -/// just return the default config. -pub fn get_or_create_config(config_path: Option<&Path>) -> OptionResult { - match &config_path { +/// the default config. +/// +/// We're going to use the following behaviour on when we'll return an error rather +/// than just "silently" continuing on: +/// - If the user passed in a path explicitly, then we will be loud and error out. +/// - If the user does NOT pass in a path explicitly, then just show a warning, +/// but continue. This is in case they do not want to write a default config file at +/// the XDG locations, for example. +pub fn get_or_create_config(config_path: Option<&Path>) -> anyhow::Result { + let adjusted_config_path = get_config_path(config_path); + + match &adjusted_config_path { Some(path) => { if let Ok(config_string) = fs::read_to_string(path) { Ok(toml_edit::de::from_str(&config_string)?) } else { - if let Some(parent_path) = path.parent() { - fs::create_dir_all(parent_path)?; - } + match create_config_at_path(path) { + Ok(cfg) => Ok(cfg), + Err(err) => { + if config_path.is_some() { + Err(err.context(format!( + "bottom could not create a new config file at '{}'.", + path.display() + ))) + } else { + indoc::eprintdoc!( + "Note: bottom couldn't create a default config file at '{}', and the \ + application has fallen back to the default configuration. + + Caused by: + {err} + ", + path.display() + ); - fs::File::create(path)?.write_all(CONFIG_TEXT.as_bytes())?; - Ok(Config::default()) + Ok(Config::default()) + } + } + } } } None => { // If we somehow don't have any config path, then just assume the default config // but don't write to any file. // - // TODO: Maybe make this "show" an error, but don't crash. + // TODO: For now, just print a message to stderr indicating this. In the future, + // probably show in-app (too). + + eprintln!( + "Note: bottom couldn't find a location to create or read a config file, so \ + the application has fallen back to the default configuration. \ + This could be for a variety of reasons, such as issues with file permissions." + ); + Ok(Config::default()) } }