| 
									
										
										
										
											2016-10-17 18:37:56 +01:00
										 |  |  | package auto | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2021-11-20 00:19:25 +08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2016-10-17 18:37:56 +01:00
										 |  |  | 	"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 | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-02-15 13:25:02 -05:00
										 |  |  | 		if err := a.Notify(); err != nil { | 
					
						
							|  |  |  | 			log.Warning(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-11-20 00:19:25 +08:00
										 |  |  | 		if a.loader.ReloadInterval == 0 { | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-10-17 18:37:56 +01:00
										 |  |  | 		go func() { | 
					
						
							| 
									
										
										
										
											2019-04-01 14:28:01 +08:00
										 |  |  | 			ticker := time.NewTicker(a.loader.ReloadInterval) | 
					
						
							| 
									
										
										
										
											2022-10-11 09:27:20 -04:00
										 |  |  | 			defer ticker.Stop() | 
					
						
							| 
									
										
										
										
											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() | 
					
						
							| 
									
										
										
										
											2023-02-15 13:25:02 -05:00
										 |  |  | 					if err := a.Notify(); err != nil { | 
					
						
							|  |  |  | 						log.Warning(err) | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											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...] | 
					
						
							|  |  |  | 		args := c.RemainingArgs() | 
					
						
							| 
									
										
										
										
											2021-05-17 22:19:54 +02:00
										 |  |  | 		a.Zones.origins = plugin.OriginsFromArgsOrServerBlock(args, c.ServerBlockKeys) | 
					
						
							| 
									
										
										
										
											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": | 
					
						
							| 
									
										
										
										
											2021-11-20 00:19:25 +08:00
										 |  |  | 				t := c.RemainingArgs() | 
					
						
							|  |  |  | 				if len(t) < 1 { | 
					
						
							|  |  |  | 					return a, errors.New("reload duration value is expected") | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				d, err := time.ParseDuration(t[0]) | 
					
						
							|  |  |  | 				if d < 0 { | 
					
						
							|  |  |  | 					err = errors.New("invalid duration") | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-09-29 17:50:49 +02:00
										 |  |  | 				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 | 
					
						
							|  |  |  | } |