diff --git a/wait/host_port.go b/wait/host_port.go index e44f50ae62c..c0f686e27d8 100644 --- a/wait/host_port.go +++ b/wait/host_port.go @@ -13,6 +13,13 @@ import ( "github.com/docker/go-connections/nat" ) +type exitStatus = int + +const ( + exitEaccess exitStatus = 126 // container cmd can't be invoked (permission denied) + exitCmdNotFound exitStatus = 127 // container cmd not found/does not exist or invalid bind-mount +) + // Implement interface var ( _ Strategy = (*HostPortStrategy)(nil) @@ -20,6 +27,7 @@ var ( ) var errShellNotExecutable = errors.New("/bin/sh command not executable") +var errShellNotFound = errors.New("/bin/sh command not found") type HostPortStrategy struct { // Port is a string containing port number and protocol in the format "80/tcp" @@ -207,10 +215,16 @@ func internalCheck(ctx context.Context, internalPort nat.Port, target StrategyTa return fmt.Errorf("%w, host port waiting failed", err) } + /* + Docker maps exit code 127 to 126, while Podman treats them separately. + Handle both to ensure compatibility with Docker and Podman. + */ if exitCode == 0 { break - } else if exitCode == 126 || exitCode == 127 { + } else if exitCode == exitEaccess { return errShellNotExecutable + } else if exitCode == exitCmdNotFound { + return errShellNotFound } } return nil diff --git a/wait/host_port_test.go b/wait/host_port_test.go index bf349f05a71..3ef1476260e 100644 --- a/wait/host_port_test.go +++ b/wait/host_port_test.go @@ -497,7 +497,7 @@ func TestHostPortStrategySucceedsGivenShellIsNotInstalled(t *testing.T) { }, ExecImpl: func(_ context.Context, _ []string, _ ...exec.ProcessOption) (int, io.Reader, error) { // This is the error that would be returned if the shell is not installed. - return 126, nil, nil + return exitEaccess, nil, nil }, }