From 149f67a8af039763362c91d59bfa69fcd890e0cb Mon Sep 17 00:00:00 2001 From: Johannes Schindelin Date: Thu, 1 Jul 2021 13:30:01 +0200 Subject: [PATCH 1/2] Allow overriding which POSIX shell to use Right now, the compiled-in default is used always, which can be overridden at compile time via the `SHELL_PATH` variable. However, in Git for Windows' context, there is a scenario where this is not good enough: the BusyBox flavor of MinGit wants to use BusyBox' `ash` instead. To allow for that, let's introduce a new config variable: `core.shell`. Its value is expected to be the absolute path to a valid, working POSIX shell. This shell will be then used by `git.exe` whenever it needs to execute something via shell, such as moderately complex aliases. And it is not only Git that is expected to use that `ash`, it is also OpenSSH (e.g. when running any configured `ProxyCommand`): programs spawned from `git.exe` may need to run a shell and look at the environment variable `SHELL` for that. Therefore, let's set that, too. Signed-off-by: Johannes Schindelin --- Documentation/config/core.txt | 4 ++++ help.c | 2 +- run-command.c | 20 +++++++++++++++++--- run-command.h | 5 +++++ t/t0061-run-command.sh | 7 +++++++ 5 files changed, 34 insertions(+), 4 deletions(-) diff --git a/Documentation/config/core.txt b/Documentation/config/core.txt index c04f62a54a154c..2d3f790a90fb67 100644 --- a/Documentation/config/core.txt +++ b/Documentation/config/core.txt @@ -628,3 +628,7 @@ core.abbrev:: If set to "no", no abbreviation is made and the object names are shown in their full length. The minimum length is 4. + +core.shell:: + Set the absolute path to the executable to use as a POSIX shell; + This will also set/override the environment variable `SHELL`. diff --git a/help.c b/help.c index 3c3bdec21356d9..4690f7241df946 100644 --- a/help.c +++ b/help.c @@ -662,7 +662,7 @@ void get_version_info(struct strbuf *buf, int show_build_options) strbuf_addstr(buf, "no commit associated with this build\n"); strbuf_addf(buf, "sizeof-long: %d\n", (int)sizeof(long)); strbuf_addf(buf, "sizeof-size_t: %d\n", (int)sizeof(size_t)); - strbuf_addf(buf, "shell-path: %s\n", SHELL_PATH); + strbuf_addf(buf, "shell-path: %s\n", get_shell_path(SHELL_PATH)); /* NEEDSWORK: also save and output GIT-BUILD_OPTIONS? */ } } diff --git a/run-command.c b/run-command.c index be6bc128cd9df2..38cb5b112cb58c 100644 --- a/run-command.c +++ b/run-command.c @@ -9,6 +9,20 @@ #include "quote.h" #include "config.h" +const char *get_shell_path(const char *fallback) +{ + static const char *shell; + static int initialized; + + if (!initialized) { + if (!git_config_get_pathname("core.shell", &shell)) + setenv("SHELL", shell, 1); + initialized = 1; + } + + return shell ? shell : fallback; +} + void child_process_init(struct child_process *child) { memset(child, 0, sizeof(*child)); @@ -271,9 +285,9 @@ static const char **prepare_shell_cmd(struct strvec *out, const char **argv) if (strcspn(argv[0], "|&;<>()$`\\\"' \t\n*?[#~=%") != strlen(argv[0])) { #ifndef GIT_WINDOWS_NATIVE - strvec_push(out, SHELL_PATH); + strvec_push(out, get_shell_path(SHELL_PATH)); #else - strvec_push(out, "sh"); + strvec_push(out, get_shell_path("sh")); #endif strvec_push(out, "-c"); @@ -411,7 +425,7 @@ static int prepare_cmd(struct strvec *out, const struct child_process *cmd) * Add SHELL_PATH so in the event exec fails with ENOEXEC we can * attempt to interpret the command with 'sh'. */ - strvec_push(out, SHELL_PATH); + strvec_push(out, get_shell_path(SHELL_PATH)); if (cmd->git_cmd) { prepare_git_cmd(out, cmd->argv); diff --git a/run-command.h b/run-command.h index d08414a92e734c..4f10b63184dfad 100644 --- a/run-command.h +++ b/run-command.h @@ -483,4 +483,9 @@ int run_processes_parallel_tr2(int n, get_next_task_fn, start_failure_fn, task_finished_fn, void *pp_cb, const char *tr2_category, const char *tr2_label); +/* + * Get the path of the POSIX shell to use in `start_command()`. + */ +const char *get_shell_path(const char *fallback); + #endif diff --git a/t/t0061-run-command.sh b/t/t0061-run-command.sh index 7d599675e35a75..366720a174a458 100755 --- a/t/t0061-run-command.sh +++ b/t/t0061-run-command.sh @@ -233,4 +233,11 @@ test_expect_success MINGW 'can spawn .bat with argv[0] containing spaces' ' grep "git-upload-pack" out ' +SQ="'" +test_expect_success 'core.shell' ' + test_config_global core.shell "$GIT_EXEC_PATH/git$X" && + test_must_fail git -c alias.eq="!a.b=c" eq 2>actual && + grep "${SQ}a.b=c${SQ} is not a git command" actual +' + test_done From acbd1f3de838ec2ad7b45356bf28f10b3b66024f Mon Sep 17 00:00:00 2001 From: Johannes Schindelin Date: Thu, 1 Jul 2021 13:49:36 +0200 Subject: [PATCH 2/2] mingw: respect core.shell when executing scripts On Windows, we have to emulate that Linux/Unix/macOS feature where a file starting with a `#!` line and being marked as executable is run as a script through the interpreter specified by said `#!` line. Traditionally, we ignore the actual path specified in that line because it will be a Unix-style path anyway, something that `git.exe` is not even supposed to understand. We then go on to look up the actual path of the interpreter by iterating over the components in the environment variable `PATH`. Let's special-case `sh` in that scenario when the config setting `core.shell` exists: in this case, we want to use it instead. This allows us to configure BusyBox' `ash` to be used for all of the shell scripting needs of the BusyBox flavor of MinGit. While at it, assume that any shell configured via `core.shell` is _not_ an MSYS2 shell, i.e. that we should use regular Win32 command-line quoting, not MSYS2/Cygwin one. Signed-off-by: Johannes Schindelin --- compat/mingw.c | 10 ++++++++++ t/t0061-run-command.sh | 7 +++++++ 2 files changed, 17 insertions(+) diff --git a/compat/mingw.c b/compat/mingw.c index aa647b367b0fab..5d9cc59a707e0a 100644 --- a/compat/mingw.c +++ b/compat/mingw.c @@ -1277,6 +1277,10 @@ static char *path_lookup(const char *cmd, int exe_only) if (strpbrk(cmd, "/\\")) return xstrdup(cmd); + if (!strcmp(cmd, "sh") && + (prog = xstrdup_or_null(get_shell_path(NULL)))) + return prog; + path = mingw_getenv("PATH"); if (!path) return NULL; @@ -1463,6 +1467,12 @@ static int is_msys2_sh(const char *cmd) if (ret >= 0) return ret; + if (get_shell_path(NULL)) { + /* Assume an overridden shell is not MSYS2 */ + ret = 0; + return ret; + } + p = path_lookup(cmd, 0); if (!p) ret = 0; diff --git a/t/t0061-run-command.sh b/t/t0061-run-command.sh index 366720a174a458..8f7618476db183 100755 --- a/t/t0061-run-command.sh +++ b/t/t0061-run-command.sh @@ -240,4 +240,11 @@ test_expect_success 'core.shell' ' grep "${SQ}a.b=c${SQ} is not a git command" actual ' +test_expect_success MINGW 'core.shell executes scripts' ' + test_config_global core.shell "$GIT_EXEC_PATH/git$X" && + echo "#!/bin/sh" >script && + test_must_fail git -c alias.s="!./script" s 2>actual && + grep "${SQ}./script${SQ} is not a git command" actual +' + test_done