#![allow(unused_imports)] #[cfg(unix)] use std::os::unix::prelude::MetadataExt; use std::path::PathBuf; use std::process::Command; use anyhow::Result; use directories::BaseDirs; use log::debug; #[cfg(unix)] use nix::unistd::Uid; use semver::Version; use crate::executor::{CommandExt, RunType}; use crate::terminal::print_separator; use crate::utils::{require, PathExt}; use crate::{error::SkipStep, execution_context::ExecutionContext}; #[allow(clippy::upper_case_acronyms)] struct NPM { command: PathBuf, pnpm: Option, } impl NPM { fn new(command: PathBuf) -> Self { Self { command, pnpm: require("pnpm").ok(), } } #[cfg(target_os = "linux")] fn root(&self) -> Result { let version = self.version()?; let args = if version < Version::new(8, 11, 0) { ["root", "-g"] } else { ["root", "--location=global"] }; Command::new(&self.command) .args(args) .check_output() .map(|s| PathBuf::from(s.trim())) } fn version(&self) -> Result { let version_str = Command::new(&self.command) .args(&["--version"]) .check_output() .map(|s| s.trim().to_owned()); Version::parse(&version_str?).map_err(|err| err.into()) } fn upgrade(&self, run_type: RunType, use_sudo: bool) -> Result<()> { print_separator("Node Package Manager"); let version = self.version()?; let args = if version < Version::new(8, 11, 0) || self.pnpm.is_some() { ["update", "-g"] } else { ["update", "--location=global"] }; if use_sudo { run_type .execute("sudo") .arg(self.pnpm.as_ref().unwrap_or(&self.command)) .args(args) .check_run()?; } else { run_type.execute(&self.command).args(args).check_run()?; } Ok(()) } #[cfg(target_os = "linux")] pub fn should_use_sudo(&self) -> Result { let npm_root = self.root()?; if !npm_root.exists() { return Err(SkipStep(format!("NPM root at {} doesn't exist", npm_root.display(),)).into()); } let metadata = std::fs::metadata(&npm_root)?; let uid = Uid::effective(); Ok(metadata.uid() != uid.as_raw() && metadata.uid() == 0) } } struct Yarn { command: PathBuf, yarn: Option, } impl Yarn { fn new(command: PathBuf) -> Self { Self { command, yarn: require("yarn").ok(), } } #[cfg(target_os = "linux")] fn root(&self) -> Result { let args = ["global", "dir"]; Command::new(&self.command) .args(args) .check_output() .map(|s| PathBuf::from(s.trim())) } fn upgrade(&self, run_type: RunType, use_sudo: bool) -> Result<()> { print_separator("Yarn Package Manager"); let args = ["global", "upgrade"]; if use_sudo { run_type .execute("sudo") .arg(self.yarn.as_ref().unwrap_or(&self.command)) .args(args) .check_run()?; } else { run_type.execute(&self.command).args(args).check_run()?; } Ok(()) } #[cfg(target_os = "linux")] pub fn should_use_sudo(&self) -> Result { let yarn_root = self.root()?; if !yarn_root.exists() { return Err(SkipStep(format!("NPM root at {} doesn't exist", yarn_root.display(),)).into()); } let metadata = std::fs::metadata(&yarn_root)?; let uid = Uid::effective(); Ok(metadata.uid() != uid.as_raw() && metadata.uid() == 0) } } #[cfg(target_os = "linux")] fn should_use_sudo(npm: &NPM, ctx: &ExecutionContext) -> Result { if npm.should_use_sudo()? { if ctx.config().npm_use_sudo() { Ok(true) } else { Err(SkipStep("NPM root is owned by another user which is not the current user. Set use_sudo = true under the NPM section in your configuration to run NPM as sudo".to_string()) .into()) } } else { Ok(false) } } #[cfg(target_os = "linux")] fn should_use_sudo_yarn(yarn: &Yarn, ctx: &ExecutionContext) -> Result { if yarn.should_use_sudo()? { if ctx.config().yarn_use_sudo() { Ok(true) } else { Err(SkipStep("NPM root is owned by another user which is not the current user. Set use_sudo = true under the NPM section in your configuration to run NPM as sudo".to_string()) .into()) } } else { Ok(false) } } pub fn run_npm_upgrade(ctx: &ExecutionContext) -> Result<()> { let npm = require("npm").map(NPM::new)?; #[cfg(target_os = "linux")] { npm.upgrade(ctx.run_type(), should_use_sudo(&npm, ctx)?) } #[cfg(not(target_os = "linux"))] { npm.upgrade(ctx.run_type(), false) } } pub fn run_yarn_upgrade(ctx: &ExecutionContext) -> Result<()> { let yarn = require("yarn").map(Yarn::new)?; #[cfg(target_os = "linux")] { yarn.upgrade(ctx.run_type(), should_use_sudo_yarn(&yarn, ctx)?) } #[cfg(not(target_os = "linux"))] { yarn.upgrade(ctx.run_type(), false) } } pub fn deno_upgrade(ctx: &ExecutionContext) -> Result<()> { let deno = require("deno")?; let deno_dir = ctx.base_dirs().home_dir().join(".deno"); if !deno.canonicalize()?.is_descendant_of(&deno_dir) { let skip_reason = SkipStep("Deno installed outside of .deno directory".to_string()); return Err(skip_reason.into()); } print_separator("Deno"); ctx.run_type().execute(&deno).arg("upgrade").check_run() }