mirror of
https://github.com/pesde-pkg/pesde.git
synced 2025-04-05 11:20:55 +01:00
feat(engines): create linkers at install time
Some checks are pending
Debug / Get build version (push) Waiting to run
Debug / Build for linux-x86_64 (push) Blocked by required conditions
Debug / Build for macos-aarch64 (push) Blocked by required conditions
Debug / Build for macos-x86_64 (push) Blocked by required conditions
Debug / Build for windows-x86_64 (push) Blocked by required conditions
Test & Lint / lint (push) Waiting to run
Some checks are pending
Debug / Get build version (push) Waiting to run
Debug / Build for linux-x86_64 (push) Blocked by required conditions
Debug / Build for macos-aarch64 (push) Blocked by required conditions
Debug / Build for macos-x86_64 (push) Blocked by required conditions
Debug / Build for windows-x86_64 (push) Blocked by required conditions
Test & Lint / lint (push) Waiting to run
Additionally fixes engines being executed as scripts, and fixes downloading pesde from GitHub.
This commit is contained in:
parent
e3177eeb75
commit
4946a19f8b
8 changed files with 89 additions and 46 deletions
|
@ -1,8 +1,7 @@
|
||||||
use crate::cli::{version::replace_bin_exe, HOME_DIR};
|
use crate::cli::{version::replace_pesde_bin_exe, HOME_DIR};
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
use clap::Args;
|
use clap::Args;
|
||||||
use colored::Colorize;
|
use colored::Colorize;
|
||||||
use pesde::engine::EngineKind;
|
|
||||||
use std::env::current_exe;
|
use std::env::current_exe;
|
||||||
|
|
||||||
#[derive(Debug, Args)]
|
#[derive(Debug, Args)]
|
||||||
|
@ -72,11 +71,7 @@ and then restart your shell.
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
replace_bin_exe(
|
replace_pesde_bin_exe(¤t_exe().context("failed to get current exe path")?).await?;
|
||||||
EngineKind::Pesde,
|
|
||||||
¤t_exe().context("failed to get current exe path")?,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
use crate::cli::{
|
use crate::{
|
||||||
config::read_config,
|
cli::{
|
||||||
version::{
|
config::read_config,
|
||||||
current_version, find_latest_version, get_or_download_engine, no_build_metadata,
|
version::{
|
||||||
replace_bin_exe,
|
current_version, find_latest_version, get_or_download_engine, replace_pesde_bin_exe,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
util::no_build_metadata,
|
||||||
};
|
};
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
use clap::Args;
|
use clap::Args;
|
||||||
|
@ -54,7 +56,7 @@ impl SelfUpgradeCommand {
|
||||||
VersionReq::parse(&format!("={latest_version}")).unwrap(),
|
VersionReq::parse(&format!("={latest_version}")).unwrap(),
|
||||||
)
|
)
|
||||||
.await?;
|
.await?;
|
||||||
replace_bin_exe(EngineKind::Pesde, &path).await?;
|
replace_pesde_bin_exe(&path).await?;
|
||||||
|
|
||||||
println!("upgraded to version {display_latest_version}!");
|
println!("upgraded to version {display_latest_version}!");
|
||||||
|
|
||||||
|
|
|
@ -200,6 +200,21 @@ pub async fn install(
|
||||||
let root_progress = root_progress;
|
let root_progress = root_progress;
|
||||||
|
|
||||||
root_progress.set_prefix(format!("{} {}: ", manifest.name, manifest.target));
|
root_progress.set_prefix(format!("{} {}: ", manifest.name, manifest.target));
|
||||||
|
#[cfg(feature = "version-management")]
|
||||||
|
{
|
||||||
|
root_progress.set_message("update engine linkers");
|
||||||
|
|
||||||
|
let mut tasks = manifest
|
||||||
|
.engines
|
||||||
|
.keys()
|
||||||
|
.map(|engine| crate::cli::version::make_linker_if_needed(*engine))
|
||||||
|
.collect::<JoinSet<_>>();
|
||||||
|
|
||||||
|
while let Some(task) = tasks.join_next().await {
|
||||||
|
task.unwrap()?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
root_progress.set_message("clean");
|
root_progress.set_message("clean");
|
||||||
|
|
||||||
if options.write {
|
if options.write {
|
||||||
|
|
|
@ -109,7 +109,7 @@ pub struct CliDownloadProgressReporter<W> {
|
||||||
impl<W: Write + Send + Sync + 'static> DownloadsReporter for CliReporter<W> {
|
impl<W: Write + Send + Sync + 'static> DownloadsReporter for CliReporter<W> {
|
||||||
type DownloadProgressReporter = CliDownloadProgressReporter<W>;
|
type DownloadProgressReporter = CliDownloadProgressReporter<W>;
|
||||||
|
|
||||||
fn report_download<'b>(self: Arc<Self>, name: String) -> Self::DownloadProgressReporter {
|
fn report_download(self: Arc<Self>, name: String) -> Self::DownloadProgressReporter {
|
||||||
self.root_progress.inc_length(1);
|
self.root_progress.inc_length(1);
|
||||||
|
|
||||||
CliDownloadProgressReporter {
|
CliDownloadProgressReporter {
|
||||||
|
|
|
@ -1,9 +1,12 @@
|
||||||
use crate::cli::{
|
use crate::{
|
||||||
bin_dir,
|
cli::{
|
||||||
config::{read_config, write_config, CliConfig},
|
bin_dir,
|
||||||
files::make_executable,
|
config::{read_config, write_config, CliConfig},
|
||||||
home_dir,
|
files::make_executable,
|
||||||
reporters::run_with_reporter,
|
home_dir,
|
||||||
|
reporters::run_with_reporter,
|
||||||
|
},
|
||||||
|
util::no_build_metadata,
|
||||||
};
|
};
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
use colored::Colorize;
|
use colored::Colorize;
|
||||||
|
@ -32,12 +35,6 @@ pub fn current_version() -> Version {
|
||||||
Version::parse(env!("CARGO_PKG_VERSION")).unwrap()
|
Version::parse(env!("CARGO_PKG_VERSION")).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn no_build_metadata(version: &Version) -> Version {
|
|
||||||
let mut version = version.clone();
|
|
||||||
version.build = semver::BuildMetadata::EMPTY;
|
|
||||||
version
|
|
||||||
}
|
|
||||||
|
|
||||||
const CHECK_INTERVAL: chrono::Duration = chrono::Duration::hours(6);
|
const CHECK_INTERVAL: chrono::Duration = chrono::Duration::hours(6);
|
||||||
|
|
||||||
pub async fn find_latest_version(reqwest: &reqwest::Client) -> anyhow::Result<Version> {
|
pub async fn find_latest_version(reqwest: &reqwest::Client) -> anyhow::Result<Version> {
|
||||||
|
@ -249,23 +246,18 @@ pub async fn get_or_download_engine(
|
||||||
.await
|
.await
|
||||||
.context("failed to make downloaded version executable")?;
|
.context("failed to make downloaded version executable")?;
|
||||||
|
|
||||||
// replace the executable if there isn't any installed, or the one installed is out of date
|
if engine != EngineKind::Pesde {
|
||||||
if installed_versions.pop_last().is_none_or(|v| version > v) {
|
make_linker_if_needed(engine).await?;
|
||||||
replace_bin_exe(
|
|
||||||
engine,
|
|
||||||
¤t_exe().context("failed to get current exe path")?,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(path)
|
Ok(path)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[instrument(level = "trace")]
|
#[instrument(level = "trace")]
|
||||||
pub async fn replace_bin_exe(engine: EngineKind, with: &Path) -> anyhow::Result<()> {
|
pub async fn replace_pesde_bin_exe(with: &Path) -> anyhow::Result<()> {
|
||||||
let bin_exe_path = bin_dir()
|
let bin_exe_path = bin_dir()
|
||||||
.await?
|
.await?
|
||||||
.join(engine.to_string())
|
.join(EngineKind::Pesde.to_string())
|
||||||
.with_extension(std::env::consts::EXE_EXTENSION);
|
.with_extension(std::env::consts::EXE_EXTENSION);
|
||||||
|
|
||||||
let exists = bin_exe_path.exists();
|
let exists = bin_exe_path.exists();
|
||||||
|
@ -295,3 +287,25 @@ pub async fn replace_bin_exe(engine: EngineKind, with: &Path) -> anyhow::Result<
|
||||||
|
|
||||||
make_executable(&bin_exe_path).await
|
make_executable(&bin_exe_path).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[instrument(level = "trace")]
|
||||||
|
pub async fn make_linker_if_needed(engine: EngineKind) -> anyhow::Result<()> {
|
||||||
|
let bin_dir = bin_dir().await?;
|
||||||
|
let linker = bin_dir
|
||||||
|
.join(engine.to_string())
|
||||||
|
.with_extension(std::env::consts::EXE_EXTENSION);
|
||||||
|
let exists = linker.exists();
|
||||||
|
|
||||||
|
if !exists {
|
||||||
|
let exe = current_exe().context("failed to get current exe path")?;
|
||||||
|
|
||||||
|
#[cfg(windows)]
|
||||||
|
let result = fs::symlink_file(exe, linker);
|
||||||
|
#[cfg(not(windows))]
|
||||||
|
let result = fs::symlink(exe, linker);
|
||||||
|
|
||||||
|
result.await.context("failed to create symlink")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ use crate::{
|
||||||
traits::{DownloadOptions, EngineSource, ResolveOptions},
|
traits::{DownloadOptions, EngineSource, ResolveOptions},
|
||||||
},
|
},
|
||||||
reporters::{response_to_async_read, DownloadProgressReporter},
|
reporters::{response_to_async_read, DownloadProgressReporter},
|
||||||
|
util::no_build_metadata,
|
||||||
version_matches,
|
version_matches,
|
||||||
};
|
};
|
||||||
use reqwest::header::ACCEPT;
|
use reqwest::header::ACCEPT;
|
||||||
|
@ -80,14 +81,21 @@ impl EngineSource for GitHubEngineSource {
|
||||||
..
|
..
|
||||||
} = options;
|
} = options;
|
||||||
|
|
||||||
let desired_asset_name = self
|
let desired_asset_names = [
|
||||||
.asset_template
|
self.asset_template
|
||||||
.replace("{VERSION}", &version.to_string());
|
.replace("{VERSION}", &version.to_string()),
|
||||||
|
self.asset_template
|
||||||
|
.replace("{VERSION}", &no_build_metadata(version).to_string()),
|
||||||
|
];
|
||||||
|
|
||||||
let asset = engine_ref
|
let asset = engine_ref
|
||||||
.assets
|
.assets
|
||||||
.iter()
|
.iter()
|
||||||
.find(|asset| asset.name.eq_ignore_ascii_case(&desired_asset_name))
|
.find(|asset| {
|
||||||
|
desired_asset_names
|
||||||
|
.iter()
|
||||||
|
.any(|name| asset.name.eq_ignore_ascii_case(name))
|
||||||
|
})
|
||||||
.ok_or(errors::DownloadError::AssetNotFound)?;
|
.ok_or(errors::DownloadError::AssetNotFound)?;
|
||||||
|
|
||||||
reporter.report_start();
|
reporter.report_start();
|
||||||
|
|
16
src/main.rs
16
src/main.rs
|
@ -138,12 +138,17 @@ impl<'a> MakeWriter<'a> for IndicatifWriter {
|
||||||
async fn run() -> anyhow::Result<()> {
|
async fn run() -> anyhow::Result<()> {
|
||||||
let cwd = std::env::current_dir().expect("failed to get current working directory");
|
let cwd = std::env::current_dir().expect("failed to get current working directory");
|
||||||
let current_exe = std::env::current_exe().expect("failed to get current executable path");
|
let current_exe = std::env::current_exe().expect("failed to get current executable path");
|
||||||
let exe_name = current_exe.file_stem().unwrap();
|
let exe_name = current_exe
|
||||||
|
.file_stem()
|
||||||
|
.unwrap()
|
||||||
|
.to_str()
|
||||||
|
.expect("exe name is not valid utf-8");
|
||||||
|
let exe_name_engine = EngineKind::from_str(exe_name);
|
||||||
|
|
||||||
#[cfg(windows)]
|
#[cfg(windows)]
|
||||||
'scripts: {
|
'scripts: {
|
||||||
// we're called the same as the binary, so we're not a (legal) script
|
// if we're an engine, we don't want to run any scripts
|
||||||
if exe_name == env!("CARGO_PKG_NAME") {
|
if exe_name_engine.is_ok() {
|
||||||
break 'scripts;
|
break 'scripts;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -273,10 +278,7 @@ async fn run() -> anyhow::Result<()> {
|
||||||
|
|
||||||
#[cfg(feature = "version-management")]
|
#[cfg(feature = "version-management")]
|
||||||
'engines: {
|
'engines: {
|
||||||
let Some(engine) = exe_name
|
let Ok(engine) = exe_name_engine else {
|
||||||
.to_str()
|
|
||||||
.and_then(|str| EngineKind::from_str(str).ok())
|
|
||||||
else {
|
|
||||||
break 'engines;
|
break 'engines;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
use crate::AuthConfig;
|
use crate::AuthConfig;
|
||||||
use gix::bstr::BStr;
|
use gix::bstr::BStr;
|
||||||
|
use semver::Version;
|
||||||
use serde::{Deserialize, Deserializer, Serializer};
|
use serde::{Deserialize, Deserializer, Serializer};
|
||||||
use sha2::{Digest, Sha256};
|
use sha2::{Digest, Sha256};
|
||||||
use std::collections::{BTreeMap, HashSet};
|
use std::collections::{BTreeMap, HashSet};
|
||||||
|
@ -88,3 +89,9 @@ pub fn hash<S: AsRef<[u8]>>(struc: S) -> String {
|
||||||
pub fn is_default<T: Default + Eq>(t: &T) -> bool {
|
pub fn is_default<T: Default + Eq>(t: &T) -> bool {
|
||||||
t == &T::default()
|
t == &T::default()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn no_build_metadata(version: &Version) -> Version {
|
||||||
|
let mut version = version.clone();
|
||||||
|
version.build = semver::BuildMetadata::EMPTY;
|
||||||
|
version
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue