// Package auto implements an on-the-fly loading file backend. package auto import ( "errors" "regexp" "time" "github.com/coredns/coredns/middleware" "github.com/coredns/coredns/middleware/file" "github.com/coredns/coredns/middleware/metrics" "github.com/coredns/coredns/middleware/proxy" "github.com/coredns/coredns/request" "github.com/miekg/dns" "golang.org/x/net/context" ) type ( // Auto holds the zones and the loader configuration for automatically loading zones. Auto struct { Next middleware.Handler *Zones metrics *metrics.Metrics loader } loader struct { directory string template string re *regexp.Regexp // In the future this should be something like ZoneMeta that contains all this stuff. transferTo []string noReload bool proxy proxy.Proxy // Proxy for looking up names during the resolution process duration time.Duration } ) // ServeDNS implements the middleware.Handle interface. func (a Auto) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) { state := request.Request{W: w, Req: r} if state.QClass() != dns.ClassINET { return dns.RcodeServerFailure, middleware.Error(a.Name(), errors.New("can only deal with ClassINET")) } qname := state.Name() // TODO(miek): match the qname better in the map // Precheck with the origins, i.e. are we allowed to looks here. zone := middleware.Zones(a.Zones.Origins()).Matches(qname) if zone == "" { return middleware.NextOrFailure(a.Name(), a.Next, ctx, w, r) } // Now the real zone. zone = middleware.Zones(a.Zones.Names()).Matches(qname) a.Zones.RLock() z, ok := a.Zones.Z[zone] a.Zones.RUnlock() if !ok || z == nil { return dns.RcodeServerFailure, nil } if state.QType() == dns.TypeAXFR || state.QType() == dns.TypeIXFR { xfr := file.Xfr{Zone: z} return xfr.ServeDNS(ctx, w, r) } answer, ns, extra, result := z.Lookup(state, qname) m := new(dns.Msg) m.SetReply(r) m.Authoritative, m.RecursionAvailable, m.Compress = true, true, true m.Answer, m.Ns, m.Extra = answer, ns, extra switch result { case file.Success: case file.NoData: case file.NameError: m.Rcode = dns.RcodeNameError case file.Delegation: m.Authoritative = false case file.ServerFailure: return dns.RcodeServerFailure, nil } state.SizeAndDo(m) m, _ = state.Scrub(m) w.WriteMsg(m) return dns.RcodeSuccess, nil } // Name implements the Handler interface. func (a Auto) Name() string { return "auto" }