mirror of
https://github.com/pesde-pkg/pesde.git
synced 2025-04-06 03:40:59 +01:00
feat: add cas pruning command
Removes unused files from the CAS. Still needs to remove individual package index entries to be complete.
This commit is contained in:
parent
3e4ef00f4a
commit
a38da43670
6 changed files with 237 additions and 4 deletions
52
Cargo.lock
generated
52
Cargo.lock
generated
|
@ -2502,7 +2502,7 @@ checksum = "f9c7c7c8ac16c798734b8a24560c1362120597c40d5e1459f09498f8f6c8f2ba"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"libc",
|
"libc",
|
||||||
"windows",
|
"windows 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -2652,7 +2652,7 @@ dependencies = [
|
||||||
"iana-time-zone-haiku",
|
"iana-time-zone-haiku",
|
||||||
"js-sys",
|
"js-sys",
|
||||||
"wasm-bindgen",
|
"wasm-bindgen",
|
||||||
"windows-core",
|
"windows-core 0.52.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -3656,6 +3656,7 @@ dependencies = [
|
||||||
"url",
|
"url",
|
||||||
"urlencoding",
|
"urlencoding",
|
||||||
"wax",
|
"wax",
|
||||||
|
"windows 0.59.0",
|
||||||
"windows-registry 0.4.0",
|
"windows-registry 0.4.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -5669,10 +5670,20 @@ version = "0.52.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e48a53791691ab099e5e2ad123536d0fff50652600abaf43bbf952894110d0be"
|
checksum = "e48a53791691ab099e5e2ad123536d0fff50652600abaf43bbf952894110d0be"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"windows-core",
|
"windows-core 0.52.0",
|
||||||
"windows-targets 0.52.6",
|
"windows-targets 0.52.6",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "windows"
|
||||||
|
version = "0.59.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7f919aee0a93304be7f62e8e5027811bbba96bcb1de84d6618be56e43f8a32a1"
|
||||||
|
dependencies = [
|
||||||
|
"windows-core 0.59.0",
|
||||||
|
"windows-targets 0.53.0",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows-core"
|
name = "windows-core"
|
||||||
version = "0.52.0"
|
version = "0.52.0"
|
||||||
|
@ -5682,6 +5693,41 @@ dependencies = [
|
||||||
"windows-targets 0.52.6",
|
"windows-targets 0.52.6",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "windows-core"
|
||||||
|
version = "0.59.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "810ce18ed2112484b0d4e15d022e5f598113e220c53e373fb31e67e21670c1ce"
|
||||||
|
dependencies = [
|
||||||
|
"windows-implement",
|
||||||
|
"windows-interface",
|
||||||
|
"windows-result 0.3.0",
|
||||||
|
"windows-strings 0.3.0",
|
||||||
|
"windows-targets 0.53.0",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "windows-implement"
|
||||||
|
version = "0.59.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "83577b051e2f49a058c308f17f273b570a6a758386fc291b5f6a934dd84e48c1"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn 2.0.90",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "windows-interface"
|
||||||
|
version = "0.59.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "cb26fd936d991781ea39e87c3a27285081e3c0da5ca0fcbc02d368cc6f52ff01"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn 2.0.90",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows-registry"
|
name = "windows-registry"
|
||||||
version = "0.2.0"
|
version = "0.2.0"
|
||||||
|
|
|
@ -24,6 +24,7 @@ bin = [
|
||||||
"dep:paste",
|
"dep:paste",
|
||||||
"dep:serde_json",
|
"dep:serde_json",
|
||||||
"dep:windows-registry",
|
"dep:windows-registry",
|
||||||
|
"dep:windows",
|
||||||
"gix/worktree-mutation",
|
"gix/worktree-mutation",
|
||||||
"fs-err/expose_original_error",
|
"fs-err/expose_original_error",
|
||||||
"tokio/rt",
|
"tokio/rt",
|
||||||
|
@ -91,6 +92,7 @@ paste = { version = "1.0.15", optional = true }
|
||||||
|
|
||||||
[target.'cfg(target_os = "windows")'.dependencies]
|
[target.'cfg(target_os = "windows")'.dependencies]
|
||||||
windows-registry = { version = "0.4.0", optional = true }
|
windows-registry = { version = "0.4.0", optional = true }
|
||||||
|
windows = { version = "0.59.0", features = ["Win32_Storage", "Win32_Storage_FileSystem", "Win32_Security"], optional = true }
|
||||||
|
|
||||||
[workspace]
|
[workspace]
|
||||||
resolver = "2"
|
resolver = "2"
|
||||||
|
|
18
src/cli/commands/cas/mod.rs
Normal file
18
src/cli/commands/cas/mod.rs
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
use clap::Subcommand;
|
||||||
|
use pesde::Project;
|
||||||
|
|
||||||
|
mod prune;
|
||||||
|
|
||||||
|
#[derive(Debug, Subcommand)]
|
||||||
|
pub enum CasCommands {
|
||||||
|
/// Removes unused files from the CAS
|
||||||
|
Prune(prune::PruneCommand),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CasCommands {
|
||||||
|
pub async fn run(self, project: Project) -> anyhow::Result<()> {
|
||||||
|
match self {
|
||||||
|
CasCommands::Prune(prune) => prune.run(project).await,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
161
src/cli/commands/cas/prune.rs
Normal file
161
src/cli/commands/cas/prune.rs
Normal file
|
@ -0,0 +1,161 @@
|
||||||
|
use anyhow::Context;
|
||||||
|
use clap::Args;
|
||||||
|
use fs_err::tokio as fs;
|
||||||
|
use pesde::Project;
|
||||||
|
use std::{collections::HashSet, path::Path};
|
||||||
|
use tokio::task::{spawn_blocking, JoinSet};
|
||||||
|
|
||||||
|
#[derive(Debug, Args)]
|
||||||
|
pub struct PruneCommand {}
|
||||||
|
|
||||||
|
#[allow(unreachable_code)]
|
||||||
|
async fn get_nlinks(path: &Path) -> anyhow::Result<u64> {
|
||||||
|
#[cfg(unix)]
|
||||||
|
{
|
||||||
|
use std::os::unix::fs::MetadataExt;
|
||||||
|
let metadata = fs::metadata(path).await?;
|
||||||
|
return Ok(metadata.nlink());
|
||||||
|
}
|
||||||
|
// life if rust stabilized the nightly feature from 2019
|
||||||
|
#[cfg(windows)]
|
||||||
|
{
|
||||||
|
use std::os::windows::ffi::OsStrExt;
|
||||||
|
use windows::{
|
||||||
|
core::PWSTR,
|
||||||
|
Win32::{
|
||||||
|
Foundation::CloseHandle,
|
||||||
|
Storage::FileSystem::{
|
||||||
|
CreateFileW, GetFileInformationByHandle, FILE_ATTRIBUTE_NORMAL,
|
||||||
|
FILE_GENERIC_READ, FILE_SHARE_READ, OPEN_EXISTING,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
let path = path.to_path_buf();
|
||||||
|
return spawn_blocking(move || unsafe {
|
||||||
|
let handle = CreateFileW(
|
||||||
|
PWSTR(
|
||||||
|
path.as_os_str()
|
||||||
|
.encode_wide()
|
||||||
|
.chain(std::iter::once(0))
|
||||||
|
.collect::<Vec<_>>()
|
||||||
|
.as_mut_ptr(),
|
||||||
|
),
|
||||||
|
FILE_GENERIC_READ.0,
|
||||||
|
FILE_SHARE_READ,
|
||||||
|
None,
|
||||||
|
OPEN_EXISTING,
|
||||||
|
FILE_ATTRIBUTE_NORMAL,
|
||||||
|
None,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
let mut info =
|
||||||
|
windows::Win32::Storage::FileSystem::BY_HANDLE_FILE_INFORMATION::default();
|
||||||
|
let res = GetFileInformationByHandle(handle, &mut info);
|
||||||
|
CloseHandle(handle)?;
|
||||||
|
res?;
|
||||||
|
|
||||||
|
Ok(info.nNumberOfLinks as u64)
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
#[cfg(not(any(unix, windows)))]
|
||||||
|
{
|
||||||
|
compile_error!("unsupported platform");
|
||||||
|
}
|
||||||
|
anyhow::bail!("unsupported platform")
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn remove_hashes(cas_dir: &Path) -> anyhow::Result<HashSet<String>> {
|
||||||
|
let mut tasks = JoinSet::new();
|
||||||
|
|
||||||
|
let mut cas_entries = fs::read_dir(cas_dir)
|
||||||
|
.await
|
||||||
|
.context("failed to read directory")?;
|
||||||
|
|
||||||
|
while let Some(cas_entry) = cas_entries
|
||||||
|
.next_entry()
|
||||||
|
.await
|
||||||
|
.context("failed to read dir entry")?
|
||||||
|
{
|
||||||
|
let prefix = cas_entry.file_name();
|
||||||
|
let Some(prefix) = prefix.to_str() else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
// we only want hash directories
|
||||||
|
if prefix.len() != 2 {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let prefix = prefix.to_string();
|
||||||
|
|
||||||
|
tasks.spawn(async move {
|
||||||
|
let mut hash_entries = fs::read_dir(cas_entry.path())
|
||||||
|
.await
|
||||||
|
.context("failed to read hash directory")?;
|
||||||
|
|
||||||
|
let mut tasks = JoinSet::new();
|
||||||
|
|
||||||
|
while let Some(hash_entry) = hash_entries
|
||||||
|
.next_entry()
|
||||||
|
.await
|
||||||
|
.context("failed to read hash dir entry")?
|
||||||
|
{
|
||||||
|
let hash = hash_entry.file_name();
|
||||||
|
let hash = hash.to_str().expect("non-UTF-8 hash").to_string();
|
||||||
|
let hash = format!("{prefix}{hash}");
|
||||||
|
|
||||||
|
let path = hash_entry.path();
|
||||||
|
tasks.spawn(async move {
|
||||||
|
let nlinks = get_nlinks(&path)
|
||||||
|
.await
|
||||||
|
.context("failed to count file usage")?;
|
||||||
|
if nlinks != 1 {
|
||||||
|
return Ok::<_, anyhow::Error>(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
fs::remove_file(path)
|
||||||
|
.await
|
||||||
|
.context("failed to remove unused file")?;
|
||||||
|
|
||||||
|
Ok::<_, anyhow::Error>(Some(hash))
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut removed_hashes = HashSet::new();
|
||||||
|
|
||||||
|
while let Some(removed_hash) = tasks.join_next().await {
|
||||||
|
let Some(hash) = removed_hash.unwrap()? else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
|
||||||
|
removed_hashes.insert(hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok::<_, anyhow::Error>(removed_hashes)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut res = HashSet::new();
|
||||||
|
|
||||||
|
while let Some(removed_hashes) = tasks.join_next().await {
|
||||||
|
res.extend(removed_hashes.unwrap()?);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(res)
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PruneCommand {
|
||||||
|
pub async fn run(self, project: Project) -> anyhow::Result<()> {
|
||||||
|
// CAS structure:
|
||||||
|
// /2 first chars of hash/rest of hash
|
||||||
|
// /index/hash/name/version/target
|
||||||
|
// /wally_index/hash/name/version
|
||||||
|
// /git_index/hash/hash
|
||||||
|
// the last thing in the path is the serialized PackageFs
|
||||||
|
let _ = remove_hashes(project.cas_dir()).await?;
|
||||||
|
|
||||||
|
todo!("remove unused index entries");
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,6 +2,7 @@ use pesde::Project;
|
||||||
|
|
||||||
mod add;
|
mod add;
|
||||||
mod auth;
|
mod auth;
|
||||||
|
mod cas;
|
||||||
mod config;
|
mod config;
|
||||||
mod deprecate;
|
mod deprecate;
|
||||||
mod execute;
|
mod execute;
|
||||||
|
@ -30,6 +31,10 @@ pub enum Subcommand {
|
||||||
#[command(subcommand)]
|
#[command(subcommand)]
|
||||||
Config(config::ConfigCommands),
|
Config(config::ConfigCommands),
|
||||||
|
|
||||||
|
/// CAS-related commands
|
||||||
|
#[command(subcommand)]
|
||||||
|
Cas(cas::CasCommands),
|
||||||
|
|
||||||
/// Initializes a manifest file in the current directory
|
/// Initializes a manifest file in the current directory
|
||||||
Init(init::InitCommand),
|
Init(init::InitCommand),
|
||||||
|
|
||||||
|
@ -83,6 +88,7 @@ impl Subcommand {
|
||||||
match self {
|
match self {
|
||||||
Subcommand::Auth(auth) => auth.run(project, reqwest).await,
|
Subcommand::Auth(auth) => auth.run(project, reqwest).await,
|
||||||
Subcommand::Config(config) => config.run().await,
|
Subcommand::Config(config) => config.run().await,
|
||||||
|
Subcommand::Cas(cas) => cas.run(project).await,
|
||||||
Subcommand::Init(init) => init.run(project).await,
|
Subcommand::Init(init) => init.run(project).await,
|
||||||
Subcommand::Run(run) => run.run(project).await,
|
Subcommand::Run(run) => run.run(project).await,
|
||||||
Subcommand::Install(install) => install.run(project, reqwest).await,
|
Subcommand::Install(install) => install.run(project, reqwest).await,
|
||||||
|
|
|
@ -297,7 +297,7 @@ pub fn display_err(result: anyhow::Result<()>, prefix: &str) {
|
||||||
if !cause.is_empty() {
|
if !cause.is_empty() {
|
||||||
eprintln!("{}:", ERROR_STYLE.apply_to("caused by"));
|
eprintln!("{}:", ERROR_STYLE.apply_to("caused by"));
|
||||||
for err in cause {
|
for err in cause {
|
||||||
eprintln!("\t- {}", ERROR_STYLE.apply_to(err));
|
eprintln!("\t- {err}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue