ls now collects metadata in a separate thread (#14627)

Closes #6174

# Description

This PR aims to improve the performance of `ls` within large
directories. `ls` now delegates the metadata collection to
a thread in its thread pool.

Before:

![image](https://github.com/user-attachments/assets/1967ab78-177c-485f-9b2f-f9d625678171)

Now:

![image](https://github.com/user-attachments/assets/fc215d0a-4b26-4791-a3a1-77cecff133e2)

# User-Facing Changes

If an error occurs while file metadata is being collected in another
thread, the `ls` command now notifies the user about this error by
sending an error value through a channel (which then gets collected into
an iterator and shown to the user later on).

However, if an error occurs _while_ sending this error value to the
channel (i.e the resulting value iterator has been dropped), then the
user is not notified of this error. I think this behavior is acceptable,
since behavior only occurs when the `ls` pipeline has been dropped and
the user is no longer interested in output from `ls`.

# Tests + Formatting

I do not know if it is a good idea to test this performance with
`timeit`, since it can be unreliable.
This commit is contained in:
Renan Ribeiro 2024-12-25 10:36:02 -03:00 committed by GitHub
parent 6ebc0fc3ff
commit 81baf53814
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -316,6 +316,7 @@ fn ls_for_one_pattern(
}; };
let hidden_dir_specified = is_hidden_dir(pattern_arg.as_ref()); let hidden_dir_specified = is_hidden_dir(pattern_arg.as_ref());
let path = pattern_arg.into_spanned(p_tag); let path = pattern_arg.into_spanned(p_tag);
let (prefix, paths) = if just_read_dir { let (prefix, paths) = if just_read_dir {
let expanded = nu_path::expand_path_with(path.item.as_ref(), &cwd, path.item.is_expand()); let expanded = nu_path::expand_path_with(path.item.as_ref(), &cwd, path.item.is_expand());
@ -358,7 +359,8 @@ fn ls_for_one_pattern(
}; };
pool.install(|| { pool.install(|| {
paths_peek rayon::spawn(move || {
let result = paths_peek
.par_bridge() .par_bridge()
.filter_map(move |x| match x { .filter_map(move |x| match x {
Ok(path) => { Ok(path) => {
@ -390,8 +392,9 @@ fn ls_for_one_pattern(
if let Ok(remainder) = path.strip_prefix(prefix) { if let Ok(remainder) = path.strip_prefix(prefix) {
if directory { if directory {
// When the path is the same as the cwd, path_diff should be "." // When the path is the same as the cwd, path_diff should be "."
let path_diff = let path_diff = if let Some(path_diff_not_dot) =
if let Some(path_diff_not_dot) = diff_paths(&path, &cwd) { diff_paths(&path, &cwd)
{
let path_diff_not_dot = path_diff_not_dot.to_string_lossy(); let path_diff_not_dot = path_diff_not_dot.to_string_lossy();
if path_diff_not_dot.is_empty() { if path_diff_not_dot.is_empty() {
".".to_string() ".".to_string()
@ -458,14 +461,19 @@ fn ls_for_one_pattern(
inner: vec![], inner: vec![],
}) })
}) })
})
.map_err(|err| ShellError::GenericError { .map_err(|err| ShellError::GenericError {
error: "Unable to create a rayon pool".into(), error: "Unable to create a rayon pool".into(),
msg: err.to_string(), msg: err.to_string(),
span: Some(call_span), span: Some(call_span),
help: None, help: None,
inner: vec![], inner: vec![],
})?; });
if let Err(error) = result {
let _ = tx.send(Value::error(error, call_span));
}
});
});
Ok(rx Ok(rx
.into_iter() .into_iter()