mirror of
				https://github.com/traefik/traefik.git
				synced 2025-10-26 22:11:40 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			134 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package requestdecorator
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"net"
 | |
| 	"sort"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/miekg/dns"
 | |
| 	"github.com/patrickmn/go-cache"
 | |
| 	"github.com/rs/zerolog/log"
 | |
| )
 | |
| 
 | |
| type cnameResolv struct {
 | |
| 	TTL    time.Duration
 | |
| 	Record string
 | |
| }
 | |
| 
 | |
| type byTTL []*cnameResolv
 | |
| 
 | |
| func (a byTTL) Len() int           { return len(a) }
 | |
| func (a byTTL) Swap(i, j int)      { a[i], a[j] = a[j], a[i] }
 | |
| func (a byTTL) Less(i, j int) bool { return a[i].TTL > a[j].TTL }
 | |
| 
 | |
| // Resolver used for host resolver.
 | |
| type Resolver struct {
 | |
| 	CnameFlattening bool
 | |
| 	ResolvConfig    string
 | |
| 	ResolvDepth     int
 | |
| 	cache           *cache.Cache
 | |
| }
 | |
| 
 | |
| // CNAMEFlatten check if CNAME record exists, flatten if possible.
 | |
| func (hr *Resolver) CNAMEFlatten(ctx context.Context, host string) string {
 | |
| 	if hr.cache == nil {
 | |
| 		hr.cache = cache.New(30*time.Minute, 5*time.Minute)
 | |
| 	}
 | |
| 
 | |
| 	result := host
 | |
| 	request := host
 | |
| 
 | |
| 	value, found := hr.cache.Get(host)
 | |
| 	if found {
 | |
| 		return value.(string)
 | |
| 	}
 | |
| 
 | |
| 	logger := log.Ctx(ctx)
 | |
| 	cacheDuration := 0 * time.Second
 | |
| 	for depth := range hr.ResolvDepth {
 | |
| 		resolv, err := cnameResolve(ctx, request, hr.ResolvConfig)
 | |
| 		if err != nil {
 | |
| 			logger.Error().Err(err).Send()
 | |
| 			break
 | |
| 		}
 | |
| 		if resolv == nil {
 | |
| 			break
 | |
| 		}
 | |
| 
 | |
| 		result = resolv.Record
 | |
| 		if depth == 0 {
 | |
| 			cacheDuration = resolv.TTL
 | |
| 		}
 | |
| 		request = resolv.Record
 | |
| 	}
 | |
| 
 | |
| 	hr.cache.Set(host, result, cacheDuration)
 | |
| 
 | |
| 	return result
 | |
| }
 | |
| 
 | |
| // cnameResolve resolves CNAME if exists, and return with the highest TTL.
 | |
| func cnameResolve(ctx context.Context, host, resolvPath string) (*cnameResolv, error) {
 | |
| 	config, err := dns.ClientConfigFromFile(resolvPath)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("invalid resolver configuration file: %s", resolvPath)
 | |
| 	}
 | |
| 
 | |
| 	if net.ParseIP(host) != nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	client := &dns.Client{Timeout: 30 * time.Second}
 | |
| 
 | |
| 	m := &dns.Msg{}
 | |
| 	m.SetQuestion(dns.Fqdn(host), dns.TypeCNAME)
 | |
| 
 | |
| 	var result []*cnameResolv
 | |
| 	for _, server := range config.Servers {
 | |
| 		tempRecord, err := getRecord(client, m, server, config.Port)
 | |
| 		if err != nil {
 | |
| 			if errors.Is(err, errNoCNAMERecord) {
 | |
| 				log.Ctx(ctx).Debug().Err(err).Msgf("CNAME lookup for hostname %q", host)
 | |
| 				continue
 | |
| 			}
 | |
| 			log.Ctx(ctx).Error().Err(err).Msgf("CNAME lookup for hostname %q", host)
 | |
| 			continue
 | |
| 		}
 | |
| 		result = append(result, tempRecord)
 | |
| 	}
 | |
| 
 | |
| 	if len(result) == 0 {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	sort.Sort(byTTL(result))
 | |
| 	return result[0], nil
 | |
| }
 | |
| 
 | |
| var errNoCNAMERecord = errors.New("no CNAME record for host")
 | |
| 
 | |
| func getRecord(client *dns.Client, msg *dns.Msg, server, port string) (*cnameResolv, error) {
 | |
| 	resp, _, err := client.Exchange(msg, net.JoinHostPort(server, port))
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("exchange error for server %s: %w", server, err)
 | |
| 	}
 | |
| 
 | |
| 	if resp == nil || len(resp.Answer) == 0 {
 | |
| 		return nil, fmt.Errorf("%w: %s", errNoCNAMERecord, server)
 | |
| 	}
 | |
| 
 | |
| 	rr, ok := resp.Answer[0].(*dns.CNAME)
 | |
| 	if !ok {
 | |
| 		return nil, fmt.Errorf("invalid response type for server %s", server)
 | |
| 	}
 | |
| 
 | |
| 	return &cnameResolv{
 | |
| 		TTL:    time.Duration(rr.Hdr.Ttl) * time.Second,
 | |
| 		Record: strings.TrimSuffix(rr.Target, "."),
 | |
| 	}, nil
 | |
| }
 |