package home import ( "encoding/binary" "net" "strings" "time" "github.com/AdguardTeam/AdGuardHome/internal/dnsforward" "github.com/AdguardTeam/golibs/cache" "github.com/AdguardTeam/golibs/log" "github.com/miekg/dns" ) // RDNS - module context type RDNS struct { dnsServer *dnsforward.Server clients *clientsContainer ipChannel chan net.IP // pass data from DNS request handling thread to rDNS thread // Contains IP addresses of clients to be resolved by rDNS // If IP address is resolved, it stays here while it's inside Clients. // If it's removed from Clients, this IP address will be resolved once again. // If IP address couldn't be resolved, it stays here for some time to prevent further attempts to resolve the same IP. ipAddrs cache.Cache } // InitRDNS - create module context func InitRDNS(dnsServer *dnsforward.Server, clients *clientsContainer) *RDNS { r := &RDNS{ dnsServer: dnsServer, clients: clients, ipAddrs: cache.New(cache.Config{ EnableLRU: true, MaxCount: 10000, }), ipChannel: make(chan net.IP, 256), } go r.workerLoop() return r } // Begin - add IP address to rDNS queue func (r *RDNS) Begin(ip net.IP) { now := uint64(time.Now().Unix()) expire := r.ipAddrs.Get(ip) if len(expire) != 0 { exp := binary.BigEndian.Uint64(expire) if exp > now { return } // TTL expired } expire = make([]byte, 8) const ttl = 1 * 60 * 60 binary.BigEndian.PutUint64(expire, now+ttl) _ = r.ipAddrs.Set(ip, expire) id := ip.String() if r.clients.Exists(id, ClientSourceRDNS) { return } log.Tracef("rDNS: adding %s", ip) select { case r.ipChannel <- ip: // default: log.Tracef("rDNS: queue is full") } } // Use rDNS to get hostname by IP address func (r *RDNS) resolve(ip net.IP) string { log.Tracef("Resolving host for %s", ip) name, err := dns.ReverseAddr(ip.String()) if err != nil { log.Debug("Error while calling dns.ReverseAddr(%s): %s", ip, err) return "" } resp, err := r.dnsServer.Exchange(&dns.Msg{ MsgHdr: dns.MsgHdr{ Id: dns.Id(), RecursionDesired: true, }, Question: []dns.Question{{ Name: name, Qtype: dns.TypePTR, Qclass: dns.ClassINET, }}, }) if err != nil { log.Debug("Error while making an rDNS lookup for %s: %s", ip, err) return "" } if len(resp.Answer) == 0 { log.Debug("No answer for rDNS lookup of %s", ip) return "" } ptr, ok := resp.Answer[0].(*dns.PTR) if !ok { log.Debug("not a PTR response for %s", ip) return "" } log.Tracef("PTR response for %s: %s", ip, ptr.String()) if strings.HasSuffix(ptr.Ptr, ".") { ptr.Ptr = ptr.Ptr[:len(ptr.Ptr)-1] } return ptr.Ptr } // Wait for a signal and then synchronously resolve hostname by IP address // Add the hostname:IP pair to "Clients" array func (r *RDNS) workerLoop() { for { ip := <-r.ipChannel host := r.resolve(ip) if len(host) == 0 { continue } _, _ = r.clients.AddHost(ip.String(), host, ClientSourceRDNS) } }