@@ -10,13 +10,15 @@ package native
10
10
import (
11
11
"bytes"
12
12
"encoding/binary"
13
+ "fmt"
13
14
"io"
14
15
"log"
15
16
"os"
16
17
)
17
18
18
19
var (
19
20
DefaultLogger = log .New (os .Stderr , "" , 0 )
21
+ CloseError = fmt .Errorf ("native connection closed" )
20
22
)
21
23
22
24
// A Handler responds to a native message.
@@ -78,7 +80,7 @@ func alwaysAccept(uint32) bool {
78
80
return true
79
81
}
80
82
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
82
84
// in a new service goroutine.
83
85
func (s * Server ) ListenAndServe () error {
84
86
var messageAccepter = alwaysAccept
@@ -90,16 +92,29 @@ func (s *Server) ListenAndServe() error {
90
92
for {
91
93
// first read the message length
92
94
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
96
111
io .Copy (io .Discard , os .Stdin )
97
112
}
98
113
// NOTE(edoput) without reading the full body of the message
99
114
// once we kick off the goroutine we are then free to read
100
115
// some more. That would consume the message 4 bytes at a time
101
116
// and spawn goroutines with meaningless messages.
102
- var body = make ([]byte , n )
117
+ var body = make ([]byte , size )
103
118
io .ReadFull (os .Stdin , body )
104
119
go s .serve (& Message {bytes .NewReader (body ), binary .LittleEndian .Uint32 (b )})
105
120
}
0 commit comments