From acdaf3292a6f170a2937982fe3142fbb146f528a Mon Sep 17 00:00:00 2001 From: Roland Schaer Date: Fri, 10 Jan 2025 11:23:16 -0300 Subject: [PATCH] fix: mise does not operate well under Git Bash on Windows --- src/shell/bash.rs | 82 +++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 79 insertions(+), 3 deletions(-) diff --git a/src/shell/bash.rs b/src/shell/bash.rs index c1d967679f..72daa42a25 100644 --- a/src/shell/bash.rs +++ b/src/shell/bash.rs @@ -1,8 +1,10 @@ #![allow(unknown_lints)] #![allow(clippy::literal_string_with_formatting_args)] +use std::env; use std::fmt::Display; use indoc::formatdoc; +use xx::regex; use crate::config::Settings; use crate::shell::{ActivateOptions, Shell}; @@ -15,7 +17,7 @@ impl Shell for Bash { let exe = opts.exe; let flags = opts.flags; let settings = Settings::get(); - let exe = exe.to_string_lossy(); + let exe = get_cygpath(&exe.to_string_lossy()); let mut out = formatdoc! {r#" export MISE_SHELL=bash export __MISE_ORIG_PATH="$PATH" @@ -56,7 +58,7 @@ impl Shell for Bash { {chpwd_load} chpwd_functions+=(_mise_hook) _mise_hook - "#, + "#, chpwd_functions = include_str!("../assets/bash_zsh_support/chpwd/function.sh"), chpwd_load = include_str!("../assets/bash_zsh_support/chpwd/load.sh") }); @@ -101,13 +103,21 @@ impl Shell for Bash { } fn set_env(&self, k: &str, v: &str) -> String { + let v = env::split_paths(&v) + .map(|p| get_cygpath(p.to_str().unwrap())) + .collect::>() + .join(":"); let k = shell_escape::unix::escape(k.into()); let v = shell_escape::unix::escape(v.into()); format!("export {k}={v}\n") } fn prepend_env(&self, k: &str, v: &str) -> String { - format!("export {k}=\"{v}:${k}\"\n") + let path = env::split_paths(&v) + .map(|p| get_cygpath(p.to_str().unwrap())) + .collect::>() + .join(":"); + format!("export {k}=\"{path}:${k}\"\n") } fn unset_env(&self, k: &str) -> String { @@ -121,6 +131,44 @@ impl Display for Bash { } } +fn get_cygpath(s: &str) -> String { + match () { + _ if is_windows_cygwin() => get_cygwinpath(s), + _ if is_windows_msys() => get_msyspath(s), + _ => s.to_string(), + } +} + +fn is_windows_msys() -> bool { + cfg!(windows) && env::var("MSYSTEM").is_ok() +} + +fn get_msyspath(s: &str) -> String { + let regex_path = regex!(r"^([A-Za-z]):\\"); + let mut p = regex_path + .replace_all(s, "/$1/") + .replace("\\", "/") + .to_string(); + let regex_git = regex!(r#"/[A-Za-z](/.*)?/Git(/mingw64|/usr)/bin"#); + p = regex_git.replace_all(&p, "$2/bin").to_string(); + p +} + +fn is_windows_cygwin() -> bool { + cfg!(windows) && env::var("PWD").map_or(false, |v| v.starts_with("/")) +} + +fn get_cygwinpath(s: &str) -> String { + let regex_path = regex!(r"^([A-Za-z]):\\"); + let mut p = regex_path + .replace_all(s, "/cygdrive/$1/") + .replace("\\", "/") + .to_string(); + let regex_git = regex!(r#"/cygdrive/[A-Za-z](/.*?)?/cygwin64(/usr/local)?/bin"#); + p = regex_git.replace_all(&p, "$2/bin").to_string(); + p +} + #[cfg(test)] mod tests { use insta::assert_snapshot; @@ -164,4 +212,32 @@ mod tests { let deactivate = Bash::default().deactivate(); assert_snapshot!(replace_path(&deactivate)); } + + #[test] + fn test_get_msyspath() { + assert_eq!( + get_msyspath("C:\\Users\\User\\Program"), + "/C/Users/User/Program" + ); + assert_eq!( + get_msyspath("C:\\Program Files\\Git\\mingw64\\bin"), + "/mingw64/bin" + ); + assert_eq!(get_msyspath("D:\\Git\\mingw64\\bin"), "/mingw64/bin"); + assert_eq!(get_msyspath("C:\\Program Files\\Git\\usr\\bin"), "/usr/bin"); + } + + #[test] + fn test_get_cygwinpath() { + assert_eq!( + get_cygwinpath("C:\\Users\\User\\Program"), + "/cygdrive/C/Users/User/Program" + ); + assert_eq!( + get_cygwinpath("C:\\Program Files\\cygwin64\\usr\\local\\bin"), + "/usr/local/bin" + ); + assert_eq!(get_cygwinpath("C:\\Program Files\\cygwin64\\bin"), "/bin"); + assert_eq!(get_cygwinpath("D:\\cygwin64\\bin"), "/bin"); + } }