2019-09-30 12:10:35 +00:00
|
|
|
use crate::configs::StarshipRootConfig;
|
|
|
|
use crate::utils;
|
|
|
|
use ansi_term::{Color, Style};
|
2020-10-13 19:06:41 +00:00
|
|
|
use indexmap::IndexMap;
|
2021-03-31 15:31:55 +00:00
|
|
|
use serde::Serialize;
|
2019-09-30 12:10:35 +00:00
|
|
|
|
|
|
|
use std::clone::Clone;
|
2019-12-20 17:30:47 +00:00
|
|
|
use std::collections::HashMap;
|
2020-12-13 15:24:48 +00:00
|
|
|
use std::io::ErrorKind;
|
2019-09-30 12:10:35 +00:00
|
|
|
use std::marker::Sized;
|
|
|
|
|
|
|
|
use std::env;
|
|
|
|
use toml::Value;
|
|
|
|
|
|
|
|
/// Root config of a module.
|
|
|
|
pub trait RootModuleConfig<'a>
|
|
|
|
where
|
2021-03-15 10:40:52 +00:00
|
|
|
Self: ModuleConfig<'a> + Default,
|
2019-09-30 12:10:35 +00:00
|
|
|
{
|
|
|
|
/// Load root module config from given Value and fill unset variables with default
|
|
|
|
/// values.
|
|
|
|
fn load(config: &'a Value) -> Self {
|
2021-03-31 18:13:23 +00:00
|
|
|
let mut out = Self::default();
|
|
|
|
out.load_config(config);
|
|
|
|
out
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Helper function that will call RootModuleConfig::load(config) if config is Some,
|
|
|
|
/// or RootModuleConfig::new() if config is None.
|
|
|
|
fn try_load(config: Option<&'a Value>) -> Self {
|
|
|
|
if let Some(config) = config {
|
|
|
|
Self::load(config)
|
|
|
|
} else {
|
2021-03-15 10:40:52 +00:00
|
|
|
Self::default()
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 10:40:52 +00:00
|
|
|
impl<'a, T: ModuleConfig<'a> + Default> RootModuleConfig<'a> for T {}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
/// Parsable config.
|
|
|
|
pub trait ModuleConfig<'a>
|
|
|
|
where
|
|
|
|
Self: Sized + Clone,
|
|
|
|
{
|
|
|
|
/// Construct a `ModuleConfig` from a toml value.
|
|
|
|
fn from_config(_config: &'a Value) -> Option<Self> {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Merge `self` with config from a toml table.
|
2021-03-31 18:13:23 +00:00
|
|
|
fn load_config(&mut self, config: &'a Value) {
|
|
|
|
if let Some(value) = Self::from_config(config) {
|
|
|
|
let _ = std::mem::replace(self, value);
|
|
|
|
}
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Add logging to default implementations
|
|
|
|
impl<'a> ModuleConfig<'a> for &'a str {
|
|
|
|
fn from_config(config: &'a Value) -> Option<Self> {
|
|
|
|
config.as_str()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> ModuleConfig<'a> for Style {
|
|
|
|
fn from_config(config: &Value) -> Option<Self> {
|
|
|
|
parse_style_string(config.as_str()?)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> ModuleConfig<'a> for bool {
|
|
|
|
fn from_config(config: &Value) -> Option<Self> {
|
|
|
|
config.as_bool()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> ModuleConfig<'a> for i64 {
|
|
|
|
fn from_config(config: &Value) -> Option<Self> {
|
|
|
|
config.as_integer()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-28 13:41:16 +00:00
|
|
|
impl<'a> ModuleConfig<'a> for u64 {
|
|
|
|
fn from_config(config: &Value) -> Option<Self> {
|
|
|
|
match config {
|
|
|
|
Value::Integer(value) => {
|
|
|
|
// Converting i64 to u64
|
|
|
|
if *value > 0 {
|
|
|
|
Some(*value as u64)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Value::String(value) => value.parse::<u64>().ok(),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
impl<'a> ModuleConfig<'a> for f64 {
|
|
|
|
fn from_config(config: &Value) -> Option<Self> {
|
|
|
|
config.as_float()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-06 16:57:42 +00:00
|
|
|
impl<'a> ModuleConfig<'a> for usize {
|
|
|
|
fn from_config(config: &Value) -> Option<Self> {
|
|
|
|
match config {
|
|
|
|
Value::Integer(value) => {
|
|
|
|
if *value > 0 {
|
|
|
|
Some(*value as usize)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Value::String(value) => value.parse::<usize>().ok(),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
impl<'a, T> ModuleConfig<'a> for Vec<T>
|
|
|
|
where
|
|
|
|
T: ModuleConfig<'a>,
|
|
|
|
{
|
|
|
|
fn from_config(config: &'a Value) -> Option<Self> {
|
|
|
|
config
|
|
|
|
.as_array()?
|
|
|
|
.iter()
|
|
|
|
.map(|value| T::from_config(value))
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 17:30:47 +00:00
|
|
|
impl<'a, T, S: ::std::hash::BuildHasher + Default> ModuleConfig<'a> for HashMap<String, T, S>
|
|
|
|
where
|
|
|
|
T: ModuleConfig<'a>,
|
|
|
|
S: Clone,
|
|
|
|
{
|
|
|
|
fn from_config(config: &'a Value) -> Option<Self> {
|
|
|
|
let mut hm = HashMap::default();
|
|
|
|
|
|
|
|
for (x, y) in config.as_table()?.iter() {
|
|
|
|
hm.insert(x.clone(), T::from_config(y)?);
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(hm)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-13 19:06:41 +00:00
|
|
|
impl<'a, T, S: ::std::hash::BuildHasher + Default> ModuleConfig<'a> for IndexMap<String, T, S>
|
|
|
|
where
|
|
|
|
T: ModuleConfig<'a>,
|
|
|
|
S: Clone,
|
|
|
|
{
|
|
|
|
fn from_config(config: &'a Value) -> Option<Self> {
|
|
|
|
let mut im = IndexMap::default();
|
|
|
|
|
|
|
|
for (x, y) in config.as_table()?.iter() {
|
|
|
|
im.insert(x.clone(), T::from_config(y)?);
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(im)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
impl<'a, T> ModuleConfig<'a> for Option<T>
|
|
|
|
where
|
|
|
|
T: ModuleConfig<'a> + Sized,
|
|
|
|
{
|
|
|
|
fn from_config(config: &'a Value) -> Option<Self> {
|
|
|
|
Some(T::from_config(config))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-27 07:38:05 +00:00
|
|
|
/// A wrapper around `Vec<T>` that implements `ModuleConfig`, and either
|
|
|
|
/// accepts a value of type `T` or a list of values of type `T`.
|
2021-03-31 15:31:55 +00:00
|
|
|
#[derive(Clone, Default, Serialize)]
|
2020-05-27 07:38:05 +00:00
|
|
|
pub struct VecOr<T>(pub Vec<T>);
|
|
|
|
|
|
|
|
impl<'a, T> ModuleConfig<'a> for VecOr<T>
|
|
|
|
where
|
|
|
|
T: ModuleConfig<'a> + Sized,
|
|
|
|
{
|
|
|
|
fn from_config(config: &'a Value) -> Option<Self> {
|
|
|
|
if let Some(item) = T::from_config(config) {
|
|
|
|
return Some(VecOr(vec![item]));
|
|
|
|
}
|
|
|
|
|
|
|
|
let vec = config
|
|
|
|
.as_array()?
|
|
|
|
.iter()
|
|
|
|
.map(|value| T::from_config(value))
|
|
|
|
.collect::<Option<Vec<T>>>()?;
|
|
|
|
|
|
|
|
Some(VecOr(vec))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
/// Root config of starship.
|
|
|
|
pub struct StarshipConfig {
|
|
|
|
pub config: Option<Value>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl StarshipConfig {
|
|
|
|
/// Initialize the Config struct
|
|
|
|
pub fn initialize() -> Self {
|
|
|
|
if let Some(file_data) = Self::config_from_file() {
|
|
|
|
StarshipConfig {
|
|
|
|
config: Some(file_data),
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
StarshipConfig {
|
|
|
|
config: Some(Value::Table(toml::value::Table::new())),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a config from a starship configuration file
|
|
|
|
fn config_from_file() -> Option<Value> {
|
|
|
|
let file_path = if let Ok(path) = env::var("STARSHIP_CONFIG") {
|
|
|
|
// Use $STARSHIP_CONFIG as the config path if available
|
2020-09-28 20:38:50 +00:00
|
|
|
log::debug!("STARSHIP_CONFIG is set: {}", &path);
|
2019-09-30 12:10:35 +00:00
|
|
|
path
|
|
|
|
} else {
|
|
|
|
// Default to using ~/.config/starship.toml
|
|
|
|
log::debug!("STARSHIP_CONFIG is not set");
|
2020-06-20 17:59:35 +00:00
|
|
|
let config_path = dirs_next::home_dir()?.join(".config/starship.toml");
|
2019-09-30 12:10:35 +00:00
|
|
|
let config_path_str = config_path.to_str()?.to_owned();
|
|
|
|
log::debug!("Using default config path: {}", config_path_str);
|
|
|
|
config_path_str
|
|
|
|
};
|
|
|
|
|
|
|
|
let toml_content = match utils::read_file(&file_path) {
|
|
|
|
Ok(content) => {
|
2020-09-28 20:38:50 +00:00
|
|
|
log::trace!("Config file content: \"\n{}\"", &content);
|
2019-09-30 12:10:35 +00:00
|
|
|
Some(content)
|
|
|
|
}
|
|
|
|
Err(e) => {
|
2020-12-13 15:24:48 +00:00
|
|
|
let level = if e.kind() == ErrorKind::NotFound {
|
|
|
|
log::Level::Debug
|
|
|
|
} else {
|
|
|
|
log::Level::Error
|
|
|
|
};
|
|
|
|
|
|
|
|
log::log!(level, "Unable to read config file content: {}", &e);
|
2019-09-30 12:10:35 +00:00
|
|
|
None
|
|
|
|
}
|
|
|
|
}?;
|
|
|
|
|
2020-06-24 20:11:05 +00:00
|
|
|
match toml::from_str(&toml_content) {
|
|
|
|
Ok(parsed) => {
|
2020-09-28 20:38:50 +00:00
|
|
|
log::debug!("Config parsed: {:?}", &parsed);
|
2020-06-24 20:11:05 +00:00
|
|
|
Some(parsed)
|
|
|
|
}
|
|
|
|
Err(error) => {
|
2020-09-28 20:38:50 +00:00
|
|
|
log::error!("Unable to parse the config file: {}", error);
|
2020-06-24 20:11:05 +00:00
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the subset of the table for a module by its name
|
|
|
|
pub fn get_module_config(&self, module_name: &str) -> Option<&Value> {
|
2020-07-07 22:45:32 +00:00
|
|
|
let module_config = self.get_config(&[module_name]);
|
2019-09-30 12:10:35 +00:00
|
|
|
if module_config.is_some() {
|
|
|
|
log::debug!(
|
2020-09-28 20:38:50 +00:00
|
|
|
"Config found for \"{}\": {:?}",
|
2019-09-30 12:10:35 +00:00
|
|
|
&module_name,
|
|
|
|
&module_config
|
|
|
|
);
|
|
|
|
}
|
|
|
|
module_config
|
|
|
|
}
|
|
|
|
|
2020-07-07 22:45:32 +00:00
|
|
|
/// Get the value of the config in a specific path
|
|
|
|
pub fn get_config(&self, path: &[&str]) -> Option<&Value> {
|
|
|
|
let mut prev_table = self.config.as_ref()?.as_table()?;
|
|
|
|
|
|
|
|
assert_ne!(
|
|
|
|
path.len(),
|
|
|
|
0,
|
|
|
|
"Starship::get_config called with an empty path"
|
|
|
|
);
|
|
|
|
|
|
|
|
let (table_options, _) = path.split_at(path.len() - 1);
|
|
|
|
|
|
|
|
// Assumes all keys except the last in path has a table
|
|
|
|
for option in table_options {
|
|
|
|
match prev_table.get(*option) {
|
|
|
|
Some(value) => match value.as_table() {
|
|
|
|
Some(value) => {
|
|
|
|
prev_table = value;
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
log::trace!(
|
|
|
|
"No config found for \"{}\": \"{}\" is not a table",
|
|
|
|
path.join("."),
|
|
|
|
&option
|
|
|
|
);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
None => {
|
|
|
|
log::trace!(
|
|
|
|
"No config found for \"{}\": Option \"{}\" not found",
|
|
|
|
path.join("."),
|
|
|
|
&option
|
|
|
|
);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let last_option = path.last().unwrap();
|
|
|
|
let value = prev_table.get(*last_option);
|
|
|
|
if value.is_none() {
|
|
|
|
log::trace!(
|
|
|
|
"No config found for \"{}\": Option \"{}\" not found",
|
|
|
|
path.join("."),
|
|
|
|
&last_option
|
|
|
|
);
|
|
|
|
};
|
|
|
|
value
|
|
|
|
}
|
|
|
|
|
2020-04-11 16:37:24 +00:00
|
|
|
/// Get the subset of the table for a custom module by its name
|
|
|
|
pub fn get_custom_module_config(&self, module_name: &str) -> Option<&Value> {
|
2020-07-07 22:45:32 +00:00
|
|
|
let module_config = self.get_config(&["custom", module_name]);
|
2020-04-11 16:37:24 +00:00
|
|
|
if module_config.is_some() {
|
|
|
|
log::debug!(
|
2020-09-28 20:38:50 +00:00
|
|
|
"Custom config found for \"{}\": {:?}",
|
2020-04-11 16:37:24 +00:00
|
|
|
&module_name,
|
|
|
|
&module_config
|
|
|
|
);
|
|
|
|
}
|
|
|
|
module_config
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the table of all the registered custom modules, if any
|
|
|
|
pub fn get_custom_modules(&self) -> Option<&toml::value::Table> {
|
2020-07-07 22:45:32 +00:00
|
|
|
self.get_config(&["custom"])?.as_table()
|
2020-04-11 16:37:24 +00:00
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
pub fn get_root_config(&self) -> StarshipRootConfig {
|
|
|
|
if let Some(root_config) = &self.config {
|
|
|
|
StarshipRootConfig::load(root_config)
|
|
|
|
} else {
|
2021-03-15 10:40:52 +00:00
|
|
|
StarshipRootConfig::default()
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Parse a style string which represents an ansi style. Valid tokens in the style
|
|
|
|
string include the following:
|
|
|
|
- 'fg:<color>' (specifies that the color read should be a foreground color)
|
|
|
|
- 'bg:<color>' (specifies that the color read should be a background color)
|
|
|
|
- 'underline'
|
|
|
|
- 'bold'
|
|
|
|
- 'italic'
|
2021-04-22 16:09:21 +00:00
|
|
|
- 'inverted'
|
|
|
|
- '<color>' (see the parse_color_string doc for valid color strings)
|
2019-09-30 12:10:35 +00:00
|
|
|
*/
|
2020-04-06 17:16:18 +00:00
|
|
|
pub fn parse_style_string(style_string: &str) -> Option<ansi_term::Style> {
|
2019-09-30 12:10:35 +00:00
|
|
|
style_string
|
|
|
|
.split_whitespace()
|
|
|
|
.fold(Some(ansi_term::Style::new()), |maybe_style, token| {
|
|
|
|
maybe_style.and_then(|style| {
|
|
|
|
let token = token.to_lowercase();
|
|
|
|
|
|
|
|
// Check for FG/BG identifiers and strip them off if appropriate
|
|
|
|
// If col_fg is true, color the foreground. If it's false, color the background.
|
|
|
|
let (token, col_fg) = if token.as_str().starts_with("fg:") {
|
|
|
|
(token.trim_start_matches("fg:").to_owned(), true)
|
|
|
|
} else if token.as_str().starts_with("bg:") {
|
|
|
|
(token.trim_start_matches("bg:").to_owned(), false)
|
|
|
|
} else {
|
|
|
|
(token, true) // Bare colors are assumed to color the foreground
|
|
|
|
};
|
|
|
|
|
|
|
|
match token.as_str() {
|
|
|
|
"underline" => Some(style.underline()),
|
|
|
|
"bold" => Some(style.bold()),
|
|
|
|
"italic" => Some(style.italic()),
|
|
|
|
"dimmed" => Some(style.dimmed()),
|
2021-04-22 16:09:21 +00:00
|
|
|
"inverted" => Some(style.reverse()),
|
2020-12-02 06:40:15 +00:00
|
|
|
// When the string is supposed to be a color:
|
|
|
|
// Decide if we yield none, reset background or set color.
|
|
|
|
color_string => {
|
|
|
|
if color_string == "none" && col_fg {
|
|
|
|
None // fg:none yields no style.
|
2019-09-30 12:10:35 +00:00
|
|
|
} else {
|
2020-12-02 06:40:15 +00:00
|
|
|
// Either bg or valid color or both.
|
|
|
|
let parsed = parse_color_string(color_string);
|
|
|
|
// bg + invalid color = reset the background to default.
|
|
|
|
if !col_fg && parsed.is_none() {
|
|
|
|
let mut new_style = style;
|
|
|
|
new_style.background = Option::None;
|
|
|
|
Some(new_style)
|
|
|
|
} else {
|
|
|
|
// Valid color, apply color to either bg or fg
|
|
|
|
parsed.map(|ansi_color| {
|
|
|
|
if col_fg {
|
|
|
|
style.fg(ansi_color)
|
|
|
|
} else {
|
|
|
|
style.on(ansi_color)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
2020-12-02 06:40:15 +00:00
|
|
|
}
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Parse a string that represents a color setting, returning None if this fails
|
|
|
|
There are three valid color formats:
|
|
|
|
- #RRGGBB (a hash followed by an RGB hex)
|
|
|
|
- u8 (a number from 0-255, representing an ANSI color)
|
|
|
|
- colstring (one of the 16 predefined color strings)
|
|
|
|
*/
|
|
|
|
fn parse_color_string(color_string: &str) -> Option<ansi_term::Color> {
|
|
|
|
// Parse RGB hex values
|
|
|
|
log::trace!("Parsing color_string: {}", color_string);
|
|
|
|
if color_string.starts_with('#') {
|
|
|
|
log::trace!(
|
|
|
|
"Attempting to read hexadecimal color string: {}",
|
|
|
|
color_string
|
|
|
|
);
|
2020-07-14 21:11:43 +00:00
|
|
|
if color_string.len() != 7 {
|
|
|
|
log::debug!("Could not parse hexadecimal string: {}", color_string);
|
|
|
|
return None;
|
|
|
|
}
|
2019-09-30 12:10:35 +00:00
|
|
|
let r: u8 = u8::from_str_radix(&color_string[1..3], 16).ok()?;
|
|
|
|
let g: u8 = u8::from_str_radix(&color_string[3..5], 16).ok()?;
|
|
|
|
let b: u8 = u8::from_str_radix(&color_string[5..7], 16).ok()?;
|
|
|
|
log::trace!("Read RGB color string: {},{},{}", r, g, b);
|
|
|
|
return Some(Color::RGB(r, g, b));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse a u8 (ansi color)
|
|
|
|
if let Result::Ok(ansi_color_num) = color_string.parse::<u8>() {
|
|
|
|
log::trace!("Read ANSI color string: {}", ansi_color_num);
|
|
|
|
return Some(Color::Fixed(ansi_color_num));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for any predefined color strings
|
|
|
|
// There are no predefined enums for bright colors, so we use Color::Fixed
|
|
|
|
let predefined_color = match color_string.to_lowercase().as_str() {
|
|
|
|
"black" => Some(Color::Black),
|
|
|
|
"red" => Some(Color::Red),
|
|
|
|
"green" => Some(Color::Green),
|
|
|
|
"yellow" => Some(Color::Yellow),
|
|
|
|
"blue" => Some(Color::Blue),
|
|
|
|
"purple" => Some(Color::Purple),
|
|
|
|
"cyan" => Some(Color::Cyan),
|
|
|
|
"white" => Some(Color::White),
|
|
|
|
"bright-black" => Some(Color::Fixed(8)), // "bright-black" is dark grey
|
|
|
|
"bright-red" => Some(Color::Fixed(9)),
|
|
|
|
"bright-green" => Some(Color::Fixed(10)),
|
|
|
|
"bright-yellow" => Some(Color::Fixed(11)),
|
|
|
|
"bright-blue" => Some(Color::Fixed(12)),
|
|
|
|
"bright-purple" => Some(Color::Fixed(13)),
|
|
|
|
"bright-cyan" => Some(Color::Fixed(14)),
|
|
|
|
"bright-white" => Some(Color::Fixed(15)),
|
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
|
|
|
|
if predefined_color.is_some() {
|
|
|
|
log::trace!("Read predefined color: {}", color_string);
|
|
|
|
} else {
|
|
|
|
log::debug!("Could not parse color in string: {}", color_string);
|
|
|
|
}
|
|
|
|
predefined_color
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
use starship_module_config_derive::ModuleConfig;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_load_config() {
|
2021-03-31 15:31:55 +00:00
|
|
|
#[derive(Clone, Default, ModuleConfig)]
|
2019-09-30 12:10:35 +00:00
|
|
|
struct TestConfig<'a> {
|
|
|
|
pub symbol: &'a str,
|
|
|
|
pub disabled: bool,
|
|
|
|
pub some_array: Vec<&'a str>,
|
|
|
|
}
|
|
|
|
|
|
|
|
let config = toml::toml! {
|
|
|
|
symbol = "T "
|
|
|
|
disabled = true
|
|
|
|
some_array = ["A"]
|
|
|
|
};
|
2021-03-31 18:13:23 +00:00
|
|
|
let mut rust_config = TestConfig {
|
2019-09-30 12:10:35 +00:00
|
|
|
symbol: "S ",
|
|
|
|
disabled: false,
|
|
|
|
some_array: vec!["A", "B", "C"],
|
|
|
|
};
|
2021-03-31 18:13:23 +00:00
|
|
|
rust_config.load_config(&config);
|
2019-09-30 12:10:35 +00:00
|
|
|
|
|
|
|
assert_eq!(rust_config.symbol, "T ");
|
|
|
|
assert_eq!(rust_config.disabled, true);
|
|
|
|
assert_eq!(rust_config.some_array, vec!["A"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_load_nested_config() {
|
2021-03-31 15:31:55 +00:00
|
|
|
#[derive(Clone, Default, ModuleConfig)]
|
2019-09-30 12:10:35 +00:00
|
|
|
struct TestConfig<'a> {
|
|
|
|
pub untracked: SegmentDisplayConfig<'a>,
|
|
|
|
pub modified: SegmentDisplayConfig<'a>,
|
|
|
|
}
|
|
|
|
|
2021-03-31 15:31:55 +00:00
|
|
|
#[derive(PartialEq, Debug, Clone, Default, ModuleConfig)]
|
2019-09-30 12:10:35 +00:00
|
|
|
struct SegmentDisplayConfig<'a> {
|
|
|
|
pub value: &'a str,
|
|
|
|
pub style: Style,
|
|
|
|
}
|
|
|
|
|
|
|
|
let config = toml::toml! {
|
|
|
|
untracked.value = "x"
|
|
|
|
modified = { value = "•", style = "red" }
|
|
|
|
};
|
|
|
|
|
2021-03-31 18:13:23 +00:00
|
|
|
let mut git_status_config = TestConfig {
|
2019-09-30 12:10:35 +00:00
|
|
|
untracked: SegmentDisplayConfig {
|
|
|
|
value: "?",
|
|
|
|
style: Color::Red.bold(),
|
|
|
|
},
|
|
|
|
modified: SegmentDisplayConfig {
|
|
|
|
value: "!",
|
|
|
|
style: Color::Red.bold(),
|
|
|
|
},
|
|
|
|
};
|
2021-03-31 18:13:23 +00:00
|
|
|
git_status_config.load_config(&config);
|
2019-09-30 12:10:35 +00:00
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
git_status_config.untracked,
|
|
|
|
SegmentDisplayConfig {
|
|
|
|
value: "x",
|
|
|
|
style: Color::Red.bold(),
|
|
|
|
}
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
git_status_config.modified,
|
|
|
|
SegmentDisplayConfig {
|
|
|
|
value: "•",
|
|
|
|
style: Color::Red.normal(),
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_load_optional_config() {
|
2021-03-31 15:31:55 +00:00
|
|
|
#[derive(Clone, Default, ModuleConfig)]
|
2019-09-30 12:10:35 +00:00
|
|
|
struct TestConfig<'a> {
|
|
|
|
pub optional: Option<&'a str>,
|
|
|
|
pub hidden: Option<&'a str>,
|
|
|
|
}
|
|
|
|
|
|
|
|
let config = toml::toml! {
|
|
|
|
optional = "test"
|
|
|
|
};
|
2021-03-31 18:13:23 +00:00
|
|
|
let mut rust_config = TestConfig {
|
2019-09-30 12:10:35 +00:00
|
|
|
optional: None,
|
|
|
|
hidden: None,
|
|
|
|
};
|
2021-03-31 18:13:23 +00:00
|
|
|
rust_config.load_config(&config);
|
2019-09-30 12:10:35 +00:00
|
|
|
|
|
|
|
assert_eq!(rust_config.optional, Some("test"));
|
|
|
|
assert_eq!(rust_config.hidden, None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_load_enum_config() {
|
2021-03-31 15:31:55 +00:00
|
|
|
#[derive(Clone, Default, ModuleConfig)]
|
2019-09-30 12:10:35 +00:00
|
|
|
struct TestConfig {
|
|
|
|
pub switch_a: Switch,
|
|
|
|
pub switch_b: Switch,
|
|
|
|
pub switch_c: Switch,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, PartialEq, Clone)]
|
|
|
|
enum Switch {
|
2021-03-25 20:03:19 +00:00
|
|
|
On,
|
|
|
|
Off,
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
|
2021-03-31 15:31:55 +00:00
|
|
|
impl Default for Switch {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::Off
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
impl<'a> ModuleConfig<'a> for Switch {
|
|
|
|
fn from_config(config: &'a Value) -> Option<Self> {
|
|
|
|
match config.as_str()? {
|
2021-03-25 20:03:19 +00:00
|
|
|
"on" => Some(Self::On),
|
|
|
|
"off" => Some(Self::Off),
|
2019-09-30 12:10:35 +00:00
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let config = toml::toml! {
|
|
|
|
switch_a = "on"
|
|
|
|
switch_b = "any"
|
|
|
|
};
|
2021-03-31 18:13:23 +00:00
|
|
|
let mut rust_config = TestConfig {
|
2021-03-25 20:03:19 +00:00
|
|
|
switch_a: Switch::Off,
|
|
|
|
switch_b: Switch::Off,
|
|
|
|
switch_c: Switch::Off,
|
2019-09-30 12:10:35 +00:00
|
|
|
};
|
2021-03-31 18:13:23 +00:00
|
|
|
rust_config.load_config(&config);
|
2019-09-30 12:10:35 +00:00
|
|
|
|
2021-03-25 20:03:19 +00:00
|
|
|
assert_eq!(rust_config.switch_a, Switch::On);
|
|
|
|
assert_eq!(rust_config.switch_b, Switch::Off);
|
|
|
|
assert_eq!(rust_config.switch_c, Switch::Off);
|
2019-09-30 12:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_from_string() {
|
|
|
|
let config = Value::String(String::from("S"));
|
|
|
|
assert_eq!(<&str>::from_config(&config).unwrap(), "S");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_from_bool() {
|
|
|
|
let config = Value::Boolean(true);
|
|
|
|
assert_eq!(<bool>::from_config(&config).unwrap(), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_from_i64() {
|
|
|
|
let config = Value::Integer(42);
|
|
|
|
assert_eq!(<i64>::from_config(&config).unwrap(), 42);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_from_style() {
|
|
|
|
let config = Value::from("red bold");
|
|
|
|
assert_eq!(<Style>::from_config(&config).unwrap(), Color::Red.bold());
|
|
|
|
}
|
|
|
|
|
2020-07-14 21:11:43 +00:00
|
|
|
#[test]
|
|
|
|
fn test_from_hex_color_style() {
|
|
|
|
let config = Value::from("#00000");
|
|
|
|
assert_eq!(<Style>::from_config(&config), None);
|
|
|
|
|
|
|
|
let config = Value::from("#0000000");
|
|
|
|
assert_eq!(<Style>::from_config(&config), None);
|
|
|
|
|
|
|
|
let config = Value::from("#NOTHEX");
|
|
|
|
assert_eq!(<Style>::from_config(&config), None);
|
|
|
|
|
|
|
|
let config = Value::from("#a12BcD");
|
|
|
|
assert_eq!(
|
|
|
|
<Style>::from_config(&config).unwrap(),
|
|
|
|
Color::RGB(0xA1, 0x2B, 0xCD).into()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
#[test]
|
|
|
|
fn test_from_vec() {
|
|
|
|
let config: Value = Value::Array(vec![Value::from("S")]);
|
|
|
|
assert_eq!(<Vec<&str>>::from_config(&config).unwrap(), vec!["S"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_from_option() {
|
|
|
|
let config: Value = Value::String(String::from("S"));
|
|
|
|
assert_eq!(<Option<&str>>::from_config(&config).unwrap(), Some("S"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-04-22 16:09:21 +00:00
|
|
|
fn table_get_styles_bold_italic_underline_green_dimmed_silly_caps() {
|
2019-09-30 12:10:35 +00:00
|
|
|
let config = Value::from("bOlD ItAlIc uNdErLiNe GrEeN diMMeD");
|
|
|
|
let mystyle = <Style>::from_config(&config).unwrap();
|
|
|
|
assert!(mystyle.is_bold);
|
|
|
|
assert!(mystyle.is_italic);
|
|
|
|
assert!(mystyle.is_underline);
|
|
|
|
assert!(mystyle.is_dimmed);
|
|
|
|
assert_eq!(
|
|
|
|
mystyle,
|
|
|
|
ansi_term::Style::new()
|
|
|
|
.bold()
|
|
|
|
.italic()
|
|
|
|
.underline()
|
|
|
|
.dimmed()
|
|
|
|
.fg(Color::Green)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-04-22 16:09:21 +00:00
|
|
|
#[test]
|
|
|
|
fn table_get_styles_bold_italic_underline_green_dimmed_inverted_silly_caps() {
|
|
|
|
let config = Value::from("bOlD ItAlIc uNdErLiNe GrEeN diMMeD InVeRTed");
|
|
|
|
let mystyle = <Style>::from_config(&config).unwrap();
|
|
|
|
assert!(mystyle.is_bold);
|
|
|
|
assert!(mystyle.is_italic);
|
|
|
|
assert!(mystyle.is_underline);
|
|
|
|
assert!(mystyle.is_dimmed);
|
|
|
|
assert!(mystyle.is_reverse);
|
|
|
|
assert_eq!(
|
|
|
|
mystyle,
|
|
|
|
ansi_term::Style::new()
|
|
|
|
.bold()
|
|
|
|
.italic()
|
|
|
|
.underline()
|
|
|
|
.dimmed()
|
|
|
|
.reverse()
|
|
|
|
.fg(Color::Green)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
#[test]
|
|
|
|
fn table_get_styles_plain_and_broken_styles() {
|
|
|
|
// Test a "plain" style with no formatting
|
|
|
|
let config = Value::from("");
|
|
|
|
let plain_style = <Style>::from_config(&config).unwrap();
|
|
|
|
assert_eq!(plain_style, ansi_term::Style::new());
|
|
|
|
|
|
|
|
// Test a string that's clearly broken
|
|
|
|
let config = Value::from("djklgfhjkldhlhk;j");
|
|
|
|
assert!(<Style>::from_config(&config).is_none());
|
|
|
|
|
|
|
|
// Test a string that's nullified by `none`
|
|
|
|
let config = Value::from("fg:red bg:green bold none");
|
|
|
|
assert!(<Style>::from_config(&config).is_none());
|
|
|
|
|
|
|
|
// Test a string that's nullified by `none` at the start
|
|
|
|
let config = Value::from("none fg:red bg:green bold");
|
|
|
|
assert!(<Style>::from_config(&config).is_none());
|
|
|
|
}
|
|
|
|
|
2020-12-02 06:40:15 +00:00
|
|
|
#[test]
|
|
|
|
fn table_get_styles_with_none() {
|
|
|
|
// Test that none on the end will result in None, overriding bg:none
|
|
|
|
let config = Value::from("fg:red bg:none none");
|
|
|
|
assert!(<Style>::from_config(&config).is_none());
|
|
|
|
|
|
|
|
// Test that none in front will result in None, overriding bg:none
|
|
|
|
let config = Value::from("none fg:red bg:none");
|
|
|
|
assert!(<Style>::from_config(&config).is_none());
|
|
|
|
|
|
|
|
// Test that none in the middle will result in None, overriding bg:none
|
|
|
|
let config = Value::from("fg:red none bg:none");
|
|
|
|
assert!(<Style>::from_config(&config).is_none());
|
|
|
|
|
|
|
|
// Test that fg:none will result in None
|
|
|
|
let config = Value::from("fg:none bg:black");
|
|
|
|
assert!(<Style>::from_config(&config).is_none());
|
|
|
|
|
|
|
|
// Test that bg:none will yield a style
|
|
|
|
let config = Value::from("fg:red bg:none");
|
|
|
|
assert_eq!(<Style>::from_config(&config).unwrap(), Color::Red.normal());
|
|
|
|
|
|
|
|
// Test that bg:none will yield a style
|
|
|
|
let config = Value::from("fg:red bg:none bold");
|
|
|
|
assert_eq!(<Style>::from_config(&config).unwrap(), Color::Red.bold());
|
|
|
|
|
|
|
|
// Test that bg:none will overwrite the previous background colour
|
|
|
|
let config = Value::from("fg:red bg:green bold bg:none");
|
|
|
|
assert_eq!(<Style>::from_config(&config).unwrap(), Color::Red.bold());
|
|
|
|
}
|
|
|
|
|
2019-09-30 12:10:35 +00:00
|
|
|
#[test]
|
|
|
|
fn table_get_styles_ordered() {
|
|
|
|
// Test a background style with inverted order (also test hex + ANSI)
|
|
|
|
let config = Value::from("bg:#050505 underline fg:120");
|
|
|
|
let flipped_style = <Style>::from_config(&config).unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
flipped_style,
|
|
|
|
Style::new()
|
|
|
|
.underline()
|
|
|
|
.fg(Color::Fixed(120))
|
|
|
|
.on(Color::RGB(5, 5, 5))
|
|
|
|
);
|
|
|
|
|
|
|
|
// Test that the last color style is always the one used
|
|
|
|
let config = Value::from("bg:120 bg:125 bg:127 fg:127 122 125");
|
|
|
|
let multi_style = <Style>::from_config(&config).unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
multi_style,
|
|
|
|
Style::new().fg(Color::Fixed(125)).on(Color::Fixed(127))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|