lib/readers: add mutex to RepeatableReader - fixes #2572
This commit is contained in:
parent
9b3c951ab7
commit
ee700ec01a
1 changed files with 11 additions and 3 deletions
|
@ -2,6 +2,7 @@ package readers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
|
"sync"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
)
|
)
|
||||||
|
@ -10,6 +11,7 @@ import (
|
||||||
// back and forth within the reader but will only read data from the internal Reader as necessary
|
// back and forth within the reader but will only read data from the internal Reader as necessary
|
||||||
// and will play nicely with the Account and io.LimitedReader to reflect current speed
|
// and will play nicely with the Account and io.LimitedReader to reflect current speed
|
||||||
type RepeatableReader struct {
|
type RepeatableReader struct {
|
||||||
|
mu sync.Mutex // protect against concurrent use
|
||||||
in io.Reader // Input reader
|
in io.Reader // Input reader
|
||||||
i int64 // current reading index
|
i int64 // current reading index
|
||||||
b []byte // internal cache buffer
|
b []byte // internal cache buffer
|
||||||
|
@ -21,6 +23,9 @@ var _ io.ReadSeeker = (*RepeatableReader)(nil)
|
||||||
// If seek position is passed the cache buffer length the function will return
|
// If seek position is passed the cache buffer length the function will return
|
||||||
// the maximum offset that can be used and "fs.RepeatableReader.Seek: offset is unavailable" Error
|
// the maximum offset that can be used and "fs.RepeatableReader.Seek: offset is unavailable" Error
|
||||||
func (r *RepeatableReader) Seek(offset int64, whence int) (int64, error) {
|
func (r *RepeatableReader) Seek(offset int64, whence int) (int64, error) {
|
||||||
|
r.mu.Lock()
|
||||||
|
defer r.mu.Unlock()
|
||||||
|
|
||||||
var abs int64
|
var abs int64
|
||||||
cacheLen := int64(len(r.b))
|
cacheLen := int64(len(r.b))
|
||||||
switch whence {
|
switch whence {
|
||||||
|
@ -46,6 +51,9 @@ func (r *RepeatableReader) Seek(offset int64, whence int) (int64, error) {
|
||||||
// Read data from original Reader into bytes
|
// Read data from original Reader into bytes
|
||||||
// Data is either served from the underlying Reader or from cache if was already read
|
// Data is either served from the underlying Reader or from cache if was already read
|
||||||
func (r *RepeatableReader) Read(b []byte) (n int, err error) {
|
func (r *RepeatableReader) Read(b []byte) (n int, err error) {
|
||||||
|
r.mu.Lock()
|
||||||
|
defer r.mu.Unlock()
|
||||||
|
|
||||||
cacheLen := int64(len(r.b))
|
cacheLen := int64(len(r.b))
|
||||||
if r.i == cacheLen {
|
if r.i == cacheLen {
|
||||||
n, err = r.in.Read(b)
|
n, err = r.in.Read(b)
|
||||||
|
|
Loading…
Reference in a new issue