* filters: rework update mechanism so that UI doesn't get locked while update is in progress
This commit is contained in:
parent
d664a9de1d
commit
0647f3fe86
69
filter.go
69
filter.go
@ -167,44 +167,83 @@ func periodicallyRefreshFilters() {
|
|||||||
|
|
||||||
// Checks filters updates if necessary
|
// Checks filters updates if necessary
|
||||||
// If force is true, it ignores the filter.LastUpdated field value
|
// If force is true, it ignores the filter.LastUpdated field value
|
||||||
|
//
|
||||||
|
// Algorithm:
|
||||||
|
// . Get the list of filters to be updated
|
||||||
|
// . For each filter run the download and checksum check operation
|
||||||
|
// . If filter data hasn't changed, set new update time
|
||||||
|
// . If filter data has changed, parse it, save it on disk, set new update time
|
||||||
|
// . Apply changes to the current configuration
|
||||||
|
// . Restart server
|
||||||
func refreshFiltersIfNecessary(force bool) int {
|
func refreshFiltersIfNecessary(force bool) int {
|
||||||
config.Lock()
|
var updateFilters []filter
|
||||||
|
|
||||||
// fetch URLs
|
config.RLock()
|
||||||
updateCount := 0
|
|
||||||
for i := range config.Filters {
|
for i := range config.Filters {
|
||||||
filter := &config.Filters[i] // otherwise we will be operating on a copy
|
f := &config.Filters[i] // otherwise we will be operating on a copy
|
||||||
|
|
||||||
if !filter.Enabled {
|
if !f.Enabled {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if !force && time.Since(filter.LastUpdated) <= updatePeriod {
|
if !force && time.Since(f.LastUpdated) <= updatePeriod {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
updated, err := filter.update()
|
var uf filter
|
||||||
|
uf.ID = f.ID
|
||||||
|
uf.URL = f.URL
|
||||||
|
uf.checksum = f.checksum
|
||||||
|
updateFilters = append(updateFilters, uf)
|
||||||
|
}
|
||||||
|
config.RUnlock()
|
||||||
|
|
||||||
|
updateCount := 0
|
||||||
|
for i := range updateFilters {
|
||||||
|
uf := &updateFilters[i]
|
||||||
|
updated, err := uf.update()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Failed to update filter %s: %s\n", filter.URL, err)
|
log.Printf("Failed to update filter %s: %s\n", uf.URL, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if updated {
|
if updated {
|
||||||
// Saving it to the filters dir now
|
// Saving it to the filters dir now
|
||||||
err = filter.save()
|
err = uf.save()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Failed to save the updated filter %d: %s", filter.ID, err)
|
log.Printf("Failed to save the updated filter %d: %s", uf.ID, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
updateCount++
|
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
mtime := time.Now()
|
mtime := time.Now()
|
||||||
os.Chtimes(filter.Path(), mtime, mtime)
|
e := os.Chtimes(uf.Path(), mtime, mtime)
|
||||||
filter.LastUpdated = mtime
|
if e != nil {
|
||||||
|
log.Error("os.Chtimes(): %v", e)
|
||||||
|
}
|
||||||
|
uf.LastUpdated = mtime
|
||||||
}
|
}
|
||||||
|
|
||||||
|
config.Lock()
|
||||||
|
for k := range config.Filters {
|
||||||
|
f := &config.Filters[k]
|
||||||
|
if f.ID != uf.ID || f.URL != uf.URL {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
f.LastUpdated = uf.LastUpdated
|
||||||
|
if !updated {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info("Updated filter #%d. Rules: %d -> %d",
|
||||||
|
f.ID, f.RulesCount, uf.RulesCount)
|
||||||
|
f.Name = uf.Name
|
||||||
|
f.Rules = uf.Rules
|
||||||
|
f.RulesCount = uf.RulesCount
|
||||||
|
f.checksum = uf.checksum
|
||||||
|
updateCount++
|
||||||
|
}
|
||||||
|
config.Unlock()
|
||||||
}
|
}
|
||||||
config.Unlock()
|
|
||||||
|
|
||||||
if updateCount > 0 && isRunning() {
|
if updateCount > 0 && isRunning() {
|
||||||
err := reconfigureDNSServer()
|
err := reconfigureDNSServer()
|
||||||
|
Loading…
Reference in New Issue
Block a user