mirror of
https://github.com/nushell/nushell.git
synced 2025-01-21 13:50:11 +01: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
@ -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()
|
||||||
|
Loading…
Reference in New Issue
Block a user