mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-07 00:53:58 +00:00
lib/versioner: Reduce surface area (#6186)
* lib/versioner: Reduce surface area This is a refactor while I was anyway rooting around in the versioner. Instead of exporting every possible implementation and the factory and letting the caller do whatever, this now encapsulates all that and exposes a New() that takes a config.VersioningConfiguration. Given that and that we don't know (from the outside) how a versioner works or what state it keeps, we now just construct it once per folder and keep it around. Previously it was recreated for each restore request. * unparam * wip
This commit is contained in:
parent
f747ba6d69
commit
4e151d380c
@ -18,7 +18,6 @@ import (
|
|||||||
"github.com/syncthing/syncthing/lib/fs"
|
"github.com/syncthing/syncthing/lib/fs"
|
||||||
"github.com/syncthing/syncthing/lib/protocol"
|
"github.com/syncthing/syncthing/lib/protocol"
|
||||||
"github.com/syncthing/syncthing/lib/util"
|
"github.com/syncthing/syncthing/lib/util"
|
||||||
"github.com/syncthing/syncthing/lib/versioner"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -105,18 +104,6 @@ func (f FolderConfiguration) Filesystem() fs.Filesystem {
|
|||||||
return f.cachedFilesystem
|
return f.cachedFilesystem
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f FolderConfiguration) Versioner() versioner.Versioner {
|
|
||||||
if f.Versioning.Type == "" {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
versionerFactory, ok := versioner.Factories[f.Versioning.Type]
|
|
||||||
if !ok {
|
|
||||||
panic(fmt.Sprintf("Requested versioning type %q that does not exist", f.Versioning.Type))
|
|
||||||
}
|
|
||||||
|
|
||||||
return versionerFactory(f.ID, f.Filesystem(), f.Versioning.Params)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f FolderConfiguration) ModTimeWindow() time.Duration {
|
func (f FolderConfiguration) ModTimeWindow() time.Duration {
|
||||||
return f.cachedModTimeWindow
|
return f.cachedModTimeWindow
|
||||||
}
|
}
|
||||||
|
@ -140,6 +140,7 @@ type model struct {
|
|||||||
folderRunners map[string]service // folder -> puller or scanner
|
folderRunners map[string]service // folder -> puller or scanner
|
||||||
folderRunnerTokens map[string][]suture.ServiceToken // folder -> tokens for puller or scanner
|
folderRunnerTokens map[string][]suture.ServiceToken // folder -> tokens for puller or scanner
|
||||||
folderRestartMuts syncMutexMap // folder -> restart mutex
|
folderRestartMuts syncMutexMap // folder -> restart mutex
|
||||||
|
folderVersioners map[string]versioner.Versioner // folder -> versioner (may be nil)
|
||||||
|
|
||||||
pmut sync.RWMutex // protects the below
|
pmut sync.RWMutex // protects the below
|
||||||
conn map[protocol.DeviceID]connections.Connection
|
conn map[protocol.DeviceID]connections.Connection
|
||||||
@ -166,6 +167,7 @@ var (
|
|||||||
errFolderNotRunning = errors.New("folder is not running")
|
errFolderNotRunning = errors.New("folder is not running")
|
||||||
errFolderMissing = errors.New("no such folder")
|
errFolderMissing = errors.New("no such folder")
|
||||||
errNetworkNotAllowed = errors.New("network not allowed")
|
errNetworkNotAllowed = errors.New("network not allowed")
|
||||||
|
errNoVersioner = errors.New("folder has no versioner")
|
||||||
// errors about why a connection is closed
|
// errors about why a connection is closed
|
||||||
errIgnoredFolderRemoved = errors.New("folder no longer ignored")
|
errIgnoredFolderRemoved = errors.New("folder no longer ignored")
|
||||||
errReplacingConnection = errors.New("replacing connection")
|
errReplacingConnection = errors.New("replacing connection")
|
||||||
@ -200,6 +202,7 @@ func NewModel(cfg config.Wrapper, id protocol.DeviceID, clientName, clientVersio
|
|||||||
folderIgnores: make(map[string]*ignore.Matcher),
|
folderIgnores: make(map[string]*ignore.Matcher),
|
||||||
folderRunners: make(map[string]service),
|
folderRunners: make(map[string]service),
|
||||||
folderRunnerTokens: make(map[string][]suture.ServiceToken),
|
folderRunnerTokens: make(map[string][]suture.ServiceToken),
|
||||||
|
folderVersioners: make(map[string]versioner.Versioner),
|
||||||
conn: make(map[protocol.DeviceID]connections.Connection),
|
conn: make(map[protocol.DeviceID]connections.Connection),
|
||||||
connRequestLimiters: make(map[protocol.DeviceID]*byteSemaphore),
|
connRequestLimiters: make(map[protocol.DeviceID]*byteSemaphore),
|
||||||
closed: make(map[protocol.DeviceID]chan struct{}),
|
closed: make(map[protocol.DeviceID]chan struct{}),
|
||||||
@ -318,21 +321,29 @@ func (m *model) startFolderLocked(cfg config.FolderConfiguration) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ver := cfg.Versioner()
|
|
||||||
if service, ok := ver.(suture.Service); ok {
|
|
||||||
// The versioner implements the suture.Service interface, so
|
|
||||||
// expects to be run in the background in addition to being called
|
|
||||||
// when files are going to be archived.
|
|
||||||
token := m.Add(service)
|
|
||||||
m.folderRunnerTokens[folder] = append(m.folderRunnerTokens[folder], token)
|
|
||||||
}
|
|
||||||
|
|
||||||
ffs := fset.MtimeFS()
|
ffs := fset.MtimeFS()
|
||||||
|
|
||||||
// These are our metadata files, and they should always be hidden.
|
// These are our metadata files, and they should always be hidden.
|
||||||
ffs.Hide(config.DefaultMarkerName)
|
_ = ffs.Hide(config.DefaultMarkerName)
|
||||||
ffs.Hide(".stversions")
|
_ = ffs.Hide(".stversions")
|
||||||
ffs.Hide(".stignore")
|
_ = ffs.Hide(".stignore")
|
||||||
|
|
||||||
|
var ver versioner.Versioner
|
||||||
|
if cfg.Versioning.Type != "" {
|
||||||
|
var err error
|
||||||
|
ver, err = versioner.New(ffs, cfg.Versioning)
|
||||||
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("creating versioner: %v", err))
|
||||||
|
}
|
||||||
|
if service, ok := ver.(suture.Service); ok {
|
||||||
|
// The versioner implements the suture.Service interface, so
|
||||||
|
// expects to be run in the background in addition to being called
|
||||||
|
// when files are going to be archived.
|
||||||
|
token := m.Add(service)
|
||||||
|
m.folderRunnerTokens[folder] = append(m.folderRunnerTokens[folder], token)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m.folderVersioners[folder] = ver
|
||||||
|
|
||||||
ignores := m.folderIgnores[folder]
|
ignores := m.folderIgnores[folder]
|
||||||
|
|
||||||
@ -459,6 +470,7 @@ func (m *model) removeFolderLocked(cfg config.FolderConfiguration) {
|
|||||||
delete(m.folderIgnores, cfg.ID)
|
delete(m.folderIgnores, cfg.ID)
|
||||||
delete(m.folderRunners, cfg.ID)
|
delete(m.folderRunners, cfg.ID)
|
||||||
delete(m.folderRunnerTokens, cfg.ID)
|
delete(m.folderRunnerTokens, cfg.ID)
|
||||||
|
delete(m.folderVersioners, cfg.ID)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *model) restartFolder(from, to config.FolderConfiguration) {
|
func (m *model) restartFolder(from, to config.FolderConfiguration) {
|
||||||
@ -2444,14 +2456,14 @@ func (m *model) GlobalDirectoryTree(folder, prefix string, levels int, dirsonly
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (m *model) GetFolderVersions(folder string) (map[string][]versioner.FileVersion, error) {
|
func (m *model) GetFolderVersions(folder string) (map[string][]versioner.FileVersion, error) {
|
||||||
fcfg, ok := m.cfg.Folder(folder)
|
m.fmut.RLock()
|
||||||
|
ver, ok := m.folderVersioners[folder]
|
||||||
|
m.fmut.RUnlock()
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, errFolderMissing
|
return nil, errFolderMissing
|
||||||
}
|
}
|
||||||
|
|
||||||
ver := fcfg.Versioner()
|
|
||||||
if ver == nil {
|
if ver == nil {
|
||||||
return nil, errors.New("no versioner configured")
|
return nil, errNoVersioner
|
||||||
}
|
}
|
||||||
|
|
||||||
return ver.GetVersions()
|
return ver.GetVersions()
|
||||||
@ -2463,7 +2475,15 @@ func (m *model) RestoreFolderVersions(folder string, versions map[string]time.Ti
|
|||||||
return nil, errFolderMissing
|
return nil, errFolderMissing
|
||||||
}
|
}
|
||||||
|
|
||||||
ver := fcfg.Versioner()
|
m.fmut.RLock()
|
||||||
|
ver := m.folderVersioners[folder]
|
||||||
|
m.fmut.RUnlock()
|
||||||
|
if !ok {
|
||||||
|
return nil, errFolderMissing
|
||||||
|
}
|
||||||
|
if ver == nil {
|
||||||
|
return nil, errNoVersioner
|
||||||
|
}
|
||||||
|
|
||||||
restoreErrors := make(map[string]string)
|
restoreErrors := make(map[string]string)
|
||||||
|
|
||||||
|
@ -21,22 +21,22 @@ import (
|
|||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
// Register the constructor for this type of versioner with the name "external"
|
// Register the constructor for this type of versioner with the name "external"
|
||||||
Factories["external"] = NewExternal
|
factories["external"] = newExternal
|
||||||
}
|
}
|
||||||
|
|
||||||
type External struct {
|
type external struct {
|
||||||
command string
|
command string
|
||||||
filesystem fs.Filesystem
|
filesystem fs.Filesystem
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewExternal(folderID string, filesystem fs.Filesystem, params map[string]string) Versioner {
|
func newExternal(filesystem fs.Filesystem, params map[string]string) Versioner {
|
||||||
command := params["command"]
|
command := params["command"]
|
||||||
|
|
||||||
if runtime.GOOS == "windows" {
|
if runtime.GOOS == "windows" {
|
||||||
command = strings.Replace(command, `\`, `\\`, -1)
|
command = strings.Replace(command, `\`, `\\`, -1)
|
||||||
}
|
}
|
||||||
|
|
||||||
s := External{
|
s := external{
|
||||||
command: command,
|
command: command,
|
||||||
filesystem: filesystem,
|
filesystem: filesystem,
|
||||||
}
|
}
|
||||||
@ -47,7 +47,7 @@ func NewExternal(folderID string, filesystem fs.Filesystem, params map[string]st
|
|||||||
|
|
||||||
// Archive moves the named file away to a version archive. If this function
|
// Archive moves the named file away to a version archive. If this function
|
||||||
// returns nil, the named file does not exist any more (has been archived).
|
// returns nil, the named file does not exist any more (has been archived).
|
||||||
func (v External) Archive(filePath string) error {
|
func (v external) Archive(filePath string) error {
|
||||||
info, err := v.filesystem.Lstat(filePath)
|
info, err := v.filesystem.Lstat(filePath)
|
||||||
if fs.IsNotExist(err) {
|
if fs.IsNotExist(err) {
|
||||||
l.Debugln("not archiving nonexistent file", filePath)
|
l.Debugln("not archiving nonexistent file", filePath)
|
||||||
@ -107,10 +107,10 @@ func (v External) Archive(filePath string) error {
|
|||||||
return errors.New("Versioner: file was not removed by external script")
|
return errors.New("Versioner: file was not removed by external script")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v External) GetVersions() (map[string][]FileVersion, error) {
|
func (v external) GetVersions() (map[string][]FileVersion, error) {
|
||||||
return nil, ErrRestorationNotSupported
|
return nil, ErrRestorationNotSupported
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v External) Restore(filePath string, versionTime time.Time) error {
|
func (v external) Restore(filePath string, versionTime time.Time) error {
|
||||||
return ErrRestorationNotSupported
|
return ErrRestorationNotSupported
|
||||||
}
|
}
|
||||||
|
@ -29,7 +29,7 @@ func TestExternalNoCommand(t *testing.T) {
|
|||||||
|
|
||||||
// The versioner should fail due to missing command.
|
// The versioner should fail due to missing command.
|
||||||
|
|
||||||
e := External{
|
e := external{
|
||||||
filesystem: fs.NewFilesystem(fs.FilesystemTypeBasic, "."),
|
filesystem: fs.NewFilesystem(fs.FilesystemTypeBasic, "."),
|
||||||
command: "nonexistent command",
|
command: "nonexistent command",
|
||||||
}
|
}
|
||||||
@ -62,7 +62,7 @@ func TestExternal(t *testing.T) {
|
|||||||
|
|
||||||
// The versioner should run successfully.
|
// The versioner should run successfully.
|
||||||
|
|
||||||
e := External{
|
e := external{
|
||||||
filesystem: fs.NewFilesystem(fs.FilesystemTypeBasic, "."),
|
filesystem: fs.NewFilesystem(fs.FilesystemTypeBasic, "."),
|
||||||
command: cmd,
|
command: cmd,
|
||||||
}
|
}
|
||||||
|
@ -15,22 +15,22 @@ import (
|
|||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
// Register the constructor for this type of versioner with the name "simple"
|
// Register the constructor for this type of versioner with the name "simple"
|
||||||
Factories["simple"] = NewSimple
|
factories["simple"] = newSimple
|
||||||
}
|
}
|
||||||
|
|
||||||
type Simple struct {
|
type simple struct {
|
||||||
keep int
|
keep int
|
||||||
folderFs fs.Filesystem
|
folderFs fs.Filesystem
|
||||||
versionsFs fs.Filesystem
|
versionsFs fs.Filesystem
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSimple(folderID string, folderFs fs.Filesystem, params map[string]string) Versioner {
|
func newSimple(folderFs fs.Filesystem, params map[string]string) Versioner {
|
||||||
keep, err := strconv.Atoi(params["keep"])
|
keep, err := strconv.Atoi(params["keep"])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
keep = 5 // A reasonable default
|
keep = 5 // A reasonable default
|
||||||
}
|
}
|
||||||
|
|
||||||
s := Simple{
|
s := simple{
|
||||||
keep: keep,
|
keep: keep,
|
||||||
folderFs: folderFs,
|
folderFs: folderFs,
|
||||||
versionsFs: fsFromParams(folderFs, params),
|
versionsFs: fsFromParams(folderFs, params),
|
||||||
@ -42,7 +42,7 @@ func NewSimple(folderID string, folderFs fs.Filesystem, params map[string]string
|
|||||||
|
|
||||||
// Archive moves the named file away to a version archive. If this function
|
// Archive moves the named file away to a version archive. If this function
|
||||||
// returns nil, the named file does not exist any more (has been archived).
|
// returns nil, the named file does not exist any more (has been archived).
|
||||||
func (v Simple) Archive(filePath string) error {
|
func (v simple) Archive(filePath string) error {
|
||||||
err := archiveFile(v.folderFs, v.versionsFs, filePath, TagFilename)
|
err := archiveFile(v.folderFs, v.versionsFs, filePath, TagFilename)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -63,10 +63,10 @@ func (v Simple) Archive(filePath string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v Simple) GetVersions() (map[string][]FileVersion, error) {
|
func (v simple) GetVersions() (map[string][]FileVersion, error) {
|
||||||
return retrieveVersions(v.versionsFs)
|
return retrieveVersions(v.versionsFs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v Simple) Restore(filepath string, versionTime time.Time) error {
|
func (v simple) Restore(filepath string, versionTime time.Time) error {
|
||||||
return restoreFile(v.versionsFs, v.folderFs, filepath, versionTime, TagFilename)
|
return restoreFile(v.versionsFs, v.folderFs, filepath, versionTime, TagFilename)
|
||||||
}
|
}
|
||||||
|
@ -41,7 +41,7 @@ func TestTaggedFilename(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Test parser
|
// Test parser
|
||||||
tag := ExtractTag(tc[2])
|
tag := extractTag(tc[2])
|
||||||
if tag != tc[1] {
|
if tag != tc[1] {
|
||||||
t.Errorf("%s != %s", tag, tc[1])
|
t.Errorf("%s != %s", tag, tc[1])
|
||||||
}
|
}
|
||||||
@ -61,7 +61,7 @@ func TestSimpleVersioningVersionCount(t *testing.T) {
|
|||||||
|
|
||||||
fs := fs.NewFilesystem(fs.FilesystemTypeBasic, dir)
|
fs := fs.NewFilesystem(fs.FilesystemTypeBasic, dir)
|
||||||
|
|
||||||
v := NewSimple("", fs, map[string]string{"keep": "2"})
|
v := newSimple(fs, map[string]string{"keep": "2"})
|
||||||
|
|
||||||
path := "test"
|
path := "test"
|
||||||
|
|
||||||
|
@ -22,26 +22,26 @@ import (
|
|||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
// Register the constructor for this type of versioner with the name "staggered"
|
// Register the constructor for this type of versioner with the name "staggered"
|
||||||
Factories["staggered"] = NewStaggered
|
factories["staggered"] = newStaggered
|
||||||
}
|
}
|
||||||
|
|
||||||
type Interval struct {
|
type interval struct {
|
||||||
step int64
|
step int64
|
||||||
end int64
|
end int64
|
||||||
}
|
}
|
||||||
|
|
||||||
type Staggered struct {
|
type staggered struct {
|
||||||
suture.Service
|
suture.Service
|
||||||
cleanInterval int64
|
cleanInterval int64
|
||||||
folderFs fs.Filesystem
|
folderFs fs.Filesystem
|
||||||
versionsFs fs.Filesystem
|
versionsFs fs.Filesystem
|
||||||
interval [4]Interval
|
interval [4]interval
|
||||||
mutex sync.Mutex
|
mutex sync.Mutex
|
||||||
|
|
||||||
testCleanDone chan struct{}
|
testCleanDone chan struct{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewStaggered(folderID string, folderFs fs.Filesystem, params map[string]string) Versioner {
|
func newStaggered(folderFs fs.Filesystem, params map[string]string) Versioner {
|
||||||
maxAge, err := strconv.ParseInt(params["maxAge"], 10, 0)
|
maxAge, err := strconv.ParseInt(params["maxAge"], 10, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
maxAge = 31536000 // Default: ~1 year
|
maxAge = 31536000 // Default: ~1 year
|
||||||
@ -55,11 +55,11 @@ func NewStaggered(folderID string, folderFs fs.Filesystem, params map[string]str
|
|||||||
params["fsPath"] = params["versionsPath"]
|
params["fsPath"] = params["versionsPath"]
|
||||||
versionsFs := fsFromParams(folderFs, params)
|
versionsFs := fsFromParams(folderFs, params)
|
||||||
|
|
||||||
s := &Staggered{
|
s := &staggered{
|
||||||
cleanInterval: cleanInterval,
|
cleanInterval: cleanInterval,
|
||||||
folderFs: folderFs,
|
folderFs: folderFs,
|
||||||
versionsFs: versionsFs,
|
versionsFs: versionsFs,
|
||||||
interval: [4]Interval{
|
interval: [4]interval{
|
||||||
{30, 3600}, // first hour -> 30 sec between versions
|
{30, 3600}, // first hour -> 30 sec between versions
|
||||||
{3600, 86400}, // next day -> 1 h between versions
|
{3600, 86400}, // next day -> 1 h between versions
|
||||||
{86400, 592000}, // next 30 days -> 1 day between versions
|
{86400, 592000}, // next 30 days -> 1 day between versions
|
||||||
@ -73,7 +73,7 @@ func NewStaggered(folderID string, folderFs fs.Filesystem, params map[string]str
|
|||||||
return s
|
return s
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v *Staggered) serve(ctx context.Context) {
|
func (v *staggered) serve(ctx context.Context) {
|
||||||
v.clean()
|
v.clean()
|
||||||
if v.testCleanDone != nil {
|
if v.testCleanDone != nil {
|
||||||
close(v.testCleanDone)
|
close(v.testCleanDone)
|
||||||
@ -91,7 +91,7 @@ func (v *Staggered) serve(ctx context.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v *Staggered) clean() {
|
func (v *staggered) clean() {
|
||||||
l.Debugln("Versioner clean: Waiting for lock on", v.versionsFs)
|
l.Debugln("Versioner clean: Waiting for lock on", v.versionsFs)
|
||||||
v.mutex.Lock()
|
v.mutex.Lock()
|
||||||
defer v.mutex.Unlock()
|
defer v.mutex.Unlock()
|
||||||
@ -142,7 +142,7 @@ func (v *Staggered) clean() {
|
|||||||
l.Debugln("Cleaner: Finished cleaning", v.versionsFs)
|
l.Debugln("Cleaner: Finished cleaning", v.versionsFs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v *Staggered) expire(versions []string) {
|
func (v *staggered) expire(versions []string) {
|
||||||
l.Debugln("Versioner: Expiring versions", versions)
|
l.Debugln("Versioner: Expiring versions", versions)
|
||||||
for _, file := range v.toRemove(versions, time.Now()) {
|
for _, file := range v.toRemove(versions, time.Now()) {
|
||||||
if fi, err := v.versionsFs.Lstat(file); err != nil {
|
if fi, err := v.versionsFs.Lstat(file); err != nil {
|
||||||
@ -159,7 +159,7 @@ func (v *Staggered) expire(versions []string) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v *Staggered) toRemove(versions []string, now time.Time) []string {
|
func (v *staggered) toRemove(versions []string, now time.Time) []string {
|
||||||
var prevAge int64
|
var prevAge int64
|
||||||
firstFile := true
|
firstFile := true
|
||||||
var remove []string
|
var remove []string
|
||||||
@ -168,7 +168,7 @@ func (v *Staggered) toRemove(versions []string, now time.Time) []string {
|
|||||||
sort.Strings(versions)
|
sort.Strings(versions)
|
||||||
|
|
||||||
for _, version := range versions {
|
for _, version := range versions {
|
||||||
versionTime, err := time.ParseInLocation(TimeFormat, ExtractTag(version), time.Local)
|
versionTime, err := time.ParseInLocation(TimeFormat, extractTag(version), time.Local)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
l.Debugf("Versioner: file name %q is invalid: %v", version, err)
|
l.Debugf("Versioner: file name %q is invalid: %v", version, err)
|
||||||
continue
|
continue
|
||||||
@ -190,7 +190,7 @@ func (v *Staggered) toRemove(versions []string, now time.Time) []string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Find the interval the file fits in
|
// Find the interval the file fits in
|
||||||
var usedInterval Interval
|
var usedInterval interval
|
||||||
for _, usedInterval = range v.interval {
|
for _, usedInterval = range v.interval {
|
||||||
if age < usedInterval.end {
|
if age < usedInterval.end {
|
||||||
break
|
break
|
||||||
@ -211,7 +211,7 @@ func (v *Staggered) toRemove(versions []string, now time.Time) []string {
|
|||||||
|
|
||||||
// Archive moves the named file away to a version archive. If this function
|
// Archive moves the named file away to a version archive. If this function
|
||||||
// returns nil, the named file does not exist any more (has been archived).
|
// returns nil, the named file does not exist any more (has been archived).
|
||||||
func (v *Staggered) Archive(filePath string) error {
|
func (v *staggered) Archive(filePath string) error {
|
||||||
l.Debugln("Waiting for lock on ", v.versionsFs)
|
l.Debugln("Waiting for lock on ", v.versionsFs)
|
||||||
v.mutex.Lock()
|
v.mutex.Lock()
|
||||||
defer v.mutex.Unlock()
|
defer v.mutex.Unlock()
|
||||||
@ -225,14 +225,14 @@ func (v *Staggered) Archive(filePath string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v *Staggered) GetVersions() (map[string][]FileVersion, error) {
|
func (v *staggered) GetVersions() (map[string][]FileVersion, error) {
|
||||||
return retrieveVersions(v.versionsFs)
|
return retrieveVersions(v.versionsFs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v *Staggered) Restore(filepath string, versionTime time.Time) error {
|
func (v *staggered) Restore(filepath string, versionTime time.Time) error {
|
||||||
return restoreFile(v.versionsFs, v.folderFs, filepath, versionTime, TagFilename)
|
return restoreFile(v.versionsFs, v.folderFs, filepath, versionTime, TagFilename)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v *Staggered) String() string {
|
func (v *staggered) String() string {
|
||||||
return fmt.Sprintf("Staggered/@%p", v)
|
return fmt.Sprintf("Staggered/@%p", v)
|
||||||
}
|
}
|
||||||
|
@ -57,9 +57,9 @@ func TestStaggeredVersioningVersionCount(t *testing.T) {
|
|||||||
}
|
}
|
||||||
sort.Strings(delete)
|
sort.Strings(delete)
|
||||||
|
|
||||||
v := NewStaggered("", fs.NewFilesystem(fs.FilesystemTypeFake, "testdata"), map[string]string{
|
v := newStaggered(fs.NewFilesystem(fs.FilesystemTypeFake, "testdata"), map[string]string{
|
||||||
"maxAge": strconv.Itoa(365 * 86400),
|
"maxAge": strconv.Itoa(365 * 86400),
|
||||||
}).(*Staggered)
|
}).(*staggered)
|
||||||
rem := v.toRemove(versionsWithMtime, now)
|
rem := v.toRemove(versionsWithMtime, now)
|
||||||
sort.Strings(rem)
|
sort.Strings(rem)
|
||||||
|
|
||||||
|
@ -20,21 +20,21 @@ import (
|
|||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
// Register the constructor for this type of versioner
|
// Register the constructor for this type of versioner
|
||||||
Factories["trashcan"] = NewTrashcan
|
factories["trashcan"] = newTrashcan
|
||||||
}
|
}
|
||||||
|
|
||||||
type Trashcan struct {
|
type trashcan struct {
|
||||||
suture.Service
|
suture.Service
|
||||||
folderFs fs.Filesystem
|
folderFs fs.Filesystem
|
||||||
versionsFs fs.Filesystem
|
versionsFs fs.Filesystem
|
||||||
cleanoutDays int
|
cleanoutDays int
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTrashcan(folderID string, folderFs fs.Filesystem, params map[string]string) Versioner {
|
func newTrashcan(folderFs fs.Filesystem, params map[string]string) Versioner {
|
||||||
cleanoutDays, _ := strconv.Atoi(params["cleanoutDays"])
|
cleanoutDays, _ := strconv.Atoi(params["cleanoutDays"])
|
||||||
// On error we default to 0, "do not clean out the trash can"
|
// On error we default to 0, "do not clean out the trash can"
|
||||||
|
|
||||||
s := &Trashcan{
|
s := &trashcan{
|
||||||
folderFs: folderFs,
|
folderFs: folderFs,
|
||||||
versionsFs: fsFromParams(folderFs, params),
|
versionsFs: fsFromParams(folderFs, params),
|
||||||
cleanoutDays: cleanoutDays,
|
cleanoutDays: cleanoutDays,
|
||||||
@ -47,13 +47,13 @@ func NewTrashcan(folderID string, folderFs fs.Filesystem, params map[string]stri
|
|||||||
|
|
||||||
// Archive moves the named file away to a version archive. If this function
|
// Archive moves the named file away to a version archive. If this function
|
||||||
// returns nil, the named file does not exist any more (has been archived).
|
// returns nil, the named file does not exist any more (has been archived).
|
||||||
func (t *Trashcan) Archive(filePath string) error {
|
func (t *trashcan) Archive(filePath string) error {
|
||||||
return archiveFile(t.folderFs, t.versionsFs, filePath, func(name, tag string) string {
|
return archiveFile(t.folderFs, t.versionsFs, filePath, func(name, tag string) string {
|
||||||
return name
|
return name
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Trashcan) serve(ctx context.Context) {
|
func (t *trashcan) serve(ctx context.Context) {
|
||||||
l.Debugln(t, "starting")
|
l.Debugln(t, "starting")
|
||||||
defer l.Debugln(t, "stopping")
|
defer l.Debugln(t, "stopping")
|
||||||
|
|
||||||
@ -79,11 +79,11 @@ func (t *Trashcan) serve(ctx context.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Trashcan) String() string {
|
func (t *trashcan) String() string {
|
||||||
return fmt.Sprintf("trashcan@%p", t)
|
return fmt.Sprintf("trashcan@%p", t)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Trashcan) cleanoutArchive() error {
|
func (t *trashcan) cleanoutArchive() error {
|
||||||
if _, err := t.versionsFs.Lstat("."); fs.IsNotExist(err) {
|
if _, err := t.versionsFs.Lstat("."); fs.IsNotExist(err) {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -121,11 +121,11 @@ func (t *Trashcan) cleanoutArchive() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Trashcan) GetVersions() (map[string][]FileVersion, error) {
|
func (t *trashcan) GetVersions() (map[string][]FileVersion, error) {
|
||||||
return retrieveVersions(t.versionsFs)
|
return retrieveVersions(t.versionsFs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Trashcan) Restore(filepath string, versionTime time.Time) error {
|
func (t *trashcan) Restore(filepath string, versionTime time.Time) error {
|
||||||
// If we have an untagged file A and want to restore it on top of existing file A, we can't first archive the
|
// If we have an untagged file A and want to restore it on top of existing file A, we can't first archive the
|
||||||
// existing A as we'd overwrite the old A version, therefore when we archive existing file, we archive it with a
|
// existing A as we'd overwrite the old A version, therefore when we archive existing file, we archive it with a
|
||||||
// tag but when the restoration is finished, we rename it (untag it). This is only important if when restoring A,
|
// tag but when the restoration is finished, we rename it (untag it). This is only important if when restoring A,
|
||||||
|
@ -53,7 +53,7 @@ func TestTrashcanCleanout(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
versioner := NewTrashcan("default", fs.NewFilesystem(fs.FilesystemTypeBasic, "testdata"), map[string]string{"cleanoutDays": "7"}).(*Trashcan)
|
versioner := newTrashcan(fs.NewFilesystem(fs.FilesystemTypeBasic, "testdata"), map[string]string{"cleanoutDays": "7"}).(*trashcan)
|
||||||
if err := versioner.cleanoutArchive(); err != nil {
|
if err := versioner.cleanoutArchive(); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -95,7 +95,7 @@ func TestTrashcanArchiveRestoreSwitcharoo(t *testing.T) {
|
|||||||
|
|
||||||
writeFile(t, folderFs, "file", "A")
|
writeFile(t, folderFs, "file", "A")
|
||||||
|
|
||||||
versioner := NewTrashcan("", folderFs, map[string]string{
|
versioner := newTrashcan(folderFs, map[string]string{
|
||||||
"fsType": "basic",
|
"fsType": "basic",
|
||||||
"fsPath": tmpDir2,
|
"fsPath": tmpDir2,
|
||||||
})
|
})
|
||||||
|
@ -24,7 +24,7 @@ var errDirectory = fmt.Errorf("cannot restore on top of a directory")
|
|||||||
var errNotFound = fmt.Errorf("version not found")
|
var errNotFound = fmt.Errorf("version not found")
|
||||||
var errFileAlreadyExists = fmt.Errorf("file already exists")
|
var errFileAlreadyExists = fmt.Errorf("file already exists")
|
||||||
|
|
||||||
// Inserts ~tag just before the extension of the filename.
|
// TagFilename inserts ~tag just before the extension of the filename.
|
||||||
func TagFilename(name, tag string) string {
|
func TagFilename(name, tag string) string {
|
||||||
dir, file := filepath.Dir(name), filepath.Base(name)
|
dir, file := filepath.Dir(name), filepath.Base(name)
|
||||||
ext := filepath.Ext(file)
|
ext := filepath.Ext(file)
|
||||||
@ -34,8 +34,8 @@ func TagFilename(name, tag string) string {
|
|||||||
|
|
||||||
var tagExp = regexp.MustCompile(`.*~([^~.]+)(?:\.[^.]+)?$`)
|
var tagExp = regexp.MustCompile(`.*~([^~.]+)(?:\.[^.]+)?$`)
|
||||||
|
|
||||||
// Returns the tag from a filename, whether at the end or middle.
|
// extractTag returns the tag from a filename, whether at the end or middle.
|
||||||
func ExtractTag(path string) string {
|
func extractTag(path string) string {
|
||||||
match := tagExp.FindStringSubmatch(path)
|
match := tagExp.FindStringSubmatch(path)
|
||||||
// match is []string{"whole match", "submatch"} when successful
|
// match is []string{"whole match", "submatch"} when successful
|
||||||
|
|
||||||
@ -45,9 +45,10 @@ func ExtractTag(path string) string {
|
|||||||
return match[1]
|
return match[1]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UntagFilename returns the filename without tag, and the extracted tag
|
||||||
func UntagFilename(path string) (string, string) {
|
func UntagFilename(path string) (string, string) {
|
||||||
ext := filepath.Ext(path)
|
ext := filepath.Ext(path)
|
||||||
versionTag := ExtractTag(path)
|
versionTag := extractTag(path)
|
||||||
|
|
||||||
// Files tagged with old style tags cannot be untagged.
|
// Files tagged with old style tags cannot be untagged.
|
||||||
if versionTag == "" {
|
if versionTag == "" {
|
||||||
@ -276,7 +277,7 @@ func findAllVersions(fs fs.Filesystem, filePath string) []string {
|
|||||||
file := filepath.Base(filePath)
|
file := filepath.Base(filePath)
|
||||||
|
|
||||||
// Glob according to the new file~timestamp.ext pattern.
|
// Glob according to the new file~timestamp.ext pattern.
|
||||||
pattern := filepath.Join(inFolderPath, TagFilename(file, TimeGlob))
|
pattern := filepath.Join(inFolderPath, TagFilename(file, timeGlob))
|
||||||
versions, err := fs.Glob(pattern)
|
versions, err := fs.Glob(pattern)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
l.Warnln("globbing:", err, "for", pattern)
|
l.Warnln("globbing:", err, "for", pattern)
|
||||||
|
@ -12,6 +12,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/syncthing/syncthing/lib/config"
|
||||||
"github.com/syncthing/syncthing/lib/fs"
|
"github.com/syncthing/syncthing/lib/fs"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -27,10 +28,22 @@ type FileVersion struct {
|
|||||||
Size int64 `json:"size"`
|
Size int64 `json:"size"`
|
||||||
}
|
}
|
||||||
|
|
||||||
var Factories = map[string]func(folderID string, filesystem fs.Filesystem, params map[string]string) Versioner{}
|
type factory func(filesystem fs.Filesystem, params map[string]string) Versioner
|
||||||
|
|
||||||
|
var factories = make(map[string]factory)
|
||||||
|
|
||||||
var ErrRestorationNotSupported = fmt.Errorf("version restoration not supported with the current versioner")
|
var ErrRestorationNotSupported = fmt.Errorf("version restoration not supported with the current versioner")
|
||||||
|
|
||||||
const (
|
const (
|
||||||
TimeFormat = "20060102-150405"
|
TimeFormat = "20060102-150405"
|
||||||
TimeGlob = "[0-9][0-9][0-9][0-9][0-9][0-9][0-9][0-9]-[0-9][0-9][0-9][0-9][0-9][0-9]" // glob pattern matching TimeFormat
|
timeGlob = "[0-9][0-9][0-9][0-9][0-9][0-9][0-9][0-9]-[0-9][0-9][0-9][0-9][0-9][0-9]" // glob pattern matching TimeFormat
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func New(fs fs.Filesystem, cfg config.VersioningConfiguration) (Versioner, error) {
|
||||||
|
fac, ok := factories[cfg.Type]
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("requested versioning type %q does not exist", cfg.Type)
|
||||||
|
}
|
||||||
|
|
||||||
|
return fac(fs, cfg.Params), nil
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user