-
Notifications
You must be signed in to change notification settings - Fork 8
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: create plugin path alongside repo cache
- Loading branch information
1 parent
c999d29
commit 97bdc55
Showing
6 changed files
with
429 additions
and
401 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,380 +1,4 @@ | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
use crate::error::Result; | ||
use crate::hc_error; | ||
use dialoguer::Confirm; | ||
use git2::Repository; | ||
use pathbuf::pathbuf; | ||
use regex::Regex; | ||
use serde::{Deserialize, Serialize}; | ||
use std::borrow::Borrow; | ||
use std::collections::HashMap; | ||
use std::fs; | ||
use std::path::{Path, PathBuf}; | ||
use std::result::Result as StdResult; | ||
use std::time::SystemTime; | ||
use tabled::{Table, Tabled}; | ||
use walkdir::{DirEntry, WalkDir}; | ||
|
||
static CACHE_FILE_NAME: &str = "index.json"; | ||
|
||
#[derive(Debug, Clone)] | ||
pub enum CacheSort { | ||
Oldest, | ||
Largest, | ||
Alpha, | ||
} | ||
|
||
#[derive(Debug, Clone)] | ||
pub enum CacheDeleteScope { | ||
All, | ||
Group { | ||
sort: CacheSort, | ||
invert: bool, | ||
n: usize, | ||
}, | ||
} | ||
|
||
#[derive(Debug, Clone)] | ||
pub struct CacheListScope { | ||
pub sort: CacheSort, | ||
pub invert: bool, | ||
pub n: Option<usize>, | ||
} | ||
|
||
#[derive(Debug, Clone, Tabled)] | ||
struct CacheEntry { | ||
pub name: String, | ||
#[tabled(display_with("Self::display_parent", self), rename = "path")] | ||
pub parent: PathBuf, | ||
pub commit: String, | ||
#[tabled(display_with("Self::display_size", self))] | ||
pub size: usize, | ||
#[tabled(display_with("Self::display_modified", self))] | ||
pub modified: SystemTime, | ||
} | ||
impl CacheEntry { | ||
// Helper funcs for displaying CacheEntry using `tabled` crate | ||
fn display_parent(&self) -> String { | ||
self.parent | ||
.clone() | ||
.into_os_string() | ||
.into_string() | ||
.unwrap_or("<DISPLAY_ERROR>".to_owned()) | ||
.to_string() | ||
} | ||
fn display_modified(&self) -> String { | ||
let Ok(dur) = self.modified.duration_since(SystemTime::UNIX_EPOCH) else { | ||
return "<DISPLAY_ERROR>".to_owned(); | ||
}; | ||
let Some(dt) = chrono::DateTime::<chrono::offset::Utc>::from_timestamp( | ||
dur.as_secs() as i64, | ||
dur.subsec_nanos(), | ||
) else { | ||
return "<DISPLAY_ERROR>".to_owned(); | ||
}; | ||
let chars = dt.to_rfc2822().chars().collect::<Vec<char>>(); | ||
// Remove unnecessary " +0000" from end of rfc datetime str | ||
chars[..chars.len() - 6].iter().collect() | ||
} | ||
fn display_size(&self) -> String { | ||
static ONE_KB: f64 = 1000.0; | ||
static ONE_MB: f64 = ONE_KB * 1000.0; | ||
static ONE_GB: f64 = ONE_MB * 1000.0; | ||
let e_size = self.size as f64; | ||
if e_size > ONE_GB { | ||
format!("{:.2} GB", e_size / ONE_GB) | ||
} else if e_size > ONE_MB { | ||
format!("{:.2} MB", e_size / ONE_MB) | ||
} else if e_size > ONE_KB { | ||
format!("{:.2} KB", e_size / ONE_KB) | ||
} else { | ||
format!("{:.0} B", e_size) | ||
} | ||
} | ||
} | ||
|
||
// We store a cached version of the `HcCache` object on disk at the root of the | ||
// `clones` dir in the cache folder. When instantiating an `HcCache` object, we | ||
// load this cached version and use it to short-cut entry size calculation if | ||
// the repo has not changed since the last calculation | ||
#[derive(Debug, Clone, Serialize, Deserialize)] | ||
struct HcCacheDiskEntry { | ||
pub size: usize, | ||
pub commit: String, | ||
pub modified: SystemTime, | ||
} | ||
impl From<CacheEntry> for (PathBuf, HcCacheDiskEntry) { | ||
fn from(value: CacheEntry) -> (PathBuf, HcCacheDiskEntry) { | ||
let path = pathbuf![value.parent.as_path(), value.name.as_str()]; | ||
let entry = HcCacheDiskEntry { | ||
size: value.size, | ||
commit: value.commit, | ||
modified: value.modified, | ||
}; | ||
(path, entry) | ||
} | ||
} | ||
type HcCacheDisk = HashMap<PathBuf, HcCacheDiskEntry>; | ||
fn try_load(path: &Path) -> Result<HcCacheDisk> { | ||
let data = fs::read_to_string(path)?; | ||
let disk: HcCacheDisk = serde_json::from_str(data.as_str())?; | ||
Ok(disk) | ||
} | ||
fn load_or_get_empty(path: &Path) -> HcCacheDisk { | ||
try_load(path).unwrap_or_default() | ||
} | ||
|
||
fn try_get_last_modified(path: &Path) -> Result<SystemTime> { | ||
Ok(fs::metadata(path)?.modified()?) | ||
} | ||
fn get_last_modified_or_now(path: &Path) -> SystemTime { | ||
try_get_last_modified(path).unwrap_or(SystemTime::now()) | ||
} | ||
|
||
/// Starting from a given cache dir, finds and iterates over git repos as "CacheEntry" structs | ||
struct HcCacheIterator { | ||
root: PathBuf, | ||
disk: HcCacheDisk, | ||
wd: Box<dyn Iterator<Item = StdResult<DirEntry, walkdir::Error>>>, | ||
} | ||
impl HcCacheIterator { | ||
pub fn new(dir: &Path) -> Self { | ||
let disk_path = pathbuf![dir, CACHE_FILE_NAME]; | ||
HcCacheIterator { | ||
root: PathBuf::from(dir), | ||
disk: load_or_get_empty(disk_path.as_path()), | ||
wd: Box::new( | ||
WalkDir::new(dir) | ||
.max_depth(5) // reduce time wasted churning through repos | ||
.into_iter() | ||
.filter_entry(|e| e.path().is_dir()), | ||
), | ||
} | ||
} | ||
} | ||
impl HcCacheIterator { | ||
fn path_to_cache_entry(&self, path: &Path) -> Result<CacheEntry> { | ||
let name = path | ||
.file_name() | ||
.ok_or(hc_error!("cache directory doesn't have a name"))? | ||
.to_str() | ||
.unwrap() | ||
.to_owned(); | ||
let repo = Repository::open(path)?; | ||
let commit = repo | ||
.head()? | ||
.peel_to_commit()? | ||
.as_object() | ||
.short_id()? | ||
.as_str() | ||
.unwrap() | ||
.to_owned(); | ||
let modified = get_last_modified_or_now(path); | ||
let cache_subdir = pathbuf![path.strip_prefix(self.root.as_path()).unwrap()]; | ||
let mut parent = cache_subdir.clone(); | ||
parent.pop(); | ||
let size: usize = match self.disk.get(&cache_subdir) { | ||
// If existing cache entry exists and is not outdated, use size | ||
Some(existing) => { | ||
if modified != existing.modified || commit != existing.commit { | ||
fs_extra::dir::get_size(path)? as usize | ||
} else { | ||
existing.size | ||
} | ||
} | ||
None => fs_extra::dir::get_size(path)? as usize, | ||
}; | ||
Ok(CacheEntry { | ||
name, | ||
parent, | ||
commit, | ||
modified, | ||
size, | ||
}) | ||
} | ||
} | ||
impl Iterator for HcCacheIterator { | ||
type Item = CacheEntry; | ||
fn next(&mut self) -> Option<Self::Item> { | ||
loop { | ||
if let Some(Ok(e)) = self.wd.next() { | ||
if e.file_name().to_str().map(|s| s == ".git").unwrap_or(false) { | ||
let mut path = e.into_path(); | ||
path.pop(); // Remove ".git" | ||
if let Ok(ce) = self.path_to_cache_entry(path.as_path()) { | ||
return Some(ce); | ||
} | ||
} | ||
} else { | ||
return None; | ||
} | ||
} | ||
} | ||
} | ||
|
||
pub struct HcCache { | ||
path: PathBuf, | ||
entries: Vec<CacheEntry>, | ||
} | ||
impl HcCache { | ||
pub fn new(path: &Path) -> Self { | ||
let clones_path = pathbuf![path, "clones"]; | ||
let entries: Vec<CacheEntry> = HcCacheIterator::new(clones_path.as_path()).collect(); | ||
HcCache { | ||
path: clones_path, | ||
entries, | ||
} | ||
} | ||
/// Internal function for sorting CacheEntry Vecs | ||
fn sort<A: Borrow<CacheEntry>>(entries: &mut [A], sort: CacheSort, invert: bool) { | ||
// Generic allows sort to handle both owned and borrowed lists | ||
let sort_func: fn(&CacheEntry, &CacheEntry) -> std::cmp::Ordering = match (sort, invert) { | ||
(CacheSort::Alpha, false) => |a, b| a.name.partial_cmp(&b.name).unwrap(), | ||
(CacheSort::Alpha, true) => |a, b| b.name.partial_cmp(&a.name).unwrap(), | ||
(CacheSort::Oldest, false) => |a, b| a.modified.partial_cmp(&b.modified).unwrap(), | ||
(CacheSort::Oldest, true) => |a, b| b.modified.partial_cmp(&a.modified).unwrap(), | ||
(CacheSort::Largest, false) => |a, b| b.size.partial_cmp(&a.size).unwrap(), | ||
(CacheSort::Largest, true) => |a, b| a.size.partial_cmp(&b.size).unwrap(), | ||
}; | ||
entries.sort_by(|a1: &A, a2: &A| sort_func(a1.borrow(), a2.borrow())); | ||
} | ||
/// Delete cache entries | ||
pub fn delete( | ||
&mut self, | ||
scope: CacheDeleteScope, | ||
filter: Option<String>, | ||
force: bool, | ||
) -> Result<()> { | ||
// Parse filter to regex if provided | ||
let opt_pat: Option<Regex> = match filter { | ||
Some(raw_p) => Some(Regex::new(format!("^{raw_p}$").as_str())?), | ||
None => None, | ||
}; | ||
// Drain and partition self.entries into two vecs for saving and deletion | ||
let (to_keep, to_del): (Vec<CacheEntry>, Vec<CacheEntry>) = match scope { | ||
CacheDeleteScope::All => self.entries.drain(0..).partition(|e| match &opt_pat { | ||
Some(pat) => !pat.is_match(e.name.as_str()), | ||
None => true, | ||
}), | ||
CacheDeleteScope::Group { sort, invert, n } => { | ||
// First sort entries in-place in self.entries | ||
HcCache::sort(&mut self.entries, sort, invert); | ||
let mut hits = 0; | ||
// Now get the first N entries that pass filter | ||
self.entries.drain(0..).partition(|e| { | ||
let del = match &opt_pat { | ||
Some(pat) => pat.is_match(e.name.as_str()), | ||
None => true, | ||
}; | ||
// passes filter and below max threshold, delete | ||
if del && hits < n { | ||
hits += 1; | ||
false | ||
// put in do_keep | ||
} else { | ||
true | ||
} | ||
}) | ||
} | ||
}; | ||
// At this point self.entries is empty and we have separated the set of | ||
// entries to delete | ||
if !to_del.is_empty() { | ||
if !force { | ||
// Ask user for confirmation | ||
println!("You will delete the following entries:"); | ||
self.display(to_del.iter().collect()); | ||
let conf = Confirm::new() | ||
.with_prompt("Are you sure you want to delete?") | ||
.interact() | ||
.unwrap(); | ||
if !conf { | ||
// Cleanup by returning entries to storage | ||
self.entries.extend(to_del); | ||
self.entries.extend(to_keep); | ||
return Ok(()); | ||
} | ||
} | ||
// Delete entries, returning failures back to the self.entries list | ||
for entry in to_del { | ||
if let Err(e) = self.internal_delete(&entry) { | ||
println!("Failed to delete entry '{}': {e}", entry.name); | ||
self.entries.push(entry) | ||
} | ||
} | ||
} | ||
self.entries.extend(to_keep); | ||
// self.entries contains `to_keep` plus any entries that were unsuccessfully | ||
// deleted | ||
Ok(()) | ||
} | ||
/// List cache entries | ||
pub fn list(&mut self, scope: CacheListScope, filter: Option<String>) -> Result<()> { | ||
// Parse filter to a regex if provided | ||
let opt_pat: Option<Regex> = match filter { | ||
Some(raw_p) => Some(Regex::new(format!("^{raw_p}$").as_str())?), | ||
None => None, | ||
}; | ||
// Get a vec of &CacheEntry, filtering if applicable | ||
let mut filt_entries = self | ||
.entries | ||
.iter() | ||
.filter(|e| match &opt_pat { | ||
Some(pat) => pat.is_match(e.name.as_str()), | ||
None => true, | ||
}) | ||
.collect::<Vec<&CacheEntry>>(); | ||
|
||
// Sort filtered entries | ||
HcCache::sort(&mut filt_entries, scope.sort, scope.invert); | ||
|
||
// Limit to first N if specified | ||
if let Some(n) = scope.n { | ||
filt_entries.truncate(n); | ||
} | ||
|
||
// Display | ||
self.display(filt_entries); | ||
|
||
Ok(()) | ||
} | ||
/// Internal helper that performs the actual dir deletion | ||
fn internal_delete(&mut self, entry: &CacheEntry) -> Result<()> { | ||
let path = pathbuf![self.path.as_path(), entry.parent.as_path(), &entry.name]; | ||
std::fs::remove_dir_all(path)?; | ||
Ok(()) | ||
} | ||
/// Internal helper for list functions | ||
fn display(&self, to_show: Vec<&CacheEntry>) { | ||
println!("{}", Table::new(to_show)); | ||
} | ||
} | ||
// This causes the current state of the cache to be written to the cache index | ||
// file when the HcCache instance is dropped. On instantiation, the index file | ||
// is referenced opportunistically, it is not treated as ground-truth about the | ||
// contents of the Hipcheck cache. Therefore, while it is preferable to ensure | ||
// `drop` gets called before program termination, the next `HcCache` instance | ||
// will not be corrupted if a hard-kill occurs. | ||
impl Drop for HcCache { | ||
fn drop(&mut self) { | ||
let file_path = pathbuf![self.path.as_path(), CACHE_FILE_NAME]; | ||
// Create a file-oriented representation of the cache | ||
let mut hc_disk_cache = HcCacheDisk::new(); | ||
for e in self.entries.drain(0..) { | ||
let (key, val): (PathBuf, HcCacheDiskEntry) = e.into(); | ||
hc_disk_cache.insert(key, val); | ||
} | ||
// Jsonify cache and write to file | ||
let data = match serde_json::to_string(&hc_disk_cache) { | ||
Ok(d) => d, | ||
Err(e) => { | ||
log::debug!("Failed to jsonify Hipcheck cache info: {e}"); | ||
return; | ||
} | ||
}; | ||
if let Err(e) = fs::write(file_path, data) { | ||
log::debug!("Failed to write Hipcheck cache info: {e}"); | ||
} | ||
} | ||
} | ||
#[allow(clippy::module_inception)] | ||
#[allow(dead_code)] | ||
pub mod plugin_cache; | ||
pub mod repo_cache; |
Oops, something went wrong.