perf: Speed up path_to_string in cases where the path is already in the proper format

This commit is contained in:
Chris Hennick 2024-05-03 14:05:39 -07:00
parent 0e97f9eebe
commit efbea6f747
No known key found for this signature in database
GPG key ID: DA47AABA4961C509

View file

@ -217,6 +217,8 @@ impl Zip64CentralDirectoryEnd {
/// Converts a path to the ZIP format (forward-slash-delimited and normalized). /// Converts a path to the ZIP format (forward-slash-delimited and normalized).
pub(crate) fn path_to_string<T: AsRef<Path>>(path: T) -> String { pub(crate) fn path_to_string<T: AsRef<Path>>(path: T) -> String {
let original = path.as_ref().to_str();
let mut recreate = original.is_some();
let mut normalized_components = Vec::new(); let mut normalized_components = Vec::new();
// Empty element ensures the path has a leading slash, with no extra allocation after the join // Empty element ensures the path has a leading slash, with no extra allocation after the join
@ -225,15 +227,37 @@ pub(crate) fn path_to_string<T: AsRef<Path>>(path: T) -> String {
for component in path.as_ref().components() { for component in path.as_ref().components() {
match component { match component {
Component::Normal(os_str) => { Component::Normal(os_str) => {
match os_str.to_str() {
Some(valid_str) => normalized_components.push(Cow::Borrowed(valid_str)),
None => {
recreate = true;
normalized_components.push(os_str.to_string_lossy()); normalized_components.push(os_str.to_string_lossy());
} }
}
}
Component::ParentDir => { Component::ParentDir => {
recreate = true;
if normalized_components.len() > 1 { if normalized_components.len() > 1 {
normalized_components.pop(); normalized_components.pop();
} }
} }
_ => {} _ => {
recreate = true;
} }
} }
}
if recreate {
normalized_components.join("/") normalized_components.join("/")
} else {
let original = original.unwrap();
if !original.starts_with('/') {
let mut slash_original = String::with_capacity(original.len() + 1);
slash_original.push('/');
slash_original.push_str(original);
slash_original
} else {
original.to_string()
}
}
} }