/add_url -- it fetches the URL and checks if contents are valid filter, fails if it is not, and returns number of rules if it is

This commit is contained in:
Eugene Bujak 2018-09-14 04:33:54 +03:00
parent 5554643cd0
commit 3b44efc8e3

View File

@ -17,6 +17,7 @@ import (
"syscall" "syscall"
"time" "time"
"github.com/AdguardTeam/AdguardDNS/dnsfilter"
"gopkg.in/asaskevich/govalidator.v4" "gopkg.in/asaskevich/govalidator.v4"
) )
@ -593,11 +594,43 @@ func handleFilteringAddURL(w http.ResponseWriter, r *http.Request) {
http.Error(w, "URL parameter is not valid request URL", 400) http.Error(w, "URL parameter is not valid request URL", 400)
return return
} }
// TODO: check for duplicates
// check for duplicates
for i := range config.Filters {
filter := &config.Filters[i]
if filter.URL == url {
errortext := fmt.Sprintf("Filter URL already added -- %s", url)
log.Println(errortext)
http.Error(w, errortext, http.StatusBadRequest)
return
}
}
var filter = filter{ var filter = filter{
Enabled: true, Enabled: true,
URL: url, URL: url,
} }
ok, err = filter.update(time.Now())
if err != nil {
errortext := fmt.Sprintf("Couldn't fetch filter from url %s: %s", url, err)
log.Println(errortext)
http.Error(w, errortext, http.StatusBadRequest)
return
}
if filter.RulesCount == 0 {
errortext := fmt.Sprintf("Filter at url %s has no rules (maybe it points to blank page?)", url)
log.Println(errortext)
http.Error(w, errortext, http.StatusBadRequest)
return
}
if !ok {
errortext := fmt.Sprintf("Filter at url %s is invalid (maybe it points to blank page?)", url)
log.Println(errortext)
http.Error(w, errortext, http.StatusBadRequest)
return
}
// URL is deemed valid, append it to filters, update config, write new filter file and tell coredns to reload it
config.Filters = append(config.Filters, filter) config.Filters = append(config.Filters, filter)
err = writeAllConfigs() err = writeAllConfigs()
if err != nil { if err != nil {
@ -606,8 +639,6 @@ func handleFilteringAddURL(w http.ResponseWriter, r *http.Request) {
http.Error(w, errortext, http.StatusInternalServerError) http.Error(w, errortext, http.StatusInternalServerError)
return return
} }
// kick off refresh of rules from new URLs
refreshFiltersIfNeccessary()
err = writeFilterFile() err = writeFilterFile()
if err != nil { if err != nil {
errortext := fmt.Sprintf("Couldn't write filter file: %s", err) errortext := fmt.Sprintf("Couldn't write filter file: %s", err)
@ -616,7 +647,7 @@ func handleFilteringAddURL(w http.ResponseWriter, r *http.Request) {
return return
} }
tellCoreDNSToReload() tellCoreDNSToReload()
fmt.Fprintf(w, "OK\n") fmt.Fprintf(w, "OK %d rules\n", filter.RulesCount)
} }
func handleFilteringRemoveURL(w http.ResponseWriter, r *http.Request) { func handleFilteringRemoveURL(w http.ResponseWriter, r *http.Request) {
@ -849,7 +880,7 @@ func refreshFiltersIfNeccessary() int {
updateCount := 0 updateCount := 0
for i := range config.Filters { for i := range config.Filters {
filter := &config.Filters[i] // otherwise we will be operating on a copy filter := &config.Filters[i] // otherwise we will be operating on a copy
updated, err := updateFilter(filter, now) updated, err := filter.update(now)
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", filter.URL, err)
continue continue
@ -870,7 +901,7 @@ func refreshFiltersIfNeccessary() int {
return updateCount return updateCount
} }
func updateFilter(filter *filter, now time.Time) (bool, error) { func (filter *filter) update(now time.Time) (bool, error) {
if !filter.Enabled { if !filter.Enabled {
return false, nil return false, nil
} }
@ -909,6 +940,7 @@ func updateFilter(filter *filter, now time.Time) (bool, error) {
lines := strings.Split(string(body), "\n") lines := strings.Split(string(body), "\n")
rulesCount := 0 rulesCount := 0
seenTitle := false seenTitle := false
d := dnsfilter.New()
for _, line := range lines { for _, line := range lines {
line = strings.TrimSpace(line) line = strings.TrimSpace(line)
if line[0] == '!' { if line[0] == '!' {
@ -918,9 +950,21 @@ func updateFilter(filter *filter, now time.Time) (bool, error) {
seenTitle = true seenTitle = true
} }
} else if len(line) != 0 { } else if len(line) != 0 {
err = d.AddRule(line, 0)
if err == dnsfilter.ErrInvalidSyntax {
continue
}
if err != nil {
log.Printf("Couldn't add rule %s: %s", err)
return false, err
}
rulesCount++ rulesCount++
} }
} }
if bytes.Equal(filter.contents, body) {
log.Printf("Filter contents of URL %s are same, not considering it as an update", filter.URL)
return false, nil
}
filter.RulesCount = rulesCount filter.RulesCount = rulesCount
filter.contents = body filter.contents = body
return true, nil return true, nil