2016-10-17 18:37:56 +01:00
|
|
|
package auto
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"os"
|
2018-10-21 15:59:37 +02:00
|
|
|
"path/filepath"
|
2016-10-17 18:37:56 +01:00
|
|
|
"regexp"
|
|
|
|
|
"time"
|
|
|
|
|
|
2020-09-24 18:14:41 +02:00
|
|
|
"github.com/coredns/caddy"
|
2017-02-21 22:51:47 -08:00
|
|
|
"github.com/coredns/coredns/core/dnsserver"
|
2017-09-14 09:36:06 +01:00
|
|
|
"github.com/coredns/coredns/plugin"
|
|
|
|
|
"github.com/coredns/coredns/plugin/metrics"
|
2018-04-22 21:40:33 +01:00
|
|
|
clog "github.com/coredns/coredns/plugin/pkg/log"
|
2018-02-16 03:44:50 -05:00
|
|
|
"github.com/coredns/coredns/plugin/pkg/upstream"
|
2020-09-24 11:30:39 -07:00
|
|
|
"github.com/coredns/coredns/plugin/transfer"
|
2016-10-17 18:37:56 +01:00
|
|
|
)
|
|
|
|
|
|
2018-04-22 21:40:33 +01:00
|
|
|
var log = clog.NewWithPlugin("auto")
|
|
|
|
|
|
2019-09-20 08:02:30 +01:00
|
|
|
func init() { plugin.Register("auto", setup) }
|
2016-10-17 18:37:56 +01:00
|
|
|
|
|
|
|
|
func setup(c *caddy.Controller) error {
|
|
|
|
|
a, err := autoParse(c)
|
|
|
|
|
if err != nil {
|
2017-09-14 09:36:06 +01:00
|
|
|
return plugin.Error("auto", err)
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
|
|
|
|
|
2017-08-22 14:21:42 +01:00
|
|
|
c.OnStartup(func() error {
|
|
|
|
|
m := dnsserver.GetConfig(c).Handler("prometheus")
|
2020-09-24 11:30:39 -07:00
|
|
|
if m != nil {
|
|
|
|
|
(&a).metrics = m.(*metrics.Metrics)
|
|
|
|
|
}
|
|
|
|
|
t := dnsserver.GetConfig(c).Handler("transfer")
|
|
|
|
|
if t != nil {
|
|
|
|
|
(&a).transfer = t.(*transfer.Transfer)
|
2017-08-22 14:21:42 +01:00
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
})
|
2016-10-26 10:01:52 +01:00
|
|
|
|
2016-10-17 18:37:56 +01:00
|
|
|
walkChan := make(chan bool)
|
|
|
|
|
|
|
|
|
|
c.OnStartup(func() error {
|
2016-10-26 10:01:52 +01:00
|
|
|
err := a.Walk()
|
2016-10-17 18:37:56 +01:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
go func() {
|
2019-04-01 14:28:01 +08:00
|
|
|
ticker := time.NewTicker(a.loader.ReloadInterval)
|
2016-10-17 18:37:56 +01:00
|
|
|
for {
|
|
|
|
|
select {
|
|
|
|
|
case <-walkChan:
|
|
|
|
|
return
|
|
|
|
|
case <-ticker.C:
|
2016-10-26 10:01:52 +01:00
|
|
|
a.Walk()
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
return nil
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
c.OnShutdown(func() error {
|
|
|
|
|
close(walkChan)
|
|
|
|
|
return nil
|
|
|
|
|
})
|
|
|
|
|
|
2017-09-14 09:36:06 +01:00
|
|
|
dnsserver.GetConfig(c).AddPlugin(func(next plugin.Handler) plugin.Handler {
|
2016-10-17 18:37:56 +01:00
|
|
|
a.Next = next
|
|
|
|
|
return a
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func autoParse(c *caddy.Controller) (Auto, error) {
|
2019-02-17 23:51:47 +09:00
|
|
|
nilInterval := -1 * time.Second
|
2016-10-17 18:37:56 +01:00
|
|
|
var a = Auto{
|
2019-02-17 23:51:47 +09:00
|
|
|
loader: loader{
|
|
|
|
|
template: "${1}",
|
|
|
|
|
re: regexp.MustCompile(`db\.(.*)`),
|
|
|
|
|
ReloadInterval: nilInterval,
|
|
|
|
|
},
|
|
|
|
|
Zones: &Zones{},
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
config := dnsserver.GetConfig(c)
|
|
|
|
|
|
|
|
|
|
for c.Next() {
|
2017-08-10 05:30:18 -07:00
|
|
|
// auto [ZONES...]
|
|
|
|
|
a.Zones.origins = make([]string, len(c.ServerBlockKeys))
|
|
|
|
|
copy(a.Zones.origins, c.ServerBlockKeys)
|
2016-10-17 18:37:56 +01:00
|
|
|
|
2017-08-10 05:30:18 -07:00
|
|
|
args := c.RemainingArgs()
|
|
|
|
|
if len(args) > 0 {
|
|
|
|
|
a.Zones.origins = args
|
|
|
|
|
}
|
|
|
|
|
for i := range a.Zones.origins {
|
2017-09-14 09:36:06 +01:00
|
|
|
a.Zones.origins[i] = plugin.Host(a.Zones.origins[i]).Normalize()
|
2017-08-10 05:30:18 -07:00
|
|
|
}
|
2019-07-02 16:23:47 +01:00
|
|
|
a.loader.upstream = upstream.New()
|
2016-10-17 18:37:56 +01:00
|
|
|
|
2017-08-10 05:30:18 -07:00
|
|
|
for c.NextBlock() {
|
|
|
|
|
switch c.Val() {
|
2019-04-01 14:28:01 +08:00
|
|
|
case "directory": // directory DIR [REGEXP TEMPLATE]
|
2017-08-10 05:30:18 -07:00
|
|
|
if !c.NextArg() {
|
|
|
|
|
return a, c.ArgErr()
|
|
|
|
|
}
|
|
|
|
|
a.loader.directory = c.Val()
|
2018-10-21 15:59:37 +02:00
|
|
|
if !filepath.IsAbs(a.loader.directory) && config.Root != "" {
|
|
|
|
|
a.loader.directory = filepath.Join(config.Root, a.loader.directory)
|
2017-08-10 05:30:18 -07:00
|
|
|
}
|
|
|
|
|
_, err := os.Stat(a.loader.directory)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if os.IsNotExist(err) {
|
2018-04-19 07:41:56 +01:00
|
|
|
log.Warningf("Directory does not exist: %s", a.loader.directory)
|
2017-08-10 05:30:18 -07:00
|
|
|
} else {
|
|
|
|
|
return a, c.Errf("Unable to access root path '%s': %v", a.loader.directory, err)
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
2017-08-10 05:30:18 -07:00
|
|
|
}
|
2016-10-17 18:37:56 +01:00
|
|
|
|
2019-06-13 19:07:41 +08:00
|
|
|
// regexp template
|
2017-08-10 05:30:18 -07:00
|
|
|
if c.NextArg() {
|
|
|
|
|
a.loader.re, err = regexp.Compile(c.Val())
|
|
|
|
|
if err != nil {
|
|
|
|
|
return a, err
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
2017-08-10 05:30:18 -07:00
|
|
|
if a.loader.re.NumSubexp() == 0 {
|
|
|
|
|
return a, c.Errf("Need at least one sub expression")
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
|
|
|
|
|
2019-06-13 19:07:41 +08:00
|
|
|
if !c.NextArg() {
|
|
|
|
|
return a, c.ArgErr()
|
|
|
|
|
}
|
2017-08-10 05:30:18 -07:00
|
|
|
a.loader.template = rewriteToExpand(c.Val())
|
|
|
|
|
}
|
2016-10-17 18:37:56 +01:00
|
|
|
|
2017-08-10 05:30:18 -07:00
|
|
|
if c.NextArg() {
|
2019-04-01 14:28:01 +08:00
|
|
|
return Auto{}, c.ArgErr()
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
|
|
|
|
|
2018-09-29 17:50:49 +02:00
|
|
|
case "reload":
|
|
|
|
|
d, err := time.ParseDuration(c.RemainingArgs()[0])
|
|
|
|
|
if err != nil {
|
|
|
|
|
return a, plugin.Error("file", err)
|
|
|
|
|
}
|
|
|
|
|
a.loader.ReloadInterval = d
|
|
|
|
|
|
2017-08-10 05:30:18 -07:00
|
|
|
case "upstream":
|
2019-07-02 16:23:47 +01:00
|
|
|
// remove soon
|
2019-01-13 16:54:49 +00:00
|
|
|
c.RemainingArgs() // eat remaining args
|
2017-08-10 05:30:18 -07:00
|
|
|
|
2019-04-01 14:28:01 +08:00
|
|
|
default:
|
|
|
|
|
return Auto{}, c.Errf("unknown property '%s'", c.Val())
|
2017-08-10 05:30:18 -07:00
|
|
|
}
|
2016-10-17 18:37:56 +01:00
|
|
|
}
|
|
|
|
|
}
|
2019-02-17 23:51:47 +09:00
|
|
|
|
|
|
|
|
if a.loader.ReloadInterval == nilInterval {
|
2019-04-01 14:28:01 +08:00
|
|
|
a.loader.ReloadInterval = 60 * time.Second
|
2019-02-17 23:51:47 +09:00
|
|
|
}
|
|
|
|
|
|
2016-10-17 18:37:56 +01:00
|
|
|
return a, nil
|
|
|
|
|
}
|