This checks if the next middleware to be called is nil, and if so returns ServerFailure and an error. This makes the next calling more robust and saves some lines of code. Also prefix the error with the name of the middleware to aid in debugging.
96 lines
2.4 KiB
Go
96 lines
2.4 KiB
Go
package cache
|
|
|
|
import (
|
|
"time"
|
|
|
|
"github.com/miekg/coredns/middleware"
|
|
"github.com/miekg/coredns/request"
|
|
|
|
"github.com/miekg/dns"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"golang.org/x/net/context"
|
|
)
|
|
|
|
// ServeDNS implements the middleware.Handler interface.
|
|
func (c *Cache) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
|
|
state := request.Request{W: w, Req: r}
|
|
|
|
qname := state.Name()
|
|
qtype := state.QType()
|
|
zone := middleware.Zones(c.Zones).Matches(qname)
|
|
if zone == "" {
|
|
return c.Next.ServeDNS(ctx, w, r)
|
|
}
|
|
|
|
do := state.Do() // TODO(): might need more from OPT record? Like the actual bufsize?
|
|
|
|
if i, ok, expired := c.get(qname, qtype, do); ok && !expired {
|
|
|
|
resp := i.toMsg(r)
|
|
state.SizeAndDo(resp)
|
|
w.WriteMsg(resp)
|
|
|
|
return dns.RcodeSuccess, nil
|
|
}
|
|
|
|
crr := &ResponseWriter{w, c}
|
|
return middleware.NextOrFailure(c.Name(), c.Next, ctx, crr, r)
|
|
}
|
|
|
|
// Name implements the Handler interface.
|
|
func (c *Cache) Name() string { return "cache" }
|
|
|
|
func (c *Cache) get(qname string, qtype uint16, do bool) (*item, bool, bool) {
|
|
k := rawKey(qname, qtype, do)
|
|
|
|
if i, ok := c.ncache.Get(k); ok {
|
|
cacheHits.WithLabelValues(Denial).Inc()
|
|
return i.(*item), ok, i.(*item).expired(time.Now())
|
|
}
|
|
|
|
if i, ok := c.pcache.Get(k); ok {
|
|
cacheHits.WithLabelValues(Success).Inc()
|
|
return i.(*item), ok, i.(*item).expired(time.Now())
|
|
}
|
|
cacheMisses.Inc()
|
|
return nil, false, false
|
|
}
|
|
|
|
var (
|
|
cacheSize = prometheus.NewGaugeVec(prometheus.GaugeOpts{
|
|
Namespace: middleware.Namespace,
|
|
Subsystem: subsystem,
|
|
Name: "size",
|
|
Help: "The number of elements in the cache.",
|
|
}, []string{"type"})
|
|
|
|
cacheCapacity = prometheus.NewGaugeVec(prometheus.GaugeOpts{
|
|
Namespace: middleware.Namespace,
|
|
Subsystem: subsystem,
|
|
Name: "capacity",
|
|
Help: "The cache's capacity.",
|
|
}, []string{"type"})
|
|
|
|
cacheHits = prometheus.NewCounterVec(prometheus.CounterOpts{
|
|
Namespace: middleware.Namespace,
|
|
Subsystem: subsystem,
|
|
Name: "hits_total",
|
|
Help: "The count of cache hits.",
|
|
}, []string{"type"})
|
|
|
|
cacheMisses = prometheus.NewCounter(prometheus.CounterOpts{
|
|
Namespace: middleware.Namespace,
|
|
Subsystem: subsystem,
|
|
Name: "misses_total",
|
|
Help: "The count of cache misses.",
|
|
})
|
|
)
|
|
|
|
const subsystem = "cache"
|
|
|
|
func init() {
|
|
prometheus.MustRegister(cacheSize)
|
|
prometheus.MustRegister(cacheCapacity)
|
|
prometheus.MustRegister(cacheHits)
|
|
prometheus.MustRegister(cacheMisses)
|
|
}
|