diff --git a/pkg/storage/donut/erasure.go b/pkg/storage/donut/erasure.go index 69d46c5cf..341082023 100644 --- a/pkg/storage/donut/erasure.go +++ b/pkg/storage/donut/erasure.go @@ -11,8 +11,10 @@ import ( "crypto/md5" "encoding/hex" + "github.com/minio-io/iodine" "github.com/minio-io/minio/pkg/encoding/erasure" "github.com/minio-io/minio/pkg/utils/split" + "hash" ) // getErasureTechnique - convert technique string into Technique type @@ -23,7 +25,7 @@ func getErasureTechnique(technique string) (erasure.Technique, error) { case technique == "Vandermonde": return erasure.Cauchy, nil default: - return erasure.None, errors.New("Invalid erasure technique") + return erasure.None, iodine.Error(errors.New("Invalid erasure technique: "+technique), nil) } } @@ -31,82 +33,113 @@ func getErasureTechnique(technique string) (erasure.Technique, error) { func erasureReader(readers []io.ReadCloser, donutMetadata map[string]string, writer *io.PipeWriter) { totalChunks, err := strconv.Atoi(donutMetadata["chunkCount"]) if err != nil { - writer.CloseWithError(err) + writer.CloseWithError(iodine.Error(err, donutMetadata)) return } - totalLeft, err := strconv.Atoi(donutMetadata["size"]) + totalLeft, err := strconv.ParseInt(donutMetadata["size"], 10, 64) if err != nil { - writer.CloseWithError(err) + writer.CloseWithError(iodine.Error(err, donutMetadata)) return } blockSize, err := strconv.Atoi(donutMetadata["blockSize"]) if err != nil { - writer.CloseWithError(err) + writer.CloseWithError(iodine.Error(err, donutMetadata)) return } - k, err := strconv.Atoi(donutMetadata["erasureK"]) + parsedk, err := strconv.ParseUint(donutMetadata["erasureK"], 10, 8) if err != nil { - writer.CloseWithError(err) + writer.CloseWithError(iodine.Error(err, donutMetadata)) return } - m, err := strconv.Atoi(donutMetadata["erasureM"]) + k := uint8(parsedk) + parsedm, err := strconv.ParseUint(donutMetadata["erasureM"], 10, 8) if err != nil { - writer.CloseWithError(err) + writer.CloseWithError(iodine.Error(err, donutMetadata)) return } + m := uint8(parsedm) expectedMd5sum, err := hex.DecodeString(donutMetadata["md5"]) if err != nil { - writer.CloseWithError(err) + writer.CloseWithError(iodine.Error(err, donutMetadata)) return } technique, err := getErasureTechnique(donutMetadata["erasureTechnique"]) if err != nil { - writer.CloseWithError(err) + writer.CloseWithError(iodine.Error(err, donutMetadata)) return } - summer := md5.New() - params, _ := erasure.ParseEncoderParams(uint8(k), uint8(m), technique) + hasher := md5.New() + params, err := erasure.ParseEncoderParams(k, m, technique) + if err != nil { + writer.CloseWithError(iodine.Error(err, donutMetadata)) + } encoder := erasure.NewEncoder(params) for i := 0; i < totalChunks; i++ { - curBlockSize := totalLeft - if blockSize < totalLeft { - curBlockSize = blockSize - } - - curChunkSize := erasure.GetEncodedBlockLen(curBlockSize, uint8(k)) - encodedBytes := make([][]byte, 16) - for i, reader := range readers { - defer reader.Close() - var bytesBuffer bytes.Buffer - _, err := io.CopyN(&bytesBuffer, reader, int64(curChunkSize)) - if err != nil { - writer.CloseWithError(err) - return + totalLeft, err = decodeChunk(writer, readers, encoder, hasher, k, m, totalLeft, blockSize) + if err != nil { + errParams := map[string]string{ + "totalLeft": strconv.FormatInt(totalLeft, 10), } - encodedBytes[i] = bytesBuffer.Bytes() + for k, v := range donutMetadata { + errParams[k] = v + } + writer.CloseWithError(iodine.Error(err, errParams)) } - decodedData, err := encoder.Decode(encodedBytes, curBlockSize) - if err != nil { - writer.CloseWithError(err) - return - } - summer.Write(decodedData) - _, err = io.Copy(writer, bytes.NewBuffer(decodedData)) - if err != nil { - writer.CloseWithError(err) - return - } - totalLeft = totalLeft - blockSize } - actualMd5sum := summer.Sum(nil) + actualMd5sum := hasher.Sum(nil) if bytes.Compare(expectedMd5sum, actualMd5sum) != 0 { - writer.CloseWithError(errors.New("decoded md5sum did not match")) + writer.CloseWithError(iodine.Error(errors.New("decoded md5sum did not match. expected: "+string(expectedMd5sum)+" actual: "+string(actualMd5sum)), donutMetadata)) return } writer.Close() return } +func decodeChunk(writer *io.PipeWriter, readers []io.ReadCloser, encoder *erasure.Encoder, hasher hash.Hash, k, m uint8, totalLeft int64, blockSize int) (int64, error) { + curBlockSize := 0 + if int64(blockSize) < totalLeft { + curBlockSize = blockSize + } else { + curBlockSize = int(totalLeft) // cast is safe, blockSize in if protects + } + + curChunkSize := erasure.GetEncodedBlockLen(curBlockSize, uint8(k)) + encodedBytes := make([][]byte, 16) + for i, reader := range readers { + defer reader.Close() + var bytesBuffer bytes.Buffer + written, err := io.CopyN(&bytesBuffer, reader, int64(curChunkSize)) + if err != nil { + errParams := map[string]string{} + errParams["part"] = strconv.FormatInt(written, 10) + errParams["block.written"] = strconv.FormatInt(written, 10) + errParams["block.length"] = strconv.Itoa(curChunkSize) + return totalLeft, iodine.Error(err, errParams) + } + encodedBytes[i] = bytesBuffer.Bytes() + } + decodedData, err := encoder.Decode(encodedBytes, curBlockSize) + if err != nil { + errParams := map[string]string{} + errParams["block.length"] = strconv.Itoa(curChunkSize) + return totalLeft, iodine.Error(err, errParams) + } + _, err = hasher.Write(decodedData) // not expecting errors from hash, will also catch further down on .Sum mismatch in parent + if err != nil { + errParams := map[string]string{} + errParams["block.length"] = strconv.Itoa(curChunkSize) + return totalLeft, iodine.Error(err, errParams) + } + _, err = io.Copy(writer, bytes.NewBuffer(decodedData)) + if err != nil { + errParams := map[string]string{} + errParams["block.length"] = strconv.Itoa(curChunkSize) + return totalLeft, iodine.Error(err, errParams) + } + totalLeft = totalLeft - int64(blockSize) + return totalLeft, nil +} + // erasure writer type erasureWriter struct {