mirror of
https://github.com/cargo-bins/cargo-binstall.git
synced 2025-04-22 13:38:43 +00:00
Refactor mod drivers
: Extract out sub mods
Signed-off-by: Jiahao XU <Jiahao_XU@outlook.com>
This commit is contained in:
parent
f8c8c66f57
commit
6c6055da69
3 changed files with 133 additions and 119 deletions
124
src/drivers.rs
124
src/drivers.rs
|
@ -1,126 +1,12 @@
|
||||||
use std::collections::BTreeSet;
|
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
use std::time::Duration;
|
|
||||||
|
|
||||||
use crates_io_api::AsyncClient;
|
use crate::BinstallError;
|
||||||
use log::debug;
|
|
||||||
use semver::{Version, VersionReq};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
use crate::{helpers::*, BinstallError, TarBasedFmt};
|
mod cratesio;
|
||||||
|
pub use cratesio::*;
|
||||||
|
|
||||||
fn find_version<'a, V: Iterator<Item = &'a String>>(
|
mod version;
|
||||||
requirement: &str,
|
use version::find_version;
|
||||||
version_iter: V,
|
|
||||||
) -> Result<Version, BinstallError> {
|
|
||||||
// Parse version requirement
|
|
||||||
let version_req = VersionReq::parse(requirement).map_err(|err| BinstallError::VersionReq {
|
|
||||||
req: requirement.into(),
|
|
||||||
err,
|
|
||||||
})?;
|
|
||||||
|
|
||||||
// Filter for matching versions
|
|
||||||
let filtered: BTreeSet<_> = version_iter
|
|
||||||
.filter_map(|v| {
|
|
||||||
// Remove leading `v` for git tags
|
|
||||||
let ver_str = match v.strip_prefix('s') {
|
|
||||||
Some(v) => v,
|
|
||||||
None => v,
|
|
||||||
};
|
|
||||||
|
|
||||||
// Parse out version
|
|
||||||
let ver = Version::parse(ver_str).ok()?;
|
|
||||||
debug!("Version: {:?}", ver);
|
|
||||||
|
|
||||||
// Filter by version match
|
|
||||||
if version_req.matches(&ver) {
|
|
||||||
Some(ver)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
debug!("Filtered: {:?}", filtered);
|
|
||||||
|
|
||||||
// Return highest version
|
|
||||||
filtered
|
|
||||||
.iter()
|
|
||||||
.max()
|
|
||||||
.cloned()
|
|
||||||
.ok_or(BinstallError::VersionMismatch { req: version_req })
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Fetch a crate Cargo.toml by name and version from crates.io
|
|
||||||
pub async fn fetch_crate_cratesio(
|
|
||||||
name: &str,
|
|
||||||
version_req: &str,
|
|
||||||
temp_dir: &Path,
|
|
||||||
) -> Result<PathBuf, BinstallError> {
|
|
||||||
// Fetch / update index
|
|
||||||
debug!("Looking up crate information");
|
|
||||||
|
|
||||||
// Build crates.io api client
|
|
||||||
let api_client = AsyncClient::new(
|
|
||||||
"cargo-binstall (https://github.com/ryankurte/cargo-binstall)",
|
|
||||||
Duration::from_millis(100),
|
|
||||||
)
|
|
||||||
.expect("bug: invalid user agent");
|
|
||||||
|
|
||||||
// Fetch online crate information
|
|
||||||
let base_info =
|
|
||||||
api_client
|
|
||||||
.get_crate(name.as_ref())
|
|
||||||
.await
|
|
||||||
.map_err(|err| BinstallError::CratesIoApi {
|
|
||||||
crate_name: name.into(),
|
|
||||||
err,
|
|
||||||
})?;
|
|
||||||
|
|
||||||
// Locate matching version
|
|
||||||
let version_iter =
|
|
||||||
base_info
|
|
||||||
.versions
|
|
||||||
.iter()
|
|
||||||
.filter_map(|v| if !v.yanked { Some(&v.num) } else { None });
|
|
||||||
let version_name = find_version(version_req, version_iter)?;
|
|
||||||
|
|
||||||
// Fetch information for the filtered version
|
|
||||||
let version = base_info
|
|
||||||
.versions
|
|
||||||
.iter()
|
|
||||||
.find(|v| v.num == version_name.to_string())
|
|
||||||
.ok_or_else(|| BinstallError::VersionUnavailable {
|
|
||||||
crate_name: name.into(),
|
|
||||||
v: version_name.clone(),
|
|
||||||
})?;
|
|
||||||
|
|
||||||
debug!("Found information for crate version: '{}'", version.num);
|
|
||||||
|
|
||||||
// Download crate to temporary dir (crates.io or git?)
|
|
||||||
let crate_url = format!("https://crates.io/{}", version.dl_path);
|
|
||||||
|
|
||||||
debug!("Fetching crate from: {crate_url} and extracting Cargo.toml from it");
|
|
||||||
|
|
||||||
let crate_dir: PathBuf = format!("{name}-{version_name}").into();
|
|
||||||
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_with_filter(
|
|
||||||
Url::parse(&crate_url)?,
|
|
||||||
TarBasedFmt::Tgz,
|
|
||||||
&temp_dir,
|
|
||||||
move |path: &Path| path == cargo_toml || path == main || path.starts_with(&bin),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
// Return crate directory
|
|
||||||
Ok(crate_path)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Fetch a crate by name and version from github
|
/// Fetch a crate by name and version from github
|
||||||
/// TODO: implement this
|
/// TODO: implement this
|
||||||
|
|
80
src/drivers/cratesio.rs
Normal file
80
src/drivers/cratesio.rs
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
use std::time::Duration;
|
||||||
|
|
||||||
|
use crates_io_api::AsyncClient;
|
||||||
|
use log::debug;
|
||||||
|
use url::Url;
|
||||||
|
|
||||||
|
use super::find_version;
|
||||||
|
use crate::{helpers::*, BinstallError, TarBasedFmt};
|
||||||
|
|
||||||
|
/// Fetch a crate Cargo.toml by name and version from crates.io
|
||||||
|
pub async fn fetch_crate_cratesio(
|
||||||
|
name: &str,
|
||||||
|
version_req: &str,
|
||||||
|
temp_dir: &Path,
|
||||||
|
) -> Result<PathBuf, BinstallError> {
|
||||||
|
// Fetch / update index
|
||||||
|
debug!("Looking up crate information");
|
||||||
|
|
||||||
|
// Build crates.io api client
|
||||||
|
let api_client = AsyncClient::new(
|
||||||
|
"cargo-binstall (https://github.com/ryankurte/cargo-binstall)",
|
||||||
|
Duration::from_millis(100),
|
||||||
|
)
|
||||||
|
.expect("bug: invalid user agent");
|
||||||
|
|
||||||
|
// Fetch online crate information
|
||||||
|
let base_info =
|
||||||
|
api_client
|
||||||
|
.get_crate(name.as_ref())
|
||||||
|
.await
|
||||||
|
.map_err(|err| BinstallError::CratesIoApi {
|
||||||
|
crate_name: name.into(),
|
||||||
|
err,
|
||||||
|
})?;
|
||||||
|
|
||||||
|
// Locate matching version
|
||||||
|
let version_iter =
|
||||||
|
base_info
|
||||||
|
.versions
|
||||||
|
.iter()
|
||||||
|
.filter_map(|v| if !v.yanked { Some(&v.num) } else { None });
|
||||||
|
let version_name = find_version(version_req, version_iter)?;
|
||||||
|
|
||||||
|
// Fetch information for the filtered version
|
||||||
|
let version = base_info
|
||||||
|
.versions
|
||||||
|
.iter()
|
||||||
|
.find(|v| v.num == version_name.to_string())
|
||||||
|
.ok_or_else(|| BinstallError::VersionUnavailable {
|
||||||
|
crate_name: name.into(),
|
||||||
|
v: version_name.clone(),
|
||||||
|
})?;
|
||||||
|
|
||||||
|
debug!("Found information for crate version: '{}'", version.num);
|
||||||
|
|
||||||
|
// Download crate to temporary dir (crates.io or git?)
|
||||||
|
let crate_url = format!("https://crates.io/{}", version.dl_path);
|
||||||
|
|
||||||
|
debug!("Fetching crate from: {crate_url} and extracting Cargo.toml from it");
|
||||||
|
|
||||||
|
let crate_dir: PathBuf = format!("{name}-{version_name}").into();
|
||||||
|
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_with_filter(
|
||||||
|
Url::parse(&crate_url)?,
|
||||||
|
TarBasedFmt::Tgz,
|
||||||
|
&temp_dir,
|
||||||
|
move |path: &Path| path == cargo_toml || path == main || path.starts_with(&bin),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
// Return crate directory
|
||||||
|
Ok(crate_path)
|
||||||
|
}
|
48
src/drivers/version.rs
Normal file
48
src/drivers/version.rs
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
use std::collections::BTreeSet;
|
||||||
|
|
||||||
|
use log::debug;
|
||||||
|
use semver::{Version, VersionReq};
|
||||||
|
|
||||||
|
use crate::BinstallError;
|
||||||
|
|
||||||
|
pub(crate) fn find_version<'a, V: Iterator<Item = &'a String>>(
|
||||||
|
requirement: &str,
|
||||||
|
version_iter: V,
|
||||||
|
) -> Result<Version, BinstallError> {
|
||||||
|
// Parse version requirement
|
||||||
|
let version_req = VersionReq::parse(requirement).map_err(|err| BinstallError::VersionReq {
|
||||||
|
req: requirement.into(),
|
||||||
|
err,
|
||||||
|
})?;
|
||||||
|
|
||||||
|
// Filter for matching versions
|
||||||
|
let filtered: BTreeSet<_> = version_iter
|
||||||
|
.filter_map(|v| {
|
||||||
|
// Remove leading `v` for git tags
|
||||||
|
let ver_str = match v.strip_prefix('s') {
|
||||||
|
Some(v) => v,
|
||||||
|
None => v,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Parse out version
|
||||||
|
let ver = Version::parse(ver_str).ok()?;
|
||||||
|
debug!("Version: {:?}", ver);
|
||||||
|
|
||||||
|
// Filter by version match
|
||||||
|
if version_req.matches(&ver) {
|
||||||
|
Some(ver)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
debug!("Filtered: {:?}", filtered);
|
||||||
|
|
||||||
|
// Return highest version
|
||||||
|
filtered
|
||||||
|
.iter()
|
||||||
|
.max()
|
||||||
|
.cloned()
|
||||||
|
.ok_or(BinstallError::VersionMismatch { req: version_req })
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue