mirror of
https://github.com/octoleo/syncthing.git
synced 2024-11-19 03:25:16 +00:00
lib/api: Improve folder summary event, verbose service (#9370)
This makes a couple of small improvements to the folder summary mechanism: - The folder summary includes the local and remote sequence numbers in clear text, rather than some odd sum that I'm not sure what it was intended to represent. - The folder summary event is generated when appropriate, regardless of whether there is an event listener. We did this before because generating it was expensive, and we wanted to avoid doing it unnecessarily. Nowadays, however, it's mostly just reading out pre-calculated metadata, and anyway, it's nice if it shows up reliably when running with -verbose. The point of all this is to make it easier to use these events to judge when devices are, in fact, in sync. As-is, if I'm looking at two devices, it's very difficult to reliably determine if they are in sync or not. The reason is that while we can ask device A if it thinks it's in sync, we can't see if the answer is "yes" because it has processed all changes from B, or if it just doesn't know about the changes from B yet. With proper sequence numbers in the event we can compare the two and determine the truth. This makes testing a lot easier.
This commit is contained in:
parent
bda4016109
commit
f16817632f
@ -355,12 +355,7 @@ func (s *service) Serve(ctx context.Context) error {
|
|||||||
|
|
||||||
// Handle Prometheus metrics
|
// Handle Prometheus metrics
|
||||||
promHttpHandler := promhttp.Handler()
|
promHttpHandler := promhttp.Handler()
|
||||||
mux.HandleFunc("/metrics", func(w http.ResponseWriter, req *http.Request) {
|
mux.Handle("/metrics", promHttpHandler)
|
||||||
// fetching metrics counts as an event, for the purpose of whether
|
|
||||||
// we should prepare folder summaries etc.
|
|
||||||
s.fss.OnEventRequest()
|
|
||||||
promHttpHandler.ServeHTTP(w, req)
|
|
||||||
})
|
|
||||||
|
|
||||||
guiCfg := s.cfg.GUI()
|
guiCfg := s.cfg.GUI()
|
||||||
|
|
||||||
@ -1395,11 +1390,7 @@ func (s *service) getDiskEvents(w http.ResponseWriter, r *http.Request) {
|
|||||||
s.getEvents(w, r, sub)
|
s.getEvents(w, r, sub)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *service) getEvents(w http.ResponseWriter, r *http.Request, eventSub events.BufferedSubscription) {
|
func (*service) getEvents(w http.ResponseWriter, r *http.Request, eventSub events.BufferedSubscription) {
|
||||||
if eventSub.Mask()&(events.FolderSummary|events.FolderCompletion) != 0 {
|
|
||||||
s.fss.OnEventRequest()
|
|
||||||
}
|
|
||||||
|
|
||||||
qs := r.URL.Query()
|
qs := r.URL.Query()
|
||||||
sinceStr := qs.Get("since")
|
sinceStr := qs.Get("since")
|
||||||
limitStr := qs.Get("limit")
|
limitStr := qs.Get("limit")
|
||||||
|
@ -338,17 +338,22 @@ func (s *Snapshot) Sequence(device protocol.DeviceID) int64 {
|
|||||||
return s.meta.Counts(device, 0).Sequence
|
return s.meta.Counts(device, 0).Sequence
|
||||||
}
|
}
|
||||||
|
|
||||||
// RemoteSequence returns the change version for the given folder, as
|
// RemoteSequences returns a map of the sequence numbers seen for each
|
||||||
// sent by remote peers. This is guaranteed to increment if the contents of
|
// remote device sharing this folder.
|
||||||
// the remote or global folder has changed.
|
func (s *Snapshot) RemoteSequences() map[protocol.DeviceID]int64 {
|
||||||
func (s *Snapshot) RemoteSequence() int64 {
|
res := make(map[protocol.DeviceID]int64)
|
||||||
var ver int64
|
|
||||||
|
|
||||||
for _, device := range s.meta.devices() {
|
for _, device := range s.meta.devices() {
|
||||||
ver += s.Sequence(device)
|
switch device {
|
||||||
|
case protocol.EmptyDeviceID, protocol.LocalDeviceID, protocol.GlobalDeviceID:
|
||||||
|
continue
|
||||||
|
default:
|
||||||
|
if seq := s.Sequence(device); seq > 0 {
|
||||||
|
res[device] = seq
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ver
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Snapshot) LocalSize() Counts {
|
func (s *Snapshot) LocalSize() Counts {
|
||||||
|
@ -25,12 +25,9 @@ import (
|
|||||||
"github.com/syncthing/syncthing/lib/sync"
|
"github.com/syncthing/syncthing/lib/sync"
|
||||||
)
|
)
|
||||||
|
|
||||||
const maxDurationSinceLastEventReq = time.Minute
|
|
||||||
|
|
||||||
type FolderSummaryService interface {
|
type FolderSummaryService interface {
|
||||||
suture.Service
|
suture.Service
|
||||||
Summary(folder string) (*FolderSummary, error)
|
Summary(folder string) (*FolderSummary, error)
|
||||||
OnEventRequest()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// The folderSummaryService adds summary information events (FolderSummary and
|
// The folderSummaryService adds summary information events (FolderSummary and
|
||||||
@ -47,23 +44,18 @@ type folderSummaryService struct {
|
|||||||
// For keeping track of folders to recalculate for
|
// For keeping track of folders to recalculate for
|
||||||
foldersMut sync.Mutex
|
foldersMut sync.Mutex
|
||||||
folders map[string]struct{}
|
folders map[string]struct{}
|
||||||
|
|
||||||
// For keeping track of when the last event request on the API was
|
|
||||||
lastEventReq time.Time
|
|
||||||
lastEventReqMut sync.Mutex
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewFolderSummaryService(cfg config.Wrapper, m Model, id protocol.DeviceID, evLogger events.Logger) FolderSummaryService {
|
func NewFolderSummaryService(cfg config.Wrapper, m Model, id protocol.DeviceID, evLogger events.Logger) FolderSummaryService {
|
||||||
service := &folderSummaryService{
|
service := &folderSummaryService{
|
||||||
Supervisor: suture.New("folderSummaryService", svcutil.SpecWithDebugLogger(l)),
|
Supervisor: suture.New("folderSummaryService", svcutil.SpecWithDebugLogger(l)),
|
||||||
cfg: cfg,
|
cfg: cfg,
|
||||||
model: m,
|
model: m,
|
||||||
id: id,
|
id: id,
|
||||||
evLogger: evLogger,
|
evLogger: evLogger,
|
||||||
immediate: make(chan string),
|
immediate: make(chan string),
|
||||||
folders: make(map[string]struct{}),
|
folders: make(map[string]struct{}),
|
||||||
foldersMut: sync.NewMutex(),
|
foldersMut: sync.NewMutex(),
|
||||||
lastEventReqMut: sync.NewMutex(),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
service.Add(svcutil.AsService(service.listenForUpdates, fmt.Sprintf("%s/listenForUpdates", service)))
|
service.Add(svcutil.AsService(service.listenForUpdates, fmt.Sprintf("%s/listenForUpdates", service)))
|
||||||
@ -119,8 +111,9 @@ type FolderSummary struct {
|
|||||||
StateChanged time.Time `json:"stateChanged"`
|
StateChanged time.Time `json:"stateChanged"`
|
||||||
Error string `json:"error"`
|
Error string `json:"error"`
|
||||||
|
|
||||||
Version int64 `json:"version"` // deprecated
|
Version int64 `json:"version"` // deprecated
|
||||||
Sequence int64 `json:"sequence"`
|
Sequence int64 `json:"sequence"`
|
||||||
|
RemoteSequence map[protocol.DeviceID]int64 `json:"remoteSequence"`
|
||||||
|
|
||||||
IgnorePatterns bool `json:"ignorePatterns"`
|
IgnorePatterns bool `json:"ignorePatterns"`
|
||||||
WatchError string `json:"watchError"`
|
WatchError string `json:"watchError"`
|
||||||
@ -130,7 +123,8 @@ func (c *folderSummaryService) Summary(folder string) (*FolderSummary, error) {
|
|||||||
res := new(FolderSummary)
|
res := new(FolderSummary)
|
||||||
|
|
||||||
var local, global, need, ro db.Counts
|
var local, global, need, ro db.Counts
|
||||||
var ourSeq, remoteSeq int64
|
var ourSeq int64
|
||||||
|
var remoteSeq map[protocol.DeviceID]int64
|
||||||
errors, err := c.model.FolderErrors(folder)
|
errors, err := c.model.FolderErrors(folder)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
var snap *db.Snapshot
|
var snap *db.Snapshot
|
||||||
@ -140,7 +134,7 @@ func (c *folderSummaryService) Summary(folder string) (*FolderSummary, error) {
|
|||||||
need = snap.NeedSize(protocol.LocalDeviceID)
|
need = snap.NeedSize(protocol.LocalDeviceID)
|
||||||
ro = snap.ReceiveOnlyChangedSize()
|
ro = snap.ReceiveOnlyChangedSize()
|
||||||
ourSeq = snap.Sequence(protocol.LocalDeviceID)
|
ourSeq = snap.Sequence(protocol.LocalDeviceID)
|
||||||
remoteSeq = snap.Sequence(protocol.GlobalDeviceID)
|
remoteSeq = snap.RemoteSequences()
|
||||||
snap.Release()
|
snap.Release()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -192,8 +186,9 @@ func (c *folderSummaryService) Summary(folder string) (*FolderSummary, error) {
|
|||||||
res.Error = err.Error()
|
res.Error = err.Error()
|
||||||
}
|
}
|
||||||
|
|
||||||
res.Version = ourSeq + remoteSeq // legacy
|
res.Version = ourSeq // legacy
|
||||||
res.Sequence = ourSeq + remoteSeq // new name
|
res.Sequence = ourSeq
|
||||||
|
res.RemoteSequence = remoteSeq
|
||||||
|
|
||||||
ignorePatterns, _, _ := c.model.CurrentIgnores(folder)
|
ignorePatterns, _, _ := c.model.CurrentIgnores(folder)
|
||||||
res.IgnorePatterns = false
|
res.IgnorePatterns = false
|
||||||
@ -212,12 +207,6 @@ func (c *folderSummaryService) Summary(folder string) (*FolderSummary, error) {
|
|||||||
return res, nil
|
return res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *folderSummaryService) OnEventRequest() {
|
|
||||||
c.lastEventReqMut.Lock()
|
|
||||||
c.lastEventReq = time.Now()
|
|
||||||
c.lastEventReqMut.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
// listenForUpdates subscribes to the event bus and makes note of folders that
|
// listenForUpdates subscribes to the event bus and makes note of folders that
|
||||||
// need their data recalculated.
|
// need their data recalculated.
|
||||||
func (c *folderSummaryService) listenForUpdates(ctx context.Context) error {
|
func (c *folderSummaryService) listenForUpdates(ctx context.Context) error {
|
||||||
@ -357,17 +346,6 @@ func (c *folderSummaryService) calculateSummaries(ctx context.Context) error {
|
|||||||
// foldersToHandle returns the list of folders needing a summary update, and
|
// foldersToHandle returns the list of folders needing a summary update, and
|
||||||
// clears the list.
|
// clears the list.
|
||||||
func (c *folderSummaryService) foldersToHandle() []string {
|
func (c *folderSummaryService) foldersToHandle() []string {
|
||||||
// We only recalculate summaries if someone is listening to events
|
|
||||||
// (a request to /rest/events has been made within the last
|
|
||||||
// pingEventInterval).
|
|
||||||
|
|
||||||
c.lastEventReqMut.Lock()
|
|
||||||
last := c.lastEventReq
|
|
||||||
c.lastEventReqMut.Unlock()
|
|
||||||
if time.Since(last) > maxDurationSinceLastEventReq {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
c.foldersMut.Lock()
|
c.foldersMut.Lock()
|
||||||
res := make([]string, 0, len(c.folders))
|
res := make([]string, 0, len(c.folders))
|
||||||
for folder := range c.folders {
|
for folder := range c.folders {
|
||||||
|
@ -52,7 +52,7 @@ var folderSummaryRemoveDeprecatedRe = regexp.MustCompile(`(Invalid|IgnorePattern
|
|||||||
|
|
||||||
func (*verboseService) formatEvent(ev events.Event) string {
|
func (*verboseService) formatEvent(ev events.Event) string {
|
||||||
switch ev.Type {
|
switch ev.Type {
|
||||||
case events.DownloadProgress, events.LocalIndexUpdated:
|
case events.DownloadProgress:
|
||||||
// Skip
|
// Skip
|
||||||
return ""
|
return ""
|
||||||
|
|
||||||
@ -86,9 +86,13 @@ func (*verboseService) formatEvent(ev events.Event) string {
|
|||||||
data := ev.Data.(map[string]string)
|
data := ev.Data.(map[string]string)
|
||||||
return fmt.Sprintf("Remote change detected in folder %q: %s %s %s", data["folder"], data["action"], data["type"], data["path"])
|
return fmt.Sprintf("Remote change detected in folder %q: %s %s %s", data["folder"], data["action"], data["type"], data["path"])
|
||||||
|
|
||||||
|
case events.LocalIndexUpdated:
|
||||||
|
data := ev.Data.(map[string]interface{})
|
||||||
|
return fmt.Sprintf("Local index update for %q with %d items (seq: %d)", data["folder"], data["items"], data["sequence"])
|
||||||
|
|
||||||
case events.RemoteIndexUpdated:
|
case events.RemoteIndexUpdated:
|
||||||
data := ev.Data.(map[string]interface{})
|
data := ev.Data.(map[string]interface{})
|
||||||
return fmt.Sprintf("Device %v sent an index update for %q with %d items", data["device"], data["folder"], data["items"])
|
return fmt.Sprintf("Device %v sent an index update for %q with %d items (seq: %d)", data["device"], data["folder"], data["items"], data["sequence"])
|
||||||
|
|
||||||
case events.DeviceRejected:
|
case events.DeviceRejected:
|
||||||
data := ev.Data.(map[string]string)
|
data := ev.Data.(map[string]string)
|
||||||
@ -117,7 +121,7 @@ func (*verboseService) formatEvent(ev events.Event) string {
|
|||||||
|
|
||||||
case events.FolderCompletion:
|
case events.FolderCompletion:
|
||||||
data := ev.Data.(map[string]interface{})
|
data := ev.Data.(map[string]interface{})
|
||||||
return fmt.Sprintf("Completion for folder %q on device %v is %v%% (state: %s)", data["folder"], data["device"], data["completion"], data["remoteState"])
|
return fmt.Sprintf("Completion for folder %q on device %v is %v%% (state: %s, seq: %d)", data["folder"], data["device"], data["completion"], data["remoteState"], data["sequence"])
|
||||||
|
|
||||||
case events.FolderSummary:
|
case events.FolderSummary:
|
||||||
data := ev.Data.(model.FolderSummaryEventData)
|
data := ev.Data.(model.FolderSummaryEventData)
|
||||||
|
Loading…
Reference in New Issue
Block a user