Fix panic (not fatal) on connection drops (#13811)

Fix more regressions from #13597 with double closed channels.

```
panic: "POST /minio/storage/data/distxl-plain/s1/d2/v42/createfile?disk-id=c789f7e1-2b52-442a-b518-aa2dac03f3a1&file-path=f6161668-b939-4543-9873-91b9da4cdff6%2F5eafa986-a3bf-4b1c-8bc0-03a37de390a3%2Fpart.1&length=2621760&volume=.minio.sys%2Ftmp": send on closed channel
goroutine 1977 [running]:
runtime/debug.Stack()
        c:/go/src/runtime/debug/stack.go:24 +0x65
github.com/minio/minio/cmd.setCriticalErrorHandler.func1.1()
        d:/minio/minio/cmd/generic-handlers.go:468 +0x8e
panic({0x2928860, 0x4fb17e0})
        c:/go/src/runtime/panic.go:1038 +0x215
github.com/minio/minio/cmd.keepHTTPReqResponseAlive.func2({0x4fe4ea0, 0xc02737d8a0})
        d:/minio/minio/cmd/storage-rest-server.go:818 +0x48
github.com/minio/minio/cmd.(*storageRESTServer).CreateFileHandler(0xc0015a8510, {0x50073e0, 0xc0273ec460}, 0xc029b9a400)
        d:/minio/minio/cmd/storage-rest-server.go:334 +0x1d2
net/http.HandlerFunc.ServeHTTP(...)
        c:/go/src/net/http/server.go:2046
github.com/minio/minio/cmd.httpTraceHdrs.func1({0x50073e0, 0xc0273ec460}, 0x0)
        d:/minio/minio/cmd/handler-utils.go:372 +0x53
net/http.HandlerFunc.ServeHTTP(0x5007380, {0x50073e0, 0xc0273ec460}, 0x10)
        c:/go/src/net/http/server.go:2046 +0x2f
github.com/minio/minio/cmd.addCustomHeaders.func1({0x5007380, 0xc0273dcf00}, 0xc0273f7340)
```

Reverts but adds write checks.
This commit is contained in:
Klaus Post 2021-12-02 11:22:32 -08:00 committed by GitHub
parent 21c868a646
commit 8309ddd486
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -760,51 +760,48 @@ func keepHTTPReqResponseAlive(w http.ResponseWriter, r *http.Request) (resp func
doneCh := make(chan error) doneCh := make(chan error)
ctx := r.Context() ctx := r.Context()
go func() { go func() {
defer close(doneCh) var canWrite = true
write := func(b []byte) {
if canWrite {
n, err := w.Write(b)
if err != nil || n != len(b) {
canWrite = false
}
}
}
// Wait for body to be read. // Wait for body to be read.
select { select {
case <-ctx.Done(): case <-ctx.Done():
return
case <-bodyDoneCh: case <-bodyDoneCh:
case err, ok := <-doneCh: case err := <-doneCh:
if !ok {
return
}
if err != nil { if err != nil {
_, werr := w.Write([]byte{1}) write([]byte{1})
if werr != nil { write([]byte(err.Error()))
return
}
w.Write([]byte(err.Error()))
} else { } else {
w.Write([]byte{0}) write([]byte{0})
} }
close(doneCh)
return return
} }
defer close(doneCh)
// Initiate ticker after body has been read. // Initiate ticker after body has been read.
ticker := time.NewTicker(time.Second * 10) ticker := time.NewTicker(time.Second * 10)
defer ticker.Stop()
for { for {
select { select {
case <-ticker.C: case <-ticker.C:
// Response not ready, write a filler byte. // Response not ready, write a filler byte.
if _, err := w.Write([]byte{32}); err != nil { write([]byte{32})
return if canWrite {
} w.(http.Flusher).Flush()
w.(http.Flusher).Flush()
case err, ok := <-doneCh:
if !ok {
return
} }
case err := <-doneCh:
if err != nil { if err != nil {
_, werr := w.Write([]byte{1}) write([]byte{1})
if werr != nil { write([]byte(err.Error()))
return
}
w.Write([]byte(err.Error()))
} else { } else {
w.Write([]byte{0}) write([]byte{0})
} }
ticker.Stop()
return return
} }
} }
@ -837,6 +834,15 @@ func keepHTTPReqResponseAlive(w http.ResponseWriter, r *http.Request) (resp func
func keepHTTPResponseAlive(w http.ResponseWriter) func(error) { func keepHTTPResponseAlive(w http.ResponseWriter) func(error) {
doneCh := make(chan error) doneCh := make(chan error)
go func() { go func() {
var canWrite = true
write := func(b []byte) {
if canWrite {
n, err := w.Write(b)
if err != nil || n != len(b) {
canWrite = false
}
}
}
defer close(doneCh) defer close(doneCh)
ticker := time.NewTicker(time.Second * 10) ticker := time.NewTicker(time.Second * 10)
defer ticker.Stop() defer ticker.Stop()
@ -844,19 +850,16 @@ func keepHTTPResponseAlive(w http.ResponseWriter) func(error) {
select { select {
case <-ticker.C: case <-ticker.C:
// Response not ready, write a filler byte. // Response not ready, write a filler byte.
if _, err := w.Write([]byte{32}); err != nil { write([]byte{32})
return if canWrite {
w.(http.Flusher).Flush()
} }
w.(http.Flusher).Flush()
case err := <-doneCh: case err := <-doneCh:
if err != nil { if err != nil {
_, werr := w.Write([]byte{1}) write([]byte{1})
if werr != nil { write([]byte(err.Error()))
return
}
w.Write([]byte(err.Error()))
} else { } else {
w.Write([]byte{0}) write([]byte{0})
} }
return return
} }