mirror of
https://github.com/nushell/nushell
synced 2025-01-14 14:14:13 +00:00
Add decimals to int when using into string --decimals
(#6085)
* Add decimals to int when using `into string --decimals` * Add tests for `into string` when converting int with `--decimals` * Apply formatting * Merge `into_str` test files * Comment out unused code and add TODOs * Use decimal separator depending on system locale * Add test helper to run closure in different locale * Add tests for int-to-string conversion using different locales * Add utils function to get system locale * Add panic message when locking mutex fails * Catch and resume panic later to prevent Mutex poisoning when test fails * Move test to `nu-test-support` to keep `nu-utils` free of `nu-*` dependencies See https://github.com/nushell/nushell/pull/6085#issuecomment-1193131694 * Rename test support fn `with_fake_locale` to `with_locale_override` * Move `get_system_locale()` to `locale` module * Allow overriding locale with special env variable (when not in release) * Use special env var to override locale during testing * Allow callback to return a value in `with_locale_override()` * Allow multiple options in `nu!` macro * Allow to set locale as `nu!` macro option * Use new `locale` option of `nu!` macro instead of `with_locale_override` Using the `locale` options does not lock the `LOCALE_OVERRIDE_MUTEX` mutex in `nu-test-support::locale_override` but instead calls the `nu` command directly with the `NU_LOCALE_OVERRIDE` environment variable. This allows for parallel test excecution. * Fix: Add option identifier for `cwd` in usage of `nu!` macro * Rely on `Display` trait for formatting `nu!` macro command - Removed the `DisplayPath` trait - Implement `Display` for `AbsolutePath`, `RelativePath` and `AbsoluteFile` * Default to locale `en_US.UTF-8` for tests when using `nu!` macro * Add doc comment to `nu!` macro * Format code using `cargo fmt --all` * Pass function directly instead of wrapping the call in a closure https://rust-lang.github.io/rust-clippy/master/index.html#redundant_closure * Pass function to `or_else()` instead of calling it inside `or()` https://rust-lang.github.io/rust-clippy/master/index.html#or_fun_call * Fix: Add option identifier for `cwd` in usage of `nu!` macro
This commit is contained in:
parent
ccebdd7a7f
commit
cb18dd5200
21 changed files with 390 additions and 122 deletions
6
Cargo.lock
generated
6
Cargo.lock
generated
|
@ -2636,6 +2636,7 @@ dependencies = [
|
||||||
"nu-test-support",
|
"nu-test-support",
|
||||||
"nu-utils",
|
"nu-utils",
|
||||||
"num 0.4.0",
|
"num 0.4.0",
|
||||||
|
"num-format",
|
||||||
"num-traits",
|
"num-traits",
|
||||||
"pathdiff",
|
"pathdiff",
|
||||||
"polars",
|
"polars",
|
||||||
|
@ -2814,9 +2815,12 @@ version = "0.66.4"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"getset",
|
"getset",
|
||||||
"hamcrest2",
|
"hamcrest2",
|
||||||
|
"lazy_static",
|
||||||
"nu-glob",
|
"nu-glob",
|
||||||
"nu-path",
|
"nu-path",
|
||||||
|
"nu-utils",
|
||||||
"num-bigint 0.4.3",
|
"num-bigint 0.4.3",
|
||||||
|
"num-format",
|
||||||
"tempfile",
|
"tempfile",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -2826,6 +2830,8 @@ version = "0.66.4"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"crossterm_winapi",
|
"crossterm_winapi",
|
||||||
"lscolors",
|
"lscolors",
|
||||||
|
"num-format",
|
||||||
|
"sys-locale",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|
|
@ -24,6 +24,7 @@ nu-term-grid = { path = "../nu-term-grid", version = "0.66.4" }
|
||||||
nu-test-support = { path = "../nu-test-support", version = "0.66.4" }
|
nu-test-support = { path = "../nu-test-support", version = "0.66.4" }
|
||||||
nu-utils = { path = "../nu-utils", version = "0.66.4" }
|
nu-utils = { path = "../nu-utils", version = "0.66.4" }
|
||||||
nu-ansi-term = "0.46.0"
|
nu-ansi-term = "0.46.0"
|
||||||
|
num-format = { version = "0.4.0" }
|
||||||
|
|
||||||
# Potential dependencies for extras
|
# Potential dependencies for extras
|
||||||
alphanumeric-sort = "1.4.4"
|
alphanumeric-sort = "1.4.4"
|
||||||
|
|
|
@ -5,8 +5,8 @@ use nu_protocol::{
|
||||||
into_code, Category, Config, Example, IntoPipelineData, PipelineData, ShellError, Signature,
|
into_code, Category, Config, Example, IntoPipelineData, PipelineData, ShellError, Signature,
|
||||||
Span, SyntaxShape, Value,
|
Span, SyntaxShape, Value,
|
||||||
};
|
};
|
||||||
|
use nu_utils::get_system_locale;
|
||||||
// TODO num_format::SystemLocale once platform-specific dependencies are stable (see Cargo.toml)
|
use num_format::ToFormattedString;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct SubCommand;
|
pub struct SubCommand;
|
||||||
|
@ -216,21 +216,8 @@ pub fn action(
|
||||||
) -> Value {
|
) -> Value {
|
||||||
match input {
|
match input {
|
||||||
Value::Int { val, .. } => {
|
Value::Int { val, .. } => {
|
||||||
let res = if group_digits {
|
let decimal_value = digits.unwrap_or(0) as usize;
|
||||||
format_int(*val) // int.to_formatted_string(*locale)
|
let res = format_int(*val, group_digits, decimal_value);
|
||||||
} else if let Some(dig) = digits {
|
|
||||||
let mut val_with_trailing_zeroes = val.to_string();
|
|
||||||
if dig != 0 {
|
|
||||||
val_with_trailing_zeroes.push('.');
|
|
||||||
}
|
|
||||||
for _ in 0..dig {
|
|
||||||
val_with_trailing_zeroes.push('0');
|
|
||||||
}
|
|
||||||
val_with_trailing_zeroes
|
|
||||||
} else {
|
|
||||||
val.to_string()
|
|
||||||
};
|
|
||||||
|
|
||||||
Value::String { val: res, span }
|
Value::String { val: res, span }
|
||||||
}
|
}
|
||||||
Value::Float { val, .. } => {
|
Value::Float { val, .. } => {
|
||||||
|
@ -305,21 +292,29 @@ pub fn action(
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn format_int(int: i64) -> String {
|
|
||||||
int.to_string()
|
|
||||||
|
|
||||||
// TODO once platform-specific dependencies are stable (see Cargo.toml)
|
fn format_int(int: i64, group_digits: bool, decimals: usize) -> String {
|
||||||
// #[cfg(windows)]
|
let locale = get_system_locale();
|
||||||
// {
|
|
||||||
// int.to_formatted_string(&Locale::en)
|
let str = if group_digits {
|
||||||
// }
|
int.to_formatted_string(&locale)
|
||||||
// #[cfg(not(windows))]
|
} else {
|
||||||
// {
|
int.to_string()
|
||||||
// match SystemLocale::default() {
|
};
|
||||||
// Ok(locale) => int.to_formatted_string(&locale),
|
|
||||||
// Err(_) => int.to_formatted_string(&Locale::en),
|
if decimals > 0 {
|
||||||
// }
|
let decimal_point = locale.decimal();
|
||||||
// }
|
|
||||||
|
format!(
|
||||||
|
"{}{decimal_point}{dummy:0<decimals$}",
|
||||||
|
str,
|
||||||
|
decimal_point = decimal_point,
|
||||||
|
dummy = "",
|
||||||
|
decimals = decimals
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
str
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|
|
@ -31,7 +31,7 @@ fn filesystem_change_from_current_directory_using_absolute_path() {
|
||||||
cd "{}"
|
cd "{}"
|
||||||
echo (pwd)
|
echo (pwd)
|
||||||
"#,
|
"#,
|
||||||
dirs.formats()
|
dirs.formats().display()
|
||||||
);
|
);
|
||||||
|
|
||||||
assert_eq!(PathBuf::from(actual.out), dirs.formats());
|
assert_eq!(PathBuf::from(actual.out), dirs.formats());
|
||||||
|
@ -52,7 +52,7 @@ fn filesystem_switch_back_to_previous_working_directory() {
|
||||||
cd -
|
cd -
|
||||||
echo (pwd)
|
echo (pwd)
|
||||||
"#,
|
"#,
|
||||||
dirs.test()
|
dirs.test().display()
|
||||||
);
|
);
|
||||||
|
|
||||||
assert_eq!(PathBuf::from(actual.out), dirs.test().join("odin"));
|
assert_eq!(PathBuf::from(actual.out), dirs.test().join("odin"));
|
||||||
|
@ -132,7 +132,7 @@ fn filesystem_change_current_directory_to_parent_directory_after_delete_cwd() {
|
||||||
cd ..
|
cd ..
|
||||||
echo (pwd)
|
echo (pwd)
|
||||||
"#,
|
"#,
|
||||||
dirs.test()
|
dirs.test().display()
|
||||||
);
|
);
|
||||||
|
|
||||||
let actual = actual.out.split(',').nth(1).unwrap();
|
let actual = actual.out.split(',').nth(1).unwrap();
|
||||||
|
|
|
@ -10,7 +10,7 @@ fn copies_a_file() {
|
||||||
nu!(
|
nu!(
|
||||||
cwd: dirs.root(),
|
cwd: dirs.root(),
|
||||||
"cp `{}` cp_test_1/sample.ini",
|
"cp `{}` cp_test_1/sample.ini",
|
||||||
dirs.formats().join("sample.ini")
|
dirs.formats().join("sample.ini").display()
|
||||||
);
|
);
|
||||||
|
|
||||||
assert!(dirs.test().join("sample.ini").exists());
|
assert!(dirs.test().join("sample.ini").exists());
|
||||||
|
@ -37,7 +37,7 @@ fn error_if_attempting_to_copy_a_directory_to_another_directory() {
|
||||||
Playground::setup("cp_test_3", |dirs, _| {
|
Playground::setup("cp_test_3", |dirs, _| {
|
||||||
let actual = nu!(
|
let actual = nu!(
|
||||||
cwd: dirs.formats(),
|
cwd: dirs.formats(),
|
||||||
"cp ../formats {}", dirs.test()
|
"cp ../formats {}", dirs.test().display()
|
||||||
);
|
);
|
||||||
|
|
||||||
assert!(actual.err.contains("../formats"));
|
assert!(actual.err.contains("../formats"));
|
||||||
|
@ -128,7 +128,7 @@ fn copies_using_path_with_wildcard() {
|
||||||
Playground::setup("cp_test_6", |dirs, _| {
|
Playground::setup("cp_test_6", |dirs, _| {
|
||||||
nu!(
|
nu!(
|
||||||
cwd: dirs.formats(),
|
cwd: dirs.formats(),
|
||||||
"cp -r ../formats/* {}", dirs.test()
|
"cp -r ../formats/* {}", dirs.test().display()
|
||||||
);
|
);
|
||||||
|
|
||||||
assert!(files_exist_at(
|
assert!(files_exist_at(
|
||||||
|
@ -150,7 +150,7 @@ fn copies_using_a_glob() {
|
||||||
Playground::setup("cp_test_7", |dirs, _| {
|
Playground::setup("cp_test_7", |dirs, _| {
|
||||||
nu!(
|
nu!(
|
||||||
cwd: dirs.formats(),
|
cwd: dirs.formats(),
|
||||||
"cp -r * {}", dirs.test()
|
"cp -r * {}", dirs.test().display()
|
||||||
);
|
);
|
||||||
|
|
||||||
assert!(files_exist_at(
|
assert!(files_exist_at(
|
||||||
|
@ -173,13 +173,13 @@ fn copies_same_file_twice() {
|
||||||
nu!(
|
nu!(
|
||||||
cwd: dirs.root(),
|
cwd: dirs.root(),
|
||||||
"cp `{}` cp_test_8/sample.ini",
|
"cp `{}` cp_test_8/sample.ini",
|
||||||
dirs.formats().join("sample.ini")
|
dirs.formats().join("sample.ini").display()
|
||||||
);
|
);
|
||||||
|
|
||||||
nu!(
|
nu!(
|
||||||
cwd: dirs.root(),
|
cwd: dirs.root(),
|
||||||
"cp `{}` cp_test_8/sample.ini",
|
"cp `{}` cp_test_8/sample.ini",
|
||||||
dirs.formats().join("sample.ini")
|
dirs.formats().join("sample.ini").display()
|
||||||
);
|
);
|
||||||
|
|
||||||
assert!(dirs.test().join("sample.ini").exists());
|
assert!(dirs.test().join("sample.ini").exists());
|
||||||
|
|
|
@ -531,7 +531,7 @@ fn list_directory_contains_invalid_utf8() {
|
||||||
|
|
||||||
std::fs::create_dir_all(&path).expect("failed to create directory");
|
std::fs::create_dir_all(&path).expect("failed to create directory");
|
||||||
|
|
||||||
let actual = nu!(cwd, "ls");
|
let actual = nu!(cwd: cwd, "ls");
|
||||||
|
|
||||||
assert!(actual.out.contains("warning: get non-utf8 filename"));
|
assert!(actual.out.contains("warning: get non-utf8 filename"));
|
||||||
assert!(actual.err.contains("No matches found for"));
|
assert!(actual.err.contains("No matches found for"));
|
||||||
|
|
|
@ -384,7 +384,7 @@ fn mv_directory_with_same_name() {
|
||||||
|
|
||||||
let cwd = sandbox.cwd().join("testdir");
|
let cwd = sandbox.cwd().join("testdir");
|
||||||
let actual = nu!(
|
let actual = nu!(
|
||||||
cwd,
|
cwd: cwd,
|
||||||
r#"
|
r#"
|
||||||
mv testdir ..
|
mv testdir ..
|
||||||
"#
|
"#
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use nu_test_support::fs::{AbsolutePath, DisplayPath, Stub::FileWithContent};
|
use nu_test_support::fs::{AbsolutePath, Stub::FileWithContent};
|
||||||
use nu_test_support::nu;
|
use nu_test_support::nu;
|
||||||
use nu_test_support::pipeline;
|
use nu_test_support::pipeline;
|
||||||
use nu_test_support::playground::Playground;
|
use nu_test_support::playground::Playground;
|
||||||
|
@ -18,7 +18,7 @@ fn sources_also_files_under_custom_lib_dirs_path() {
|
||||||
lib_dirs = ["{}"]
|
lib_dirs = ["{}"]
|
||||||
skip_welcome_message = true
|
skip_welcome_message = true
|
||||||
"#,
|
"#,
|
||||||
library_path.display_path()
|
library_path
|
||||||
),
|
),
|
||||||
)]);
|
)]);
|
||||||
|
|
||||||
|
|
|
@ -183,3 +183,87 @@ fn from_error() {
|
||||||
|
|
||||||
assert_eq!(actual.out, "nu::shell::name_not_found");
|
assert_eq!(actual.out, "nu::shell::name_not_found");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn int_into_string() {
|
||||||
|
let actual = nu!(
|
||||||
|
cwd: ".", pipeline(
|
||||||
|
r#"
|
||||||
|
10 | into string
|
||||||
|
"#
|
||||||
|
));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "10");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn int_into_string_decimals_0() {
|
||||||
|
let actual = nu!(
|
||||||
|
locale: "en_US.UTF-8",
|
||||||
|
pipeline(
|
||||||
|
r#"
|
||||||
|
10 | into string --decimals 0
|
||||||
|
"#
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "10");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn int_into_string_decimals_1() {
|
||||||
|
let actual = nu!(
|
||||||
|
locale: "en_US.UTF-8",
|
||||||
|
pipeline(
|
||||||
|
r#"
|
||||||
|
10 | into string --decimals 1
|
||||||
|
"#
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "10.0");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn int_into_string_decimals_10() {
|
||||||
|
let actual = nu!(
|
||||||
|
locale: "en_US.UTF-8",
|
||||||
|
pipeline(
|
||||||
|
r#"
|
||||||
|
10 | into string --decimals 10
|
||||||
|
"#
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "10.0000000000");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn int_into_string_decimals_respects_system_locale_de() {
|
||||||
|
// Set locale to `de_DE`, which uses `,` (comma) as decimal separator
|
||||||
|
let actual = nu!(
|
||||||
|
locale: "de_DE.UTF-8",
|
||||||
|
pipeline(
|
||||||
|
r#"
|
||||||
|
10 | into string --decimals 1
|
||||||
|
"#
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "10,0");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn int_into_string_decimals_respects_system_locale_en() {
|
||||||
|
// Set locale to `en_US`, which uses `.` (period) as decimal separator
|
||||||
|
let actual = nu!(
|
||||||
|
locale: "en_US.UTF-8",
|
||||||
|
pipeline(
|
||||||
|
r#"
|
||||||
|
10 | into string --decimals 1
|
||||||
|
"#
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "10.0");
|
||||||
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use nu_test_support::fs::{AbsolutePath, DisplayPath, Stub::FileWithContent};
|
use nu_test_support::fs::{AbsolutePath, Stub::FileWithContent};
|
||||||
use nu_test_support::nu;
|
use nu_test_support::nu;
|
||||||
use nu_test_support::pipeline;
|
use nu_test_support::pipeline;
|
||||||
use nu_test_support::playground::Playground;
|
use nu_test_support::playground::Playground;
|
||||||
|
@ -9,7 +9,7 @@ fn use_module_file_within_block() {
|
||||||
let file = AbsolutePath::new(dirs.test().join("spam.nu"));
|
let file = AbsolutePath::new(dirs.test().join("spam.nu"));
|
||||||
|
|
||||||
nu.with_files(vec![FileWithContent(
|
nu.with_files(vec![FileWithContent(
|
||||||
&file.display_path(),
|
&file.to_string(),
|
||||||
r#"
|
r#"
|
||||||
export def foo [] {
|
export def foo [] {
|
||||||
echo "hello world"
|
echo "hello world"
|
||||||
|
@ -39,7 +39,7 @@ fn use_keeps_doc_comments() {
|
||||||
let file = AbsolutePath::new(dirs.test().join("spam.nu"));
|
let file = AbsolutePath::new(dirs.test().join("spam.nu"));
|
||||||
|
|
||||||
nu.with_files(vec![FileWithContent(
|
nu.with_files(vec![FileWithContent(
|
||||||
&file.display_path(),
|
&file.to_string(),
|
||||||
r#"
|
r#"
|
||||||
# this is my foo command
|
# this is my foo command
|
||||||
export def foo [
|
export def foo [
|
||||||
|
|
|
@ -16,7 +16,8 @@ pub use custom_value::CustomValue;
|
||||||
use fancy_regex::Regex;
|
use fancy_regex::Regex;
|
||||||
pub use from_value::FromValue;
|
pub use from_value::FromValue;
|
||||||
use indexmap::map::IndexMap;
|
use indexmap::map::IndexMap;
|
||||||
use num_format::{Locale, ToFormattedString};
|
use nu_utils::get_system_locale;
|
||||||
|
use num_format::ToFormattedString;
|
||||||
pub use range::*;
|
pub use range::*;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use std::{
|
use std::{
|
||||||
|
@ -28,7 +29,6 @@ use std::{
|
||||||
{cmp::Ordering, fmt::Debug},
|
{cmp::Ordering, fmt::Debug},
|
||||||
};
|
};
|
||||||
pub use stream::*;
|
pub use stream::*;
|
||||||
use sys_locale::get_locale;
|
|
||||||
pub use unit::*;
|
pub use unit::*;
|
||||||
|
|
||||||
/// Core structured values that pass through the pipeline in Nushell.
|
/// Core structured values that pass through the pipeline in Nushell.
|
||||||
|
@ -2760,25 +2760,7 @@ pub fn format_filesize(num_bytes: i64, format_value: &str, filesize_metric: bool
|
||||||
|
|
||||||
match adj_byte.get_unit() {
|
match adj_byte.get_unit() {
|
||||||
byte_unit::ByteUnit::B => {
|
byte_unit::ByteUnit::B => {
|
||||||
let locale_string = get_locale().unwrap_or_else(|| String::from("en-US"));
|
let locale = get_system_locale();
|
||||||
// Since get_locale() and Locale::from_name() don't always return the same items
|
|
||||||
// we need to try and parse it to match. For instance, a valid locale is de_DE
|
|
||||||
// however Locale::from_name() wants only de so we split and parse it out.
|
|
||||||
let locale_string = locale_string.replace('_', "-"); // en_AU -> en-AU
|
|
||||||
let locale = match Locale::from_name(&locale_string) {
|
|
||||||
Ok(loc) => loc,
|
|
||||||
_ => {
|
|
||||||
let all = num_format::Locale::available_names();
|
|
||||||
let locale_prefix = &locale_string.split('-').collect::<Vec<&str>>();
|
|
||||||
if all.contains(&locale_prefix[0]) {
|
|
||||||
// eprintln!("Found alternate: {}", &locale_prefix[0]);
|
|
||||||
Locale::from_name(locale_prefix[0]).unwrap_or(Locale::en)
|
|
||||||
} else {
|
|
||||||
// eprintln!("Unable to find matching locale. Defaulting to en-US");
|
|
||||||
Locale::en
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
let locale_byte = adj_byte.get_value() as u64;
|
let locale_byte = adj_byte.get_value() as u64;
|
||||||
let locale_byte_string = locale_byte.to_formatted_string(&locale);
|
let locale_byte_string = locale_byte.to_formatted_string(&locale);
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,9 @@ doctest = false
|
||||||
[dependencies]
|
[dependencies]
|
||||||
nu-path = { path="../nu-path", version = "0.66.4" }
|
nu-path = { path="../nu-path", version = "0.66.4" }
|
||||||
nu-glob = { path = "../nu-glob", version = "0.66.4" }
|
nu-glob = { path = "../nu-glob", version = "0.66.4" }
|
||||||
|
nu-utils = { path="../nu-utils", version = "0.66.4" }
|
||||||
|
lazy_static = "1.4.0"
|
||||||
|
num-format = "0.4.0"
|
||||||
|
|
||||||
getset = "0.1.1"
|
getset = "0.1.1"
|
||||||
num-bigint = { version="0.4.3", features=["serde"] }
|
num-bigint = { version="0.4.3", features=["serde"] }
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
use std::fmt::Display;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
use std::ops::Div;
|
use std::ops::Div;
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
@ -113,45 +114,25 @@ impl<T: AsRef<str>> Div<T> for &RelativePath {
|
||||||
RelativePath::new(result)
|
RelativePath::new(result)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub trait DisplayPath {
|
|
||||||
fn display_path(&self) -> String;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DisplayPath for AbsolutePath {
|
impl Display for AbsoluteFile {
|
||||||
fn display_path(&self) -> String {
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
self.inner.display().to_string()
|
write!(f, "{}", self.inner.display())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DisplayPath for PathBuf {
|
impl Display for AbsolutePath {
|
||||||
fn display_path(&self) -> String {
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
self.display().to_string()
|
write!(f, "{}", self.inner.display())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DisplayPath for str {
|
impl Display for RelativePath {
|
||||||
fn display_path(&self) -> String {
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
self.to_string()
|
write!(f, "{}", self.inner.display())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DisplayPath for &str {
|
|
||||||
fn display_path(&self) -> String {
|
|
||||||
(*self).to_string()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DisplayPath for String {
|
|
||||||
fn display_path(&self) -> String {
|
|
||||||
self.clone()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DisplayPath for &String {
|
|
||||||
fn display_path(&self) -> String {
|
|
||||||
(*self).to_string()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub enum Stub<'a> {
|
pub enum Stub<'a> {
|
||||||
FileWithContent(&'a str, &'a str),
|
FileWithContent(&'a str, &'a str),
|
||||||
FileWithContentToBeTrimmed(&'a str, &'a str),
|
FileWithContentToBeTrimmed(&'a str, &'a str),
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
pub mod commands;
|
pub mod commands;
|
||||||
pub mod fs;
|
pub mod fs;
|
||||||
|
pub mod locale_override;
|
||||||
pub mod macros;
|
pub mod macros;
|
||||||
pub mod playground;
|
pub mod playground;
|
||||||
|
|
||||||
|
|
46
crates/nu-test-support/src/locale_override.rs
Normal file
46
crates/nu-test-support/src/locale_override.rs
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
#![cfg(debug_assertions)]
|
||||||
|
|
||||||
|
use std::sync::{Arc, Mutex};
|
||||||
|
|
||||||
|
use lazy_static::lazy_static;
|
||||||
|
use nu_utils::locale::LOCALE_OVERRIDE_ENV_VAR;
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
static ref LOCALE_OVERRIDE_MUTEX: Arc<Mutex<()>> = Arc::new(Mutex::new(()));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Run a closure in a fake locale environment.
|
||||||
|
///
|
||||||
|
/// Before the closure is executed, an environment variable whose name is
|
||||||
|
/// defined in `nu_utils::locale::LOCALE_OVERRIDE_ENV_VAR` is set to the value
|
||||||
|
/// provided by `locale_string`. When the closure is done, the previous value is
|
||||||
|
/// restored.
|
||||||
|
///
|
||||||
|
/// Environment variables are global values. So when they are changed by one
|
||||||
|
/// thread they are changed for all others. To prevent a test from overwriting
|
||||||
|
/// the environment variable of another test, a mutex is used.
|
||||||
|
pub fn with_locale_override<T>(locale_string: &str, func: fn() -> T) -> T {
|
||||||
|
let result = {
|
||||||
|
let _lock = LOCALE_OVERRIDE_MUTEX
|
||||||
|
.lock()
|
||||||
|
.expect("Failed to get mutex lock for locale override");
|
||||||
|
|
||||||
|
let saved = std::env::var(LOCALE_OVERRIDE_ENV_VAR).ok();
|
||||||
|
std::env::set_var(LOCALE_OVERRIDE_ENV_VAR, locale_string);
|
||||||
|
|
||||||
|
let result = std::panic::catch_unwind(func);
|
||||||
|
|
||||||
|
if let Some(locale_str) = saved {
|
||||||
|
std::env::set_var(LOCALE_OVERRIDE_ENV_VAR, locale_str);
|
||||||
|
} else {
|
||||||
|
std::env::remove_var(LOCALE_OVERRIDE_ENV_VAR);
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
|
};
|
||||||
|
|
||||||
|
match result {
|
||||||
|
Ok(result) => result,
|
||||||
|
Err(err) => std::panic::resume_unwind(err),
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,21 +1,120 @@
|
||||||
|
/// Run a command in nu and get it's output
|
||||||
|
///
|
||||||
|
/// The `nu!` macro accepts a number of options like the `cwd` in which the
|
||||||
|
/// command should be run. It is also possible to specify a different `locale`
|
||||||
|
/// to test locale dependent commands.
|
||||||
|
///
|
||||||
|
/// Pass options as the first arguments in the form of `key_1: value_1, key_1:
|
||||||
|
/// value_2, ...`. The options are defined in the `NuOpts` struct inside the
|
||||||
|
/// `nu!` macro.
|
||||||
|
///
|
||||||
|
/// The command can be formatted using `{}` just like `println!` or `format!`.
|
||||||
|
/// Pass the format arguments comma separated after the command itself.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```no_run
|
||||||
|
/// # // NOTE: The `nu!` macro needs the `nu` binary to exist. The test are
|
||||||
|
/// # // therefore only compiled but not run (thats what the `no_run` at
|
||||||
|
/// # // the beginning of this code block is for).
|
||||||
|
/// #
|
||||||
|
/// use nu_test_support::nu;
|
||||||
|
///
|
||||||
|
/// let outcome = nu!(
|
||||||
|
/// "date now | date to-record | get year"
|
||||||
|
/// );
|
||||||
|
///
|
||||||
|
/// let dir = "/";
|
||||||
|
/// let outcome = nu!(
|
||||||
|
/// "ls {} | get name",
|
||||||
|
/// dir,
|
||||||
|
/// );
|
||||||
|
///
|
||||||
|
/// let outcome = nu!(
|
||||||
|
/// cwd: "/",
|
||||||
|
/// "ls | get name",
|
||||||
|
/// );
|
||||||
|
///
|
||||||
|
/// let cell = "size";
|
||||||
|
/// let outcome = nu!(
|
||||||
|
/// locale: "de_DE.UTF-8",
|
||||||
|
/// "ls | into int {}",
|
||||||
|
/// cell,
|
||||||
|
/// );
|
||||||
|
///
|
||||||
|
/// let decimals = 2;
|
||||||
|
/// let outcome = nu!(
|
||||||
|
/// locale: "de_DE.UTF-8",
|
||||||
|
/// "10 | into string --decimals {}",
|
||||||
|
/// decimals,
|
||||||
|
/// );
|
||||||
|
/// ```
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! nu {
|
macro_rules! nu {
|
||||||
(cwd: $cwd:expr, $path:expr, $($part:expr),*) => {{
|
// In the `@options` phase, we restucture all the
|
||||||
use $crate::fs::DisplayPath;
|
// `$field_1: $value_1, $field_2: $value_2, ...`
|
||||||
|
// pairs to a structure like
|
||||||
|
// `@options[ $field_1 => $value_1 ; $field_2 => $value_2 ; ... ]`.
|
||||||
|
// We do this to later distinguish the options from the `$path` and `$part`s.
|
||||||
|
// (See
|
||||||
|
// https://users.rust-lang.org/t/i-dont-think-this-local-ambiguity-when-calling-macro-is-ambiguous/79401?u=x3ro
|
||||||
|
// )
|
||||||
|
//
|
||||||
|
// If there is any special treatment needed for the `$value`, we can just
|
||||||
|
// match for the specific `field` name.
|
||||||
|
(
|
||||||
|
@options [ $($options:tt)* ]
|
||||||
|
cwd: $value:expr,
|
||||||
|
$($rest:tt)*
|
||||||
|
) => {
|
||||||
|
nu!(@options [ $($options)* cwd => $crate::fs::in_directory($value) ; ] $($rest)*)
|
||||||
|
};
|
||||||
|
// For all other options, we call `.into()` on the `$value` and hope for the best. ;)
|
||||||
|
(
|
||||||
|
@options [ $($options:tt)* ]
|
||||||
|
$field:ident : $value:expr,
|
||||||
|
$($rest:tt)*
|
||||||
|
) => {
|
||||||
|
nu!(@options [ $($options)* $field => $value.into() ; ] $($rest)*)
|
||||||
|
};
|
||||||
|
|
||||||
let path = format!($path, $(
|
// When the `$field: $value,` pairs are all parsed, the next tokens are the `$path` and any
|
||||||
$part.display_path()
|
// number of `$part`s, potentially followed by a trailing comma.
|
||||||
),*);
|
(
|
||||||
|
@options [ $($options:tt)* ]
|
||||||
nu!($cwd, &path)
|
$path:expr
|
||||||
|
$(, $part:expr)*
|
||||||
|
$(,)*
|
||||||
|
) => {{
|
||||||
|
// Here we parse the options into a `NuOpts` struct
|
||||||
|
let opts = nu!(@nu_opts $($options)*);
|
||||||
|
// and format the `$path` using the `$part`s
|
||||||
|
let path = nu!(@format_path $path, $($part),*);
|
||||||
|
// Then finally we go to the `@main` phase, where the actual work is done.
|
||||||
|
nu!(@main opts, path)
|
||||||
}};
|
}};
|
||||||
|
|
||||||
(cwd: $cwd:expr, $path:expr) => {{
|
// Create the NuOpts struct from the `field => value ;` pairs
|
||||||
nu!($cwd, $path)
|
(@nu_opts $( $field:ident => $value:expr ; )*) => {
|
||||||
|
NuOpts{
|
||||||
|
$(
|
||||||
|
$field: Some($value),
|
||||||
|
)*
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Helper to format `$path`.
|
||||||
|
(@format_path $path:expr $(,)?) => {
|
||||||
|
// When there are no `$part`s, do not format anything
|
||||||
|
$path
|
||||||
|
};
|
||||||
|
(@format_path $path:expr, $($part:expr),* $(,)?) => {{
|
||||||
|
format!($path, $( $part ),*)
|
||||||
}};
|
}};
|
||||||
|
|
||||||
($cwd:expr, $path:expr) => {{
|
// Do the actual work.
|
||||||
pub use itertools::Itertools;
|
(@main $opts:expr, $path:expr) => {{
|
||||||
pub use std::error::Error;
|
pub use std::error::Error;
|
||||||
pub use std::io::prelude::*;
|
pub use std::io::prelude::*;
|
||||||
pub use std::process::{Command, Stdio};
|
pub use std::process::{Command, Stdio};
|
||||||
|
@ -36,13 +135,6 @@ macro_rules! nu {
|
||||||
output
|
output
|
||||||
}
|
}
|
||||||
|
|
||||||
// let commands = &*format!(
|
|
||||||
// "
|
|
||||||
// {}
|
|
||||||
// exit",
|
|
||||||
// $crate::fs::DisplayPath::display_path(&$path)
|
|
||||||
// );
|
|
||||||
|
|
||||||
let test_bins = $crate::fs::binaries();
|
let test_bins = $crate::fs::binaries();
|
||||||
|
|
||||||
let cwd = std::env::current_dir().expect("Could not get current working directory.");
|
let cwd = std::env::current_dir().expect("Could not get current working directory.");
|
||||||
|
@ -64,21 +156,25 @@ macro_rules! nu {
|
||||||
Err(_) => panic!("Couldn't join paths for PATH var."),
|
Err(_) => panic!("Couldn't join paths for PATH var."),
|
||||||
};
|
};
|
||||||
|
|
||||||
let target_cwd = $crate::fs::in_directory(&$cwd);
|
let target_cwd = $opts.cwd.unwrap_or(".".to_string());
|
||||||
|
let locale = $opts.locale.unwrap_or("en_US.UTF-8".to_string());
|
||||||
|
|
||||||
let mut process = match Command::new($crate::fs::executable_path())
|
let mut command = Command::new($crate::fs::executable_path());
|
||||||
|
command
|
||||||
.env("PWD", &target_cwd)
|
.env("PWD", &target_cwd)
|
||||||
|
.env(nu_utils::locale::LOCALE_OVERRIDE_ENV_VAR, locale)
|
||||||
.current_dir(target_cwd)
|
.current_dir(target_cwd)
|
||||||
.env(NATIVE_PATH_ENV_VAR, paths_joined)
|
.env(NATIVE_PATH_ENV_VAR, paths_joined)
|
||||||
// .arg("--skip-plugins")
|
// .arg("--skip-plugins")
|
||||||
// .arg("--no-history")
|
// .arg("--no-history")
|
||||||
// .arg("--config-file")
|
// .arg("--config-file")
|
||||||
// .arg($crate::fs::DisplayPath::display_path(&$crate::fs::fixtures().join("playground/config/default.toml")))
|
// .arg($crate::fs::DisplayPath::display_path(&$crate::fs::fixtures().join("playground/config/default.toml")))
|
||||||
.arg(format!("-c {}", escape_quote_string($crate::fs::DisplayPath::display_path(&path))))
|
.arg(format!("-c {}", escape_quote_string(path)))
|
||||||
.stdout(Stdio::piped())
|
.stdout(Stdio::piped())
|
||||||
// .stdin(Stdio::piped())
|
// .stdin(Stdio::piped())
|
||||||
.stderr(Stdio::piped())
|
.stderr(Stdio::piped());
|
||||||
.spawn()
|
|
||||||
|
let mut process = match command.spawn()
|
||||||
{
|
{
|
||||||
Ok(child) => child,
|
Ok(child) => child,
|
||||||
Err(why) => panic!("Can't run test {:?} {}", $crate::fs::executable_path(), why.to_string()),
|
Err(why) => panic!("Can't run test {:?} {}", $crate::fs::executable_path(), why.to_string()),
|
||||||
|
@ -100,6 +196,17 @@ macro_rules! nu {
|
||||||
|
|
||||||
$crate::Outcome::new(out,err.into_owned())
|
$crate::Outcome::new(out,err.into_owned())
|
||||||
}};
|
}};
|
||||||
|
|
||||||
|
// This is the entrypoint for this macro.
|
||||||
|
($($token:tt)*) => {{
|
||||||
|
#[derive(Default)]
|
||||||
|
struct NuOpts {
|
||||||
|
cwd: Option<String>,
|
||||||
|
locale: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
nu!(@options [ ] $($token)*)
|
||||||
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
|
|
19
crates/nu-test-support/tests/get_system_locale.rs
Normal file
19
crates/nu-test-support/tests/get_system_locale.rs
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
use nu_test_support::locale_override::with_locale_override;
|
||||||
|
use nu_utils::get_system_locale;
|
||||||
|
use num_format::Grouping;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_get_system_locale_en() {
|
||||||
|
let locale = with_locale_override("en_US.UTF-8", || get_system_locale());
|
||||||
|
|
||||||
|
assert_eq!(locale.name(), "en");
|
||||||
|
assert_eq!(locale.grouping(), Grouping::Standard)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_get_system_locale_de() {
|
||||||
|
let locale = with_locale_override("de_DE.UTF-8", || get_system_locale());
|
||||||
|
|
||||||
|
assert_eq!(locale.name(), "de");
|
||||||
|
assert_eq!(locale.grouping(), Grouping::Standard)
|
||||||
|
}
|
|
@ -13,6 +13,8 @@ path = "src/main.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
lscolors = { version = "0.12.0", features = ["crossterm"]}
|
lscolors = { version = "0.12.0", features = ["crossterm"]}
|
||||||
|
num-format = { version = "0.4.0" }
|
||||||
|
sys-locale = "0.2.1"
|
||||||
|
|
||||||
[target.'cfg(windows)'.dependencies]
|
[target.'cfg(windows)'.dependencies]
|
||||||
crossterm_winapi = "0.9.0"
|
crossterm_winapi = "0.9.0"
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
|
pub mod locale;
|
||||||
pub mod utils;
|
pub mod utils;
|
||||||
|
|
||||||
|
pub use locale::get_system_locale;
|
||||||
pub use utils::{
|
pub use utils::{
|
||||||
enable_vt_processing, get_default_config, get_default_env, get_ls_colors,
|
enable_vt_processing, get_default_config, get_default_env, get_ls_colors,
|
||||||
stderr_write_all_and_flush, stdout_write_all_and_flush,
|
stderr_write_all_and_flush, stdout_write_all_and_flush,
|
||||||
|
|
39
crates/nu-utils/src/locale.rs
Normal file
39
crates/nu-utils/src/locale.rs
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
use num_format::Locale;
|
||||||
|
|
||||||
|
#[cfg(debug_assertions)]
|
||||||
|
pub const LOCALE_OVERRIDE_ENV_VAR: &str = "NU_TEST_LOCALE_OVERRIDE";
|
||||||
|
|
||||||
|
pub fn get_system_locale() -> Locale {
|
||||||
|
let locale_string = get_system_locale_string().unwrap_or_else(|| String::from("en-US"));
|
||||||
|
// Since get_locale() and Locale::from_name() don't always return the same items
|
||||||
|
// we need to try and parse it to match. For instance, a valid locale is de_DE
|
||||||
|
// however Locale::from_name() wants only de so we split and parse it out.
|
||||||
|
let locale_string = locale_string.replace('_', "-"); // en_AU -> en-AU
|
||||||
|
|
||||||
|
match Locale::from_name(&locale_string) {
|
||||||
|
Ok(loc) => loc,
|
||||||
|
_ => {
|
||||||
|
let all = num_format::Locale::available_names();
|
||||||
|
let locale_prefix = &locale_string.split('-').collect::<Vec<&str>>();
|
||||||
|
if all.contains(&locale_prefix[0]) {
|
||||||
|
// eprintln!("Found alternate: {}", &locale_prefix[0]);
|
||||||
|
Locale::from_name(locale_prefix[0]).unwrap_or(Locale::en)
|
||||||
|
} else {
|
||||||
|
// eprintln!("Unable to find matching locale. Defaulting to en-US");
|
||||||
|
Locale::en
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(debug_assertions)]
|
||||||
|
fn get_system_locale_string() -> Option<String> {
|
||||||
|
std::env::var(LOCALE_OVERRIDE_ENV_VAR)
|
||||||
|
.ok()
|
||||||
|
.or_else(sys_locale::get_locale)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(debug_assertions))]
|
||||||
|
fn get_system_locale_string() -> Option<String> {
|
||||||
|
sys_locale::get_locale()
|
||||||
|
}
|
|
@ -12,9 +12,9 @@ pub mod support {
|
||||||
pub fn in_path(dirs: &Dirs, block: impl FnOnce() -> Outcome) -> Outcome {
|
pub fn in_path(dirs: &Dirs, block: impl FnOnce() -> Outcome) -> Outcome {
|
||||||
let for_env_manifest = dirs.test().to_string_lossy();
|
let for_env_manifest = dirs.test().to_string_lossy();
|
||||||
|
|
||||||
nu!(cwd: dirs.root(), format!("autoenv trust \"{}\"", for_env_manifest));
|
nu!(cwd: dirs.root(), "autoenv trust \"{}\"", for_env_manifest);
|
||||||
let out = block();
|
let out = block();
|
||||||
nu!(cwd: dirs.root(), format!("autoenv untrust \"{}\"", for_env_manifest));
|
nu!(cwd: dirs.root(), "autoenv untrust \"{}\"", for_env_manifest);
|
||||||
|
|
||||||
out
|
out
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue