2014-11-16 20:13:20 +00:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
2014-09-29 19:43:32 +00:00
|
|
|
//
|
2015-03-07 20:36:35 +00: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,
|
2017-02-09 06:52:18 +00:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2014-09-04 06:31:38 +00:00
|
|
|
|
2015-01-12 13:50:30 +00:00
|
|
|
package db
|
2014-07-06 12:46:48 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2014-10-09 08:44:18 +00:00
|
|
|
"fmt"
|
2014-07-06 12:46:48 +00:00
|
|
|
|
2015-09-22 17:38:46 +00:00
|
|
|
"github.com/syncthing/syncthing/lib/protocol"
|
2014-07-06 12:46:48 +00:00
|
|
|
)
|
|
|
|
|
2018-06-02 13:08:32 +00:00
|
|
|
func (vl VersionList) String() string {
|
2014-10-09 08:44:18 +00:00
|
|
|
var b bytes.Buffer
|
|
|
|
var id protocol.DeviceID
|
|
|
|
b.WriteString("{")
|
2018-06-02 13:08:32 +00:00
|
|
|
for i, v := range vl.Versions {
|
2014-10-09 08:44:18 +00:00
|
|
|
if i > 0 {
|
|
|
|
b.WriteString(", ")
|
|
|
|
}
|
2016-07-04 10:40:29 +00:00
|
|
|
copy(id[:], v.Device)
|
2018-09-26 21:30:22 +00:00
|
|
|
fmt.Fprintf(&b, "{%v, %v, %v}", v.Version, id, v.Invalid)
|
2014-10-09 08:44:18 +00:00
|
|
|
}
|
|
|
|
b.WriteString("}")
|
|
|
|
return b.String()
|
|
|
|
}
|
|
|
|
|
2018-06-02 13:08:32 +00:00
|
|
|
// update brings the VersionList up to date with file. It returns the updated
|
|
|
|
// VersionList, a potentially removed old FileVersion and its index, as well as
|
|
|
|
// the index where the new FileVersion was inserted.
|
|
|
|
func (vl VersionList) update(folder, device []byte, file protocol.FileInfo, db *Instance) (_ VersionList, removedFV FileVersion, removedAt int, insertedAt int) {
|
|
|
|
removedAt, insertedAt = -1, -1
|
|
|
|
for i, v := range vl.Versions {
|
|
|
|
if bytes.Equal(v.Device, device) {
|
|
|
|
removedAt = i
|
|
|
|
removedFV = v
|
|
|
|
vl.Versions = append(vl.Versions[:i], vl.Versions[i+1:]...)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nv := FileVersion{
|
|
|
|
Device: device,
|
|
|
|
Version: file.Version,
|
2018-06-24 07:50:18 +00:00
|
|
|
Invalid: file.IsInvalid(),
|
2018-06-02 13:08:32 +00:00
|
|
|
}
|
|
|
|
for i, v := range vl.Versions {
|
|
|
|
switch v.Version.Compare(file.Version) {
|
|
|
|
case protocol.Equal:
|
|
|
|
if nv.Invalid {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
fallthrough
|
|
|
|
|
|
|
|
case protocol.Lesser:
|
|
|
|
// The version at this point in the list is equal to or lesser
|
|
|
|
// ("older") than us. We insert ourselves in front of it.
|
|
|
|
vl = vl.insertAt(i, nv)
|
|
|
|
return vl, removedFV, removedAt, i
|
|
|
|
|
|
|
|
case protocol.ConcurrentLesser, protocol.ConcurrentGreater:
|
|
|
|
// The version at this point is in conflict with us. We must pull
|
|
|
|
// the actual file metadata to determine who wins. If we win, we
|
|
|
|
// insert ourselves in front of the loser here. (The "Lesser" and
|
|
|
|
// "Greater" in the condition above is just based on the device
|
|
|
|
// IDs in the version vector, which is not the only thing we use
|
|
|
|
// to determine the winner.)
|
|
|
|
//
|
|
|
|
// A surprise missing file entry here is counted as a win for us.
|
2018-09-18 08:41:06 +00:00
|
|
|
if of, ok := db.getFile(db.keyer.GenerateDeviceFileKey(nil, folder, v.Device, []byte(file.Name))); !ok || file.WinsConflict(of) {
|
2018-06-02 13:08:32 +00:00
|
|
|
vl = vl.insertAt(i, nv)
|
|
|
|
return vl, removedFV, removedAt, i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We didn't find a position for an insert above, so append to the end.
|
|
|
|
vl.Versions = append(vl.Versions, nv)
|
|
|
|
|
|
|
|
return vl, removedFV, removedAt, len(vl.Versions) - 1
|
|
|
|
}
|
|
|
|
|
|
|
|
func (vl VersionList) insertAt(i int, v FileVersion) VersionList {
|
|
|
|
vl.Versions = append(vl.Versions, FileVersion{})
|
|
|
|
copy(vl.Versions[i+1:], vl.Versions[i:])
|
|
|
|
vl.Versions[i] = v
|
|
|
|
return vl
|
|
|
|
}
|
|
|
|
|
|
|
|
func (vl VersionList) Get(device []byte) (FileVersion, bool) {
|
|
|
|
for _, v := range vl.Versions {
|
|
|
|
if bytes.Equal(v.Device, device) {
|
|
|
|
return v, true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FileVersion{}, false
|
|
|
|
}
|
|
|
|
|
2014-07-12 21:06:48 +00:00
|
|
|
type fileList []protocol.FileInfo
|
2014-07-06 12:46:48 +00:00
|
|
|
|
2018-06-02 13:08:32 +00:00
|
|
|
func (fl fileList) Len() int {
|
|
|
|
return len(fl)
|
2014-07-06 12:46:48 +00:00
|
|
|
}
|
|
|
|
|
2018-06-02 13:08:32 +00:00
|
|
|
func (fl fileList) Swap(a, b int) {
|
|
|
|
fl[a], fl[b] = fl[b], fl[a]
|
2014-07-06 12:46:48 +00:00
|
|
|
}
|
|
|
|
|
2018-06-02 13:08:32 +00:00
|
|
|
func (fl fileList) Less(a, b int) bool {
|
|
|
|
return fl[a].Name < fl[b].Name
|
2014-07-06 12:46:48 +00:00
|
|
|
}
|
|
|
|
|
2015-02-10 19:12:17 +00:00
|
|
|
// Flush batches to disk when they contain this many records.
|
|
|
|
const batchFlushSize = 64
|