#![allow(unused_imports)] use std::ffi::OsStr; use std::path::PathBuf; use std::process::Command; use std::{env, path::Path}; use std::{fs, io::Write}; use color_eyre::eyre::eyre; use color_eyre::eyre::Context; use color_eyre::eyre::Result; use rust_i18n::t; use semver::Version; use tempfile::tempfile_in; use tracing::{debug, error}; use crate::command::{CommandExt, Utf8Output}; use crate::execution_context::ExecutionContext; use crate::executor::ExecutorOutput; use crate::terminal::{print_separator, shell}; use crate::utils::{self, check_is_python_2_or_shim, get_require_sudo_string, require, require_option, which, PathExt}; use crate::Step; use crate::HOME_DIR; use crate::{ error::{SkipStep, StepFailed, TopgradeError}, terminal::print_warning, }; #[cfg(target_os = "linux")] pub fn is_wsl() -> Result { let output = Command::new("uname").arg("-r").output_checked_utf8()?.stdout; debug!("Uname output: {}", output); Ok(output.contains("microsoft")) } #[cfg(not(target_os = "linux"))] pub fn is_wsl() -> Result { Ok(false) } pub fn run_cargo_update(ctx: &ExecutionContext) -> Result<()> { let cargo_dir = env::var_os("CARGO_HOME") .map_or_else(|| HOME_DIR.join(".cargo"), PathBuf::from) .require()?; require("cargo").or_else(|_| { require_option( cargo_dir.join("bin/cargo").if_exists(), String::from("No cargo detected"), ) })?; let toml_file = cargo_dir.join(".crates.toml").require()?; if fs::metadata(&toml_file)?.len() == 0 { return Err(SkipStep(format!("{} exists but empty", &toml_file.display())).into()); } print_separator("Cargo"); let cargo_update = require("cargo-install-update") .ok() .or_else(|| cargo_dir.join("bin/cargo-install-update").if_exists()); let Some(cargo_update) = cargo_update else { let message = String::from("cargo-update isn't installed so Topgrade can't upgrade cargo packages.\nInstall cargo-update by running `cargo install cargo-update`"); print_warning(&message); return Err(SkipStep(message).into()); }; ctx.run_type() .execute(cargo_update) .args(["install-update", "--git", "--all"]) .status_checked()?; if ctx.config().cleanup() { let cargo_cache = require("cargo-cache") .ok() .or_else(|| cargo_dir.join("bin/cargo-cache").if_exists()); if let Some(e) = cargo_cache { ctx.run_type().execute(e).args(["-a"]).status_checked()?; } else { let message = String::from("cargo-cache isn't installed so Topgrade can't cleanup cargo packages.\nInstall cargo-cache by running `cargo install cargo-cache`"); print_warning(message); } } Ok(()) } pub fn run_flutter_upgrade(ctx: &ExecutionContext) -> Result<()> { let flutter = require("flutter")?; print_separator("Flutter"); ctx.run_type().execute(flutter).arg("upgrade").status_checked() } pub fn run_gem(ctx: &ExecutionContext) -> Result<()> { let gem = require("gem")?; HOME_DIR.join(".gem").require()?; print_separator("Gems"); let mut command = ctx.run_type().execute(gem); command.arg("update"); if env::var_os("RBENV_SHELL").is_none() { debug!("Detected rbenv. Avoiding --user-install"); command.arg("--user-install"); } command.status_checked() } pub fn run_rubygems(ctx: &ExecutionContext) -> Result<()> { HOME_DIR.join(".gem").require()?; let gem = require("gem")?; print_separator("RubyGems"); let gem_path_str = gem.as_os_str(); if gem_path_str.to_str().unwrap().contains("asdf") || gem_path_str.to_str().unwrap().contains("mise") || gem_path_str.to_str().unwrap().contains(".rbenv") || gem_path_str.to_str().unwrap().contains(".rvm") { ctx.run_type() .execute(gem) .args(["update", "--system"]) .status_checked()?; } else { let sudo = require_option(ctx.sudo().as_ref(), get_require_sudo_string())?; if !Path::new("/usr/lib/ruby/vendor_ruby/rubygems/defaults/operating_system.rb").exists() { ctx.run_type() .execute(sudo) .arg("-EH") .arg(gem) .args(["update", "--system"]) .status_checked()?; } } Ok(()) } pub fn run_haxelib_update(ctx: &ExecutionContext) -> Result<()> { let haxelib = require("haxelib")?; let haxelib_dir = PathBuf::from(std::str::from_utf8(&Command::new(&haxelib).arg("config").output_checked()?.stdout)?.trim()) .require()?; let directory_writable = tempfile_in(&haxelib_dir).is_ok(); debug!("{:?} writable: {}", haxelib_dir, directory_writable); print_separator("haxelib"); let mut command = if directory_writable { ctx.run_type().execute(&haxelib) } else { let sudo = require_option(ctx.sudo().as_ref(), get_require_sudo_string())?; let mut c = ctx.run_type().execute(sudo); c.arg(&haxelib); c }; command.arg("update").status_checked() } pub fn run_sheldon(ctx: &ExecutionContext) -> Result<()> { let sheldon = require("sheldon")?; print_separator("Sheldon"); ctx.run_type() .execute(sheldon) .args(["lock", "--update"]) .status_checked() } pub fn run_fossil(ctx: &ExecutionContext) -> Result<()> { let fossil = require("fossil")?; print_separator("Fossil"); ctx.run_type().execute(fossil).args(["all", "sync"]).status_checked() } pub fn run_micro(ctx: &ExecutionContext) -> Result<()> { let micro = require("micro")?; print_separator("micro"); let stdout = ctx .run_type() .execute(micro) .args(["-plugin", "update"]) .output_checked_utf8()? .stdout; std::io::stdout().write_all(stdout.as_bytes())?; if stdout.contains("Nothing to install / update") || stdout.contains("One or more plugins installed") { Ok(()) } else { Err(eyre!("micro output does not indicate success: {}", stdout)) } } #[cfg(not(any( target_os = "freebsd", target_os = "openbsd", target_os = "netbsd", target_os = "dragonfly" )))] pub fn run_apm(ctx: &ExecutionContext) -> Result<()> { let apm = require("apm")?; print_separator("Atom Package Manager"); ctx.run_type() .execute(apm) .args(["upgrade", "--confirm=false"]) .status_checked() } pub fn run_aqua(ctx: &ExecutionContext) -> Result<()> { let aqua = require("aqua")?; print_separator("Aqua"); if ctx.run_type().dry() { println!("{}", t!("Updating aqua ...")); println!("{}", t!("Updating aqua installed cli tools ...")); Ok(()) } else { ctx.run_type().execute(&aqua).arg("update-aqua").status_checked()?; ctx.run_type().execute(&aqua).arg("update").status_checked() } } pub fn run_rustup(ctx: &ExecutionContext) -> Result<()> { let rustup = require("rustup")?; print_separator("rustup"); ctx.run_type().execute(rustup).arg("update").status_checked() } pub fn run_rye(ctx: &ExecutionContext) -> Result<()> { let rye = require("rye")?; print_separator("Rye"); ctx.run_type().execute(rye).args(["self", "update"]).status_checked() } pub fn run_elan(ctx: &ExecutionContext) -> Result<()> { let elan = require("elan")?; print_separator("elan"); let disabled_error_msg = "self-update is disabled"; let executor_output = ctx.run_type().execute(&elan).args(["self", "update"]).output()?; match executor_output { ExecutorOutput::Wet(command_output) => { if command_output.status.success() { // Flush the captured output std::io::stdout().lock().write_all(&command_output.stdout).unwrap(); std::io::stderr().lock().write_all(&command_output.stderr).unwrap(); } else { let stderr_as_str = std::str::from_utf8(&command_output.stderr).unwrap(); if stderr_as_str.contains(disabled_error_msg) { // `elan` is externally managed, we cannot do the update. Users // won't see any error message because Topgrade captures them // all. } else { // `elan` is NOT externally managed, `elan self update` can // be performed, but the invocation failed, so we report the // error to the user and error out. std::io::stdout().lock().write_all(&command_output.stdout).unwrap(); std::io::stderr().lock().write_all(&command_output.stderr).unwrap(); return Err(StepFailed.into()); } } } ExecutorOutput::Dry => { /* nothing needed because in a dry run */ } } ctx.run_type().execute(&elan).arg("update").status_checked() } pub fn run_juliaup(ctx: &ExecutionContext) -> Result<()> { let juliaup = require("juliaup")?; print_separator("juliaup"); if juliaup.canonicalize()?.is_descendant_of(&HOME_DIR) { ctx.run_type() .execute(&juliaup) .args(["self", "update"]) .status_checked()?; } ctx.run_type().execute(&juliaup).arg("update").status_checked() } pub fn run_choosenim(ctx: &ExecutionContext) -> Result<()> { let choosenim = require("choosenim")?; print_separator("choosenim"); let run_type = ctx.run_type(); run_type.execute(&choosenim).args(["update", "self"]).status_checked()?; run_type.execute(&choosenim).args(["update", "stable"]).status_checked() } pub fn run_krew_upgrade(ctx: &ExecutionContext) -> Result<()> { let krew = require("kubectl-krew")?; print_separator("Krew"); ctx.run_type().execute(krew).args(["upgrade"]).status_checked() } pub fn run_gcloud_components_update(ctx: &ExecutionContext) -> Result<()> { let gcloud = require("gcloud")?; if gcloud.starts_with("/snap") { Ok(()) } else { print_separator("gcloud"); ctx.run_type() .execute(gcloud) .args(["components", "update", "--quiet"]) .status_checked() } } pub fn run_jetpack(ctx: &ExecutionContext) -> Result<()> { let jetpack = require("jetpack")?; print_separator("Jetpack"); ctx.run_type() .execute(jetpack) .args(["global", "update"]) .status_checked() } pub fn run_rtcl(ctx: &ExecutionContext) -> Result<()> { let rupdate = require("rupdate")?; print_separator("rtcl"); ctx.run_type().execute(rupdate).status_checked() } pub fn run_opam_update(ctx: &ExecutionContext) -> Result<()> { let opam = require("opam")?; print_separator("OCaml Package Manager"); ctx.run_type().execute(&opam).arg("update").status_checked()?; let mut command = ctx.run_type().execute(&opam); command.arg("upgrade"); if ctx.config().yes(Step::Opam) { command.arg("--yes"); } command.status_checked()?; if ctx.config().cleanup() { ctx.run_type().execute(&opam).arg("clean").status_checked()?; } Ok(()) } pub fn run_vcpkg_update(ctx: &ExecutionContext) -> Result<()> { let vcpkg = require("vcpkg")?; print_separator("vcpkg"); #[cfg(unix)] let is_root_install = !&vcpkg.starts_with("/home"); #[cfg(not(unix))] let is_root_install = false; let mut command = if is_root_install { ctx.run_type().execute(&vcpkg) } else { let sudo = require_option(ctx.sudo().as_ref(), get_require_sudo_string())?; let mut c = ctx.run_type().execute(sudo); c.arg(&vcpkg); c }; command.args(["upgrade", "--no-dry-run"]).status_checked() } /// Make VSCodium a separate step because: /// /// 1. Users could use both VSCode and VSCoium /// 2. Just in case, VSCodium could have incompatible changes with VSCode pub fn run_vscodium_extensions_update(ctx: &ExecutionContext) -> Result<()> { // Calling vscodoe in WSL may install a server instead of updating extensions (https://github.com/topgrade-rs/topgrade/issues/594#issuecomment-1782157367) if is_wsl()? { return Err(SkipStep(String::from("Should not run in WSL")).into()); } let vscodium = require("codium")?; // VSCode has update command only since 1.86 version ("january 2024" update), disable the update for prior versions // Use command `code --version` which returns 3 lines: version, git commit, instruction set. We parse only the first one // // This should apply to VSCodium as well. let version: Result = match Command::new(&vscodium) .arg("--version") .output_checked_utf8()? .stdout .lines() .next() { Some(item) => Version::parse(item).map_err(std::convert::Into::into), _ => return Err(SkipStep(String::from("Cannot find vscodium version")).into()), }; if !matches!(version, Ok(version) if version >= Version::new(1, 86, 0)) { return Err(SkipStep(String::from( "Too old vscodium version to have update extensions command", )) .into()); } print_separator("VSCodium extensions"); ctx.run_type() .execute(vscodium) .arg("--update-extensions") .status_checked() } pub fn run_vscode_extensions_update(ctx: &ExecutionContext) -> Result<()> { // Calling vscode in WSL may install a server instead of updating extensions (https://github.com/topgrade-rs/topgrade/issues/594#issuecomment-1782157367) if is_wsl()? { return Err(SkipStep(String::from("Should not run in WSL")).into()); } let vscode = require("code")?; // Vscode has update command only since 1.86 version ("january 2024" update), disable the update for prior versions // Use command `code --version` which returns 3 lines: version, git commit, instruction set. We parse only the first one let version: Result = match Command::new(&vscode) .arg("--version") .output_checked_utf8()? .stdout .lines() .next() { Some(item) => Version::parse(item).map_err(std::convert::Into::into), _ => return Err(SkipStep(String::from("Cannot find vscode version")).into()), }; if !matches!(version, Ok(version) if version >= Version::new(1, 86, 0)) { return Err(SkipStep(String::from("Too old vscode version to have update extensions command")).into()); } print_separator("Visual Studio Code extensions"); ctx.run_type() .execute(vscode) .arg("--update-extensions") .status_checked() } pub fn run_pipx_update(ctx: &ExecutionContext) -> Result<()> { let pipx = require("pipx")?; print_separator("pipx"); let mut command_args = vec!["upgrade-all", "--include-injected"]; // pipx version 1.4.0 introduced a new command argument `pipx upgrade-all --quiet` // (see https://pipx.pypa.io/stable/docs/#pipx-upgrade-all) let version_str = Command::new(&pipx) .args(["--version"]) .output_checked_utf8() .map(|s| s.stdout.trim().to_owned()); let version = Version::parse(&version_str?); if matches!(version, Ok(version) if version >= Version::new(1, 4, 0)) { command_args.push("--quiet"); } ctx.run_type().execute(pipx).args(command_args).status_checked() } pub fn run_conda_update(ctx: &ExecutionContext) -> Result<()> { let conda = require("conda")?; let output = Command::new(&conda) .args(["config", "--show", "auto_activate_base"]) .output_checked_utf8()?; debug!("Conda output: {}", output.stdout); if output.stdout.contains("False") { return Err(SkipStep("auto_activate_base is set to False".to_string()).into()); } print_separator("Conda"); let mut command = ctx.run_type().execute(conda); command.args(["update", "--all", "-n", "base"]); if ctx.config().yes(Step::Conda) { command.arg("--yes"); } command.status_checked() } pub fn run_pixi_update(ctx: &ExecutionContext) -> Result<()> { let pixi = require("pixi")?; print_separator("Pixi"); ctx.run_type().execute(pixi).args(["self-update"]).status_checked() } pub fn run_mamba_update(ctx: &ExecutionContext) -> Result<()> { let mamba = require("mamba")?; print_separator("Mamba"); let mut command = ctx.run_type().execute(mamba); command.args(["update", "--all", "-n", "base"]); if ctx.config().yes(Step::Mamba) { command.arg("--yes"); } command.status_checked() } pub fn run_miktex_packages_update(ctx: &ExecutionContext) -> Result<()> { let miktex = require("miktex")?; print_separator("miktex"); ctx.run_type() .execute(miktex) .args(["packages", "update"]) .status_checked() } pub fn run_pip3_update(ctx: &ExecutionContext) -> Result<()> { let py = require("python").and_then(check_is_python_2_or_shim); let py3 = require("python3").and_then(check_is_python_2_or_shim); let python3 = match (py, py3) { // prefer `python` if it is available and is a valid Python 3. (Ok(py), _) => py, (Err(_), Ok(py3)) => py3, (Err(py_err), Err(py3_err)) => { return Err(SkipStep(format!("Skip due to following reasons: {py_err} {py3_err}")).into()); } }; Command::new(&python3) .args(["-m", "pip"]) .output_checked_utf8() .map_err(|_| SkipStep("pip does not exist".to_string()))?; let check_extern_managed_script = "import sysconfig; from os import path; print('Y') if path.isfile(path.join(sysconfig.get_path('stdlib'), 'EXTERNALLY-MANAGED')) else print('N')"; let output = Command::new(&python3) .args(["-c", check_extern_managed_script]) .output_checked_utf8()?; let stdout = output.stdout.trim(); let extern_managed = match stdout { "N" => false, "Y" => true, _ => unreachable!("unexpected output from `check_extern_managed_script`"), }; let allow_break_sys_pkg = match Command::new(&python3) .args(["-m", "pip", "config", "get", "global.break-system-packages"]) .output_checked_utf8() { Ok(output) => { let stdout = output.stdout.trim(); stdout .parse::() .expect("unexpected output that is not `true` or `false`") } // it can fail because this key may not be set // // ```sh // $ pip --version // pip 23.0.1 from /usr/lib/python3/dist-packages/pip (python 3.11) // // $ pip config get global.break-system-packages // ERROR: No such key - global.break-system-packages // // $ echo $? // 1 // ``` Err(_) => false, }; debug!("pip3 externally managed: {} ", extern_managed); debug!("pip3 global.break-system-packages: {}", allow_break_sys_pkg); // Even though pip3 is externally managed, we should still update it if // `global.break-system-packages` is true. if extern_managed && !allow_break_sys_pkg { return Err(SkipStep( "Skip pip3 update as it is externally managed and global.break-system-packages is not true".to_string(), ) .into()); } print_separator("pip3"); if env::var("VIRTUAL_ENV").is_ok() { print_warning("This step is skipped when running inside a virtual environment"); return Err(SkipStep("Does not run inside a virtual environment".to_string()).into()); } ctx.run_type() .execute(&python3) .args(["-m", "pip", "install", "--upgrade", "--user", "pip"]) .status_checked() } pub fn run_pip_review_update(ctx: &ExecutionContext) -> Result<()> { let pip_review = require("pip-review")?; print_separator("pip-review"); if !ctx.config().enable_pip_review() { print_warning( "Pip-review is disabled by default. Enable it by setting enable_pip_review=true in the configuration.", ); return Err(SkipStep(String::from("Pip-review is disabled by default")).into()); } ctx.run_type() .execute(pip_review) .arg("--auto") .status_checked_with_codes(&[1])?; Ok(()) } pub fn run_pip_review_local_update(ctx: &ExecutionContext) -> Result<()> { let pip_review = require("pip-review")?; print_separator("pip-review (local)"); if !ctx.config().enable_pip_review_local() { print_warning( "Pip-review (local) is disabled by default. Enable it by setting enable_pip_review_local=true in the configuration.", ); return Err(SkipStep(String::from("Pip-review (local) is disabled by default")).into()); } ctx.run_type() .execute(pip_review) .arg("--local") .arg("--auto") .status_checked_with_codes(&[1])?; Ok(()) } pub fn run_pipupgrade_update(ctx: &ExecutionContext) -> Result<()> { let pipupgrade = require("pipupgrade")?; print_separator("Pipupgrade"); if !ctx.config().enable_pipupgrade() { print_warning( "Pipupgrade is disabled by default. Enable it by setting enable_pipupgrade=true in the configuration.", ); return Err(SkipStep(String::from("Pipupgrade is disabled by default")).into()); } ctx.run_type() .execute(pipupgrade) .args(ctx.config().pipupgrade_arguments().split_whitespace()) .status_checked()?; Ok(()) } pub fn run_stack_update(ctx: &ExecutionContext) -> Result<()> { if require("ghcup").is_ok() { // `ghcup` is present and probably(?) being used to install `stack`. // Don't upgrade `stack`, let `ghcup` handle it. Per `ghcup install stack`: // !!! Additionally, you should upgrade stack only through ghcup and not use 'stack upgrade' !!! return Ok(()); } let stack = require("stack")?; print_separator("stack"); ctx.run_type().execute(stack).arg("upgrade").status_checked() } pub fn run_ghcup_update(ctx: &ExecutionContext) -> Result<()> { let ghcup = require("ghcup")?; print_separator("ghcup"); ctx.run_type().execute(ghcup).arg("upgrade").status_checked() } pub fn run_tlmgr_update(ctx: &ExecutionContext) -> Result<()> { cfg_if::cfg_if! { if #[cfg(any(target_os = "linux", target_os = "android"))] { if !ctx.config().enable_tlmgr_linux() { return Err(SkipStep(String::from("tlmgr must be explicity enabled in the configuration to run in Android/Linux")).into()); } } } let tlmgr = require("tlmgr")?; let kpsewhich = require("kpsewhich")?; let tlmgr_directory = { let mut d = PathBuf::from( &Command::new(kpsewhich) .arg("-var-value=SELFAUTOPARENT") .output_checked_utf8()? .stdout .trim(), ); d.push("tlpkg"); d } .require()?; let directory_writable = tempfile_in(&tlmgr_directory).is_ok(); debug!("{:?} writable: {}", tlmgr_directory, directory_writable); print_separator("TeX Live package manager"); let mut command = if directory_writable { ctx.run_type().execute(&tlmgr) } else { let sudo = require_option(ctx.sudo().as_ref(), get_require_sudo_string())?; let mut c = ctx.run_type().execute(sudo); c.arg(&tlmgr); c }; command.args(["update", "--self", "--all"]); command.status_checked() } pub fn run_chezmoi_update(ctx: &ExecutionContext) -> Result<()> { let chezmoi = require("chezmoi")?; HOME_DIR.join(".local/share/chezmoi").require()?; print_separator("chezmoi"); ctx.run_type().execute(chezmoi).arg("update").status_checked() } pub fn run_myrepos_update(ctx: &ExecutionContext) -> Result<()> { let myrepos = require("mr")?; HOME_DIR.join(".mrconfig").require()?; print_separator("myrepos"); ctx.run_type() .execute(&myrepos) .arg("--directory") .arg(&*HOME_DIR) .arg("checkout") .status_checked()?; ctx.run_type() .execute(&myrepos) .arg("--directory") .arg(&*HOME_DIR) .arg("update") .status_checked() } pub fn run_custom_command(name: &str, command: &str, ctx: &ExecutionContext) -> Result<()> { print_separator(name); let mut exec = ctx.run_type().execute(shell()); #[cfg(unix)] let command = if let Some(command) = command.strip_prefix("-i ") { exec.arg("-i"); command } else { command }; exec.arg("-c").arg(command).status_checked() } pub fn run_composer_update(ctx: &ExecutionContext) -> Result<()> { let composer = require("composer")?; let composer_home = Command::new(&composer) .args(["global", "config", "--absolute", "--quiet", "home"]) .output_checked_utf8() .map_err(|e| (SkipStep(t!("Error getting the composer directory: {error}", error = e).to_string()))) .map(|s| PathBuf::from(s.stdout.trim()))? .require()?; if !composer_home.is_descendant_of(&HOME_DIR) { return Err(SkipStep( t!( "Composer directory {composer_home} isn't a descendant of the user's home directory", composer_home = composer_home.display() ) .to_string(), ) .into()); } print_separator(t!("Composer")); if ctx.config().composer_self_update() { cfg_if::cfg_if! { if #[cfg(unix)] { // If self-update fails without sudo then there's probably an update let has_update = match ctx.run_type().execute(&composer).arg("self-update").output()? { ExecutorOutput::Wet(output) => !output.status.success(), _ => false }; if has_update { let sudo = require_option(ctx.sudo().as_ref(), get_require_sudo_string())?; ctx.run_type() .execute(sudo) .arg(&composer) .arg("self-update") .status_checked()?; } } else { ctx.run_type().execute(&composer).arg("self-update").status_checked()?; } } } let output = ctx.run_type().execute(&composer).args(["global", "update"]).output()?; if let ExecutorOutput::Wet(output) = output { let output: Utf8Output = output.try_into()?; print!("{}\n{}", output.stdout, output.stderr); if output.stdout.contains("valet") || output.stderr.contains("valet") { if let Some(valet) = which("valet") { ctx.run_type().execute(valet).arg("install").status_checked()?; } } } Ok(()) } pub fn run_dotnet_upgrade(ctx: &ExecutionContext) -> Result<()> { let dotnet = require("dotnet")?; // Skip when the `dotnet tool list` subcommand fails. // (This is expected when a dotnet runtime is installed but no SDK.) let output = match ctx .run_type() .execute(&dotnet) .args(["tool", "list", "--global"]) // dotnet will print a greeting message on its first run, from this question: // https://stackoverflow.com/q/70493706/14092446 // Setting `DOTNET_NOLOGO` to `true` should disable it .env("DOTNET_NOLOGO", "true") .output_checked_utf8() { Ok(output) => output, Err(_) => { return Err(SkipStep( t!("Error running `dotnet tool list`. This is expected when a dotnet runtime is installed but no SDK.") .to_string(), ) .into()); } }; let mut in_header = true; let mut packages = output .stdout .lines() // Skip the header: // // Package Id Version Commands // ------------------------------------- .skip_while(|line| { // The .NET SDK respects locale, so the header can be printed // in languages other than English. The separator should hopefully // always be at least 10 -'s long. if in_header && line.starts_with("----------") { in_header = false; true } else { in_header } }) .filter(|line| !line.is_empty()) .peekable(); if packages.peek().is_none() { return Err(SkipStep(t!("No dotnet global tools installed").to_string()).into()); } print_separator(".NET"); for package in packages { let package_name = package.split_whitespace().next().unwrap(); ctx.run_type() .execute(&dotnet) .args(["tool", "update", package_name, "--global"]) .status_checked() .with_context(|| format!("Failed to update .NET package {package_name:?}"))?; } Ok(()) } pub fn run_helix_grammars(ctx: &ExecutionContext) -> Result<()> { let helix = require("helix").or(require("hx"))?; print_separator("Helix"); ctx.run_type() .execute(&helix) .args(["--grammar", "fetch"]) .status_checked() .with_context(|| "Failed to download helix grammars!")?; ctx.run_type() .execute(&helix) .args(["--grammar", "build"]) .status_checked() .with_context(|| "Failed to build helix grammars!")?; Ok(()) } pub fn run_raco_update(ctx: &ExecutionContext) -> Result<()> { let raco = require("raco")?; print_separator(t!("Racket Package Manager")); ctx.run_type() .execute(raco) .args(["pkg", "update", "--all"]) .status_checked() } pub fn bin_update(ctx: &ExecutionContext) -> Result<()> { let bin = require("bin")?; print_separator("Bin"); ctx.run_type().execute(bin).arg("update").status_checked() } pub fn spicetify_upgrade(ctx: &ExecutionContext) -> Result<()> { // As of 04-07-2023 NixOS packages Spicetify with the `spicetify-cli` binary name let spicetify = require("spicetify").or(require("spicetify-cli"))?; print_separator("Spicetify"); ctx.run_type().execute(spicetify).arg("upgrade").status_checked() } pub fn run_ghcli_extensions_upgrade(ctx: &ExecutionContext) -> Result<()> { let gh = require("gh")?; let result = Command::new(&gh).args(["extensions", "list"]).output_checked_utf8(); if result.is_err() { debug!("GH result {:?}", result); return Err(SkipStep(t!("GH failed").to_string()).into()); } print_separator(t!("GitHub CLI Extensions")); ctx.run_type() .execute(&gh) .args(["extension", "upgrade", "--all"]) .status_checked() } pub fn update_julia_packages(ctx: &ExecutionContext) -> Result<()> { let julia = require("julia")?; print_separator(t!("Julia Packages")); let mut executor = ctx.run_type().execute(julia); executor.arg(if ctx.config().julia_use_startup_file() { "--startup-file=yes" } else { "--startup-file=no" }); executor.args(["-e", "using Pkg; Pkg.update()"]).status_checked() } pub fn run_helm_repo_update(ctx: &ExecutionContext) -> Result<()> { let helm = require("helm")?; print_separator("Helm"); let no_repo = "no repositories found"; let mut success = true; let mut exec = ctx.run_type().execute(helm); if let Err(e) = exec.arg("repo").arg("update").status_checked() { error!("Updating repositories failed: {e}"); success = match exec.output_checked_utf8() { Ok(s) => s.stdout.contains(no_repo) || s.stderr.contains(no_repo), Err(e) => match e.downcast_ref::() { Some(TopgradeError::ProcessFailedWithOutput(_, _, stderr)) => stderr.contains(no_repo), _ => false, }, }; } if success { Ok(()) } else { Err(eyre!(StepFailed)) } } pub fn run_stew(ctx: &ExecutionContext) -> Result<()> { let stew = require("stew")?; print_separator("stew"); ctx.run_type().execute(stew).args(["upgrade", "--all"]).status_checked() } pub fn run_bob(ctx: &ExecutionContext) -> Result<()> { let bob = require("bob")?; print_separator("Bob"); ctx.run_type().execute(bob).args(["update", "--all"]).status_checked() } pub fn run_certbot(ctx: &ExecutionContext) -> Result<()> { let sudo = require_option(ctx.sudo().as_ref(), get_require_sudo_string())?; let certbot = require("certbot")?; print_separator("Certbot"); let mut cmd = ctx.run_type().execute(sudo); cmd.arg(certbot); cmd.arg("renew"); cmd.status_checked() } /// Run `$ freshclam` to update ClamAV signature database /// /// doc: https://docs.clamav.net/manual/Usage/SignatureManagement.html#freshclam pub fn run_freshclam(ctx: &ExecutionContext) -> Result<()> { let freshclam = require("freshclam")?; print_separator(t!("Update ClamAV Database(FreshClam)")); ctx.run_type().execute(freshclam).status_checked() } /// Involve `pio upgrade` to update PlatformIO core. pub fn run_platform_io(ctx: &ExecutionContext) -> Result<()> { // We use the full path because by default the binary is not in `PATH`: // https://github.com/topgrade-rs/topgrade/issues/754#issuecomment-2020537559 #[cfg(unix)] fn bin_path() -> PathBuf { HOME_DIR.join(".platformio/penv/bin/pio") } #[cfg(windows)] fn bin_path() -> PathBuf { HOME_DIR.join(".platformio/penv/Scripts/pio.exe") } let bin_path = require(bin_path())?; print_separator("PlatformIO Core"); ctx.run_type().execute(bin_path).arg("upgrade").status_checked() } /// Run `lensfun-update-data` to update lensfun database. /// /// `sudo` will be used if `use_sudo` configuration entry is set to true. pub fn run_lensfun_update_data(ctx: &ExecutionContext) -> Result<()> { const SEPARATOR: &str = "Lensfun's database update"; let lensfun_update_data = require("lensfun-update-data")?; const EXIT_CODE_WHEN_NO_UPDATE: i32 = 1; if ctx.config().lensfun_use_sudo() { let sudo = require_option(ctx.sudo().as_ref(), get_require_sudo_string())?; print_separator(SEPARATOR); ctx.run_type() .execute(sudo) .arg(lensfun_update_data) // `lensfun-update-data` returns 1 when there is no update available // which should be considered success .status_checked_with_codes(&[EXIT_CODE_WHEN_NO_UPDATE]) } else { print_separator(SEPARATOR); ctx.run_type() .execute(lensfun_update_data) .status_checked_with_codes(&[EXIT_CODE_WHEN_NO_UPDATE]) } } 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()) } 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 = "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() } pub fn run_uv(ctx: &ExecutionContext) -> Result<()> { let uv_exec = require("uv")?; print_separator("uv"); // try uv self --help first - if it succeeds, we call uv self update let result = ctx .run_type() .execute(&uv_exec) .args(["self", "--help"]) .output_checked(); if result.is_ok() { ctx.run_type() .execute(&uv_exec) .args(["self", "update"]) .status_checked()?; } ctx.run_type() .execute(&uv_exec) .args(["tool", "upgrade", "--all"]) .status_checked() } /// Involve `zvm upgrade` to update ZVM pub fn run_zvm(ctx: &ExecutionContext) -> Result<()> { let zvm = require("zvm")?; print_separator("ZVM"); ctx.run_type().execute(zvm).arg("upgrade").status_checked() } pub fn run_bun(ctx: &ExecutionContext) -> Result<()> { let bun = require("bun")?; print_separator("Bun"); ctx.run_type().execute(bun).arg("upgrade").status_checked() }