mirror of
https://github.com/nushell/nushell
synced 2025-01-12 13:19:01 +00:00
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:
parent
6ebc0fc3ff
commit
81baf53814
1 changed files with 91 additions and 83 deletions
|
@ -316,6 +316,7 @@ fn ls_for_one_pattern(
|
|||
};
|
||||
|
||||
let hidden_dir_specified = is_hidden_dir(pattern_arg.as_ref());
|
||||
|
||||
let path = pattern_arg.into_spanned(p_tag);
|
||||
let (prefix, paths) = if just_read_dir {
|
||||
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(|| {
|
||||
paths_peek
|
||||
rayon::spawn(move || {
|
||||
let result = paths_peek
|
||||
.par_bridge()
|
||||
.filter_map(move |x| match x {
|
||||
Ok(path) => {
|
||||
|
@ -390,8 +392,9 @@ fn ls_for_one_pattern(
|
|||
if let Ok(remainder) = path.strip_prefix(prefix) {
|
||||
if directory {
|
||||
// When the path is the same as the cwd, path_diff should be "."
|
||||
let path_diff =
|
||||
if let Some(path_diff_not_dot) = diff_paths(&path, &cwd) {
|
||||
let path_diff = if let Some(path_diff_not_dot) =
|
||||
diff_paths(&path, &cwd)
|
||||
{
|
||||
let path_diff_not_dot = path_diff_not_dot.to_string_lossy();
|
||||
if path_diff_not_dot.is_empty() {
|
||||
".".to_string()
|
||||
|
@ -458,14 +461,19 @@ fn ls_for_one_pattern(
|
|||
inner: vec![],
|
||||
})
|
||||
})
|
||||
})
|
||||
.map_err(|err| ShellError::GenericError {
|
||||
error: "Unable to create a rayon pool".into(),
|
||||
msg: err.to_string(),
|
||||
span: Some(call_span),
|
||||
help: None,
|
||||
inner: vec![],
|
||||
})?;
|
||||
});
|
||||
|
||||
if let Err(error) = result {
|
||||
let _ = tx.send(Value::error(error, call_span));
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
Ok(rx
|
||||
.into_iter()
|
||||
|
|
Loading…
Reference in a new issue