Skip to content

Commit

Permalink
fix: mise does not operate well under Git Bash on Windows
Browse files Browse the repository at this point in the history
  • Loading branch information
roele committed Jan 10, 2025
1 parent a19ec94 commit f90a9ea
Show file tree
Hide file tree
Showing 3 changed files with 81 additions and 5 deletions.
2 changes: 1 addition & 1 deletion src/assets/bash_zsh_support/chpwd/function.sh
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
# shellcheck shell=bash
export -a chpwd_functions
declare -a chpwd_functions
function __zsh_like_cd()
{
\typeset __zsh_like_cd_hook
Expand Down
82 changes: 79 additions & 3 deletions src/shell/bash.rs
Original file line number Diff line number Diff line change
@@ -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};
Expand All @@ -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"
Expand Down Expand Up @@ -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")
});
Expand Down Expand Up @@ -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::<Vec<String>>()
.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::<Vec<String>>()
.join(":");
format!("export {k}=\"{path}:${k}\"\n")
}

fn unset_env(&self, k: &str) -> String {
Expand All @@ -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](/.*)?(/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](/.*?)?(/usr/local)?/bin"#);
p = regex_git.replace_all(&p, "$2/bin").to_string();
p
}

#[cfg(test)]
mod tests {
use insta::assert_snapshot;
Expand Down Expand Up @@ -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:\\Random\\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");
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ if [[ ";${PROMPT_COMMAND:-};" != *";_mise_hook;"* ]]; then
PROMPT_COMMAND="_mise_hook${PROMPT_COMMAND:+;$PROMPT_COMMAND}"
fi
# shellcheck shell=bash
export -a chpwd_functions
declare -a chpwd_functions
function __zsh_like_cd()
{
\typeset __zsh_like_cd_hook
Expand Down

0 comments on commit f90a9ea

Please sign in to comment.