From 2eb4455df706a8ec2a4ccbf6c3739b2abc92a9fb Mon Sep 17 00:00:00 2001 From: Ramkumar Chinchani Date: Sat, 4 Jan 2020 18:20:06 -0800 Subject: [PATCH] routes: improve error handling when returning blob data We have noticed that very intermittently zot returns a 500 when copying blob data. This is likely happening due to a io.ErrShortWrite --- pkg/api/routes.go | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/pkg/api/routes.go b/pkg/api/routes.go index 0f7e6433..db71db65 100644 --- a/pkg/api/routes.go +++ b/pkg/api/routes.go @@ -22,6 +22,7 @@ import ( _ "github.com/anuvu/zot/docs" // nolint (golint) - as required by swaggo "github.com/anuvu/zot/errors" + "github.com/anuvu/zot/pkg/log" "github.com/gorilla/mux" jsoniter "github.com/json-iterator/go" ispec "github.com/opencontainers/image-spec/specs-go/v1" @@ -437,7 +438,7 @@ func (rh *RouteHandler) GetBlob(w http.ResponseWriter, r *http.Request) { w.Header().Set("Content-Length", fmt.Sprintf("%d", blen)) w.Header().Set(DistContentDigestKey, digest) // return the blob data - WriteDataFromReader(w, http.StatusOK, blen, mediaType, br) + WriteDataFromReader(w, http.StatusOK, blen, mediaType, br, rh.c.Log) } // DeleteBlob godoc @@ -905,22 +906,21 @@ func WriteData(w http.ResponseWriter, status int, mediaType string, data []byte) _, _ = w.Write(data) } -func WriteDataFromReader(w http.ResponseWriter, status int, length int64, mediaType string, reader io.Reader) { +func WriteDataFromReader(w http.ResponseWriter, status int, length int64, mediaType string, + reader io.Reader, logger log.Logger) { w.Header().Set("Content-Type", mediaType) w.Header().Set("Content-Length", strconv.FormatInt(length, 10)) + w.WriteHeader(status) const maxSize = 10 * 1024 * 1024 for { - size, err := io.CopyN(w, reader, maxSize) - if size == 0 { - if err != io.EOF { - w.WriteHeader(http.StatusInternalServerError) - return - } - + _, err := io.CopyN(w, reader, maxSize) + if err == io.EOF { break + } else if err != nil { + // other kinds of intermittent errors can occur, e.g, io.ErrShortWrite + logger.Error().Err(err).Msg("copying data into http response") } } - w.WriteHeader(status) }