mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-05 16:12:20 +00:00
Suppress frequent changes to files (fixes #12)
This commit is contained in:
parent
e3bc33dc88
commit
340c9095dd
@ -50,6 +50,9 @@ type Model struct {
|
|||||||
delete bool
|
delete bool
|
||||||
|
|
||||||
trace map[string]bool
|
trace map[string]bool
|
||||||
|
|
||||||
|
fileLastChanged map[string]time.Time
|
||||||
|
fileWasSuppressed map[string]int
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -57,6 +60,9 @@ const (
|
|||||||
|
|
||||||
idxBcastHoldtime = 15 * time.Second // Wait at least this long after the last index modification
|
idxBcastHoldtime = 15 * time.Second // Wait at least this long after the last index modification
|
||||||
idxBcastMaxDelay = 120 * time.Second // Unless we've already waited this long
|
idxBcastMaxDelay = 120 * time.Second // Unless we've already waited this long
|
||||||
|
|
||||||
|
minFileHoldTimeS = 60 // Never allow file changes more often than this
|
||||||
|
maxFileHoldTimeS = 600 // Always allow file changes at least this often
|
||||||
)
|
)
|
||||||
|
|
||||||
var ErrNoSuchFile = errors.New("no such file")
|
var ErrNoSuchFile = errors.New("no such file")
|
||||||
@ -66,15 +72,17 @@ var ErrNoSuchFile = errors.New("no such file")
|
|||||||
// for file data without altering the local repository in any way.
|
// for file data without altering the local repository in any way.
|
||||||
func NewModel(dir string) *Model {
|
func NewModel(dir string) *Model {
|
||||||
m := &Model{
|
m := &Model{
|
||||||
dir: dir,
|
dir: dir,
|
||||||
global: make(map[string]File),
|
global: make(map[string]File),
|
||||||
local: make(map[string]File),
|
local: make(map[string]File),
|
||||||
remote: make(map[string]map[string]File),
|
remote: make(map[string]map[string]File),
|
||||||
need: make(map[string]bool),
|
need: make(map[string]bool),
|
||||||
nodes: make(map[string]*protocol.Connection),
|
nodes: make(map[string]*protocol.Connection),
|
||||||
rawConn: make(map[string]io.ReadWriteCloser),
|
rawConn: make(map[string]io.ReadWriteCloser),
|
||||||
lastIdxBcast: time.Now(),
|
lastIdxBcast: time.Now(),
|
||||||
trace: make(map[string]bool),
|
trace: make(map[string]bool),
|
||||||
|
fileLastChanged: make(map[string]time.Time),
|
||||||
|
fileWasSuppressed: make(map[string]int),
|
||||||
}
|
}
|
||||||
|
|
||||||
go m.broadcastIndexLoop()
|
go m.broadcastIndexLoop()
|
||||||
@ -304,6 +312,7 @@ func (m *Model) Request(nodeID, name string, offset uint64, size uint32, hash []
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ReplaceLocal replaces the local repository index with the given list of files.
|
// ReplaceLocal replaces the local repository index with the given list of files.
|
||||||
|
// Change suppression is applied to files changing too often.
|
||||||
func (m *Model) ReplaceLocal(fs []File) {
|
func (m *Model) ReplaceLocal(fs []File) {
|
||||||
m.Lock()
|
m.Lock()
|
||||||
defer m.Unlock()
|
defer m.Unlock()
|
||||||
@ -391,6 +400,28 @@ func (m *Model) AddConnection(conn io.ReadWriteCloser, nodeID string) {
|
|||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *Model) shouldSuppressChange(name string) bool {
|
||||||
|
sup := shouldSuppressChange(m.fileLastChanged[name], m.fileWasSuppressed[name])
|
||||||
|
if sup {
|
||||||
|
m.fileWasSuppressed[name]++
|
||||||
|
} else {
|
||||||
|
m.fileWasSuppressed[name] = 0
|
||||||
|
m.fileLastChanged[name] = time.Now()
|
||||||
|
}
|
||||||
|
return sup
|
||||||
|
}
|
||||||
|
|
||||||
|
func shouldSuppressChange(lastChange time.Time, numChanges int) bool {
|
||||||
|
sinceLast := time.Since(lastChange)
|
||||||
|
if sinceLast > maxFileHoldTimeS*time.Second {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if sinceLast < time.Duration((numChanges+2)*minFileHoldTimeS)*time.Second {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
// protocolIndex returns the current local index in protocol data types.
|
// protocolIndex returns the current local index in protocol data types.
|
||||||
// Must be called with the read lock held.
|
// Must be called with the read lock held.
|
||||||
func (m *Model) protocolIndex() []protocol.FileInfo {
|
func (m *Model) protocolIndex() []protocol.FileInfo {
|
||||||
|
@ -340,3 +340,28 @@ func TestRequest(t *testing.T) {
|
|||||||
t.Errorf("Unexpected non nil data on insecure file read: %q", string(bs))
|
t.Errorf("Unexpected non nil data on insecure file read: %q", string(bs))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestSuppression(t *testing.T) {
|
||||||
|
var testdata = []struct {
|
||||||
|
lastChange time.Time
|
||||||
|
hold int
|
||||||
|
result bool
|
||||||
|
}{
|
||||||
|
{time.Unix(0, 0), 0, false}, // First change
|
||||||
|
{time.Now().Add(-1 * time.Second), 0, true}, // Changed once one second ago, suppress
|
||||||
|
{time.Now().Add(-119 * time.Second), 0, true}, // Changed once 119 seconds ago, suppress
|
||||||
|
{time.Now().Add(-121 * time.Second), 0, false}, // Changed once 121 seconds ago, permit
|
||||||
|
|
||||||
|
{time.Now().Add(-179 * time.Second), 1, true}, // Suppressed once 179 seconds ago, suppress again
|
||||||
|
{time.Now().Add(-181 * time.Second), 1, false}, // Suppressed once 181 seconds ago, permit
|
||||||
|
|
||||||
|
{time.Now().Add(-599 * time.Second), 99, true}, // Suppressed lots of times, last allowed 599 seconds ago, suppress again
|
||||||
|
{time.Now().Add(-601 * time.Second), 99, false}, // Suppressed lots of times, last allowed 601 seconds ago, permit
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, tc := range testdata {
|
||||||
|
if shouldSuppressChange(tc.lastChange, tc.hold) != tc.result {
|
||||||
|
t.Errorf("Incorrect result for test #%d: %v", i, tc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -9,6 +9,7 @@ import (
|
|||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
const BlockSize = 128 * 1024
|
const BlockSize = 128 * 1024
|
||||||
@ -81,17 +82,38 @@ func (m *Model) genWalker(res *[]File, ign map[string][]string) filepath.WalkFun
|
|||||||
// No change
|
// No change
|
||||||
*res = append(*res, hf)
|
*res = append(*res, hf)
|
||||||
} else {
|
} else {
|
||||||
|
m.Lock()
|
||||||
|
if m.shouldSuppressChange(rn) {
|
||||||
|
if m.trace["file"] {
|
||||||
|
log.Println("FILE: SUPPRESS:", rn, m.fileWasSuppressed[rn], time.Since(m.fileLastChanged[rn]))
|
||||||
|
}
|
||||||
|
|
||||||
|
if ok {
|
||||||
|
// Files that are ignored will be suppressed but don't actually exist in the local model
|
||||||
|
*res = append(*res, hf)
|
||||||
|
}
|
||||||
|
m.Unlock()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
m.Unlock()
|
||||||
|
|
||||||
if m.trace["file"] {
|
if m.trace["file"] {
|
||||||
log.Printf("FILE: Hash %q", p)
|
log.Printf("FILE: Hash %q", p)
|
||||||
}
|
}
|
||||||
fd, err := os.Open(p)
|
fd, err := os.Open(p)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if m.trace["file"] {
|
||||||
|
log.Printf("FILE: %q: %v", p, err)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
defer fd.Close()
|
defer fd.Close()
|
||||||
|
|
||||||
blocks, err := Blocks(fd, BlockSize)
|
blocks, err := Blocks(fd, BlockSize)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if m.trace["file"] {
|
||||||
|
log.Printf("FILE: %q: %v", p, err)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
f := File{
|
f := File{
|
||||||
|
Loading…
Reference in New Issue
Block a user