2022-02-20 14:27:59 +00:00
|
|
|
use crate::{ShellError, Span, Value};
|
2021-11-20 13:12:35 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use std::collections::HashMap;
|
2021-11-14 19:25:57 +00:00
|
|
|
|
2021-12-31 22:41:29 +00:00
|
|
|
const ANIMATE_PROMPT_DEFAULT: bool = true;
|
2021-12-02 07:10:40 +00:00
|
|
|
|
2022-01-18 08:48:28 +00:00
|
|
|
/// Definition of a parsed keybinding from the config object
|
|
|
|
#[derive(Serialize, Deserialize, Clone, Debug)]
|
|
|
|
pub struct ParsedKeybinding {
|
2022-01-19 13:28:08 +00:00
|
|
|
pub modifier: Value,
|
|
|
|
pub keycode: Value,
|
|
|
|
pub event: Value,
|
|
|
|
pub mode: Value,
|
2022-01-18 08:48:28 +00:00
|
|
|
}
|
|
|
|
|
2022-04-04 14:54:48 +00:00
|
|
|
/// Definition of a parsed menu from the config object
|
|
|
|
#[derive(Serialize, Deserialize, Clone, Debug)]
|
|
|
|
pub struct ParsedMenu {
|
|
|
|
pub name: Value,
|
|
|
|
pub marker: Value,
|
|
|
|
pub only_buffer_difference: Value,
|
|
|
|
pub style: Value,
|
|
|
|
pub menu_type: Value,
|
|
|
|
pub source: Value,
|
|
|
|
}
|
|
|
|
|
2021-11-14 19:25:57 +00:00
|
|
|
#[derive(Serialize, Deserialize, Clone, Debug)]
|
|
|
|
pub struct Config {
|
|
|
|
pub filesize_metric: bool,
|
|
|
|
pub table_mode: String,
|
2021-11-15 20:09:17 +00:00
|
|
|
pub use_ls_colors: bool,
|
2022-01-15 17:01:44 +00:00
|
|
|
pub color_config: HashMap<String, Value>,
|
2021-11-29 20:37:09 +00:00
|
|
|
pub use_grid_icons: bool,
|
2021-12-01 19:20:23 +00:00
|
|
|
pub footer_mode: FooterMode,
|
2021-12-02 07:10:40 +00:00
|
|
|
pub animate_prompt: bool,
|
2021-12-07 20:06:14 +00:00
|
|
|
pub float_precision: i64,
|
2021-12-09 19:19:36 +00:00
|
|
|
pub filesize_format: String,
|
2021-12-09 22:06:26 +00:00
|
|
|
pub use_ansi_coloring: bool,
|
2022-02-04 15:30:21 +00:00
|
|
|
pub quick_completions: bool,
|
2022-03-03 09:13:44 +00:00
|
|
|
pub partial_completions: bool,
|
2022-04-24 21:43:18 +00:00
|
|
|
pub completion_algorithm: String,
|
2021-12-23 09:31:16 +00:00
|
|
|
pub edit_mode: String,
|
2021-12-23 19:59:00 +00:00
|
|
|
pub max_history_size: i64,
|
2022-03-31 21:25:48 +00:00
|
|
|
pub sync_history_on_enter: bool,
|
2022-01-01 21:42:50 +00:00
|
|
|
pub log_level: String,
|
2022-01-18 08:48:28 +00:00
|
|
|
pub keybindings: Vec<ParsedKeybinding>,
|
2022-04-04 14:54:48 +00:00
|
|
|
pub menus: Vec<ParsedMenu>,
|
2022-02-09 14:56:27 +00:00
|
|
|
pub rm_always_trash: bool,
|
2022-04-24 00:53:12 +00:00
|
|
|
pub shell_integration: bool,
|
2022-04-30 14:40:41 +00:00
|
|
|
pub buffer_editor: String,
|
2022-04-30 14:07:46 +00:00
|
|
|
pub disable_table_indexes: bool,
|
2022-05-06 12:58:32 +00:00
|
|
|
pub cd_with_abbreviations: bool,
|
2021-11-14 19:25:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Config {
|
|
|
|
fn default() -> Config {
|
|
|
|
Config {
|
|
|
|
filesize_metric: false,
|
|
|
|
table_mode: "rounded".into(),
|
2021-11-15 20:09:17 +00:00
|
|
|
use_ls_colors: true,
|
2021-11-20 13:12:35 +00:00
|
|
|
color_config: HashMap::new(),
|
2021-11-29 20:37:09 +00:00
|
|
|
use_grid_icons: false,
|
2022-02-20 20:20:41 +00:00
|
|
|
footer_mode: FooterMode::RowCount(25),
|
2021-12-02 07:10:40 +00:00
|
|
|
animate_prompt: ANIMATE_PROMPT_DEFAULT,
|
2021-12-07 20:06:14 +00:00
|
|
|
float_precision: 4,
|
2021-12-09 19:19:36 +00:00
|
|
|
filesize_format: "auto".into(),
|
2021-12-09 22:06:26 +00:00
|
|
|
use_ansi_coloring: true,
|
2022-02-20 20:20:41 +00:00
|
|
|
quick_completions: true,
|
2022-03-03 09:13:44 +00:00
|
|
|
partial_completions: true,
|
2022-04-24 21:43:18 +00:00
|
|
|
completion_algorithm: "prefix".into(),
|
2021-12-23 09:31:16 +00:00
|
|
|
edit_mode: "emacs".into(),
|
2021-12-23 19:59:00 +00:00
|
|
|
max_history_size: 1000,
|
2022-03-31 21:25:48 +00:00
|
|
|
sync_history_on_enter: true,
|
2022-01-01 21:42:50 +00:00
|
|
|
log_level: String::new(),
|
2022-03-27 13:01:04 +00:00
|
|
|
keybindings: Vec::new(),
|
2022-04-04 14:54:48 +00:00
|
|
|
menus: Vec::new(),
|
2022-02-09 14:56:27 +00:00
|
|
|
rm_always_trash: false,
|
2022-04-24 00:53:12 +00:00
|
|
|
shell_integration: false,
|
2022-04-30 14:40:41 +00:00
|
|
|
buffer_editor: String::new(),
|
2022-04-30 14:07:46 +00:00
|
|
|
disable_table_indexes: false,
|
2022-05-06 12:58:32 +00:00
|
|
|
cd_with_abbreviations: false,
|
2021-11-14 19:25:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-01 19:20:23 +00:00
|
|
|
#[derive(Serialize, Deserialize, Clone, Debug)]
|
|
|
|
pub enum FooterMode {
|
|
|
|
/// Never show the footer
|
|
|
|
Never,
|
|
|
|
/// Always show the footer
|
|
|
|
Always,
|
|
|
|
/// Only show the footer if there are more than RowCount rows
|
|
|
|
RowCount(u64),
|
|
|
|
/// Calculate the screen height, calculate row count, if display will be bigger than screen, add the footer
|
|
|
|
Auto,
|
|
|
|
}
|
|
|
|
|
2021-11-14 19:25:57 +00:00
|
|
|
impl Value {
|
|
|
|
pub fn into_config(self) -> Result<Config, ShellError> {
|
2022-01-19 16:42:12 +00:00
|
|
|
let v = self.as_record();
|
2021-11-14 19:25:57 +00:00
|
|
|
|
|
|
|
let mut config = Config::default();
|
|
|
|
|
2022-01-19 16:42:12 +00:00
|
|
|
if let Ok(v) = v {
|
|
|
|
for (key, value) in v.0.iter().zip(v.1) {
|
|
|
|
match key.as_str() {
|
|
|
|
"filesize_metric" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.filesize_metric = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.filesize_metric is not a bool")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"table_mode" => {
|
|
|
|
if let Ok(v) = value.as_string() {
|
|
|
|
config.table_mode = v;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.table_mode is not a string")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"use_ls_colors" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.use_ls_colors = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.use_ls_colors is not a bool")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"color_config" => {
|
|
|
|
if let Ok(map) = create_map(value, &config) {
|
|
|
|
config.color_config = map;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.color_config is not a record")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"use_grid_icons" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.use_grid_icons = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.use_grid_icons is not a bool")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"footer_mode" => {
|
|
|
|
if let Ok(b) = value.as_string() {
|
|
|
|
let val_str = b.to_lowercase();
|
|
|
|
config.footer_mode = match val_str.as_ref() {
|
|
|
|
"auto" => FooterMode::Auto,
|
|
|
|
"never" => FooterMode::Never,
|
|
|
|
"always" => FooterMode::Always,
|
|
|
|
_ => match &val_str.parse::<u64>() {
|
|
|
|
Ok(number) => FooterMode::RowCount(*number),
|
|
|
|
_ => FooterMode::Never,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.footer_mode is not a string")
|
|
|
|
}
|
2021-12-17 01:04:54 +00:00
|
|
|
}
|
2022-01-19 16:42:12 +00:00
|
|
|
"animate_prompt" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.animate_prompt = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.animate_prompt is not a bool")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"float_precision" => {
|
|
|
|
if let Ok(i) = value.as_integer() {
|
|
|
|
config.float_precision = i;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.float_precision is not an integer")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"use_ansi_coloring" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.use_ansi_coloring = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.use_ansi_coloring is not a bool")
|
|
|
|
}
|
|
|
|
}
|
2022-02-04 15:30:21 +00:00
|
|
|
"quick_completions" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.quick_completions = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.quick_completions is not a bool")
|
|
|
|
}
|
|
|
|
}
|
2022-03-03 09:13:44 +00:00
|
|
|
"partial_completions" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.partial_completions = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.partial_completions is not a bool")
|
|
|
|
}
|
|
|
|
}
|
2022-04-24 21:43:18 +00:00
|
|
|
"completion_algorithm" => {
|
|
|
|
if let Ok(v) = value.as_string() {
|
|
|
|
config.completion_algorithm = v.to_lowercase();
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.completion_algorithm is not a string")
|
|
|
|
}
|
|
|
|
}
|
2022-02-09 14:56:27 +00:00
|
|
|
"rm_always_trash" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.rm_always_trash = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.rm_always_trash is not a bool")
|
|
|
|
}
|
|
|
|
}
|
2022-01-19 16:42:12 +00:00
|
|
|
"filesize_format" => {
|
|
|
|
if let Ok(v) = value.as_string() {
|
|
|
|
config.filesize_format = v.to_lowercase();
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.filesize_format is not a string")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"edit_mode" => {
|
|
|
|
if let Ok(v) = value.as_string() {
|
|
|
|
config.edit_mode = v.to_lowercase();
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.edit_mode is not a string")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"max_history_size" => {
|
|
|
|
if let Ok(i) = value.as_i64() {
|
|
|
|
config.max_history_size = i;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.max_history_size is not an integer")
|
|
|
|
}
|
|
|
|
}
|
2022-03-31 21:25:48 +00:00
|
|
|
"sync_history_on_enter" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.sync_history_on_enter = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.sync_history_on_enter is not a bool")
|
|
|
|
}
|
|
|
|
}
|
2022-01-19 16:42:12 +00:00
|
|
|
"log_level" => {
|
|
|
|
if let Ok(v) = value.as_string() {
|
|
|
|
config.log_level = v.to_lowercase();
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.log_level is not a string")
|
|
|
|
}
|
|
|
|
}
|
2022-04-04 14:54:48 +00:00
|
|
|
"menus" => match create_menus(value, &config) {
|
|
|
|
Ok(map) => config.menus = map,
|
|
|
|
Err(e) => {
|
|
|
|
eprintln!("$config.menus is not a valid list of menus");
|
|
|
|
eprintln!("{:?}", e);
|
2022-01-19 16:42:12 +00:00
|
|
|
}
|
2022-04-04 14:54:48 +00:00
|
|
|
},
|
|
|
|
"keybindings" => match create_keybindings(value, &config) {
|
|
|
|
Ok(keybindings) => config.keybindings = keybindings,
|
|
|
|
Err(e) => {
|
|
|
|
eprintln!("$config.keybindings is not a valid keybindings list");
|
|
|
|
eprintln!("{:?}", e);
|
2022-01-25 09:39:22 +00:00
|
|
|
}
|
2022-04-04 14:54:48 +00:00
|
|
|
},
|
2022-04-24 00:53:12 +00:00
|
|
|
"shell_integration" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.shell_integration = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.shell_integration is not a bool")
|
|
|
|
}
|
|
|
|
}
|
2022-04-30 14:40:41 +00:00
|
|
|
"buffer_editor" => {
|
|
|
|
if let Ok(v) = value.as_string() {
|
|
|
|
config.buffer_editor = v.to_lowercase();
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.buffer_editor is not a string")
|
|
|
|
}
|
|
|
|
}
|
2022-04-30 14:07:46 +00:00
|
|
|
"disable_table_indexes" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.disable_table_indexes = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.disable_table_indexes is not a bool")
|
2022-05-06 12:58:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
"cd_with_abbreviations" => {
|
|
|
|
if let Ok(b) = value.as_bool() {
|
|
|
|
config.cd_with_abbreviations = b;
|
|
|
|
} else {
|
|
|
|
eprintln!("$config.disable_table_indexes is not a bool")
|
2022-04-30 14:07:46 +00:00
|
|
|
}
|
|
|
|
}
|
2022-01-19 16:42:12 +00:00
|
|
|
x => {
|
|
|
|
eprintln!("$config.{} is an unknown config setting", x)
|
|
|
|
}
|
2022-01-15 17:01:44 +00:00
|
|
|
}
|
2021-11-14 19:25:57 +00:00
|
|
|
}
|
2022-01-19 16:42:12 +00:00
|
|
|
} else {
|
|
|
|
eprintln!("$config is not a record");
|
2021-11-14 19:25:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(config)
|
|
|
|
}
|
|
|
|
}
|
2022-01-15 17:01:44 +00:00
|
|
|
|
|
|
|
fn create_map(value: &Value, config: &Config) -> Result<HashMap<String, Value>, ShellError> {
|
|
|
|
let (cols, inner_vals) = value.as_record()?;
|
|
|
|
let mut hm: HashMap<String, Value> = HashMap::new();
|
|
|
|
|
|
|
|
for (k, v) in cols.iter().zip(inner_vals) {
|
|
|
|
match &v {
|
|
|
|
Value::Record {
|
|
|
|
cols: inner_cols,
|
|
|
|
vals: inner_vals,
|
|
|
|
span,
|
|
|
|
} => {
|
2022-01-18 08:48:28 +00:00
|
|
|
let val = color_value_string(span, inner_cols, inner_vals, config);
|
|
|
|
hm.insert(k.to_string(), val);
|
2022-01-15 17:01:44 +00:00
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
hm.insert(k.to_string(), v.clone());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(hm)
|
|
|
|
}
|
2022-01-18 08:48:28 +00:00
|
|
|
|
2022-04-04 21:36:48 +00:00
|
|
|
pub fn color_value_string(
|
2022-01-18 08:48:28 +00:00
|
|
|
span: &Span,
|
|
|
|
inner_cols: &[String],
|
|
|
|
inner_vals: &[Value],
|
|
|
|
config: &Config,
|
|
|
|
) -> Value {
|
|
|
|
// make a string from our config.color_config section that
|
|
|
|
// looks like this: { fg: "#rrggbb" bg: "#rrggbb" attr: "abc", }
|
|
|
|
// the real key here was to have quotes around the values but not
|
|
|
|
// require them around the keys.
|
|
|
|
|
|
|
|
// maybe there's a better way to generate this but i'm not sure
|
|
|
|
// what it is.
|
|
|
|
let val: String = inner_cols
|
|
|
|
.iter()
|
|
|
|
.zip(inner_vals)
|
2022-01-19 13:28:08 +00:00
|
|
|
.map(|(x, y)| format!("{}: \"{}\" ", x, y.into_string(", ", config)))
|
2022-01-18 08:48:28 +00:00
|
|
|
.collect();
|
|
|
|
|
|
|
|
// now insert the braces at the front and the back to fake the json string
|
|
|
|
Value::String {
|
|
|
|
val: format!("{{{}}}", val),
|
|
|
|
span: *span,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parses the config object to extract the strings that will compose a keybinding for reedline
|
|
|
|
fn create_keybindings(value: &Value, config: &Config) -> Result<Vec<ParsedKeybinding>, ShellError> {
|
|
|
|
match value {
|
2022-01-19 13:28:08 +00:00
|
|
|
Value::Record { cols, vals, span } => {
|
|
|
|
// Finding the modifier value in the record
|
2022-04-04 14:54:48 +00:00
|
|
|
let modifier = extract_value("modifier", cols, vals, span)?.clone();
|
|
|
|
let keycode = extract_value("keycode", cols, vals, span)?.clone();
|
|
|
|
let mode = extract_value("mode", cols, vals, span)?.clone();
|
|
|
|
let event = extract_value("event", cols, vals, span)?.clone();
|
2022-01-19 13:28:08 +00:00
|
|
|
|
|
|
|
let keybinding = ParsedKeybinding {
|
2022-04-04 14:54:48 +00:00
|
|
|
modifier,
|
|
|
|
keycode,
|
|
|
|
mode,
|
|
|
|
event,
|
2022-01-19 13:28:08 +00:00
|
|
|
};
|
2022-01-18 08:48:28 +00:00
|
|
|
|
2022-04-04 14:54:48 +00:00
|
|
|
// We return a menu to be able to do recursion on the same function
|
2022-01-18 08:48:28 +00:00
|
|
|
Ok(vec![keybinding])
|
|
|
|
}
|
|
|
|
Value::List { vals, .. } => {
|
|
|
|
let res = vals
|
|
|
|
.iter()
|
|
|
|
.map(|inner_value| create_keybindings(inner_value, config))
|
|
|
|
.collect::<Result<Vec<Vec<ParsedKeybinding>>, ShellError>>();
|
|
|
|
|
|
|
|
let res = res?
|
|
|
|
.into_iter()
|
|
|
|
.flatten()
|
|
|
|
.collect::<Vec<ParsedKeybinding>>();
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
_ => Ok(Vec::new()),
|
|
|
|
}
|
|
|
|
}
|
2022-01-19 13:28:08 +00:00
|
|
|
|
2022-04-04 14:54:48 +00:00
|
|
|
// Parses the config object to extract the strings that will compose a keybinding for reedline
|
|
|
|
pub fn create_menus(value: &Value, config: &Config) -> Result<Vec<ParsedMenu>, ShellError> {
|
|
|
|
match value {
|
|
|
|
Value::Record { cols, vals, span } => {
|
|
|
|
// Finding the modifier value in the record
|
|
|
|
let name = extract_value("name", cols, vals, span)?.clone();
|
|
|
|
let marker = extract_value("marker", cols, vals, span)?.clone();
|
|
|
|
let only_buffer_difference =
|
|
|
|
extract_value("only_buffer_difference", cols, vals, span)?.clone();
|
|
|
|
let style = extract_value("style", cols, vals, span)?.clone();
|
|
|
|
let menu_type = extract_value("type", cols, vals, span)?.clone();
|
|
|
|
|
|
|
|
// Source is an optional value
|
|
|
|
let source = match extract_value("source", cols, vals, span) {
|
|
|
|
Ok(source) => source.clone(),
|
|
|
|
Err(_) => Value::Nothing { span: *span },
|
|
|
|
};
|
|
|
|
|
|
|
|
let menu = ParsedMenu {
|
|
|
|
name,
|
|
|
|
only_buffer_difference,
|
|
|
|
marker,
|
|
|
|
style,
|
|
|
|
menu_type,
|
|
|
|
source,
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(vec![menu])
|
|
|
|
}
|
|
|
|
Value::List { vals, .. } => {
|
|
|
|
let res = vals
|
|
|
|
.iter()
|
|
|
|
.map(|inner_value| create_menus(inner_value, config))
|
|
|
|
.collect::<Result<Vec<Vec<ParsedMenu>>, ShellError>>();
|
|
|
|
|
|
|
|
let res = res?.into_iter().flatten().collect::<Vec<ParsedMenu>>();
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
_ => Ok(Vec::new()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-19 13:28:08 +00:00
|
|
|
pub fn extract_value<'record>(
|
|
|
|
name: &str,
|
|
|
|
cols: &'record [String],
|
|
|
|
vals: &'record [Value],
|
|
|
|
span: &Span,
|
|
|
|
) -> Result<&'record Value, ShellError> {
|
|
|
|
cols.iter()
|
|
|
|
.position(|col| col.as_str() == name)
|
|
|
|
.and_then(|index| vals.get(index))
|
|
|
|
.ok_or_else(|| ShellError::MissingConfigValue(name.to_string(), *span))
|
|
|
|
}
|