Merge pull request #1800 from calmh/ursvc

Break out usage reporting into a service
This commit is contained in:
Audrius Butkevicius 2015-05-13 15:41:44 +03:00
commit 2324d7420c
3 changed files with 66 additions and 31 deletions

View File

@ -485,22 +485,16 @@ func (s *apiSvc) postSystemConfig(w http.ResponseWriter, r *http.Request) {
} }
} }
// Start or stop usage reporting as appropriate // Fixup usage reporting settings
if curAcc := cfg.Options().URAccepted; newCfg.Options.URAccepted > curAcc { if curAcc := cfg.Options().URAccepted; newCfg.Options.URAccepted > curAcc {
// UR was enabled // UR was enabled
newCfg.Options.URAccepted = usageReportVersion newCfg.Options.URAccepted = usageReportVersion
newCfg.Options.URUniqueID = randomString(8) newCfg.Options.URUniqueID = randomString(8)
err := sendUsageReport(s.model)
if err != nil {
l.Infoln("Usage report:", err)
}
go usageReportingLoop(s.model)
} else if newCfg.Options.URAccepted < curAcc { } else if newCfg.Options.URAccepted < curAcc {
// UR was disabled // UR was disabled
newCfg.Options.URAccepted = -1 newCfg.Options.URAccepted = -1
newCfg.Options.URUniqueID = "" newCfg.Options.URUniqueID = ""
stopUsageReporting()
} }
// Activate and save // Activate and save

View File

@ -677,16 +677,13 @@ func syncthingMain() {
cfg.SetOptions(opts) cfg.SetOptions(opts)
cfg.Save() cfg.Save()
} }
go usageReportingLoop(m)
go func() {
time.Sleep(10 * time.Minute)
err := sendUsageReport(m)
if err != nil {
l.Infoln("Usage report:", err)
}
}()
} }
// The usageReportingManager registers itself to listen to configuration
// changes, and there's nothing more we need to tell it from the outside.
// Hence we don't keep the returned pointer.
newUsageReportingManager(m, cfg)
if opts.RestartOnWakeup { if opts.RestartOnWakeup {
go standbyMonitor() go standbyMonitor()
} }

View File

@ -16,7 +16,9 @@ import (
"runtime" "runtime"
"time" "time"
"github.com/syncthing/syncthing/internal/config"
"github.com/syncthing/syncthing/internal/model" "github.com/syncthing/syncthing/internal/model"
"github.com/thejerf/suture"
) )
// Current version number of the usage report, for acceptance purposes. If // Current version number of the usage report, for acceptance purposes. If
@ -24,8 +26,45 @@ import (
// are prompted for acceptance of the new report. // are prompted for acceptance of the new report.
const usageReportVersion = 1 const usageReportVersion = 1
var stopUsageReportingCh = make(chan struct{}) type usageReportingManager struct {
model *model.Model
sup *suture.Supervisor
}
func newUsageReportingManager(m *model.Model, cfg *config.Wrapper) *usageReportingManager {
mgr := &usageReportingManager{
model: m,
}
// Start UR if it's enabled.
mgr.Changed(cfg.Raw())
// Listen to future config changes so that we can start and stop as
// appropriate.
cfg.Subscribe(mgr)
return mgr
}
func (m *usageReportingManager) Changed(cfg config.Configuration) error {
if cfg.Options.URAccepted >= usageReportVersion && m.sup == nil {
// Usage reporting was turned on; lets start it.
svc := &usageReportingService{
model: m.model,
}
m.sup = suture.NewSimple("usageReporting")
m.sup.Add(svc)
m.sup.ServeBackground()
} else if cfg.Options.URAccepted < usageReportVersion && m.sup != nil {
// Usage reporting was turned off
m.sup.Stop()
m.sup = nil
}
return nil
}
// reportData returns the data to be sent in a usage report. It's used in
// various places, so not part of the usageReportingSvc object.
func reportData(m *model.Model) map[string]interface{} { func reportData(m *model.Model) map[string]interface{} {
res := make(map[string]interface{}) res := make(map[string]interface{})
res["uniqueID"] = cfg.Options().URUniqueID res["uniqueID"] = cfg.Options().URUniqueID
@ -75,8 +114,13 @@ func reportData(m *model.Model) map[string]interface{} {
return res return res
} }
func sendUsageReport(m *model.Model) error { type usageReportingService struct {
d := reportData(m) model *model.Model
stop chan struct{}
}
func (s *usageReportingService) sendUsageReport() error {
d := reportData(s.model)
var b bytes.Buffer var b bytes.Buffer
json.NewEncoder(&b).Encode(d) json.NewEncoder(&b).Encode(d)
@ -94,32 +138,32 @@ func sendUsageReport(m *model.Model) error {
return err return err
} }
func usageReportingLoop(m *model.Model) { func (s *usageReportingService) Serve() {
s.stop = make(chan struct{})
l.Infoln("Starting usage reporting") l.Infoln("Starting usage reporting")
t := time.NewTicker(86400 * time.Second) defer l.Infoln("Stopping usage reporting")
loop:
t := time.NewTimer(10 * time.Minute) // time to initial report at start
for { for {
select { select {
case <-stopUsageReportingCh: case <-s.stop:
break loop return
case <-t.C: case <-t.C:
err := sendUsageReport(m) err := s.sendUsageReport()
if err != nil { if err != nil {
l.Infoln("Usage report:", err) l.Infoln("Usage report:", err)
} }
t.Reset(24 * time.Hour) // next report tomorrow
} }
} }
l.Infoln("Stopping usage reporting")
}
func stopUsageReporting() {
select {
case stopUsageReportingCh <- struct{}{}:
default:
}
} }
// Returns CPU performance as a measure of single threaded SHA-256 MiB/s func (s *usageReportingService) Stop() {
close(s.stop)
}
// cpuBench returns CPU performance as a measure of single threaded SHA-256 MiB/s
func cpuBench() float64 { func cpuBench() float64 {
chunkSize := 100 * 1 << 10 chunkSize := 100 * 1 << 10
h := sha256.New() h := sha256.New()