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:
Jakob Borg 2019-11-26 08:39:31 +01:00 committed by Audrius Butkevicius
parent f747ba6d69
commit 4e151d380c
12 changed files with 107 additions and 86 deletions

View File

@ -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
} }

View File

@ -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,7 +321,20 @@ func (m *model) startFolderLocked(cfg config.FolderConfiguration) {
} }
} }
ver := cfg.Versioner() ffs := fset.MtimeFS()
// These are our metadata files, and they should always be hidden.
_ = ffs.Hide(config.DefaultMarkerName)
_ = ffs.Hide(".stversions")
_ = 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 { if service, ok := ver.(suture.Service); ok {
// The versioner implements the suture.Service interface, so // The versioner implements the suture.Service interface, so
// expects to be run in the background in addition to being called // expects to be run in the background in addition to being called
@ -326,13 +342,8 @@ func (m *model) startFolderLocked(cfg config.FolderConfiguration) {
token := m.Add(service) token := m.Add(service)
m.folderRunnerTokens[folder] = append(m.folderRunnerTokens[folder], token) m.folderRunnerTokens[folder] = append(m.folderRunnerTokens[folder], token)
} }
}
ffs := fset.MtimeFS() m.folderVersioners[folder] = ver
// These are our metadata files, and they should always be hidden.
ffs.Hide(config.DefaultMarkerName)
ffs.Hide(".stversions")
ffs.Hide(".stignore")
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)

View File

@ -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
} }

View File

@ -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,
} }

View File

@ -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)
} }

View File

@ -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"

View File

@ -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)
} }

View File

@ -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)

View File

@ -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,

View File

@ -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,
}) })

View File

@ -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)

View File

@ -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
}