diff --git a/src/cli/commands/cas/prune.rs b/src/cli/commands/cas/prune.rs index 448cd0b..0bafeea 100644 --- a/src/cli/commands/cas/prune.rs +++ b/src/cli/commands/cas/prune.rs @@ -1,6 +1,9 @@ -use crate::cli::{ - reporters::run_with_reporter, - style::{INFO_STYLE, SUCCESS_STYLE}, +use crate::{ + cli::{ + reporters::run_with_reporter, + style::{INFO_STYLE, SUCCESS_STYLE}, + }, + util::remove_empty_dir, }; use anyhow::Context; use async_stream::try_stream; @@ -150,9 +153,15 @@ async fn discover_cas_packages(cas_dir: &Path) -> anyhow::Result tasks, + Err(e) if e.kind() == std::io::ErrorKind::NotFound => return Ok(res), + Err(e) => return Err(e).context("failed to read cas index directory"), + }; + + let mut tasks = tasks .map(|entry| async move { read_entry(entry.context("failed to read cas index dir entry")?).await }) @@ -160,8 +169,6 @@ async fn discover_cas_packages(cas_dir: &Path) -> anyhow::Result anyhow::Result anyhow::Result> { - let mut tasks = read_dir_stream(cas_dir) - .await? + let mut res = HashSet::new(); + + let tasks = match read_dir_stream(cas_dir).await { + Ok(tasks) => tasks, + Err(e) if e.kind() == std::io::ErrorKind::NotFound => return Ok(res), + Err(e) => return Err(e).context("failed to read cas directory"), + }; + + let mut tasks = tasks .map(|cas_entry| async move { let cas_entry = cas_entry.context("failed to read cas dir entry")?; let prefix = cas_entry.file_name(); @@ -212,10 +226,14 @@ async fn remove_hashes(cas_dir: &Path) -> anyhow::Result> { return Ok(None); } - fs::remove_file(path) + fs::remove_file(&path) .await .context("failed to remove unused file")?; + if let Some(parent) = path.parent() { + remove_empty_dir(parent).await?; + } + Ok(Some(hash)) } }) @@ -238,8 +256,6 @@ async fn remove_hashes(cas_dir: &Path) -> anyhow::Result> { .await .0; - let mut res = HashSet::new(); - while let Some(removed_hashes) = tasks.join_next().await { let Some(removed_hashes) = removed_hashes.unwrap()? else { continue; @@ -288,10 +304,24 @@ impl PruneCommand { }; if removed_hashes.contains(&hash) { + let cas_dir = project.cas_dir().to_path_buf(); tasks.spawn(async move { - fs::remove_file(path) + fs::remove_file(dbg!(&path)) .await - .context("failed to remove unused file") + .context("failed to remove unused file")?; + + // remove empty directories up to the cas dir + let mut path = &*path; + while let Some(parent) = path.parent() { + if parent == cas_dir { + break; + } + + remove_empty_dir(parent).await?; + path = parent; + } + + Ok::<_, anyhow::Error>(()) }); removed_packages += 1; // if at least one file is removed, the package is not used diff --git a/src/linking/incremental.rs b/src/linking/incremental.rs index bf352f9..631a891 100644 --- a/src/linking/incremental.rs +++ b/src/linking/incremental.rs @@ -1,6 +1,6 @@ use crate::{ - all_packages_dirs, graph::DependencyGraphWithTarget, manifest::Alias, Project, - PACKAGES_CONTAINER_NAME, SCRIPTS_LINK_FOLDER, + all_packages_dirs, graph::DependencyGraphWithTarget, manifest::Alias, util::remove_empty_dir, + Project, PACKAGES_CONTAINER_NAME, SCRIPTS_LINK_FOLDER, }; use fs_err::tokio as fs; use futures::FutureExt; @@ -11,15 +11,6 @@ use std::{ }; use tokio::task::JoinSet; -async fn remove_empty_dir(path: &Path) -> std::io::Result<()> { - match fs::remove_dir(path).await { - Ok(()) => Ok(()), - Err(e) if e.kind() == std::io::ErrorKind::NotFound => Ok(()), - Err(e) if e.kind() == std::io::ErrorKind::DirectoryNotEmpty => Ok(()), - Err(e) => Err(e), - } -} - fn index_entry( entry: fs::DirEntry, packages_index_dir: &Path, diff --git a/src/util.rs b/src/util.rs index 5cf4e20..540a338 100644 --- a/src/util.rs +++ b/src/util.rs @@ -1,9 +1,13 @@ use crate::AuthConfig; +use fs_err::tokio as fs; use gix::bstr::BStr; use semver::Version; use serde::{Deserialize, Deserializer, Serializer}; use sha2::{Digest, Sha256}; -use std::collections::{BTreeMap, HashSet}; +use std::{ + collections::{BTreeMap, HashSet}, + path::Path, +}; pub fn authenticate_conn( conn: &mut gix::remote::Connection< @@ -95,3 +99,16 @@ pub fn no_build_metadata(version: &Version) -> Version { version.build = semver::BuildMetadata::EMPTY; version } + +pub async fn remove_empty_dir(path: &Path) -> std::io::Result<()> { + match fs::remove_dir(path).await { + Ok(()) => Ok(()), + Err(e) if e.kind() == std::io::ErrorKind::NotFound => Ok(()), + Err(e) if e.kind() == std::io::ErrorKind::DirectoryNotEmpty => Ok(()), + // concurrent removal on Windows seems to fail with PermissionDenied + // TODO: investigate why this happens and whether we can avoid it without ignoring all PermissionDenied errors + #[cfg(windows)] + Err(e) if e.kind() == std::io::ErrorKind::PermissionDenied => Ok(()), + Err(e) => Err(e), + } +}