mirror of
https://github.com/cargo-bins/cargo-binstall.git
synced 2025-06-07 03:06:37 +00:00
Merge branch 'main' into feature/bin-multi-targets
This commit is contained in:
commit
6f7c8fa8ab
5 changed files with 69 additions and 30 deletions
|
@ -1,8 +1,10 @@
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
pub use gh_crate_meta::*;
|
pub use gh_crate_meta::*;
|
||||||
pub use log::debug;
|
pub use log::debug;
|
||||||
pub use quickinstall::*;
|
pub use quickinstall::*;
|
||||||
|
use tokio::task::JoinHandle;
|
||||||
|
|
||||||
use crate::{BinstallError, PkgFmt, PkgMeta};
|
use crate::{BinstallError, PkgFmt, PkgMeta};
|
||||||
|
|
||||||
|
@ -10,9 +12,9 @@ mod gh_crate_meta;
|
||||||
mod quickinstall;
|
mod quickinstall;
|
||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
pub trait Fetcher {
|
pub trait Fetcher: Send + Sync {
|
||||||
/// Create a new fetcher from some data
|
/// Create a new fetcher from some data
|
||||||
async fn new(data: &Data) -> Box<Self>
|
async fn new(data: &Data) -> Arc<Self>
|
||||||
where
|
where
|
||||||
Self: Sized;
|
Self: Sized;
|
||||||
|
|
||||||
|
@ -47,30 +49,59 @@ pub struct Data {
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
pub struct MultiFetcher {
|
pub struct MultiFetcher {
|
||||||
fetchers: Vec<Box<dyn Fetcher>>,
|
fetchers: Vec<Arc<dyn Fetcher>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MultiFetcher {
|
impl MultiFetcher {
|
||||||
pub fn add(&mut self, fetcher: Box<dyn Fetcher>) {
|
pub fn add(&mut self, fetcher: Arc<dyn Fetcher>) {
|
||||||
self.fetchers.push(fetcher);
|
self.fetchers.push(fetcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn first_available(&self) -> Option<&dyn Fetcher> {
|
pub async fn first_available(&self) -> Option<Arc<dyn Fetcher>> {
|
||||||
for fetcher in &self.fetchers {
|
let handles: Vec<_> = self
|
||||||
let available = fetcher.check().await.unwrap_or_else(|err| {
|
.fetchers
|
||||||
debug!(
|
.iter()
|
||||||
"Error while checking fetcher {}: {}",
|
.cloned()
|
||||||
fetcher.source_name(),
|
.map(|fetcher| {
|
||||||
err
|
let fetcher_cloned = fetcher.clone();
|
||||||
);
|
|
||||||
false
|
|
||||||
});
|
|
||||||
|
|
||||||
if available {
|
(
|
||||||
return Some(&**fetcher);
|
AutoAbortJoinHandle(tokio::spawn(async move { fetcher.check().await })),
|
||||||
|
fetcher_cloned,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
for (mut handle, fetcher) in handles {
|
||||||
|
match (&mut handle.0).await {
|
||||||
|
Ok(Ok(true)) => return Some(fetcher),
|
||||||
|
Ok(Ok(false)) => (),
|
||||||
|
Ok(Err(err)) => {
|
||||||
|
debug!(
|
||||||
|
"Error while checking fetcher {}: {}",
|
||||||
|
fetcher.source_name(),
|
||||||
|
err
|
||||||
|
);
|
||||||
|
}
|
||||||
|
Err(join_err) => {
|
||||||
|
debug!(
|
||||||
|
"Error while checking fetcher {}: {}",
|
||||||
|
fetcher.source_name(),
|
||||||
|
join_err
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct AutoAbortJoinHandle(JoinHandle<Result<bool, BinstallError>>);
|
||||||
|
|
||||||
|
impl Drop for AutoAbortJoinHandle {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
self.0.abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
use log::{debug, info, warn};
|
use log::{debug, info, warn};
|
||||||
use reqwest::Method;
|
use reqwest::Method;
|
||||||
|
@ -22,8 +23,8 @@ impl GhCrateMeta {
|
||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
impl super::Fetcher for GhCrateMeta {
|
impl super::Fetcher for GhCrateMeta {
|
||||||
async fn new(data: &Data) -> Box<Self> {
|
async fn new(data: &Data) -> Arc<Self> {
|
||||||
Box::new(Self { data: data.clone() })
|
Arc::new(Self { data: data.clone() })
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn check(&self) -> Result<bool, BinstallError> {
|
async fn check(&self) -> Result<bool, BinstallError> {
|
||||||
|
@ -36,7 +37,7 @@ impl super::Fetcher for GhCrateMeta {
|
||||||
}
|
}
|
||||||
|
|
||||||
info!("Checking for package at: '{url}'");
|
info!("Checking for package at: '{url}'");
|
||||||
remote_exists(url.as_str(), Method::HEAD).await
|
remote_exists(url, Method::HEAD).await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn fetch(&self, dst: &Path) -> Result<(), BinstallError> {
|
async fn fetch(&self, dst: &Path) -> Result<(), BinstallError> {
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
use log::info;
|
use log::info;
|
||||||
use reqwest::Method;
|
use reqwest::Method;
|
||||||
|
@ -18,11 +19,11 @@ pub struct QuickInstall {
|
||||||
|
|
||||||
#[async_trait::async_trait]
|
#[async_trait::async_trait]
|
||||||
impl super::Fetcher for QuickInstall {
|
impl super::Fetcher for QuickInstall {
|
||||||
async fn new(data: &Data) -> Box<Self> {
|
async fn new(data: &Data) -> Arc<Self> {
|
||||||
let crate_name = &data.name;
|
let crate_name = &data.name;
|
||||||
let version = &data.version;
|
let version = &data.version;
|
||||||
let target = data.target.clone();
|
let target = data.target.clone();
|
||||||
Box::new(Self {
|
Arc::new(Self {
|
||||||
package: format!("{crate_name}-{version}-{target}"),
|
package: format!("{crate_name}-{version}-{target}"),
|
||||||
target,
|
target,
|
||||||
})
|
})
|
||||||
|
@ -32,7 +33,7 @@ impl super::Fetcher for QuickInstall {
|
||||||
let url = self.package_url();
|
let url = self.package_url();
|
||||||
self.report().await?;
|
self.report().await?;
|
||||||
info!("Checking for package at: '{url}'");
|
info!("Checking for package at: '{url}'");
|
||||||
remote_exists(&url, Method::HEAD).await
|
remote_exists(Url::parse(&url)?, Method::HEAD).await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn fetch(&self, dst: &Path) -> Result<(), BinstallError> {
|
async fn fetch(&self, dst: &Path) -> Result<(), BinstallError> {
|
||||||
|
|
|
@ -32,8 +32,7 @@ pub fn load_manifest_path<P: AsRef<Path>>(
|
||||||
Ok(manifest)
|
Ok(manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn remote_exists(url: &str, method: Method) -> Result<bool, BinstallError> {
|
pub async fn remote_exists(url: Url, method: Method) -> Result<bool, BinstallError> {
|
||||||
let url = Url::parse(url)?;
|
|
||||||
let req = reqwest::Client::new()
|
let req = reqwest::Client::new()
|
||||||
.request(method.clone(), url.clone())
|
.request(method.clone(), url.clone())
|
||||||
.send()
|
.send()
|
||||||
|
|
19
src/main.rs
19
src/main.rs
|
@ -11,7 +11,7 @@ use miette::{miette, IntoDiagnostic, Result, WrapErr};
|
||||||
use simplelog::{ColorChoice, ConfigBuilder, TermLogger, TerminalMode};
|
use simplelog::{ColorChoice, ConfigBuilder, TermLogger, TerminalMode};
|
||||||
use structopt::StructOpt;
|
use structopt::StructOpt;
|
||||||
use tempfile::TempDir;
|
use tempfile::TempDir;
|
||||||
use tokio::{process::Command, runtime::Runtime};
|
use tokio::{process::Command, runtime::Runtime, task::JoinError};
|
||||||
|
|
||||||
use cargo_binstall::{
|
use cargo_binstall::{
|
||||||
bins,
|
bins,
|
||||||
|
@ -84,6 +84,7 @@ enum MainExit {
|
||||||
Success(Duration),
|
Success(Duration),
|
||||||
Error(BinstallError),
|
Error(BinstallError),
|
||||||
Report(miette::Report),
|
Report(miette::Report),
|
||||||
|
JoinErr(JoinError),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Termination for MainExit {
|
impl Termination for MainExit {
|
||||||
|
@ -99,6 +100,11 @@ impl Termination for MainExit {
|
||||||
eprintln!("{err:?}");
|
eprintln!("{err:?}");
|
||||||
ExitCode::from(16)
|
ExitCode::from(16)
|
||||||
}
|
}
|
||||||
|
Self::JoinErr(err) => {
|
||||||
|
error!("Fatal error:");
|
||||||
|
eprintln!("{err:?}");
|
||||||
|
ExitCode::from(17)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -107,19 +113,20 @@ fn main() -> MainExit {
|
||||||
let start = Instant::now();
|
let start = Instant::now();
|
||||||
|
|
||||||
let rt = Runtime::new().unwrap();
|
let rt = Runtime::new().unwrap();
|
||||||
let result = rt.block_on(entry());
|
let handle = rt.spawn(entry());
|
||||||
|
let result = rt.block_on(handle);
|
||||||
drop(rt);
|
drop(rt);
|
||||||
|
|
||||||
let done = start.elapsed();
|
let done = start.elapsed();
|
||||||
debug!("run time: {done:?}");
|
debug!("run time: {done:?}");
|
||||||
|
|
||||||
result
|
result.map_or_else(MainExit::JoinErr, |res| {
|
||||||
.map(|_| MainExit::Success(done))
|
res.map(|_| MainExit::Success(done)).unwrap_or_else(|err| {
|
||||||
.unwrap_or_else(|err| {
|
|
||||||
err.downcast::<BinstallError>()
|
err.downcast::<BinstallError>()
|
||||||
.map(MainExit::Error)
|
.map(MainExit::Error)
|
||||||
.unwrap_or_else(MainExit::Report)
|
.unwrap_or_else(MainExit::Report)
|
||||||
})
|
})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn entry() -> Result<()> {
|
async fn entry() -> Result<()> {
|
||||||
|
@ -240,7 +247,7 @@ async fn entry() -> Result<()> {
|
||||||
Some(fetcher) => {
|
Some(fetcher) => {
|
||||||
install_from_package(
|
install_from_package(
|
||||||
binaries,
|
binaries,
|
||||||
fetcher,
|
&*fetcher,
|
||||||
install_path,
|
install_path,
|
||||||
meta,
|
meta,
|
||||||
opts,
|
opts,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue