badguardhome/config.go

171 lines
5.2 KiB
Go
Raw Normal View History

2018-08-30 14:25:33 +00:00
package main
import (
"io/ioutil"
"log"
"os"
"path/filepath"
"sync"
2018-12-12 17:22:45 +00:00
"time"
"github.com/AdguardTeam/AdGuardHome/dnsfilter"
"github.com/AdguardTeam/AdGuardHome/dnsforward"
"gopkg.in/yaml.v2"
2018-08-30 14:25:33 +00:00
)
const (
dataDir = "data" // data storage
filterDir = "filters" // cache location for downloaded filters, it's under DataDir
)
2018-08-30 14:25:33 +00:00
// configuration is loaded from YAML
// field ordering is important -- yaml fields will mirror ordering from here
2018-08-30 14:25:33 +00:00
type configuration struct {
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
DNS dnsConfig `yaml:"dns"`
Filters []filter `yaml:"filters"`
UserRules []string `yaml:"user_rules"`
2018-12-12 17:22:45 +00:00
DHCP dhcpState `yaml:"dhcp"`
2018-08-30 14:25:33 +00:00
sync.RWMutex `yaml:"-"`
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
}
// field ordering is important -- yaml fields will mirror ordering from here
type dnsConfig struct {
Port int `yaml:"port"`
dnsforward.FilteringConfig `yaml:",inline"`
UpstreamDNS []string `yaml:"upstream_dns"`
2018-08-30 14:25:33 +00:00
}
var defaultDNS = []string{"tls://1.1.1.1", "tls://1.0.0.1"}
2018-08-30 14:25:33 +00:00
2018-12-12 17:22:45 +00:00
// field ordering is important -- yaml fields will mirror ordering from here
type dhcpState struct {
2018-12-13 10:17:41 +00:00
Config dhcpConfig
2018-12-12 17:22:45 +00:00
Leases []dhcpLease
}
// field ordering is important -- yaml fields will mirror ordering from here
type dhcpConfig struct {
2018-12-13 11:51:35 +00:00
Enabled bool `json:"enabled" yaml:"enabled"`
GatewayIP string `json:"gateway_ip" yaml:"gateway_ip"`
SubnetMask string `json:"subnet_mask" yaml:"subnet_mask"`
RangeStart string `json:"range_start" yaml:"range_start"`
RangeEnd string `json:"range_end" yaml:"range_end"`
LeaseDuration uint64 `json:"lease_duration" yaml:"lease_duration"` // in seconds
2018-12-12 17:22:45 +00:00
}
// field ordering is important -- yaml fields will mirror ordering from here
type dhcpLease struct {
HWAddr [6]byte `json:"mac" yaml:"hwaddr"`
IP string `json:"ip"` // json by default keeps IP uppercase but we need lowercase
Hostname string
Expires time.Time
}
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{
ourConfigFilename: "AdGuardHome.yaml",
2018-08-30 14:25:33 +00:00
BindPort: 3000,
BindHost: "127.0.0.1",
DNS: dnsConfig{
Port: 53,
FilteringConfig: dnsforward.FilteringConfig{
ProtectionEnabled: true, // whether or not use any of dnsfilter features
FilteringEnabled: true, // whether or not use filter lists
BlockedResponseTTL: 10, // in seconds
QueryLogEnabled: true,
Ratelimit: 20,
RefuseAny: true,
BootstrapDNS: "8.8.8.8:53",
},
UpstreamDNS: defaultDNS,
2018-08-30 14:25:33 +00:00
},
Filters: []filter{
{Filter: dnsfilter.Filter{ID: 1}, Enabled: true, URL: "https://adguardteam.github.io/AdGuardSDNSFilter/Filters/filter.txt", Name: "AdGuard Simplified Domain Names filter"},
{Filter: dnsfilter.Filter{ID: 2}, Enabled: false, URL: "https://adaway.org/hosts.txt", Name: "AdAway"},
{Filter: dnsfilter.Filter{ID: 3}, Enabled: false, URL: "https://hosts-file.net/ad_servers.txt", Name: "hpHosts - Ad and Tracking servers only"},
{Filter: dnsfilter.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-12-13 10:17:41 +00:00
DHCP: dhcpState{Config: dhcpConfig{
2018-12-13 11:53:25 +00:00
LeaseDuration: 12 * 60 * 60, // in seconds
2018-12-12 17:22:45 +00:00
}},
SchemaVersion: currentSchemaVersion,
2018-08-30 14:25:33 +00:00
}
// Loads configuration from the YAML file
2018-08-30 14:25:33 +00:00
func parseConfig() error {
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
log.Printf("YAML file doesn't exist, skipping: %s", configFile)
2018-08-30 14:25:33 +00:00
return nil
}
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
}
// Deduplicate filters
2018-12-06 14:18:34 +00:00
deduplicateFilters()
updateUniqueFilterID(config.Filters)
2018-08-30 14:25:33 +00:00
return nil
}
// Saves configuration to the YAML file and also saves the user filter contents to a file
func (c *configuration) write() error {
c.Lock()
defer c.Unlock()
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
}
err = safeWriteFile(configFile, yamlText)
2018-08-30 14:25:33 +00:00
if err != nil {
log.Printf("Couldn't save YAML config: %s", err)
2018-08-30 14:25:33 +00:00
return err
}
return nil
}
func writeAllConfigs() error {
err := config.write()
if err != nil {
log.Printf("Couldn't write config: %s", err)
return err
}
userFilter := userFilter()
err = userFilter.save()
if err != nil {
log.Printf("Couldn't save the user filter: %s", err)
return err
}
2018-08-30 14:25:33 +00:00
return nil
}