mirror of
https://github.com/zrepl/zrepl.git
synced 2025-01-26 16:19:24 +01:00
149 lines
2.8 KiB
Go
149 lines
2.8 KiB
Go
package util
|
|
|
|
import (
|
|
"io"
|
|
"os"
|
|
"time"
|
|
)
|
|
|
|
type ReadWriteCloserLogger struct {
|
|
RWC io.ReadWriteCloser
|
|
ReadFile *os.File
|
|
WriteFile *os.File
|
|
}
|
|
|
|
func NewReadWriteCloserLogger(rwc io.ReadWriteCloser, readlog, writelog string) (l *ReadWriteCloserLogger, err error) {
|
|
l = &ReadWriteCloserLogger{
|
|
RWC: rwc,
|
|
}
|
|
flags := os.O_CREATE | os.O_WRONLY
|
|
if readlog != "" {
|
|
if l.ReadFile, err = os.OpenFile(readlog, flags, 0600); err != nil {
|
|
return
|
|
}
|
|
}
|
|
if writelog != "" {
|
|
if l.WriteFile, err = os.OpenFile(writelog, flags, 0600); err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (c *ReadWriteCloserLogger) Read(buf []byte) (n int, err error) {
|
|
n, err = c.RWC.Read(buf)
|
|
if c.WriteFile != nil {
|
|
if _, writeErr := c.ReadFile.Write(buf[0:n]); writeErr != nil {
|
|
panic(writeErr)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (c *ReadWriteCloserLogger) Write(buf []byte) (n int, err error) {
|
|
n, err = c.RWC.Write(buf)
|
|
if c.ReadFile != nil {
|
|
if _, writeErr := c.WriteFile.Write(buf[0:n]); writeErr != nil {
|
|
panic(writeErr)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
func (c *ReadWriteCloserLogger) Close() (err error) {
|
|
err = c.RWC.Close()
|
|
if err != nil {
|
|
return
|
|
}
|
|
if c.ReadFile != nil {
|
|
if err := c.ReadFile.Close(); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
if c.WriteFile != nil {
|
|
if err := c.WriteFile.Close(); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
type ChainedReader struct {
|
|
Readers []io.Reader
|
|
curReader int
|
|
}
|
|
|
|
func NewChainedReader(reader ...io.Reader) *ChainedReader {
|
|
return &ChainedReader{
|
|
Readers: reader,
|
|
curReader: 0,
|
|
}
|
|
}
|
|
|
|
func (c *ChainedReader) Read(buf []byte) (n int, err error) {
|
|
|
|
n = 0
|
|
|
|
for c.curReader < len(c.Readers) {
|
|
n, err = c.Readers[c.curReader].Read(buf)
|
|
if err == io.EOF {
|
|
c.curReader++
|
|
continue
|
|
}
|
|
break
|
|
}
|
|
if c.curReader == len(c.Readers) {
|
|
err = io.EOF // actually, there was no gap
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
type IOProgress struct {
|
|
TotalRX uint64
|
|
}
|
|
|
|
type IOProgressCallback func(progress IOProgress)
|
|
|
|
type IOProgressWatcher struct {
|
|
Reader io.Reader
|
|
callback IOProgressCallback
|
|
callbackTicker *time.Ticker
|
|
progress IOProgress
|
|
updateChannel chan int
|
|
}
|
|
|
|
func (w *IOProgressWatcher) KickOff(callbackInterval time.Duration, callback IOProgressCallback) {
|
|
w.callback = callback
|
|
w.callbackTicker = time.NewTicker(callbackInterval)
|
|
w.updateChannel = make(chan int)
|
|
go func() {
|
|
outer:
|
|
for {
|
|
select {
|
|
case newBytes, more := <-w.updateChannel:
|
|
w.progress.TotalRX += uint64(newBytes)
|
|
if !more {
|
|
w.callbackTicker.Stop()
|
|
break outer
|
|
}
|
|
case <-w.callbackTicker.C:
|
|
w.callback(w.progress)
|
|
}
|
|
}
|
|
w.callback(w.progress)
|
|
}()
|
|
}
|
|
|
|
func (w *IOProgressWatcher) Progress() IOProgress {
|
|
return w.progress
|
|
}
|
|
|
|
func (w *IOProgressWatcher) Read(p []byte) (n int, err error) {
|
|
n, err = w.Reader.Read(p)
|
|
w.updateChannel <- n
|
|
if err != nil {
|
|
close(w.updateChannel)
|
|
}
|
|
return
|
|
}
|