From 40ba5b7f7d51b13ab1c7e2bc82141a42afa278be Mon Sep 17 00:00:00 2001 From: AThePeanut4 <49614525+AThePeanut4@users.noreply.github.com> Date: Sun, 24 Nov 2024 15:08:20 +0200 Subject: [PATCH 1/3] fix(poetry): skip if not installed with official script --- src/steps/generic.rs | 77 +++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 76 insertions(+), 1 deletion(-) diff --git a/src/steps/generic.rs b/src/steps/generic.rs index c4e323dd..b164a753 100644 --- a/src/steps/generic.rs +++ b/src/steps/generic.rs @@ -1,5 +1,6 @@ #![allow(unused_imports)] +use std::ffi::OsStr; use std::path::PathBuf; use std::process::Command; use std::{env, path::Path}; @@ -1029,8 +1030,82 @@ pub fn run_lensfun_update_data(ctx: &ExecutionContext) -> Result<()> { pub fn run_poetry(ctx: &ExecutionContext) -> Result<()> { let poetry = require("poetry")?; + + #[cfg(unix)] + fn get_interpreter(poetry: &PathBuf) -> Result { + use std::os::unix::ffi::OsStrExt; + + let script = fs::read(poetry)?; + if let Some(r) = script.iter().position(|&b| b == b'\n') { + let first_line = &script[..r]; + if first_line.starts_with(b"#!") { + return Ok(OsStr::from_bytes(&first_line[2..]).into()); + } + } + + Err(eyre!("Could not find shebang")) + } + #[cfg(windows)] + fn get_interpreter(poetry: &PathBuf) -> Result { + let data = fs::read(poetry)?; + + // https://bitbucket.org/vinay.sajip/simple_launcher/src/master/compare_launchers.py + + let pos = match data.windows(4).rposition(|b| b == b"PK\x05\x06") { + Some(i) => i, + None => return Err(eyre!("Not a ZIP archive")), + }; + + let cdr_size = match data.get(pos + 12..pos + 16) { + Some(b) => u32::from_le_bytes(b.try_into().unwrap()) as usize, + None => return Err(eyre!("Invalid CDR size")), + }; + let cdr_offset = match data.get(pos + 16..pos + 20) { + Some(b) => u32::from_le_bytes(b.try_into().unwrap()) as usize, + None => return Err(eyre!("Invalid CDR offset")), + }; + if pos < cdr_size + cdr_offset { + return Err(eyre!("Invalid ZIP archive")); + } + let arc_pos = pos - cdr_size - cdr_offset; + let shebang = match data[..arc_pos].windows(2).rposition(|b| b == b"#!") { + Some(l) => &data[l + 2..arc_pos - 1], + None => return Err(eyre!("Could not find shebang")), + }; + + // shebang line is utf8 + Ok(std::str::from_utf8(shebang)?.into()) + } + + let interpreter = match get_interpreter(&poetry) { + Ok(p) => p, + Err(e) => return Err(SkipStep(format!("Could not find interpreter for {}: {}", poetry.display(), e)).into()), + }; + debug!("poetry interpreter: {}", interpreter.display()); + + let check_official_install_script = + "import sys; from os import path; print('Y') if path.isfile(path.join(sys.prefix, 'poetry_env')) else print('N')"; + let output = Command::new(&interpreter) + .args(["-c", check_official_install_script]) + .output_checked_utf8()?; + let stdout = output.stdout.trim(); + let official_install = match stdout { + "N" => false, + "Y" => true, + _ => unreachable!("unexpected output from `check_official_install_script`"), + }; + + debug!("poetry is official install: {}", official_install); + + if !official_install { + return Err(SkipStep("Not installed with the official script".to_string()).into()); + } + print_separator("Poetry"); - ctx.run_type().execute(poetry).args(["self", "update"]).status_checked() + ctx.run_type() + .execute(&poetry) + .args(["self", "update"]) + .status_checked() } pub fn run_uv(ctx: &ExecutionContext) -> Result<()> { From 389f8406ee2051962353bfedadb13702361ad2d1 Mon Sep 17 00:00:00 2001 From: AThePeanut4 <49614525+AThePeanut4@users.noreply.github.com> Date: Sun, 24 Nov 2024 16:05:16 +0200 Subject: [PATCH 2/3] feat(poetry): add poetry_force_self_update config option --- config.example.toml | 4 ++++ src/config.rs | 8 ++++++++ src/steps/generic.rs | 42 ++++++++++++++++++++++++------------------ 3 files changed, 36 insertions(+), 18 deletions(-) diff --git a/config.example.toml b/config.example.toml index 551ebc54..f2689dde 100644 --- a/config.example.toml +++ b/config.example.toml @@ -103,6 +103,10 @@ # enable_pipupgrade = true ###disabled by default # pipupgrade_arguments = "-y -u --pip-path pip" ###disabled by default +# Run `poetry self update` even if poetry was not installed with the official script +# (default: false) +# poetry_force_self_update = true + [composer] # self_update = true diff --git a/src/config.rs b/src/config.rs index 62e3dc02..deafae4f 100644 --- a/src/config.rs +++ b/src/config.rs @@ -229,6 +229,7 @@ pub struct Python { enable_pip_review_local: Option, enable_pipupgrade: Option, pipupgrade_arguments: Option, + poetry_force_self_update: Option, } #[derive(Deserialize, Default, Debug, Merge)] @@ -1627,6 +1628,13 @@ impl Config { .and_then(|python| python.enable_pip_review_local) .unwrap_or(false) } + pub fn poetry_force_self_update(&self) -> bool { + self.config_file + .python + .as_ref() + .and_then(|python| python.poetry_force_self_update) + .unwrap_or(false) + } pub fn display_time(&self) -> bool { self.config_file diff --git a/src/steps/generic.rs b/src/steps/generic.rs index b164a753..1d125583 100644 --- a/src/steps/generic.rs +++ b/src/steps/generic.rs @@ -1077,28 +1077,34 @@ pub fn run_poetry(ctx: &ExecutionContext) -> Result<()> { Ok(std::str::from_utf8(shebang)?.into()) } - let interpreter = match get_interpreter(&poetry) { - Ok(p) => p, - Err(e) => return Err(SkipStep(format!("Could not find interpreter for {}: {}", poetry.display(), e)).into()), - }; - debug!("poetry interpreter: {}", interpreter.display()); + if ctx.config().poetry_force_self_update() { + debug!("forcing poetry self update"); + } else { + let interpreter = match get_interpreter(&poetry) { + Ok(p) => p, + Err(e) => { + return Err(SkipStep(format!("Could not find interpreter for {}: {}", poetry.display(), e)).into()) + } + }; + debug!("poetry interpreter: {}", interpreter.display()); - let check_official_install_script = + let check_official_install_script = "import sys; from os import path; print('Y') if path.isfile(path.join(sys.prefix, 'poetry_env')) else print('N')"; - let output = Command::new(&interpreter) - .args(["-c", check_official_install_script]) - .output_checked_utf8()?; - let stdout = output.stdout.trim(); - let official_install = match stdout { - "N" => false, - "Y" => true, - _ => unreachable!("unexpected output from `check_official_install_script`"), - }; + let output = Command::new(&interpreter) + .args(["-c", check_official_install_script]) + .output_checked_utf8()?; + let stdout = output.stdout.trim(); + let official_install = match stdout { + "N" => false, + "Y" => true, + _ => unreachable!("unexpected output from `check_official_install_script`"), + }; - debug!("poetry is official install: {}", official_install); + debug!("poetry is official install: {}", official_install); - if !official_install { - return Err(SkipStep("Not installed with the official script".to_string()).into()); + if !official_install { + return Err(SkipStep("Not installed with the official script".to_string()).into()); + } } print_separator("Poetry"); From 15b3d8abb0f782922e8b84734914599927745e97 Mon Sep 17 00:00:00 2001 From: Steve Lau Date: Sat, 7 Dec 2024 15:02:45 +0800 Subject: [PATCH 3/3] docs: give this config a more detailed explanation --- config.example.toml | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/config.example.toml b/config.example.toml index f2689dde..e2bb6e3c 100644 --- a/config.example.toml +++ b/config.example.toml @@ -103,7 +103,10 @@ # enable_pipupgrade = true ###disabled by default # pipupgrade_arguments = "-y -u --pip-path pip" ###disabled by default -# Run `poetry self update` even if poetry was not installed with the official script +# For the poetry step, by default, Topgrade skips its update if poetry is not +# installed with the official script. This configuration entry forces Topgrade +# to run the update in this case. +# # (default: false) # poetry_force_self_update = true