mirror of
https://github.com/octoleo/syncthing.git
synced 2025-02-02 11:58:28 +00:00
lib/model: Keep fmut locked while adding/starting/restarting folders (#6156)
This commit is contained in:
parent
e7100bc573
commit
28edf2f5bb
@ -224,8 +224,7 @@ func (m *model) Serve() {
|
|||||||
folderCfg.CreateRoot()
|
folderCfg.CreateRoot()
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
m.addFolder(folderCfg)
|
m.newFolder(folderCfg)
|
||||||
m.startFolder(folderCfg.ID)
|
|
||||||
}
|
}
|
||||||
m.Supervisor.Serve()
|
m.Supervisor.Serve()
|
||||||
}
|
}
|
||||||
@ -253,12 +252,16 @@ func (m *model) StartDeadlockDetector(timeout time.Duration) {
|
|||||||
|
|
||||||
// startFolder constructs the folder service and starts it.
|
// startFolder constructs the folder service and starts it.
|
||||||
func (m *model) startFolder(folder string) {
|
func (m *model) startFolder(folder string) {
|
||||||
|
m.fmut.RLock()
|
||||||
|
folderCfg := m.folderCfgs[folder]
|
||||||
|
m.fmut.RUnlock()
|
||||||
|
|
||||||
|
// Close connections to affected devices
|
||||||
|
m.closeConns(folderCfg.DeviceIDs(), fmt.Errorf("started folder %v", folderCfg.Description()))
|
||||||
|
|
||||||
m.fmut.Lock()
|
m.fmut.Lock()
|
||||||
defer m.fmut.Unlock()
|
defer m.fmut.Unlock()
|
||||||
folderCfg := m.folderCfgs[folder]
|
|
||||||
m.startFolderLocked(folderCfg)
|
m.startFolderLocked(folderCfg)
|
||||||
|
|
||||||
l.Infof("Ready to synchronize %s (%s)", folderCfg.Description(), folderCfg.Type)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Need to hold lock on m.fmut when calling this.
|
// Need to hold lock on m.fmut when calling this.
|
||||||
@ -288,11 +291,6 @@ func (m *model) startFolderLocked(cfg config.FolderConfiguration) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close connections to affected devices
|
|
||||||
m.fmut.Unlock()
|
|
||||||
m.closeConns(cfg.DeviceIDs(), fmt.Errorf("started folder %v", cfg.Description()))
|
|
||||||
m.fmut.Lock()
|
|
||||||
|
|
||||||
v, ok := fset.Sequence(protocol.LocalDeviceID), true
|
v, ok := fset.Sequence(protocol.LocalDeviceID), true
|
||||||
indexHasFiles := ok && v > 0
|
indexHasFiles := ok && v > 0
|
||||||
if !indexHasFiles {
|
if !indexHasFiles {
|
||||||
@ -325,28 +323,31 @@ func (m *model) startFolderLocked(cfg config.FolderConfiguration) {
|
|||||||
ffs.Hide(".stversions")
|
ffs.Hide(".stversions")
|
||||||
ffs.Hide(".stignore")
|
ffs.Hide(".stignore")
|
||||||
|
|
||||||
p := folderFactory(m, fset, m.folderIgnores[folder], cfg, ver, ffs, m.evLogger)
|
ignores := m.folderIgnores[folder]
|
||||||
|
|
||||||
|
p := folderFactory(m, fset, ignores, cfg, ver, ffs, m.evLogger)
|
||||||
|
|
||||||
m.folderRunners[folder] = p
|
m.folderRunners[folder] = p
|
||||||
|
|
||||||
m.warnAboutOverwritingProtectedFiles(folder)
|
m.warnAboutOverwritingProtectedFiles(cfg, ignores)
|
||||||
|
|
||||||
token := m.Add(p)
|
token := m.Add(p)
|
||||||
m.folderRunnerTokens[folder] = append(m.folderRunnerTokens[folder], token)
|
m.folderRunnerTokens[folder] = append(m.folderRunnerTokens[folder], token)
|
||||||
|
|
||||||
|
l.Infof("Ready to synchronize %s (%s)", cfg.Description(), cfg.Type)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *model) warnAboutOverwritingProtectedFiles(folder string) {
|
func (m *model) warnAboutOverwritingProtectedFiles(cfg config.FolderConfiguration, ignores *ignore.Matcher) {
|
||||||
if m.folderCfgs[folder].Type == config.FolderTypeSendOnly {
|
if cfg.Type == config.FolderTypeSendOnly {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is a bit of a hack.
|
// This is a bit of a hack.
|
||||||
ffs := m.folderCfgs[folder].Filesystem()
|
ffs := cfg.Filesystem()
|
||||||
if ffs.Type() != fs.FilesystemTypeBasic {
|
if ffs.Type() != fs.FilesystemTypeBasic {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
folderLocation := ffs.URI()
|
folderLocation := ffs.URI()
|
||||||
ignores := m.folderIgnores[folder]
|
|
||||||
|
|
||||||
var filesAtRisk []string
|
var filesAtRisk []string
|
||||||
for _, protectedFilePath := range m.protectedFiles {
|
for _, protectedFilePath := range m.protectedFiles {
|
||||||
@ -399,8 +400,9 @@ func (m *model) addFolderLocked(cfg config.FolderConfiguration, fset *db.FileSet
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (m *model) removeFolder(cfg config.FolderConfiguration) {
|
func (m *model) removeFolder(cfg config.FolderConfiguration) {
|
||||||
|
m.stopFolder(cfg, fmt.Errorf("removing folder %v", cfg.Description()))
|
||||||
|
|
||||||
m.fmut.Lock()
|
m.fmut.Lock()
|
||||||
defer m.fmut.Unlock()
|
|
||||||
|
|
||||||
isPathUnique := true
|
isPathUnique := true
|
||||||
for folderID, folderCfg := range m.folderCfgs {
|
for folderID, folderCfg := range m.folderCfgs {
|
||||||
@ -414,18 +416,20 @@ func (m *model) removeFolder(cfg config.FolderConfiguration) {
|
|||||||
cfg.Filesystem().RemoveAll(config.DefaultMarkerName)
|
cfg.Filesystem().RemoveAll(config.DefaultMarkerName)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.tearDownFolderLocked(cfg, fmt.Errorf("removing folder %v", cfg.Description()))
|
m.removeFolderLocked(cfg)
|
||||||
|
|
||||||
|
m.fmut.Unlock()
|
||||||
|
|
||||||
// Remove it from the database
|
// Remove it from the database
|
||||||
db.DropFolder(m.db, cfg.ID)
|
db.DropFolder(m.db, cfg.ID)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Need to hold lock on m.fmut when calling this.
|
func (m *model) stopFolder(cfg config.FolderConfiguration, err error) {
|
||||||
func (m *model) tearDownFolderLocked(cfg config.FolderConfiguration, err error) {
|
|
||||||
// Stop the services running for this folder and wait for them to finish
|
// Stop the services running for this folder and wait for them to finish
|
||||||
// stopping to prevent races on restart.
|
// stopping to prevent races on restart.
|
||||||
|
m.fmut.RLock()
|
||||||
tokens := m.folderRunnerTokens[cfg.ID]
|
tokens := m.folderRunnerTokens[cfg.ID]
|
||||||
|
m.fmut.RUnlock()
|
||||||
m.fmut.Unlock()
|
|
||||||
|
|
||||||
// Close connections to affected devices
|
// Close connections to affected devices
|
||||||
// Must happen before stopping the folder service to abort ongoing
|
// Must happen before stopping the folder service to abort ongoing
|
||||||
@ -439,9 +443,10 @@ func (m *model) tearDownFolderLocked(cfg config.FolderConfiguration, err error)
|
|||||||
// Wait for connections to stop to ensure that no more calls to methods
|
// Wait for connections to stop to ensure that no more calls to methods
|
||||||
// expecting this folder to exist happen (e.g. .IndexUpdate).
|
// expecting this folder to exist happen (e.g. .IndexUpdate).
|
||||||
w.Wait()
|
w.Wait()
|
||||||
|
}
|
||||||
|
|
||||||
m.fmut.Lock()
|
// Need to hold lock on m.fmut when calling this.
|
||||||
|
func (m *model) removeFolderLocked(cfg config.FolderConfiguration) {
|
||||||
// Clean up our config maps
|
// Clean up our config maps
|
||||||
delete(m.folderCfgs, cfg.ID)
|
delete(m.folderCfgs, cfg.ID)
|
||||||
delete(m.folderFiles, cfg.ID)
|
delete(m.folderFiles, cfg.ID)
|
||||||
@ -483,22 +488,40 @@ func (m *model) restartFolder(from, to config.FolderConfiguration) {
|
|||||||
errMsg = "restarting"
|
errMsg = "restarting"
|
||||||
}
|
}
|
||||||
|
|
||||||
m.fmut.Lock()
|
var fset *db.FileSet
|
||||||
defer m.fmut.Unlock()
|
|
||||||
|
|
||||||
m.tearDownFolderLocked(from, fmt.Errorf("%v folder %v", errMsg, to.Description()))
|
|
||||||
if !to.Paused {
|
if !to.Paused {
|
||||||
// Creating the fileset can take a long time (metadata calculation)
|
// Creating the fileset can take a long time (metadata calculation)
|
||||||
// so we do it outside of the lock.
|
// so we do it outside of the lock.
|
||||||
m.fmut.Unlock()
|
fset = db.NewFileSet(to.ID, to.Filesystem(), m.db)
|
||||||
fset := db.NewFileSet(to.ID, to.Filesystem(), m.db)
|
}
|
||||||
|
|
||||||
|
m.stopFolder(from, fmt.Errorf("%v folder %v", errMsg, to.Description()))
|
||||||
|
|
||||||
m.fmut.Lock()
|
m.fmut.Lock()
|
||||||
|
defer m.fmut.Unlock()
|
||||||
|
|
||||||
|
m.removeFolderLocked(from)
|
||||||
|
if !to.Paused {
|
||||||
m.addFolderLocked(to, fset)
|
m.addFolderLocked(to, fset)
|
||||||
m.startFolderLocked(to)
|
m.startFolderLocked(to)
|
||||||
}
|
}
|
||||||
l.Infof("%v folder %v (%v)", infoMsg, to.Description(), to.Type)
|
l.Infof("%v folder %v (%v)", infoMsg, to.Description(), to.Type)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *model) newFolder(cfg config.FolderConfiguration) {
|
||||||
|
// Creating the fileset can take a long time (metadata calculation) so
|
||||||
|
// we do it outside of the lock.
|
||||||
|
fset := db.NewFileSet(cfg.ID, cfg.Filesystem(), m.db)
|
||||||
|
|
||||||
|
// Close connections to affected devices
|
||||||
|
m.closeConns(cfg.DeviceIDs(), fmt.Errorf("started folder %v", cfg.Description()))
|
||||||
|
|
||||||
|
m.fmut.Lock()
|
||||||
|
defer m.fmut.Unlock()
|
||||||
|
m.addFolderLocked(cfg, fset)
|
||||||
|
m.startFolderLocked(cfg)
|
||||||
|
}
|
||||||
|
|
||||||
func (m *model) UsageReportingStats(version int, preview bool) map[string]interface{} {
|
func (m *model) UsageReportingStats(version int, preview bool) map[string]interface{} {
|
||||||
stats := make(map[string]interface{})
|
stats := make(map[string]interface{})
|
||||||
if version >= 3 {
|
if version >= 3 {
|
||||||
@ -2523,8 +2546,7 @@ func (m *model) CommitConfiguration(from, to config.Configuration) bool {
|
|||||||
l.Infoln("Paused folder", cfg.Description())
|
l.Infoln("Paused folder", cfg.Description())
|
||||||
} else {
|
} else {
|
||||||
l.Infoln("Adding folder", cfg.Description())
|
l.Infoln("Adding folder", cfg.Description())
|
||||||
m.addFolder(cfg)
|
m.newFolder(cfg)
|
||||||
m.startFolder(folderID)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user