cosmos-sdk/snapshots/chunk.go

164 lines
3.3 KiB
Go

package snapshots
import (
"io"
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
)
// ChunkWriter reads an input stream, splits it into fixed-size chunks, and writes them to a
// sequence of io.ReadClosers via a channel.
type ChunkWriter struct {
ch chan<- io.ReadCloser
pipe *io.PipeWriter
chunkSize uint64
written uint64
closed bool
}
// NewChunkWriter creates a new ChunkWriter. If chunkSize is 0, no chunking will be done.
func NewChunkWriter(ch chan<- io.ReadCloser, chunkSize uint64) *ChunkWriter {
return &ChunkWriter{
ch: ch,
chunkSize: chunkSize,
}
}
// chunk creates a new chunk.
func (w *ChunkWriter) chunk() error {
if w.pipe != nil {
err := w.pipe.Close()
if err != nil {
return err
}
}
pr, pw := io.Pipe()
w.ch <- pr
w.pipe = pw
w.written = 0
return nil
}
// Close implements io.Closer.
func (w *ChunkWriter) Close() error {
if !w.closed {
w.closed = true
close(w.ch)
var err error
if w.pipe != nil {
err = w.pipe.Close()
}
return err
}
return nil
}
// CloseWithError closes the writer and sends an error to the reader.
func (w *ChunkWriter) CloseWithError(err error) {
if !w.closed {
w.closed = true
close(w.ch)
if w.pipe != nil {
w.pipe.CloseWithError(err)
}
}
}
// Write implements io.Writer.
func (w *ChunkWriter) Write(data []byte) (int, error) {
if w.closed {
return 0, sdkerrors.Wrap(sdkerrors.ErrLogic, "cannot write to closed ChunkWriter")
}
nTotal := 0
for len(data) > 0 {
if w.pipe == nil || (w.written >= w.chunkSize && w.chunkSize > 0) {
err := w.chunk()
if err != nil {
return nTotal, err
}
}
var writeSize uint64
if w.chunkSize == 0 {
writeSize = uint64(len(data))
} else {
writeSize = w.chunkSize - w.written
}
if writeSize > uint64(len(data)) {
writeSize = uint64(len(data))
}
n, err := w.pipe.Write(data[:writeSize])
w.written += uint64(n)
nTotal += n
if err != nil {
return nTotal, err
}
data = data[writeSize:]
}
return nTotal, nil
}
// ChunkReader reads chunks from a channel of io.ReadClosers and outputs them as an io.Reader
type ChunkReader struct {
ch <-chan io.ReadCloser
reader io.ReadCloser
}
// NewChunkReader creates a new ChunkReader.
func NewChunkReader(ch <-chan io.ReadCloser) *ChunkReader {
return &ChunkReader{ch: ch}
}
// next fetches the next chunk from the channel, or returns io.EOF if there are no more chunks.
func (r *ChunkReader) next() error {
reader, ok := <-r.ch
if !ok {
return io.EOF
}
r.reader = reader
return nil
}
// Close implements io.ReadCloser.
func (r *ChunkReader) Close() error {
var err error
if r.reader != nil {
err = r.reader.Close()
r.reader = nil
}
for reader := range r.ch {
if e := reader.Close(); e != nil && err == nil {
err = e
}
}
return err
}
// Read implements io.Reader.
func (r *ChunkReader) Read(p []byte) (int, error) {
if r.reader == nil {
err := r.next()
if err != nil {
return 0, err
}
}
n, err := r.reader.Read(p)
if err == io.EOF {
err = r.reader.Close()
r.reader = nil
if err != nil {
return 0, err
}
return r.Read(p)
}
return n, err
}
// DrainChunks drains and closes all remaining chunks from a chunk channel.
func DrainChunks(chunks <-chan io.ReadCloser) {
for chunk := range chunks {
_ = chunk.Close()
}
}