2014-12-08 14:02:27 +01:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
|
|
|
//
|
2015-03-07 21:36:35 +01:00
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
// You can obtain one at http://mozilla.org/MPL/2.0/.
|
2014-12-08 14:02:27 +01:00
|
|
|
|
2014-11-16 23:18:59 +00:00
|
|
|
package model
|
|
|
|
|
|
|
|
import (
|
|
|
|
"path/filepath"
|
|
|
|
"reflect"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/syncthing/syncthing/internal/config"
|
|
|
|
"github.com/syncthing/syncthing/internal/events"
|
2015-04-22 23:54:31 +01:00
|
|
|
"github.com/syncthing/syncthing/internal/sync"
|
2014-11-16 23:18:59 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type ProgressEmitter struct {
|
|
|
|
registry map[string]*sharedPullerState
|
|
|
|
interval time.Duration
|
|
|
|
last map[string]map[string]*pullerProgress
|
|
|
|
mut sync.Mutex
|
|
|
|
|
|
|
|
timer *time.Timer
|
|
|
|
|
|
|
|
stop chan struct{}
|
|
|
|
}
|
|
|
|
|
2015-04-28 22:32:10 +02:00
|
|
|
// NewProgressEmitter creates a new progress emitter which emits
|
|
|
|
// DownloadProgress events every interval.
|
2014-12-08 16:39:11 +01:00
|
|
|
func NewProgressEmitter(cfg *config.Wrapper) *ProgressEmitter {
|
2014-11-16 23:18:59 +00:00
|
|
|
t := &ProgressEmitter{
|
|
|
|
stop: make(chan struct{}),
|
|
|
|
registry: make(map[string]*sharedPullerState),
|
|
|
|
last: make(map[string]map[string]*pullerProgress),
|
|
|
|
timer: time.NewTimer(time.Millisecond),
|
2015-04-22 23:54:31 +01:00
|
|
|
mut: sync.NewMutex(),
|
2014-11-16 23:18:59 +00:00
|
|
|
}
|
|
|
|
t.Changed(cfg.Raw())
|
|
|
|
cfg.Subscribe(t)
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
2015-04-28 22:32:10 +02:00
|
|
|
// Serve starts the progress emitter which starts emitting DownloadProgress
|
|
|
|
// events as the progress happens.
|
2014-11-16 23:18:59 +00:00
|
|
|
func (t *ProgressEmitter) Serve() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-t.stop:
|
|
|
|
if debug {
|
|
|
|
l.Debugln("progress emitter: stopping")
|
|
|
|
}
|
|
|
|
return
|
|
|
|
case <-t.timer.C:
|
2014-11-29 22:51:13 +01:00
|
|
|
t.mut.Lock()
|
2014-11-16 23:18:59 +00:00
|
|
|
if debug {
|
|
|
|
l.Debugln("progress emitter: timer - looking after", len(t.registry))
|
|
|
|
}
|
|
|
|
output := make(map[string]map[string]*pullerProgress)
|
|
|
|
for _, puller := range t.registry {
|
|
|
|
if output[puller.folder] == nil {
|
|
|
|
output[puller.folder] = make(map[string]*pullerProgress)
|
|
|
|
}
|
|
|
|
output[puller.folder][puller.file.Name] = puller.Progress()
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(t.last, output) {
|
|
|
|
events.Default.Log(events.DownloadProgress, output)
|
|
|
|
t.last = output
|
|
|
|
if debug {
|
|
|
|
l.Debugf("progress emitter: emitting %#v", output)
|
|
|
|
}
|
|
|
|
} else if debug {
|
|
|
|
l.Debugln("progress emitter: nothing new")
|
|
|
|
}
|
|
|
|
if len(t.registry) != 0 {
|
|
|
|
t.timer.Reset(t.interval)
|
|
|
|
}
|
|
|
|
t.mut.Unlock()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-28 22:32:10 +02:00
|
|
|
// Changed implements the config.Handler Interface to handle configuration
|
|
|
|
// changes
|
2014-11-16 23:18:59 +00:00
|
|
|
func (t *ProgressEmitter) Changed(cfg config.Configuration) error {
|
|
|
|
t.mut.Lock()
|
|
|
|
defer t.mut.Unlock()
|
|
|
|
|
|
|
|
t.interval = time.Duration(cfg.Options.ProgressUpdateIntervalS) * time.Second
|
|
|
|
if debug {
|
|
|
|
l.Debugln("progress emitter: updated interval", t.interval)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-04-28 22:32:10 +02:00
|
|
|
// Stop stops the emitter.
|
2014-11-16 23:18:59 +00:00
|
|
|
func (t *ProgressEmitter) Stop() {
|
|
|
|
t.stop <- struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register a puller with the emitter which will start broadcasting pullers
|
|
|
|
// progress.
|
|
|
|
func (t *ProgressEmitter) Register(s *sharedPullerState) {
|
|
|
|
t.mut.Lock()
|
|
|
|
defer t.mut.Unlock()
|
|
|
|
if debug {
|
|
|
|
l.Debugln("progress emitter: registering", s.folder, s.file.Name)
|
|
|
|
}
|
|
|
|
if len(t.registry) == 0 {
|
|
|
|
t.timer.Reset(t.interval)
|
|
|
|
}
|
|
|
|
t.registry[filepath.Join(s.folder, s.file.Name)] = s
|
|
|
|
}
|
|
|
|
|
2015-04-28 18:34:55 +03:00
|
|
|
// Deregister a puller which will stop broadcasting pullers state.
|
2014-11-16 23:18:59 +00:00
|
|
|
func (t *ProgressEmitter) Deregister(s *sharedPullerState) {
|
|
|
|
t.mut.Lock()
|
|
|
|
defer t.mut.Unlock()
|
|
|
|
if debug {
|
|
|
|
l.Debugln("progress emitter: deregistering", s.folder, s.file.Name)
|
|
|
|
}
|
|
|
|
delete(t.registry, filepath.Join(s.folder, s.file.Name))
|
|
|
|
}
|
|
|
|
|
2015-04-28 22:32:10 +02:00
|
|
|
// BytesCompleted returns the number of bytes completed in the given folder.
|
2014-11-16 23:18:59 +00:00
|
|
|
func (t *ProgressEmitter) BytesCompleted(folder string) (bytes int64) {
|
|
|
|
t.mut.Lock()
|
|
|
|
defer t.mut.Unlock()
|
|
|
|
|
2014-11-25 22:07:18 +00:00
|
|
|
for _, s := range t.registry {
|
|
|
|
if s.folder == folder {
|
|
|
|
bytes += s.Progress().BytesDone
|
2014-11-16 23:18:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if debug {
|
|
|
|
l.Debugf("progress emitter: bytes completed for %s: %d", folder, bytes)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|