mirror of
https://github.com/wader/fq.git
synced 2024-11-23 09:56:07 +03:00
65 lines
1.5 KiB
Go
65 lines
1.5 KiB
Go
|
package progressreadseeker
|
||
|
|
||
|
import (
|
||
|
"io"
|
||
|
)
|
||
|
|
||
|
type progressReaderSeeker struct {
|
||
|
rs io.ReadSeeker
|
||
|
pos int64
|
||
|
length int64
|
||
|
partitionSize int64
|
||
|
partitions []bool
|
||
|
partitionsReadCount int64
|
||
|
progressFn func(readBytes int64, length int64)
|
||
|
}
|
||
|
|
||
|
func New(rs io.ReadSeeker, precision int64, length int64, fn func(pos int64, length int64)) *progressReaderSeeker {
|
||
|
partitionSize := length / precision
|
||
|
if length%precision != 0 {
|
||
|
partitionSize++
|
||
|
}
|
||
|
return &progressReaderSeeker{
|
||
|
rs: rs,
|
||
|
length: length,
|
||
|
partitionSize: partitionSize,
|
||
|
partitions: make([]bool, precision),
|
||
|
progressFn: fn,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (prs *progressReaderSeeker) Read(p []byte) (n int, err error) {
|
||
|
n, err = prs.rs.Read(p)
|
||
|
newPos := prs.pos + int64(n)
|
||
|
lastPartitionsReadCount := prs.partitionsReadCount
|
||
|
|
||
|
partStart := prs.pos / prs.partitionSize
|
||
|
partEnd := newPos / prs.partitionSize
|
||
|
|
||
|
for i := partStart; i < partEnd; i++ {
|
||
|
if prs.partitions[i] {
|
||
|
continue
|
||
|
}
|
||
|
prs.partitions[i] = true
|
||
|
prs.partitionsReadCount++
|
||
|
}
|
||
|
|
||
|
if lastPartitionsReadCount != prs.partitionsReadCount {
|
||
|
readBytes := prs.partitionSize * prs.partitionsReadCount
|
||
|
if readBytes > prs.length {
|
||
|
readBytes = prs.length
|
||
|
}
|
||
|
prs.progressFn(readBytes, prs.length)
|
||
|
}
|
||
|
|
||
|
prs.pos = newPos
|
||
|
|
||
|
return n, err
|
||
|
}
|
||
|
|
||
|
func (prs *progressReaderSeeker) Seek(offset int64, whence int) (int64, error) {
|
||
|
pos, err := prs.rs.Seek(offset, whence)
|
||
|
prs.pos = pos
|
||
|
return pos, err
|
||
|
}
|