mirror of
				https://github.com/coredns/coredns.git
				synced 2025-10-31 02:03:20 -04:00 
			
		
		
		
	Move all (almost all) Go files in middleware into their own packages. This makes for better naming and discoverability. Lot of changes elsewhere to make this change. The middleware.State was renamed to request.Request which is better, but still does not cover all use-cases. It was also moved out middleware because it is used by `dnsserver` as well. A pkg/dnsutil packages was added for shared, handy, dns util functions. All normalize functions are now put in normalize.go
		
			
				
	
	
		
			143 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			143 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package cache
 | |
| 
 | |
| import (
 | |
| 	"log"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/miekg/coredns/middleware"
 | |
| 	"github.com/miekg/coredns/middleware/pkg/response"
 | |
| 
 | |
| 	"github.com/miekg/dns"
 | |
| 	gcache "github.com/patrickmn/go-cache"
 | |
| )
 | |
| 
 | |
| // Cache is middleware that looks up responses in a cache and caches replies.
 | |
| type Cache struct {
 | |
| 	Next  middleware.Handler
 | |
| 	Zones []string
 | |
| 	cache *gcache.Cache
 | |
| 	cap   time.Duration
 | |
| }
 | |
| 
 | |
| func NewCache(ttl int, zones []string, next middleware.Handler) Cache {
 | |
| 	return Cache{Next: next, Zones: zones, cache: gcache.New(defaultDuration, purgeDuration), cap: time.Duration(ttl) * time.Second}
 | |
| }
 | |
| 
 | |
| func cacheKey(m *dns.Msg, t response.Type, do bool) string {
 | |
| 	if m.Truncated {
 | |
| 		return ""
 | |
| 	}
 | |
| 
 | |
| 	qtype := m.Question[0].Qtype
 | |
| 	qname := strings.ToLower(m.Question[0].Name)
 | |
| 	switch t {
 | |
| 	case response.Success:
 | |
| 		fallthrough
 | |
| 	case response.Delegation:
 | |
| 		return successKey(qname, qtype, do)
 | |
| 	case response.NameError:
 | |
| 		return nameErrorKey(qname, do)
 | |
| 	case response.NoData:
 | |
| 		return noDataKey(qname, qtype, do)
 | |
| 	case response.OtherError:
 | |
| 		return ""
 | |
| 	}
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| type CachingResponseWriter struct {
 | |
| 	dns.ResponseWriter
 | |
| 	cache *gcache.Cache
 | |
| 	cap   time.Duration
 | |
| }
 | |
| 
 | |
| func NewCachingResponseWriter(w dns.ResponseWriter, cache *gcache.Cache, cap time.Duration) *CachingResponseWriter {
 | |
| 	return &CachingResponseWriter{w, cache, cap}
 | |
| }
 | |
| 
 | |
| func (c *CachingResponseWriter) WriteMsg(res *dns.Msg) error {
 | |
| 	do := false
 | |
| 	mt, opt := response.Classify(res)
 | |
| 	if opt != nil {
 | |
| 		do = opt.Do()
 | |
| 	}
 | |
| 
 | |
| 	key := cacheKey(res, mt, do)
 | |
| 	c.set(res, key, mt)
 | |
| 
 | |
| 	if c.cap != 0 {
 | |
| 		setCap(res, uint32(c.cap.Seconds()))
 | |
| 	}
 | |
| 
 | |
| 	return c.ResponseWriter.WriteMsg(res)
 | |
| }
 | |
| 
 | |
| func (c *CachingResponseWriter) set(m *dns.Msg, key string, mt response.Type) {
 | |
| 	if key == "" {
 | |
| 		log.Printf("[ERROR] Caching called with empty cache key")
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	duration := c.cap
 | |
| 	switch mt {
 | |
| 	case response.Success, response.Delegation:
 | |
| 		if c.cap == 0 {
 | |
| 			duration = minTtl(m.Answer, mt)
 | |
| 		}
 | |
| 		i := newItem(m, duration)
 | |
| 
 | |
| 		c.cache.Set(key, i, duration)
 | |
| 	case response.NameError, response.NoData:
 | |
| 		if c.cap == 0 {
 | |
| 			duration = minTtl(m.Ns, mt)
 | |
| 		}
 | |
| 		i := newItem(m, duration)
 | |
| 
 | |
| 		c.cache.Set(key, i, duration)
 | |
| 	case response.OtherError:
 | |
| 		// don't cache these
 | |
| 	default:
 | |
| 		log.Printf("[WARNING] Caching called with unknown middleware MsgType: %d", mt)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (c *CachingResponseWriter) Write(buf []byte) (int, error) {
 | |
| 	log.Printf("[WARNING] Caching called with Write: not caching reply")
 | |
| 	n, err := c.ResponseWriter.Write(buf)
 | |
| 	return n, err
 | |
| }
 | |
| 
 | |
| func (c *CachingResponseWriter) Hijack() {
 | |
| 	c.ResponseWriter.Hijack()
 | |
| 	return
 | |
| }
 | |
| 
 | |
| func minTtl(rrs []dns.RR, mt response.Type) time.Duration {
 | |
| 	if mt != response.Success && mt != response.NameError && mt != response.NoData {
 | |
| 		return 0
 | |
| 	}
 | |
| 
 | |
| 	minTtl := maxTtl
 | |
| 	for _, r := range rrs {
 | |
| 		switch mt {
 | |
| 		case response.NameError, response.NoData:
 | |
| 			if r.Header().Rrtype == dns.TypeSOA {
 | |
| 				return time.Duration(r.(*dns.SOA).Minttl) * time.Second
 | |
| 			}
 | |
| 		case response.Success, response.Delegation:
 | |
| 			if r.Header().Ttl < minTtl {
 | |
| 				minTtl = r.Header().Ttl
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return time.Duration(minTtl) * time.Second
 | |
| }
 | |
| 
 | |
| const (
 | |
| 	purgeDuration          = 1 * time.Minute
 | |
| 	defaultDuration        = 20 * time.Minute
 | |
| 	baseTtl                = 5 // minimum ttl that we will allow
 | |
| 	maxTtl          uint32 = 2 * 3600
 | |
| )
 |