2018-08-30 14:25:33 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"sync"
|
|
|
|
"time"
|
2018-10-30 14:16:20 +00:00
|
|
|
|
2018-11-28 14:29:48 +00:00
|
|
|
"github.com/AdguardTeam/AdGuardHome/dnsforward"
|
2018-10-30 14:16:20 +00:00
|
|
|
"gopkg.in/yaml.v2"
|
2018-08-30 14:25:33 +00:00
|
|
|
)
|
|
|
|
|
2018-11-27 17:51:12 +00:00
|
|
|
const (
|
|
|
|
currentSchemaVersion = 1 // used for upgrading from old configs to new config
|
|
|
|
dataDir = "data" // data storage
|
|
|
|
filterDir = "filters" // cache location for downloaded filters, it's under DataDir
|
|
|
|
userFilterID = 0 // special filter ID, always 0
|
|
|
|
)
|
2018-10-30 14:16:20 +00:00
|
|
|
|
2018-10-30 09:24:59 +00:00
|
|
|
// Just a counter that we use for incrementing the filter ID
|
2018-11-27 18:25:03 +00:00
|
|
|
var nextFilterID int64 = time.Now().Unix()
|
2018-10-30 09:24:59 +00:00
|
|
|
|
2018-08-30 14:25:33 +00:00
|
|
|
// configuration is loaded from YAML
|
2018-11-27 17:51:12 +00:00
|
|
|
// field ordering is important -- yaml fields will mirror ordering from here
|
2018-08-30 14:25:33 +00:00
|
|
|
type configuration struct {
|
2018-11-27 17:51:12 +00:00
|
|
|
ourConfigFilename string // Config filename (can be overriden via the command line arguments)
|
|
|
|
ourBinaryDir string // Location of our directory, used to protect against CWD being somewhere else
|
|
|
|
|
|
|
|
BindHost string `yaml:"bind_host"`
|
|
|
|
BindPort int `yaml:"bind_port"`
|
|
|
|
AuthName string `yaml:"auth_name"`
|
|
|
|
AuthPass string `yaml:"auth_pass"`
|
|
|
|
Language string `yaml:"language"` // two-letter ISO 639-1 language code
|
|
|
|
CoreDNS coreDNSConfig `yaml:"coredns"`
|
|
|
|
Filters []filter `yaml:"filters"`
|
|
|
|
UserRules []string `yaml:"user_rules,omitempty"`
|
2018-08-30 14:25:33 +00:00
|
|
|
|
2018-10-06 21:58:59 +00:00
|
|
|
sync.RWMutex `yaml:"-"`
|
2018-11-27 17:51:12 +00:00
|
|
|
|
|
|
|
SchemaVersion int `yaml:"schema_version"` // keeping last so that users will be less tempted to change it -- used when upgrading between versions
|
2018-08-30 14:25:33 +00:00
|
|
|
}
|
|
|
|
|
2018-11-27 17:51:12 +00:00
|
|
|
// field ordering is important -- yaml fields will mirror ordering from here
|
2018-08-30 14:25:33 +00:00
|
|
|
type coreDNSConfig struct {
|
2018-11-28 15:24:04 +00:00
|
|
|
binaryFile string
|
|
|
|
coreFile string
|
|
|
|
Filters []filter `yaml:"-"`
|
|
|
|
Port int `yaml:"port"`
|
|
|
|
|
|
|
|
dnsforward.FilteringConfig `yaml:",inline"`
|
|
|
|
|
|
|
|
QueryLogEnabled bool `yaml:"querylog_enabled"`
|
|
|
|
Ratelimit int `yaml:"ratelimit"`
|
|
|
|
RefuseAny bool `yaml:"refuse_any"`
|
|
|
|
Pprof string `yaml:"-"`
|
|
|
|
Cache string `yaml:"-"`
|
|
|
|
Prometheus string `yaml:"-"`
|
|
|
|
BootstrapDNS string `yaml:"bootstrap_dns"`
|
|
|
|
UpstreamDNS []string `yaml:"upstream_dns"`
|
2018-08-30 14:25:33 +00:00
|
|
|
}
|
|
|
|
|
2018-11-27 17:51:12 +00:00
|
|
|
// field ordering is important -- yaml fields will mirror ordering from here
|
2018-08-30 14:25:33 +00:00
|
|
|
type filter struct {
|
2018-11-27 17:51:12 +00:00
|
|
|
Enabled bool `json:"enabled"`
|
2018-11-27 13:48:57 +00:00
|
|
|
URL string `json:"url"`
|
|
|
|
Name string `json:"name" yaml:"name"`
|
|
|
|
RulesCount int `json:"rulesCount" yaml:"-"`
|
2018-11-27 17:51:12 +00:00
|
|
|
LastUpdated time.Time `json:"lastUpdated,omitempty" yaml:"last_updated,omitempty"`
|
|
|
|
|
2018-11-28 14:29:48 +00:00
|
|
|
dnsforward.Filter `yaml:",inline"`
|
2018-08-30 14:25:33 +00:00
|
|
|
}
|
|
|
|
|
2018-09-26 14:47:23 +00:00
|
|
|
var defaultDNS = []string{"tls://1.1.1.1", "tls://1.0.0.1"}
|
2018-08-30 14:25:33 +00:00
|
|
|
|
|
|
|
// initialize to default values, will be changed later when reading config or parsing command line
|
|
|
|
var config = configuration{
|
2018-10-15 13:02:19 +00:00
|
|
|
ourConfigFilename: "AdGuardHome.yaml",
|
2018-08-30 14:25:33 +00:00
|
|
|
BindPort: 3000,
|
|
|
|
BindHost: "127.0.0.1",
|
|
|
|
CoreDNS: coreDNSConfig{
|
2018-11-28 15:24:04 +00:00
|
|
|
Port: 53,
|
|
|
|
binaryFile: "coredns", // only filename, no path
|
|
|
|
coreFile: "Corefile", // only filename, no path
|
|
|
|
FilteringConfig: dnsforward.FilteringConfig{
|
|
|
|
ProtectionEnabled: true,
|
|
|
|
FilteringEnabled: true,
|
|
|
|
SafeBrowsingEnabled: false,
|
|
|
|
BlockedResponseTTL: 10, // in seconds
|
|
|
|
},
|
|
|
|
QueryLogEnabled: true,
|
|
|
|
Ratelimit: 20,
|
|
|
|
RefuseAny: true,
|
|
|
|
BootstrapDNS: "8.8.8.8:53",
|
|
|
|
UpstreamDNS: defaultDNS,
|
|
|
|
Cache: "cache",
|
|
|
|
Prometheus: "prometheus :9153",
|
2018-08-30 14:25:33 +00:00
|
|
|
},
|
|
|
|
Filters: []filter{
|
2018-11-28 14:29:48 +00:00
|
|
|
{Filter: dnsforward.Filter{ID: 1}, Enabled: true, URL: "https://adguardteam.github.io/AdGuardSDNSFilter/Filters/filter.txt", Name: "AdGuard Simplified Domain Names filter"},
|
|
|
|
{Filter: dnsforward.Filter{ID: 2}, Enabled: false, URL: "https://adaway.org/hosts.txt", Name: "AdAway"},
|
|
|
|
{Filter: dnsforward.Filter{ID: 3}, Enabled: false, URL: "https://hosts-file.net/ad_servers.txt", Name: "hpHosts - Ad and Tracking servers only"},
|
|
|
|
{Filter: dnsforward.Filter{ID: 4}, Enabled: false, URL: "http://www.malwaredomainlist.com/hostslist/hosts.txt", Name: "MalwareDomainList.com Hosts List"},
|
2018-08-30 14:25:33 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2018-10-29 23:17:24 +00:00
|
|
|
// Creates a helper object for working with the user rules
|
2018-11-27 13:48:57 +00:00
|
|
|
func userFilter() filter {
|
2018-11-28 13:05:24 +00:00
|
|
|
return filter{
|
2018-10-30 14:16:20 +00:00
|
|
|
// User filter always has constant ID=0
|
2018-11-28 13:05:24 +00:00
|
|
|
Enabled: true,
|
2018-11-28 14:29:48 +00:00
|
|
|
Filter: dnsforward.Filter{
|
|
|
|
ID: userFilterID,
|
|
|
|
Rules: config.UserRules,
|
|
|
|
},
|
2018-10-29 23:17:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-30 09:24:59 +00:00
|
|
|
// Loads configuration from the YAML file
|
2018-08-30 14:25:33 +00:00
|
|
|
func parseConfig() error {
|
2018-10-30 09:24:59 +00:00
|
|
|
configFile := filepath.Join(config.ourBinaryDir, config.ourConfigFilename)
|
|
|
|
log.Printf("Reading YAML file: %s", configFile)
|
|
|
|
if _, err := os.Stat(configFile); os.IsNotExist(err) {
|
2018-08-30 14:25:33 +00:00
|
|
|
// do nothing, file doesn't exist
|
2018-10-30 09:24:59 +00:00
|
|
|
log.Printf("YAML file doesn't exist, skipping: %s", configFile)
|
2018-08-30 14:25:33 +00:00
|
|
|
return nil
|
|
|
|
}
|
2018-10-30 09:24:59 +00:00
|
|
|
yamlFile, err := ioutil.ReadFile(configFile)
|
2018-08-30 14:25:33 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("Couldn't read config file: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = yaml.Unmarshal(yamlFile, &config)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Couldn't parse config file: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-10-30 09:24:59 +00:00
|
|
|
// Deduplicate filters
|
|
|
|
{
|
|
|
|
i := 0 // output index, used for deletion later
|
|
|
|
urls := map[string]bool{}
|
|
|
|
for _, filter := range config.Filters {
|
|
|
|
if _, ok := urls[filter.URL]; !ok {
|
|
|
|
// we didn't see it before, keep it
|
|
|
|
urls[filter.URL] = true // remember the URL
|
|
|
|
config.Filters[i] = filter
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// all entries we want to keep are at front, delete the rest
|
|
|
|
config.Filters = config.Filters[:i]
|
|
|
|
}
|
|
|
|
|
2018-11-27 18:25:03 +00:00
|
|
|
updateUniqueFilterID(config.Filters)
|
2018-10-30 09:24:59 +00:00
|
|
|
|
2018-08-30 14:25:33 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-10-30 09:24:59 +00:00
|
|
|
// Saves configuration to the YAML file and also saves the user filter contents to a file
|
2018-11-29 11:56:56 +00:00
|
|
|
func (c *configuration) write() error {
|
2018-11-29 10:31:50 +00:00
|
|
|
c.Lock()
|
|
|
|
defer c.Unlock()
|
2018-10-30 09:24:59 +00:00
|
|
|
configFile := filepath.Join(config.ourBinaryDir, config.ourConfigFilename)
|
|
|
|
log.Printf("Writing YAML file: %s", configFile)
|
2018-08-30 14:25:33 +00:00
|
|
|
yamlText, err := yaml.Marshal(&config)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Couldn't generate YAML file: %s", err)
|
|
|
|
return err
|
|
|
|
}
|
2018-11-27 17:39:59 +00:00
|
|
|
err = safeWriteFile(configFile, yamlText)
|
2018-08-30 14:25:33 +00:00
|
|
|
if err != nil {
|
2018-10-29 23:17:24 +00:00
|
|
|
log.Printf("Couldn't save YAML config: %s", err)
|
2018-08-30 14:25:33 +00:00
|
|
|
return err
|
|
|
|
}
|
2018-10-29 23:17:24 +00:00
|
|
|
|
2018-11-27 13:48:57 +00:00
|
|
|
userFilter := userFilter()
|
2018-10-29 23:17:24 +00:00
|
|
|
err = userFilter.save()
|
2018-09-05 23:03:03 +00:00
|
|
|
if err != nil {
|
2018-10-29 23:17:24 +00:00
|
|
|
log.Printf("Couldn't save the user filter: %s", err)
|
2018-09-05 23:03:03 +00:00
|
|
|
return err
|
|
|
|
}
|
2018-10-29 23:17:24 +00:00
|
|
|
|
2018-08-30 14:25:33 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func writeAllConfigs() error {
|
2018-11-28 13:45:30 +00:00
|
|
|
return config.write()
|
2018-08-30 14:25:33 +00:00
|
|
|
}
|
2018-11-27 18:25:03 +00:00
|
|
|
|
|
|
|
// Set the next filter ID to max(filter.ID) + 1
|
|
|
|
func updateUniqueFilterID(filters []filter) {
|
|
|
|
for _, filter := range filters {
|
|
|
|
if nextFilterID < filter.ID {
|
|
|
|
nextFilterID = filter.ID + 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func assignUniqueFilterID() int64 {
|
|
|
|
value := nextFilterID
|
|
|
|
nextFilterID += 1
|
|
|
|
return value
|
|
|
|
}
|