// Copyright 2015 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. //go:build unix package main_test import ( "bufio" "context" "internal/testenv" "io" "os" "os/exec" "slices" "strings" "syscall" "testing" ) func TestGoBuildUmask(t *testing.T) { // Do not use tg.parallel; avoid other tests seeing umask manipulation. mask := syscall.Umask(0077) // prohibit low bits defer syscall.Umask(mask) tg := testgo(t) defer tg.cleanup() tg.tempFile("x.go", `package main; func main() {}`) // We have set a umask, but if the parent directory happens to have a default // ACL, the umask may be ignored. To prevent spurious failures from an ACL, // we compare the file created by "go build" against a file written explicitly // by os.WriteFile. // // (See https://go.dev/issue/62724, https://go.dev/issue/17909.) control := tg.path("control") tg.creatingTemp(control) if err := os.WriteFile(control, []byte("#!/bin/sh\nexit 0"), 0777); err != nil { t.Fatal(err) } cfi, err := os.Stat(control) if err != nil { t.Fatal(err) } exe := tg.path("x") tg.creatingTemp(exe) tg.run("build", "-o", exe, tg.path("x.go")) fi, err := os.Stat(exe) if err != nil { t.Fatal(err) } got, want := fi.Mode(), cfi.Mode() if got == want { t.Logf("wrote x with mode %v", got) } else { t.Fatalf("wrote x with mode %v, wanted no 0077 bits (%v)", got, want) } } // TestTestInterrupt verifies the fix for issue #60203. // // If the whole process group for a 'go test' invocation receives // SIGINT (as would be sent by pressing ^C on a console), // it should return quickly, not deadlock. func TestTestInterrupt(t *testing.T) { if testing.Short() { t.Skipf("skipping in short mode: test executes many subprocesses") } // Don't run this test in parallel, for the same reason. tg := testgo(t) defer tg.cleanup() tg.setenv("GOROOT", testGOROOT) ctx, cancel := context.WithCancel(context.Background()) cmd := testenv.CommandContext(t, ctx, tg.goTool(), "test", "std", "-short", "-count=1") cmd.Dir = tg.execDir // Override $TMPDIR when running the tests: since we're terminating the tests // with a signal they might fail to clean up some temp files, and we don't // want that to cause an "unexpected files" failure at the end of the run. cmd.Env = append(slices.Clip(tg.env), tempEnvName()+"="+t.TempDir()) cmd.SysProcAttr = &syscall.SysProcAttr{ Setpgid: true, } cmd.Cancel = func() error { pgid := cmd.Process.Pid return syscall.Kill(-pgid, syscall.SIGINT) } pipe, err := cmd.StdoutPipe() if err != nil { t.Fatal(err) } t.Logf("running %v", cmd) if err := cmd.Start(); err != nil { t.Fatal(err) } stdout := new(strings.Builder) r := bufio.NewReader(pipe) line, err := r.ReadString('\n') if err != nil { t.Fatal(err) } stdout.WriteString(line) // The output line for some test was written, so we know things are in progress. // // Cancel the rest of the run by sending SIGINT to the process group: // it should finish up and exit with a nonzero status, // not have to be killed with SIGKILL. cancel() io.Copy(stdout, r) if stdout.Len() > 0 { t.Logf("stdout:\n%s", stdout) } err = cmd.Wait() ee, _ := err.(*exec.ExitError) if ee == nil { t.Fatalf("unexpectedly finished with nonzero status") } if len(ee.Stderr) > 0 { t.Logf("stderr:\n%s", ee.Stderr) } if !ee.Exited() { t.Fatalf("'go test' did not exit after interrupt: %v", err) } t.Logf("interrupted tests without deadlocking") }