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

Additionally fixes engines being executed as scripts,
and fixes downloading pesde from GitHub.
This commit is contained in:
daimond113 2025-01-15 18:33:38 +01:00
parent e3177eeb75
commit 4946a19f8b
No known key found for this signature in database
GPG key ID: 3A8ECE51328B513C
8 changed files with 89 additions and 46 deletions

View file

@ -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(&current_exe().context("failed to get current exe path")?).await?;
EngineKind::Pesde,
&current_exe().context("failed to get current exe path")?,
)
.await?;
Ok(()) Ok(())
} }

View file

@ -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}!");

View file

@ -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 {

View file

@ -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 {

View file

@ -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,
&current_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(())
}

View file

@ -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();

View file

@ -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;
}; };

View file

@ -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
}