Tests on Windows must usually be run in Git Bash or a similar environment
See https://github.com/Byron/gitoxide/issues/1359#issuecomment-2316614616 for updated information.
Current behavior 😯
In this project, tests are typically run on Windows in a Git Bash environment. On CI in Windows, bash is Git Bash. This is also the environment from which I have typically run tests. This is a native environment; it should not be confused with WSL or even something like Cygwin. Furthermore, tools maintained by rustup are not specifically connected to this environment, though they are run from it.
All tests are able to pass when run in Git Bash on Windows, and likely would pass if run in similar environments that provide a POSIX-compatible shell for Windows, set environment variables accordingly, and provide access to POSIX versions of common tools.
But running the tests from PowerShell instead produces 627 test failures. This presumably relates to the associated environment rather than the shell itself, since however the tests are run, a test runner subprocess is controlling the run.
The main problem seems to be the use of paths with backslashes in them, and the way that interacts with fixture scripts. Note, however, that this is not specific to the situation where GIX_TEST_IGNORE_ARCHIVES is set. That is to say that this is a separate issue from #1358 (which also involves far fewer failures) and occurs even when GIX_TEST_IGNORE_ARCHIVES is unset.
Even the test summary showing one failure per line is too long for GitHub to allow me to include it in this issue description. This gist has it. The full output can be seen in this other gist.
As of now, I have not figured out the cause, which I think is a prerequisite for making a decision about whether to try to support such environments or to instead document them as unsupported.
Expected behavior 🤔
Either all tests should pass even when run from PowerShell, or the need to run them in a Git Bash environment (or whatever other more precise requirement is known) should be documented.
I think which is better depends on how cumbersome it would be to enable the tests to pass when run from PowerShell and, more importantly, how cumbersome it would be to maintain this state. I suspect it may be feasible, though.
Git behavior
A direct comparison is difficult in that this is about running the tests rather than the functionality of Git and gitoxide. However, it is worth noting that Git for Windows has its own SDK environment that is used for building it and running its tests, and this environment is separate even from the more minimal Git Bash environment.
Thus, if it is infeasible to support running tests from PowerShell, documenting that would still not be very restrictive, even compared to what is needed to develop and test Git for Windows, since the non-SDK "Git Bash" environment works fine for running gitoxide's tests.
Steps to reproduce 🕹
Using Windows, make sure the tests can pass when run from Git Bash using the first of two cargo test-runner commands shown below.
Optionally clean the build with cargo clean or, if preferred, clean everything with gix clean -xde.
There are two approaches. The first approach is to display the output in the console: To do that, in PowerShell, run:
cargo nextest --all --no-fail-fast
However, that is hard to read because of the very large volume of output combined with the staggering of output lines suggesting a problem identifying the terminal width or computing the width of text written.
Therefore, you may instead wish to write both stdout and stderr to a file, and inspect the file during and/or after the run.
cargo nextest --all --no-fail-fast *> ../output.txt
This writes all output to output.txt in the directory above the current directory. This is to avoid creating a new file in the repository while running the tests, in case that were to interfere with something, though it shouldn't since that should only be able to affect a small number of journey test runs, which are not included when running cargo nextest.
The *> operator in PowerShell is similar to the &> or >& operators in some shells and to the effect of 2>&1 > in most shells.
Although Windows comes with Windows PowerShell, I suggest against using it for this, since is somewhat less intuitive, and has fewer convenience features, compared to the newer PowerShell (which is sometimes called PowerShell Core). Windows PowerShell is also less likely to be used by developers on Windows, and thus probably less important to support than recent versions of PowerShell. I used PowerShell 7.4.2 on Windows 10.
Thanks for the detailed report!
It appears that gix-testtools are unable to execute any script with backslashes in its path as these will count as escape-character. So for this to work (better), all it might take is to properly escape such paths. There might be more to it though. Maybe a Windows build of gix-testtools could have special mitigations for that built in at some point.
Sorry, I posted this comment in the wrong place! I've reposted it at #1429 and hid this.
I've opened #1559 for this. To make sure it really fixes the CI regression, I've rebased this PR onto 4f2ab5b from it, which should cause CI here to fail again (i.e. to correctly report the test failure that was already happening). Assuming #1559 is merged, the extra commit here will go away when I rebase this onto main again afterwards.
As noted in #1559, my claim in this issue is apparently overstated, since tests are able to run in PowerShell on Windows GHA runners, and a pwsh shell that does not have any Unix-style shell as an ancestor in the process tree is decisively not a "Git Bash or similar environment" in the sens that I meant it or that this phrase is likely to be understood. I'll try to narrow down the claim here when I can; for now, this comment serves as a partial correction. That the tests work in pwsh on CI may, through examination of the CI environment help reveal what is going on here.
The reason this works on CI as noted in https://github.com/Byron/gitoxide/issues/1359#issuecomment-2309382458, while not working locally, is that bash on a Windows GHA runner is Git Bash even in the PATH used in pwsh, while this is rare locally.
That the current implementation is ever able to use Git Bash on Windows, even when the tests are run from a Git Bash shell, also depends inadvertently on undocumented behavior of std::process::Command that may change in the future. So it's possible Windows tests will break suddenly at some point, even if run from Git Bash. Fixing this bug appears feasible and will address that, too. Details follow.
How the tests fail
Consider the following output, which is typical of the test failures:
FAIL [ 0.557s] gix remote::connection::fetch::refs::tests::update::local_direct_refs_are_written_with_symbolic_ones
--- STDOUT: gix remote::connection::fetch::refs::tests::update::local_direct_refs_are_written_with_symbolic_ones ---
running 1 test
test remote::connection::fetch::refs::tests::update::local_direct_refs_are_written_with_symbolic_ones ... FAILED
failures:
failures:
remote::connection::fetch::refs::tests::update::local_direct_refs_are_written_with_symbolic_ones
test result: FAILED. 0 passed; 1 failed; 0 ignored; 0 measured; 16 filtered out; finished in 0.54s
--- STDERR: gix remote::connection::fetch::refs::tests::update::local_direct_refs_are_written_with_symbolic_ones ---
Archive at 'tests\fixtures\generated-archives\make_remote_repos.tar' not found, creating fixture using script 'make_remote_repos.sh'
thread 'remote::connection::fetch::refs::tests::update::local_direct_refs_are_written_with_symbolic_ones' panicked at tests\tools\src\lib.rs:566:17:
fixture script of "bash" "C:\\Users\\ek\\source\\repos\\gitoxide\\gix\\tests\\fixtures\\make_remote_repos.sh" failed: stdout:
stderr: /bin/bash: C:Userseksourcereposgitoxidegixtestsfixturesmake_remote_repos.sh: No such file or directory
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
There are three things going on here:
- Some generated archives are (intentionally) listed in
.gitignorefiles, so fixture scripts always have to run to create them. Therefore, a number of tests will fail if fixture scripts cannot run successfully. - Running fixture scripts on Windows passes their Windows-style paths to
bash. These paths are resolved before use, so they usually begin with drive letters, e.g. withC:. They also use\(rather than/) characters as directory separators. - Whether
bashcan tolerate a script path argument that starts with a Windows drive letter, or that uses\as a directory separator, varies acrossbashimplementations. Whichbashis selected is affected by what exists on the system, the value of thePATHenvironment variable, and subtleties ofstd::process::Command.
Why is it even running bash?
The sequence is:
-
Find that a generated archive is not available, so that running the fixture script is necessary.
-
Attempt to run the script as an executable:
https://github.com/Byron/gitoxide/blob/1cfe577d461293879e91538dbc4bbfe01722e1e8/tests/tools/src/lib.rs#L553
On Windows, that always fails, because files that do not match a supported binary executable format must be batch files with a
.cmdor.batextension forstd::process::Commandto run them automatically with an interpreter (whichCommandsupports becauseCreateProcessWsupports it). The fixture scripts here arebashscripts and they have a.shsuffix. -
Check if the error was due to the inability to execute the file:
https://github.com/Byron/gitoxide/blob/1cfe577d461293879e91538dbc4bbfe01722e1e8/tests/tools/src/lib.rs#L555-L556
Windows error code 193 means this is a "not a valid Win32 application" error, so we do effectively detect that the script was not able to be executed.
-
Try again by running
bash, with the script path and the arguments for the script as its arguments:https://github.com/Byron/gitoxide/blob/1cfe577d461293879e91538dbc4bbfe01722e1e8/tests/tools/src/lib.rs#L558-L559
What is bash on Windows?
Today, there are two typical cases for a bash shell on Windows:
-
The port of
bashto Windows that is provided by MSYS2 and similar environments, including Git Bash, tolerates backslashes used as directory separators. -
The
bashassociated with WSL does not, since it delegates tobashin an installed distribution for WSL, which either fails because there is no distribution or the distribution does not havebash, or succeeds but runs that distribution's nativebashexecutable that has no custom behavior for compatibility with Windows.In addition, even if the paths were normalized so a WSL
bashwould run the fixture scripts, we should expect the scripts, if run that way, to fail to produce correct test data. This is because they would be run inside a WSL distribution, but these are the scripts that have to be run because their generated archives are in.gitignore, which should only be done if the scripts do not currently produce archives that work when used on another system, including when generated on a GNU/Linux system and used on Windows.So any fixture for which this would work reliably should just be omitted from all
.gitignorefiles and have its generated archive committed. Fixing the paths is thus not a worthwhile solution, at least not if done by itself.
Of those, which bash is chosen?
In practice, there seem to be three cases for how bash is resolved on Windows:
- In an MSYS2-like environment including Git Bash, the
PATHwill almost always be set in such a way that this environment'sbashprecedes any otherbash. - Outside such an environment, the
PATHwill usually be set so that thebashassociated with WSL, usually atC:\Windows\System32\bash.exe, precedes any otherbash. - Outside such an environment but on a GitHub Actions runner, the
PATHis set so thatbashis Git Bash.
So when the the tests are not run from Git Bash or a similar environment, they will usually find the wrong bash, the major exception being when run on CI.
Here's a simplified demonstration on my machine, in PowerShell (pwsh), where bash is the WSL-associated bash, and the scoop-installed bash is Git Bash (though the effect is in no way scoop-specific):
C:\Users\ek\tmp> (Get-Command bash).path
C:\Windows\system32\bash.exe
C:\Users\ek\tmp> Get-Content 'a\b'
echo 'Hello, world!'
C:\Users\ek\tmp> bash 'a/b'
Hello, world!
C:\Users\ek\tmp> bash 'a\b'
/bin/bash: ab: No such file or directory
C:\Users\ek\tmp> C:\Users\ek\scoop\shims\bash.exe 'a/b'
Hello, world!
C:\Users\ek\tmp> C:\Users\ek\scoop\shims\bash.exe 'a\b'
Hello, world!
But that's far from the whole story
To run Git Bash, why is it sufficient for the directory that contains the Git Bash bash.exe to precede the System32 directory (that has the WSL-associated bash.exe) in PATH? Why does it search in PATH before other locations?
The above example feels convincing, but it shouldn't. I ran those commands in PowerShell, which searches PATH first. On most OSes, a path search is a PATH search, but that is not generally the case on Windows, nor is this really an accurate characterization of what std::process::Command is doing.
- Other than shells, which are expected to perform their own path search with different rules from
CreateProcessW, most programs on Windows do not do path search themselves and instead letCreateProcessWdo it. Most libraries and frameworks that offer process creation facilities delegate fully, including for path search, toCreateProcessW. ButCreateProcessWsearches various locations beforePATH, regardless of whether or where those locations are also listed inPATH. These prioritized locations include theSystem32directory, which usually contains the WSL-associatedbash.exe. - As noted in #1432 (comment), for security, the Rust standard library performs its own
PATHsearch whenstd::process::Commandis used. It passes the resolved absolute path toCreateProcessW, rather than usingCreateProcessWfor the search. This is mainly to avoid searching in the current directory. It was implemented in rust-lang/rust#87704 and fixes the main concern of rust-lang/rust#87945. - But that is not intended to, and usually does not, prevent the
System32directory from taking precedence overPATHdirectories. As reported in rust-lang/rust#122660,std::process::Commandwill often choose thebash.exeinSystem32rather than a Git Bashbash.exethat has been placed earlier inPATH, for this very reason. - But as noted there and elaborated in rust-lang/rust#122660 (comment) (see also rust-lang/rust#37519), this is averted when methods such as
envare called on theCommandinstance to customize the child process environment. This happens because, when customizing its environment, the current (undocumented) behavior ofCommandis to search thePATHfor the child before searching special locations and thePATHof the parent. This is done even ifPATHis not one of the environment variables being changed or unset.
The latter condition applies to the way gix-testtools runs fixture scripts on Windows. Returning attention to the code shown above, it calls configure_command, which makes a few env_remove calls and numerous env calls.
https://github.com/Byron/gitoxide/blob/1cfe577d461293879e91538dbc4bbfe01722e1e8/tests/tools/src/lib.rs#L598-L604
There are many more env calls than shown in that quoted portion, but none of them set PATH, nor is that variable ever unset or the environment ever cleared. So the child PATH is the parent's PATH, but since other customizations to the environment have been made, std::process::Command searches this PATH for bash before it searches in any special locations.
A possible fix
Although I tend to think about improvements to how gix-testtools runs fixture scripts as being closely connected to improvements to gix-command and the machinery for running hooks, I think the key to identifying a fix for this bug is to notice that they are different in ways that allow changes to gix-testtools that might not be suitable elsewhere.
However gix-testtools runs fixture scripts, it:
- Does not need to be compatible with how
gitruns scripts (though it should use the same shell if possible). - Does not need to be compatible with any documented or intended behavior related to how any other gitoxide crates run scripts.
- Does not need to be able to run scripts that are not shell scripts.
- Does not need to be able to run binary executables.
- Can require
git, because nearly all of the fixture scripts would fail without it anyway.
Therefore, on Windows, gix-testtools can always safely attempt to use Git Bash to run the script first. If it can't find Git Bash based on information provided by git, then it can either just fail, or fall back to something like what it is currently doing.
Git Bash can usually be located using a git --exec-path-based method like what is done in https://github.com/gitpython-developers/GitPython/pull/1791. Conceptually, this is along the lines of
realpath -- "$(git --exec-path)/../../../bin/bash.exe"
though of course it would be in Rust and not Bash, since the point is to find which bash to use.
Thanks so much for researching this, and the very informative and interesting read!
The special-behaviour in std::process::Command is indeed something that could break a lot of people if it ever changes, and if it wasn't intended to kick-in when PATH isn't set, then it might well be considered a bug that will be fixed eventually.
By the looks of it, this does affect gix-command as well, which is exposed to the undocumented behaviour of std::process::Command currently.
Fixing this both in gix-testtools and in gix-command should be possible, I was thinking about using gix_path::env::exe_location() can be used to derive the bash.exe location, which would be nice as it's cached.
However, I wouldn't mind any fix to start with whatever is needed for gix-testtools, and then take it from there.