mirror of
https://github.com/golang/go.git
synced 2025-12-28 06:34:04 +00:00
The changes are likely to break users, and we need to make it easy to unbreak without code changes. For #43724. Fixes #53962. Change-Id: I105c5d6c801d354467e0cefd268189c18846858e Reviewed-on: https://go-review.googlesource.com/c/go/+/419794 Reviewed-by: Bryan Mills <bcmills@google.com> Reviewed-by: Ian Lance Taylor <iant@google.com> TryBot-Result: Gopher Robot <gobot@golang.org> Run-TryBot: Russ Cox <rsc@golang.org>
192 lines
6.2 KiB
Go
192 lines
6.2 KiB
Go
// Copyright 2022 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package exec_test
|
|
|
|
import (
|
|
"errors"
|
|
"internal/testenv"
|
|
"os"
|
|
. "os/exec"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
"testing"
|
|
)
|
|
|
|
var pathVar string = func() string {
|
|
if runtime.GOOS == "plan9" {
|
|
return "path"
|
|
}
|
|
return "PATH"
|
|
}()
|
|
|
|
func TestLookPath(t *testing.T) {
|
|
testenv.MustHaveExec(t)
|
|
|
|
tmpDir := filepath.Join(t.TempDir(), "testdir")
|
|
if err := os.Mkdir(tmpDir, 0777); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
executable := "execabs-test"
|
|
if runtime.GOOS == "windows" {
|
|
executable += ".exe"
|
|
}
|
|
if err := os.WriteFile(filepath.Join(tmpDir, executable), []byte{1, 2, 3}, 0777); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
cwd, err := os.Getwd()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer func() {
|
|
if err := os.Chdir(cwd); err != nil {
|
|
panic(err)
|
|
}
|
|
}()
|
|
if err = os.Chdir(tmpDir); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
t.Setenv("PWD", tmpDir)
|
|
t.Logf(". is %#q", tmpDir)
|
|
|
|
origPath := os.Getenv(pathVar)
|
|
|
|
// Add "." to PATH so that exec.LookPath looks in the current directory on all systems.
|
|
// And try to trick it with "../testdir" too.
|
|
for _, errdot := range []string{"1", "0"} {
|
|
t.Run("GODEBUG=execerrdot="+errdot, func(t *testing.T) {
|
|
t.Setenv("GODEBUG", "execerrdot="+errdot)
|
|
for _, dir := range []string{".", "../testdir"} {
|
|
t.Run(pathVar+"="+dir, func(t *testing.T) {
|
|
t.Setenv(pathVar, dir+string(filepath.ListSeparator)+origPath)
|
|
good := dir + "/execabs-test"
|
|
if found, err := LookPath(good); err != nil || !strings.HasPrefix(found, good) {
|
|
t.Fatalf(`LookPath(%#q) = %#q, %v, want "%s...", nil`, good, found, err, good)
|
|
}
|
|
if runtime.GOOS == "windows" {
|
|
good = dir + `\execabs-test`
|
|
if found, err := LookPath(good); err != nil || !strings.HasPrefix(found, good) {
|
|
t.Fatalf(`LookPath(%#q) = %#q, %v, want "%s...", nil`, good, found, err, good)
|
|
}
|
|
}
|
|
|
|
_, err := LookPath("execabs-test")
|
|
if errdot == "1" {
|
|
if err == nil {
|
|
t.Fatalf("LookPath didn't fail when finding a non-relative path")
|
|
} else if !errors.Is(err, ErrDot) {
|
|
t.Fatalf("LookPath returned unexpected error: want Is ErrDot, got %q", err)
|
|
}
|
|
} else {
|
|
if err != nil {
|
|
t.Fatalf("LookPath failed unexpectedly: %v", err)
|
|
}
|
|
}
|
|
|
|
cmd := Command("execabs-test")
|
|
if errdot == "1" {
|
|
if cmd.Err == nil {
|
|
t.Fatalf("Command didn't fail when finding a non-relative path")
|
|
} else if !errors.Is(cmd.Err, ErrDot) {
|
|
t.Fatalf("Command returned unexpected error: want Is ErrDot, got %q", cmd.Err)
|
|
}
|
|
cmd.Err = nil
|
|
} else {
|
|
if cmd.Err != nil {
|
|
t.Fatalf("Command failed unexpectedly: %v", err)
|
|
}
|
|
}
|
|
|
|
// Clearing cmd.Err should let the execution proceed,
|
|
// and it should fail because it's not a valid binary.
|
|
if err := cmd.Run(); err == nil {
|
|
t.Fatalf("Run did not fail: expected exec error")
|
|
} else if errors.Is(err, ErrDot) {
|
|
t.Fatalf("Run returned unexpected error ErrDot: want error like ENOEXEC: %q", err)
|
|
}
|
|
})
|
|
}
|
|
})
|
|
}
|
|
|
|
// Test the behavior when the first entry in PATH is an absolute name for the
|
|
// current directory.
|
|
//
|
|
// On Windows, "." may or may not be implicitly included before the explicit
|
|
// %PATH%, depending on the process environment;
|
|
// see https://go.dev/issue/4394.
|
|
//
|
|
// If the relative entry from "." resolves to the same executable as what
|
|
// would be resolved from an absolute entry in %PATH% alone, LookPath should
|
|
// return the absolute version of the path instead of ErrDot.
|
|
// (See https://go.dev/issue/53536.)
|
|
//
|
|
// If PATH does not implicitly include "." (such as on Unix platforms, or on
|
|
// Windows configured with NoDefaultCurrentDirectoryInExePath), then this
|
|
// lookup should succeed regardless of the behavior for ".", so it may be
|
|
// useful to run as a control case even on those platforms.
|
|
t.Run(pathVar+"=$PWD", func(t *testing.T) {
|
|
t.Setenv(pathVar, tmpDir+string(filepath.ListSeparator)+origPath)
|
|
good := filepath.Join(tmpDir, "execabs-test")
|
|
if found, err := LookPath(good); err != nil || !strings.HasPrefix(found, good) {
|
|
t.Fatalf(`LookPath(%#q) = %#q, %v, want \"%s...\", nil`, good, found, err, good)
|
|
}
|
|
|
|
if found, err := LookPath("execabs-test"); err != nil || !strings.HasPrefix(found, good) {
|
|
t.Fatalf(`LookPath(%#q) = %#q, %v, want \"%s...\", nil`, "execabs-test", found, err, good)
|
|
}
|
|
|
|
cmd := Command("execabs-test")
|
|
if cmd.Err != nil {
|
|
t.Fatalf("Command(%#q).Err = %v; want nil", "execabs-test", cmd.Err)
|
|
}
|
|
})
|
|
|
|
t.Run(pathVar+"=$OTHER", func(t *testing.T) {
|
|
// Control case: if the lookup returns ErrDot when PATH is empty, then we
|
|
// know that PATH implicitly includes ".". If it does not, then we don't
|
|
// expect to see ErrDot at all in this test (because the path will be
|
|
// unambiguously absolute).
|
|
wantErrDot := false
|
|
t.Setenv(pathVar, "")
|
|
if found, err := LookPath("execabs-test"); errors.Is(err, ErrDot) {
|
|
wantErrDot = true
|
|
} else if err == nil {
|
|
t.Fatalf(`with PATH='', LookPath(%#q) = %#q; want non-nil error`, "execabs-test", found)
|
|
}
|
|
|
|
// Set PATH to include an explicit directory that contains a completely
|
|
// independent executable that happens to have the same name as an
|
|
// executable in ".". If "." is included implicitly, looking up the
|
|
// (unqualified) executable name will return ErrDot; otherwise, the
|
|
// executable in "." should have no effect and the lookup should
|
|
// unambiguously resolve to the directory in PATH.
|
|
|
|
dir := t.TempDir()
|
|
executable := "execabs-test"
|
|
if runtime.GOOS == "windows" {
|
|
executable += ".exe"
|
|
}
|
|
if err := os.WriteFile(filepath.Join(dir, executable), []byte{1, 2, 3}, 0777); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
t.Setenv(pathVar, dir+string(filepath.ListSeparator)+origPath)
|
|
|
|
found, err := LookPath("execabs-test")
|
|
if wantErrDot {
|
|
wantFound := filepath.Join(".", executable)
|
|
if found != wantFound || !errors.Is(err, ErrDot) {
|
|
t.Fatalf(`LookPath(%#q) = %#q, %v, want %#q, Is ErrDot`, "execabs-test", found, err, wantFound)
|
|
}
|
|
} else {
|
|
wantFound := filepath.Join(dir, executable)
|
|
if found != wantFound || err != nil {
|
|
t.Fatalf(`LookPath(%#q) = %#q, %v, want %#q, nil`, "execabs-test", found, err, wantFound)
|
|
}
|
|
}
|
|
})
|
|
}
|