Commit 9db05830 authored by Brad Fitzpatrick's avatar Brad Fitzpatrick

os/exec: fix fd leak with Std*Pipe + LookPath

If LookPath in Command fails, sets a sticky error, and then
StdinPipe, StdoutPipe, or StderrPipe were called, those pipe
fds were never cleaned up.

Fixes #5071

R=golang-dev, rogpeppe
CC=golang-dev
https://golang.org/cl/7799046
parent 7f50c23e
...@@ -235,6 +235,8 @@ func (c *Cmd) Run() error { ...@@ -235,6 +235,8 @@ func (c *Cmd) Run() error {
// Start starts the specified command but does not wait for it to complete. // Start starts the specified command but does not wait for it to complete.
func (c *Cmd) Start() error { func (c *Cmd) Start() error {
if c.err != nil { if c.err != nil {
c.closeDescriptors(c.closeAfterStart)
c.closeDescriptors(c.closeAfterWait)
return c.err return c.err
} }
if c.Process != nil { if c.Process != nil {
......
...@@ -151,6 +151,33 @@ func TestPipes(t *testing.T) { ...@@ -151,6 +151,33 @@ func TestPipes(t *testing.T) {
check("Wait", err) check("Wait", err)
} }
// Issue 5071
func TestPipeLookPathLeak(t *testing.T) {
fd0 := numOpenFDS(t)
for i := 0; i < 4; i++ {
cmd := Command("something-that-does-not-exist-binary")
cmd.StdoutPipe()
cmd.StderrPipe()
cmd.StdinPipe()
if err := cmd.Run(); err == nil {
t.Fatal("unexpected success")
}
}
fdGrowth := numOpenFDS(t) - fd0
if fdGrowth > 2 {
t.Errorf("leaked %d fds; want ~0", fdGrowth)
}
}
func numOpenFDS(t *testing.T) int {
lsof, err := Command("lsof", "-n", "-p", strconv.Itoa(os.Getpid())).Output()
if err != nil {
t.Skip("skipping test; error finding or running lsof")
return 0
}
return bytes.Count(lsof, []byte("\n"))
}
var testedAlreadyLeaked = false var testedAlreadyLeaked = false
// basefds returns the number of expected file descriptors // basefds returns the number of expected file descriptors
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment