mirror of
https://github.com/minio/minio.git
synced 2025-01-11 15:03:22 -05:00
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:
parent
21c868a646
commit
8309ddd486
@ -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
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user