Skip to content

Commit 3331040

Browse files
committed
do not spawn service goroutines on closing of STDIN
1 parent a21df2b commit 3331040

File tree

1 file changed

+20
-5
lines changed

1 file changed

+20
-5
lines changed

native.go

+20-5
Original file line numberDiff line numberDiff line change
@@ -10,13 +10,15 @@ package native
1010
import (
1111
"bytes"
1212
"encoding/binary"
13+
"fmt"
1314
"io"
1415
"log"
1516
"os"
1617
)
1718

1819
var (
1920
DefaultLogger = log.New(os.Stderr, "", 0)
21+
CloseError = fmt.Errorf("native connection closed")
2022
)
2123

2224
// A Handler responds to a native message.
@@ -78,7 +80,7 @@ func alwaysAccept(uint32) bool {
7880
return true
7981
}
8082

81-
// ListenAndServe reads from STDIO messages and dispatch them to the server's Handler
83+
// ListenAndServe reads from STDIN messages and dispatch them to the server's Handler
8284
// in a new service goroutine.
8385
func (s *Server) ListenAndServe() error {
8486
var messageAccepter = alwaysAccept
@@ -90,16 +92,29 @@ func (s *Server) ListenAndServe() error {
9092
for {
9193
// first read the message length
9294
var b = make([]byte, 4)
93-
io.ReadFull(os.Stdin, b)
94-
var n = binary.LittleEndian.Uint32(b)
95-
if !messageAccepter(n) {
95+
_ , err := io.ReadFull(os.Stdin, b)
96+
97+
if err != nil {
98+
if err == io.EOF {
99+
// standard in has been closed
100+
// there is nothing to do except
101+
// propagating error up
102+
return CloseError
103+
}
104+
return err
105+
}
106+
107+
var size = binary.LittleEndian.Uint32(b)
108+
if !messageAccepter(size) {
109+
// discard input when not accepted
110+
// copy to next EOF
96111
io.Copy(io.Discard, os.Stdin)
97112
}
98113
// NOTE(edoput) without reading the full body of the message
99114
// once we kick off the goroutine we are then free to read
100115
// some more. That would consume the message 4 bytes at a time
101116
// and spawn goroutines with meaningless messages.
102-
var body = make([]byte, n)
117+
var body = make([]byte, size)
103118
io.ReadFull(os.Stdin, body)
104119
go s.serve(&Message{bytes.NewReader(body), binary.LittleEndian.Uint32(b)})
105120
}

0 commit comments

Comments
 (0)