2020-05-11 18:57:46 +00:00
|
|
|
// Copyright (C) 2019 Storj Labs, Inc.
|
|
|
|
// See LICENSE for copying information.
|
|
|
|
|
|
|
|
package eestream
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/spacemonkeygo/monkit/v3"
|
|
|
|
"github.com/vivint/infectious"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
mon = monkit.Package()
|
|
|
|
)
|
|
|
|
|
2020-05-29 13:08:11 +00:00
|
|
|
// StripeReader can read and decodes stripes from a set of readers.
|
2020-05-11 18:57:46 +00:00
|
|
|
type StripeReader struct {
|
|
|
|
scheme ErasureScheme
|
|
|
|
cond *sync.Cond
|
|
|
|
readerCount int
|
|
|
|
bufs map[int]*PieceBuffer
|
|
|
|
inbufs map[int][]byte
|
|
|
|
inmap map[int][]byte
|
|
|
|
errmap map[int]error
|
|
|
|
forceErrorDetection bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewStripeReader creates a new StripeReader from the given readers, erasure
|
|
|
|
// scheme and max buffer memory.
|
|
|
|
func NewStripeReader(log *zap.Logger, rs map[int]io.ReadCloser, es ErasureScheme, mbm int, forceErrorDetection bool) *StripeReader {
|
|
|
|
readerCount := len(rs)
|
|
|
|
|
|
|
|
r := &StripeReader{
|
|
|
|
scheme: es,
|
|
|
|
cond: sync.NewCond(&sync.Mutex{}),
|
|
|
|
readerCount: readerCount,
|
|
|
|
bufs: make(map[int]*PieceBuffer, readerCount),
|
|
|
|
inbufs: make(map[int][]byte, readerCount),
|
|
|
|
inmap: make(map[int][]byte, readerCount),
|
|
|
|
errmap: make(map[int]error, readerCount),
|
|
|
|
forceErrorDetection: forceErrorDetection,
|
|
|
|
}
|
|
|
|
|
|
|
|
bufSize := mbm / readerCount
|
|
|
|
bufSize -= bufSize % es.ErasureShareSize()
|
|
|
|
if bufSize < es.ErasureShareSize() {
|
|
|
|
bufSize = es.ErasureShareSize()
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range rs {
|
|
|
|
r.inbufs[i] = make([]byte, es.ErasureShareSize())
|
|
|
|
r.bufs[i] = NewPieceBuffer(log, make([]byte, bufSize), es.ErasureShareSize(), r.cond)
|
|
|
|
// Kick off a goroutine each reader to be copied into a PieceBuffer.
|
|
|
|
go func(r io.Reader, buf *PieceBuffer) {
|
|
|
|
_, err := io.Copy(buf, r)
|
|
|
|
if err != nil {
|
|
|
|
buf.SetError(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
buf.SetError(io.EOF)
|
|
|
|
}(rs[i], r.bufs[i])
|
|
|
|
}
|
|
|
|
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the StripeReader and all PieceBuffers.
|
|
|
|
func (r *StripeReader) Close() error {
|
|
|
|
errs := make(chan error, len(r.bufs))
|
|
|
|
for _, buf := range r.bufs {
|
|
|
|
go func(c io.Closer) {
|
|
|
|
errs <- c.Close()
|
|
|
|
}(buf)
|
|
|
|
}
|
|
|
|
var first error
|
|
|
|
for range r.bufs {
|
|
|
|
err := <-errs
|
|
|
|
if err != nil && first == nil {
|
|
|
|
first = Error.Wrap(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return first
|
|
|
|
}
|
|
|
|
|
|
|
|
var backcompatMon = monkit.ScopeNamed("storj.io/storj/uplink/eestream")
|
|
|
|
|
|
|
|
// ReadStripe reads and decodes the num-th stripe and concatenates it to p. The
|
|
|
|
// return value is the updated byte slice.
|
|
|
|
func (r *StripeReader) ReadStripe(ctx context.Context, num int64, p []byte) (_ []byte, err error) {
|
2020-05-29 13:08:11 +00:00
|
|
|
defer mon.Task()(&ctx, num)(&err)
|
|
|
|
|
2020-05-11 18:57:46 +00:00
|
|
|
for i := range r.inmap {
|
|
|
|
delete(r.inmap, i)
|
|
|
|
}
|
|
|
|
|
|
|
|
r.cond.L.Lock()
|
|
|
|
defer r.cond.L.Unlock()
|
|
|
|
|
|
|
|
for r.pendingReaders() {
|
|
|
|
for r.readAvailableShares(ctx, num) == 0 {
|
|
|
|
r.cond.Wait()
|
|
|
|
}
|
|
|
|
if r.hasEnoughShares() {
|
|
|
|
out, err := r.scheme.Decode(p, r.inmap)
|
|
|
|
if err != nil {
|
|
|
|
if r.shouldWaitForMore(err) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return out, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// could not read enough shares to attempt a decode
|
|
|
|
backcompatMon.Meter("download_stripe_failed_not_enough_pieces_uplink").Mark(1) //locked
|
|
|
|
return nil, r.combineErrs(num)
|
|
|
|
}
|
|
|
|
|
|
|
|
// readAvailableShares reads the available num-th erasure shares from the piece
|
|
|
|
// buffers without blocking. The return value n is the number of erasure shares
|
|
|
|
// read.
|
|
|
|
func (r *StripeReader) readAvailableShares(ctx context.Context, num int64) (n int) {
|
|
|
|
for i, buf := range r.bufs {
|
|
|
|
if r.inmap[i] != nil || r.errmap[i] != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if buf.HasShare(num) {
|
|
|
|
err := buf.ReadShare(num, r.inbufs[i])
|
|
|
|
if err != nil {
|
|
|
|
r.errmap[i] = err
|
|
|
|
} else {
|
|
|
|
r.inmap[i] = r.inbufs[i]
|
|
|
|
}
|
|
|
|
n++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
|
|
|
// pendingReaders checks if there are any pending readers to get a share from.
|
|
|
|
func (r *StripeReader) pendingReaders() bool {
|
|
|
|
goodReaders := r.readerCount - len(r.errmap)
|
|
|
|
return goodReaders >= r.scheme.RequiredCount() && goodReaders > len(r.inmap)
|
|
|
|
}
|
|
|
|
|
|
|
|
// hasEnoughShares check if there are enough erasure shares read to attempt
|
|
|
|
// a decode.
|
|
|
|
func (r *StripeReader) hasEnoughShares() bool {
|
|
|
|
return len(r.inmap) >= r.scheme.RequiredCount()+1 ||
|
|
|
|
(!r.forceErrorDetection && len(r.inmap) == r.scheme.RequiredCount() && !r.pendingReaders())
|
|
|
|
}
|
|
|
|
|
|
|
|
// shouldWaitForMore checks the returned decode error if it makes sense to wait
|
|
|
|
// for more erasure shares to attempt an error correction.
|
|
|
|
func (r *StripeReader) shouldWaitForMore(err error) bool {
|
|
|
|
// check if the error is due to error detection
|
|
|
|
if !infectious.NotEnoughShares.Contains(err) &&
|
|
|
|
!infectious.TooManyErrors.Contains(err) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
// check if there are more input buffers to wait for
|
|
|
|
return r.pendingReaders()
|
|
|
|
}
|
|
|
|
|
|
|
|
// combineErrs makes a useful error message from the errors in errmap.
|
|
|
|
// combineErrs always returns an error.
|
|
|
|
func (r *StripeReader) combineErrs(num int64) error {
|
|
|
|
if len(r.errmap) == 0 {
|
|
|
|
return Error.New("programmer error: no errors to combine")
|
|
|
|
}
|
|
|
|
errstrings := make([]string, 0, len(r.errmap))
|
|
|
|
for i, err := range r.errmap {
|
|
|
|
errstrings = append(errstrings, fmt.Sprintf("\nerror retrieving piece %02d: %v", i, err))
|
|
|
|
}
|
|
|
|
sort.Strings(errstrings)
|
|
|
|
return Error.New("failed to download stripe %d: %s", num, strings.Join(errstrings, ""))
|
|
|
|
}
|