Rewrite untar: Takes a filter fn instead of array

Signed-off-by: Jiahao XU <Jiahao_XU@outlook.com>
This commit is contained in:
Jiahao XU 2022-06-10 16:35:51 +10:00
parent a681f3a156
commit 1d139324c7
No known key found for this signature in database
GPG key ID: 591C0B03040416D6
6 changed files with 55 additions and 48 deletions

View file

@ -102,14 +102,19 @@ pub async fn fetch_crate_cratesio(
debug!("Fetching crate from: {crate_url} and extracting Cargo.toml from it"); debug!("Fetching crate from: {crate_url} and extracting Cargo.toml from it");
let crate_dir = format!("{name}-{version_name}"); let crate_dir: PathBuf = format!("{name}-{version_name}").into();
let crate_path = temp_dir.join(&crate_dir); let crate_path = temp_dir.join(&crate_dir);
let cargo_toml = crate_dir.join("Cargo.toml");
let src = crate_dir.join("src");
let main = src.join("main.rs");
let bin = src.join("bin");
download_and_extract( download_and_extract(
Url::parse(&crate_url)?, Url::parse(&crate_url)?,
PkgFmt::Tgz, PkgFmt::Tgz,
&temp_dir, &temp_dir,
Some([Path::new(&crate_dir).join("Cargo.toml").into()]), Some(move |path: &Path| path == cargo_toml || path == main || path.starts_with(&bin)),
) )
.await?; .await?;

View file

@ -43,7 +43,7 @@ impl super::Fetcher for GhCrateMeta {
async fn fetch_and_extract(&self, dst: &Path) -> Result<(), BinstallError> { async fn fetch_and_extract(&self, dst: &Path) -> Result<(), BinstallError> {
let url = self.url()?; let url = self.url()?;
info!("Downloading package from: '{url}'"); info!("Downloading package from: '{url}'");
download_and_extract::<_, 0>(url, self.pkg_fmt(), dst, None).await download_and_extract::<fn(&Path) -> bool, _>(url, self.pkg_fmt(), dst, None).await
} }
fn pkg_fmt(&self) -> PkgFmt { fn pkg_fmt(&self) -> PkgFmt {

View file

@ -40,7 +40,8 @@ impl super::Fetcher for QuickInstall {
async fn fetch_and_extract(&self, dst: &Path) -> Result<(), BinstallError> { async fn fetch_and_extract(&self, dst: &Path) -> Result<(), BinstallError> {
let url = self.package_url(); let url = self.package_url();
info!("Downloading package from: '{url}'"); info!("Downloading package from: '{url}'");
download_and_extract::<_, 0>(Url::parse(&url)?, self.pkg_fmt(), dst, None).await download_and_extract::<fn(&Path) -> bool, _>(Url::parse(&url)?, self.pkg_fmt(), dst, None)
.await
} }
fn pkg_fmt(&self) -> PkgFmt { fn pkg_fmt(&self) -> PkgFmt {

View file

@ -1,5 +1,4 @@
use std::{ use std::{
borrow::Cow,
io::{stderr, stdin, Write}, io::{stderr, stdin, Write},
path::{Path, PathBuf}, path::{Path, PathBuf},
}; };
@ -46,14 +45,15 @@ pub async fn remote_exists(url: Url, method: Method) -> Result<bool, BinstallErr
/// Download a file from the provided URL and extract it to the provided path /// Download a file from the provided URL and extract it to the provided path
/// ///
/// * `desired_outputs - If Some(_) and `fmt` is not `PkgFmt::Bin` or /// * `filter` - If Some, then it will pass the path of the file to it
/// `PkgFmt::Zip`, then it will filter the tar and only extract files /// and only extract ones which filter returns `true`.
/// specified in it. /// Note that this is a best-effort and it only works when `fmt`
pub async fn download_and_extract<P: AsRef<Path>, const N: usize>( /// is not `PkgFmt::Bin` or `PkgFmt::Zip`.
pub async fn download_and_extract<Filter: FnMut(&Path) -> bool + Send + 'static, P: AsRef<Path>>(
url: Url, url: Url,
fmt: PkgFmt, fmt: PkgFmt,
path: P, path: P,
desired_outputs: Option<[Cow<'static, Path>; N]>, filter: Option<Filter>,
) -> Result<(), BinstallError> { ) -> Result<(), BinstallError> {
debug!("Downloading from: '{url}'"); debug!("Downloading from: '{url}'");
@ -69,7 +69,7 @@ pub async fn download_and_extract<P: AsRef<Path>, const N: usize>(
let path = path.as_ref(); let path = path.as_ref();
debug!("Downloading to file: '{}'", path.display()); debug!("Downloading to file: '{}'", path.display());
extract_archive_stream(resp.bytes_stream(), path, fmt, desired_outputs).await?; extract_archive_stream(resp.bytes_stream(), path, fmt, filter).await?;
debug!("Download OK, written to file: '{}'", path.display()); debug!("Download OK, written to file: '{}'", path.display());

View file

@ -1,4 +1,3 @@
use std::borrow::Cow;
use std::fs; use std::fs;
use std::io::{self, Seek, Write}; use std::io::{self, Seek, Write};
use std::path::Path; use std::path::Path;
@ -29,12 +28,14 @@ struct AsyncExtracterInner {
} }
impl AsyncExtracterInner { impl AsyncExtracterInner {
/// * `desired_outputs - If Some(_), then it will filter the tar /// * `filter` - If Some, then it will pass the path of the file to it
/// and only extract files specified in it. /// and only extract ones which filter returns `true`.
fn new<const N: usize>( /// Note that this is a best-effort and it only works when `fmt`
/// is not `PkgFmt::Bin` or `PkgFmt::Zip`.
fn new<Filter: FnMut(&Path) -> bool + Send + 'static>(
path: &Path, path: &Path,
fmt: PkgFmt, fmt: PkgFmt,
desired_outputs: Option<[Cow<'static, Path>; N]>, filter: Option<Filter>,
) -> Self { ) -> Self {
let path = path.to_owned(); let path = path.to_owned();
let (tx, rx) = mpsc::channel::<Content>(100); let (tx, rx) = mpsc::channel::<Content>(100);
@ -73,12 +74,9 @@ impl AsyncExtracterInner {
unzip(file, &path)?; unzip(file, &path)?;
} }
_ => extract_compressed_from_readable( _ => {
ReadableRx::new(&mut rx), extract_compressed_from_readable(ReadableRx::new(&mut rx), fmt, &path, filter)?
fmt, }
&path,
desired_outputs.as_ref().map(|arr| &arr[..]),
)?,
} }
Ok(()) Ok(())
@ -181,16 +179,16 @@ impl AsyncExtracter {
/// for the bin. /// for the bin.
/// Otherwise, it is the directory where the extracted content will be put. /// Otherwise, it is the directory where the extracted content will be put.
/// * `fmt` - The format of the archive to feed in. /// * `fmt` - The format of the archive to feed in.
/// * `desired_outputs - If Some(_), then it will filter the tar and /// * `filter` - If Some, then it will pass the path of the file to it
/// only extract files specified in it. /// and only extract ones which filter returns `true`.
/// Note that this is a best-effort and it only works when `fmt` /// Note that this is a best-effort and it only works when `fmt`
/// is not `PkgFmt::Bin` or `PkgFmt::Zip`. /// is not `PkgFmt::Bin` or `PkgFmt::Zip`.
fn new<const N: usize>( fn new<Filter: FnMut(&Path) -> bool + Send + 'static>(
path: &Path, path: &Path,
fmt: PkgFmt, fmt: PkgFmt,
desired_outputs: Option<[Cow<'static, Path>; N]>, filter: Option<Filter>,
) -> Self { ) -> Self {
let inner = AsyncExtracterInner::new(path, fmt, desired_outputs); let inner = AsyncExtracterInner::new(path, fmt, filter);
Self(guard(inner, AsyncExtracterInner::abort)) Self(guard(inner, AsyncExtracterInner::abort))
} }
@ -209,20 +207,20 @@ impl AsyncExtracter {
/// for the bin. /// for the bin.
/// Otherwise, it is the directory where the extracted content will be put. /// Otherwise, it is the directory where the extracted content will be put.
/// * `fmt` - The format of the archive to feed in. /// * `fmt` - The format of the archive to feed in.
/// * `desired_outputs - If Some(_), then it will filter the tar and /// * `filter` - If Some, then it will pass the path of the file to it
/// only extract files specified in it. /// and only extract ones which filter returns `true`.
/// Note that this is a best-effort and it only works when `fmt` /// Note that this is a best-effort and it only works when `fmt`
/// is not `PkgFmt::Bin` or `PkgFmt::Zip`. /// is not `PkgFmt::Bin` or `PkgFmt::Zip`.
pub async fn extract_archive_stream<E, const N: usize>( pub async fn extract_archive_stream<Filter: FnMut(&Path) -> bool + Send + 'static, E>(
mut stream: impl Stream<Item = Result<Bytes, E>> + Unpin, mut stream: impl Stream<Item = Result<Bytes, E>> + Unpin,
output: &Path, output: &Path,
fmt: PkgFmt, fmt: PkgFmt,
desired_outputs: Option<[Cow<'static, Path>; N]>, filter: Option<Filter>,
) -> Result<(), BinstallError> ) -> Result<(), BinstallError>
where where
BinstallError: From<E>, BinstallError: From<E>,
{ {
let mut extracter = AsyncExtracter::new(output, fmt, desired_outputs); let mut extracter = AsyncExtracter::new(output, fmt, filter);
while let Some(res) = stream.next().await { while let Some(res) = stream.next().await {
extracter.feed(res?).await?; extracter.feed(res?).await?;

View file

@ -1,4 +1,3 @@
use std::borrow::Cow;
use std::fs::{self, File}; use std::fs::{self, File};
use std::io::Read; use std::io::Read;
use std::path::Path; use std::path::Path;
@ -12,23 +11,25 @@ use zstd::stream::Decoder as ZstdDecoder;
use crate::{BinstallError, PkgFmt}; use crate::{BinstallError, PkgFmt};
/// * `desired_outputs - If Some(_), then it will filter the tar /// * `filter` - If Some, then it will pass the path of the file to it
/// and only extract files specified in it. /// and only extract ones which filter returns `true`.
fn untar( /// Note that this is a best-effort and it only works when `fmt`
/// is not `PkgFmt::Bin` or `PkgFmt::Zip`.
fn untar<Filter: FnMut(&Path) -> bool>(
dat: impl Read, dat: impl Read,
path: &Path, path: &Path,
desired_outputs: Option<&[Cow<'_, Path>]>, filter: Option<Filter>,
) -> Result<(), BinstallError> { ) -> Result<(), BinstallError> {
let mut tar = Archive::new(dat); let mut tar = Archive::new(dat);
if let Some(desired_outputs) = desired_outputs { if let Some(mut filter) = filter {
debug!("Untaring only {desired_outputs:#?}"); debug!("Untaring with filter");
for res in tar.entries()? { for res in tar.entries()? {
let mut entry = res?; let mut entry = res?;
let entry_path = entry.path()?; let entry_path = entry.path()?;
if desired_outputs.contains(&entry_path) { if filter(&entry_path) {
debug!("Extracting {entry_path:#?}"); debug!("Extracting {entry_path:#?}");
let dst = path.join(entry_path); let dst = path.join(entry_path);
@ -49,34 +50,36 @@ fn untar(
/// Extract files from the specified source onto the specified path. /// Extract files from the specified source onto the specified path.
/// ///
/// * `fmt` - must not be `PkgFmt::Bin` or `PkgFmt::Zip`. /// * `fmt` - must not be `PkgFmt::Bin` or `PkgFmt::Zip`.
/// * `desired_outputs - If Some(_), then it will filter the tar /// * `filter` - If Some, then it will pass the path of the file to it
/// and only extract files specified in it. /// and only extract ones which filter returns `true`.
pub(crate) fn extract_compressed_from_readable( /// Note that this is a best-effort and it only works when `fmt`
/// is not `PkgFmt::Bin` or `PkgFmt::Zip`.
pub(crate) fn extract_compressed_from_readable<Filter: FnMut(&Path) -> bool>(
dat: impl Read, dat: impl Read,
fmt: PkgFmt, fmt: PkgFmt,
path: &Path, path: &Path,
desired_outputs: Option<&[Cow<'_, Path>]>, filter: Option<Filter>,
) -> Result<(), BinstallError> { ) -> Result<(), BinstallError> {
match fmt { match fmt {
PkgFmt::Tar => { PkgFmt::Tar => {
// Extract to install dir // Extract to install dir
debug!("Extracting from tar archive to `{path:?}`"); debug!("Extracting from tar archive to `{path:?}`");
untar(dat, path, desired_outputs)? untar(dat, path, filter)?
} }
PkgFmt::Tgz => { PkgFmt::Tgz => {
// Extract to install dir // Extract to install dir
debug!("Decompressing from tgz archive to `{path:?}`"); debug!("Decompressing from tgz archive to `{path:?}`");
let tar = GzDecoder::new(dat); let tar = GzDecoder::new(dat);
untar(tar, path, desired_outputs)?; untar(tar, path, filter)?;
} }
PkgFmt::Txz => { PkgFmt::Txz => {
// Extract to install dir // Extract to install dir
debug!("Decompressing from txz archive to `{path:?}`"); debug!("Decompressing from txz archive to `{path:?}`");
let tar = XzDecoder::new(dat); let tar = XzDecoder::new(dat);
untar(tar, path, desired_outputs)?; untar(tar, path, filter)?;
} }
PkgFmt::Tzstd => { PkgFmt::Tzstd => {
// Extract to install dir // Extract to install dir
@ -87,7 +90,7 @@ pub(crate) fn extract_compressed_from_readable(
// as &[] by ZstdDecoder::new, thus ZstdDecoder::new // as &[] by ZstdDecoder::new, thus ZstdDecoder::new
// should not return any error. // should not return any error.
let tar = ZstdDecoder::new(dat)?; let tar = ZstdDecoder::new(dat)?;
untar(tar, path, desired_outputs)?; untar(tar, path, filter)?;
} }
PkgFmt::Zip => panic!("Unexpected PkgFmt::Zip!"), PkgFmt::Zip => panic!("Unexpected PkgFmt::Zip!"),
PkgFmt::Bin => panic!("Unexpected PkgFmt::Bin!"), PkgFmt::Bin => panic!("Unexpected PkgFmt::Bin!"),