2018-08-30 17:25:33 +03:00
package dnsfilter
import (
"bufio"
"errors"
"fmt"
"log"
"net"
"os"
"strconv"
"strings"
"sync"
"time"
2018-10-15 14:31:21 +03:00
"github.com/AdguardTeam/AdguardDNS/dnsfilter"
2018-08-30 17:25:33 +03:00
"github.com/coredns/coredns/core/dnsserver"
"github.com/coredns/coredns/plugin"
"github.com/coredns/coredns/plugin/metrics"
"github.com/coredns/coredns/plugin/pkg/dnstest"
"github.com/coredns/coredns/plugin/pkg/upstream"
"github.com/coredns/coredns/request"
"github.com/mholt/caddy"
"github.com/miekg/dns"
"github.com/prometheus/client_golang/prometheus"
"golang.org/x/net/context"
)
var defaultSOA = & dns . SOA {
// values copied from verisign's nonexistent .com domain
// their exact values are not important in our use case because they are used for domain transfers between primary/secondary DNS servers
Refresh : 1800 ,
Retry : 900 ,
Expire : 604800 ,
Minttl : 86400 ,
}
func init ( ) {
caddy . RegisterPlugin ( "dnsfilter" , caddy . Plugin {
ServerType : "dns" ,
Action : setup ,
} )
}
2018-10-10 19:10:38 +03:00
type cacheEntry struct {
answer [ ] dns . RR
lastUpdated time . Time
}
var (
lookupCacheTime = time . Minute * 30
lookupCache = map [ string ] cacheEntry { }
)
2018-10-07 21:24:22 +03:00
type plugSettings struct {
SafeBrowsingBlockHost string
ParentalBlockHost string
QueryLogEnabled bool
BlockedTTL uint32 // in seconds, default 3600
}
2018-09-14 16:50:56 +03:00
type plug struct {
2018-08-30 17:25:33 +03:00
d * dnsfilter . Dnsfilter
Next plugin . Handler
upstream upstream . Upstream
hosts map [ string ] net . IP
2018-10-07 21:24:22 +03:00
settings plugSettings
2018-10-07 00:58:59 +03:00
sync . RWMutex
2018-08-30 17:25:33 +03:00
}
2018-10-07 21:24:22 +03:00
var defaultPluginSettings = plugSettings {
2018-08-30 17:25:33 +03:00
SafeBrowsingBlockHost : "safebrowsing.block.dns.adguard.com" ,
ParentalBlockHost : "family.block.dns.adguard.com" ,
2018-09-26 18:27:31 +03:00
BlockedTTL : 3600 , // in seconds
2018-08-30 17:25:33 +03:00
}
//
// coredns handling functions
//
2018-09-14 16:50:56 +03:00
func setupPlugin ( c * caddy . Controller ) ( * plug , error ) {
2018-08-30 17:25:33 +03:00
// create new Plugin and copy default values
2018-10-07 21:24:22 +03:00
p := & plug {
settings : defaultPluginSettings ,
d : dnsfilter . New ( ) ,
hosts : make ( map [ string ] net . IP ) ,
}
2018-08-30 17:25:33 +03:00
2018-09-25 19:26:26 +03:00
filterFileNames := [ ] string { }
2018-08-30 17:25:33 +03:00
for c . Next ( ) {
args := c . RemainingArgs ( )
2018-09-25 19:26:26 +03:00
if len ( args ) > 0 {
filterFileNames = append ( filterFileNames , args ... )
2018-08-30 17:25:33 +03:00
}
for c . NextBlock ( ) {
switch c . Val ( ) {
case "safebrowsing" :
2018-09-14 16:50:56 +03:00
p . d . EnableSafeBrowsing ( )
2018-08-30 17:25:33 +03:00
if c . NextArg ( ) {
if len ( c . Val ( ) ) == 0 {
return nil , c . ArgErr ( )
}
2018-09-14 16:50:56 +03:00
p . d . SetSafeBrowsingServer ( c . Val ( ) )
2018-08-30 17:25:33 +03:00
}
case "safesearch" :
2018-09-14 16:50:56 +03:00
p . d . EnableSafeSearch ( )
2018-08-30 17:25:33 +03:00
case "parental" :
if ! c . NextArg ( ) {
return nil , c . ArgErr ( )
}
sensitivity , err := strconv . Atoi ( c . Val ( ) )
if err != nil {
return nil , c . ArgErr ( )
}
2018-09-14 16:50:56 +03:00
err = p . d . EnableParental ( sensitivity )
2018-08-30 17:25:33 +03:00
if err != nil {
return nil , c . ArgErr ( )
}
if c . NextArg ( ) {
if len ( c . Val ( ) ) == 0 {
return nil , c . ArgErr ( )
}
2018-10-07 21:24:22 +03:00
p . settings . ParentalBlockHost = c . Val ( )
2018-08-30 17:25:33 +03:00
}
2018-09-26 18:27:31 +03:00
case "blocked_ttl" :
if ! c . NextArg ( ) {
return nil , c . ArgErr ( )
}
blockttl , err := strconv . ParseUint ( c . Val ( ) , 10 , 32 )
if err != nil {
return nil , c . ArgErr ( )
}
2018-10-07 21:24:22 +03:00
p . settings . BlockedTTL = uint32 ( blockttl )
2018-08-30 17:25:33 +03:00
case "querylog" :
2018-10-07 21:24:22 +03:00
p . settings . QueryLogEnabled = true
2018-08-30 17:25:33 +03:00
}
}
}
2018-09-25 19:26:26 +03:00
log . Printf ( "filterFileNames = %+v" , filterFileNames )
for i , filterFileName := range filterFileNames {
file , err := os . Open ( filterFileName )
2018-08-30 17:25:33 +03:00
if err != nil {
return nil , err
}
2018-09-25 19:26:26 +03:00
defer file . Close ( )
count := 0
scanner := bufio . NewScanner ( file )
for scanner . Scan ( ) {
text := scanner . Text ( )
if p . parseEtcHosts ( text ) {
continue
}
err = p . d . AddRule ( text , uint32 ( i ) )
if err == dnsfilter . ErrInvalidSyntax {
continue
}
if err != nil {
return nil , err
}
count ++
}
log . Printf ( "Added %d rules from %s" , count , filterFileName )
2018-08-30 17:25:33 +03:00
2018-09-25 19:26:26 +03:00
if err = scanner . Err ( ) ; err != nil {
return nil , err
}
2018-08-30 17:25:33 +03:00
}
2018-10-09 04:45:05 +03:00
log . Printf ( "Loading stats from querylog" )
err := fillStatsFromQueryLog ( )
2018-10-07 23:24:04 +03:00
if err != nil {
2018-10-09 04:45:05 +03:00
log . Printf ( "Failed to load stats from querylog: %s" , err )
2018-10-07 23:24:04 +03:00
return nil , err
}
if p . settings . QueryLogEnabled {
onceQueryLog . Do ( func ( ) {
go startQueryLogServer ( ) // TODO: how to handle errors?
} )
}
2018-10-11 20:52:23 +03:00
onceHook . Do ( func ( ) {
caddy . RegisterEventHook ( "dnsfilter-reload" , hook )
} )
2018-09-14 16:50:56 +03:00
p . upstream , err = upstream . New ( nil )
2018-08-30 17:25:33 +03:00
if err != nil {
return nil , err
}
2018-09-14 16:50:56 +03:00
return p , nil
2018-08-30 17:25:33 +03:00
}
func setup ( c * caddy . Controller ) error {
2018-09-14 16:50:56 +03:00
p , err := setupPlugin ( c )
2018-08-30 17:25:33 +03:00
if err != nil {
return err
}
config := dnsserver . GetConfig ( c )
config . AddPlugin ( func ( next plugin . Handler ) plugin . Handler {
2018-09-14 16:50:56 +03:00
p . Next = next
return p
2018-08-30 17:25:33 +03:00
} )
c . OnStartup ( func ( ) error {
2018-09-06 02:07:57 +03:00
m := dnsserver . GetConfig ( c ) . Handler ( "prometheus" )
if m == nil {
return nil
}
if x , ok := m . ( * metrics . Metrics ) ; ok {
x . MustRegister ( requests )
x . MustRegister ( filtered )
x . MustRegister ( filteredLists )
x . MustRegister ( filteredSafebrowsing )
x . MustRegister ( filteredParental )
x . MustRegister ( whitelisted )
x . MustRegister ( safesearch )
x . MustRegister ( errorsTotal )
2018-10-09 04:45:05 +03:00
x . MustRegister ( elapsedTime )
2018-09-14 16:50:56 +03:00
x . MustRegister ( p )
2018-09-06 02:07:57 +03:00
}
2018-08-30 17:25:33 +03:00
return nil
} )
2018-09-14 16:50:56 +03:00
c . OnShutdown ( p . onShutdown )
2018-10-07 00:51:44 +03:00
c . OnFinalShutdown ( p . onFinalShutdown )
2018-08-30 17:25:33 +03:00
return nil
}
2018-09-14 16:50:56 +03:00
func ( p * plug ) parseEtcHosts ( text string ) bool {
2018-08-30 17:25:33 +03:00
if pos := strings . IndexByte ( text , '#' ) ; pos != - 1 {
text = text [ 0 : pos ]
}
fields := strings . Fields ( text )
if len ( fields ) < 2 {
return false
}
addr := net . ParseIP ( fields [ 0 ] )
if addr == nil {
return false
}
for _ , host := range fields [ 1 : ] {
2018-10-11 18:33:07 +03:00
// debug logging for duplicate values, pretty common if you subscribe to many hosts files
// if val, ok := p.hosts[host]; ok {
// log.Printf("warning: host %s already has value %s, will overwrite it with %s", host, val, addr)
// }
2018-09-14 16:50:56 +03:00
p . hosts [ host ] = addr
2018-08-30 17:25:33 +03:00
}
return true
}
2018-09-14 16:50:56 +03:00
func ( p * plug ) onShutdown ( ) error {
2018-10-07 00:58:59 +03:00
p . Lock ( )
2018-09-14 16:50:56 +03:00
p . d . Destroy ( )
p . d = nil
2018-10-07 00:58:59 +03:00
p . Unlock ( )
2018-08-30 17:25:33 +03:00
return nil
}
2018-10-07 00:51:44 +03:00
func ( p * plug ) onFinalShutdown ( ) error {
2018-10-07 00:58:59 +03:00
logBufferLock . Lock ( )
2018-10-07 00:51:44 +03:00
err := flushToFile ( logBuffer )
if err != nil {
log . Printf ( "failed to flush to file: %s" , err )
return err
}
2018-10-07 00:58:59 +03:00
logBufferLock . Unlock ( )
2018-10-07 00:51:44 +03:00
return nil
}
2018-08-30 17:25:33 +03:00
type statsFunc func ( ch interface { } , name string , text string , value float64 , valueType prometheus . ValueType )
func doDesc ( ch interface { } , name string , text string , value float64 , valueType prometheus . ValueType ) {
realch , ok := ch . ( chan <- * prometheus . Desc )
2018-09-14 16:50:56 +03:00
if ! ok {
2018-08-30 17:25:33 +03:00
log . Printf ( "Couldn't convert ch to chan<- *prometheus.Desc\n" )
return
}
realch <- prometheus . NewDesc ( name , text , nil , nil )
}
func doMetric ( ch interface { } , name string , text string , value float64 , valueType prometheus . ValueType ) {
realch , ok := ch . ( chan <- prometheus . Metric )
2018-09-14 16:50:56 +03:00
if ! ok {
2018-08-30 17:25:33 +03:00
log . Printf ( "Couldn't convert ch to chan<- prometheus.Metric\n" )
return
}
desc := prometheus . NewDesc ( name , text , nil , nil )
realch <- prometheus . MustNewConstMetric ( desc , valueType , value )
}
func gen ( ch interface { } , doFunc statsFunc , name string , text string , value float64 , valueType prometheus . ValueType ) {
doFunc ( ch , name , text , value , valueType )
}
2018-09-07 16:10:11 +03:00
func doStatsLookup ( ch interface { } , doFunc statsFunc , name string , lookupstats * dnsfilter . LookupStats ) {
gen ( ch , doFunc , fmt . Sprintf ( "coredns_dnsfilter_%s_requests" , name ) , fmt . Sprintf ( "Number of %s HTTP requests that were sent" , name ) , float64 ( lookupstats . Requests ) , prometheus . CounterValue )
gen ( ch , doFunc , fmt . Sprintf ( "coredns_dnsfilter_%s_cachehits" , name ) , fmt . Sprintf ( "Number of %s lookups that didn't need HTTP requests" , name ) , float64 ( lookupstats . CacheHits ) , prometheus . CounterValue )
gen ( ch , doFunc , fmt . Sprintf ( "coredns_dnsfilter_%s_pending" , name ) , fmt . Sprintf ( "Number of currently pending %s HTTP requests" , name ) , float64 ( lookupstats . Pending ) , prometheus . GaugeValue )
gen ( ch , doFunc , fmt . Sprintf ( "coredns_dnsfilter_%s_pending_max" , name ) , fmt . Sprintf ( "Maximum number of pending %s HTTP requests" , name ) , float64 ( lookupstats . PendingMax ) , prometheus . GaugeValue )
}
2018-09-14 16:50:56 +03:00
func ( p * plug ) doStats ( ch interface { } , doFunc statsFunc ) {
2018-10-07 00:58:59 +03:00
p . RLock ( )
2018-09-14 16:50:56 +03:00
stats := p . d . GetStats ( )
2018-09-07 16:10:11 +03:00
doStatsLookup ( ch , doFunc , "safebrowsing" , & stats . Safebrowsing )
doStatsLookup ( ch , doFunc , "parental" , & stats . Parental )
2018-10-07 00:58:59 +03:00
p . RUnlock ( )
2018-08-30 17:25:33 +03:00
}
2018-09-14 16:50:56 +03:00
// Describe is called by prometheus handler to know stat types
func ( p * plug ) Describe ( ch chan <- * prometheus . Desc ) {
p . doStats ( ch , doDesc )
2018-08-30 17:25:33 +03:00
}
2018-09-14 16:50:56 +03:00
// Collect is called by prometheus handler to collect stats
func ( p * plug ) Collect ( ch chan <- prometheus . Metric ) {
p . doStats ( ch , doMetric )
2018-08-30 17:25:33 +03:00
}
2018-09-14 16:50:56 +03:00
func ( p * plug ) replaceHostWithValAndReply ( ctx context . Context , w dns . ResponseWriter , r * dns . Msg , host string , val string , question dns . Question ) ( int , error ) {
2018-08-30 17:25:33 +03:00
// check if it's a domain name or IP address
addr := net . ParseIP ( val )
var records [ ] dns . RR
2018-10-11 18:33:07 +03:00
// log.Println("Will give", val, "instead of", host) // debug logging
2018-08-30 17:25:33 +03:00
if addr != nil {
// this is an IP address, return it
2018-10-07 21:24:22 +03:00
result , err := dns . NewRR ( fmt . Sprintf ( "%s %d A %s" , host , p . settings . BlockedTTL , val ) )
2018-08-30 17:25:33 +03:00
if err != nil {
log . Printf ( "Got error %s\n" , err )
return dns . RcodeServerFailure , fmt . Errorf ( "plugin/dnsfilter: %s" , err )
}
records = append ( records , result )
} else {
// this is a domain name, need to look it up
2018-10-10 19:10:38 +03:00
cacheentry := lookupCache [ val ]
if time . Since ( cacheentry . lastUpdated ) > lookupCacheTime {
req := new ( dns . Msg )
req . SetQuestion ( dns . Fqdn ( val ) , question . Qtype )
req . RecursionDesired = true
reqstate := request . Request { W : w , Req : req , Context : ctx }
result , err := p . upstream . Lookup ( reqstate , dns . Fqdn ( val ) , reqstate . QType ( ) )
if err != nil {
log . Printf ( "Got error %s\n" , err )
return dns . RcodeServerFailure , fmt . Errorf ( "plugin/dnsfilter: %s" , err )
}
if result != nil {
for _ , answer := range result . Answer {
answer . Header ( ) . Name = question . Name
}
records = result . Answer
cacheentry . answer = result . Answer
cacheentry . lastUpdated = time . Now ( )
lookupCache [ val ] = cacheentry
2018-08-30 17:25:33 +03:00
}
2018-10-10 19:10:38 +03:00
} else {
// get from cache
records = cacheentry . answer
2018-08-30 17:25:33 +03:00
}
}
m := new ( dns . Msg )
m . SetReply ( r )
m . Authoritative , m . RecursionAvailable , m . Compress = true , true , true
m . Answer = append ( m . Answer , records ... )
state := request . Request { W : w , Req : r , Context : ctx }
state . SizeAndDo ( m )
err := state . W . WriteMsg ( m )
if err != nil {
log . Printf ( "Got error %s\n" , err )
return dns . RcodeServerFailure , fmt . Errorf ( "plugin/dnsfilter: %s" , err )
}
return dns . RcodeSuccess , nil
}
// generate SOA record that makes DNS clients cache NXdomain results
// the only value that is important is TTL in header, other values like refresh, retry, expire and minttl are irrelevant
2018-09-26 18:27:31 +03:00
func ( p * plug ) genSOA ( r * dns . Msg ) [ ] dns . RR {
2018-08-30 17:25:33 +03:00
zone := r . Question [ 0 ] . Name
2018-10-07 21:24:22 +03:00
header := dns . RR_Header { Name : zone , Rrtype : dns . TypeSOA , Ttl : p . settings . BlockedTTL , Class : dns . ClassINET }
2018-08-30 17:25:33 +03:00
Mbox := "hostmaster."
if zone [ 0 ] != '.' {
Mbox += zone
}
Ns := "fake-for-negative-caching.adguard.com."
2018-10-07 00:58:59 +03:00
soa := * defaultSOA
2018-08-30 17:25:33 +03:00
soa . Hdr = header
soa . Mbox = Mbox
soa . Ns = Ns
2018-10-07 00:58:59 +03:00
soa . Serial = 100500 // faster than uint32(time.Now().Unix())
return [ ] dns . RR { & soa }
2018-08-30 17:25:33 +03:00
}
2018-09-26 18:27:31 +03:00
func ( p * plug ) writeNXdomain ( ctx context . Context , w dns . ResponseWriter , r * dns . Msg ) ( int , error ) {
2018-08-30 17:25:33 +03:00
state := request . Request { W : w , Req : r , Context : ctx }
m := new ( dns . Msg )
m . SetRcode ( state . Req , dns . RcodeNameError )
m . Authoritative , m . RecursionAvailable , m . Compress = true , true , true
2018-09-26 18:27:31 +03:00
m . Ns = p . genSOA ( r )
2018-08-30 17:25:33 +03:00
state . SizeAndDo ( m )
err := state . W . WriteMsg ( m )
if err != nil {
log . Printf ( "Got error %s\n" , err )
return dns . RcodeServerFailure , err
}
return dns . RcodeNameError , nil
}
2018-09-14 16:50:56 +03:00
func ( p * plug ) serveDNSInternal ( ctx context . Context , w dns . ResponseWriter , r * dns . Msg ) ( int , dnsfilter . Result , error ) {
2018-08-30 17:25:33 +03:00
if len ( r . Question ) != 1 {
// google DNS, bind and others do the same
2018-09-14 16:50:56 +03:00
return dns . RcodeFormatError , dnsfilter . Result { } , fmt . Errorf ( "Got DNS request with != 1 questions" )
2018-08-30 17:25:33 +03:00
}
for _ , question := range r . Question {
host := strings . ToLower ( strings . TrimSuffix ( question . Name , "." ) )
// is it a safesearch domain?
2018-10-07 00:58:59 +03:00
p . RLock ( )
2018-09-14 16:50:56 +03:00
if val , ok := p . d . SafeSearchDomain ( host ) ; ok {
rcode , err := p . replaceHostWithValAndReply ( ctx , w , r , host , val , question )
2018-08-30 17:25:33 +03:00
if err != nil {
2018-10-07 00:58:59 +03:00
p . RUnlock ( )
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { } , err
2018-08-30 17:25:33 +03:00
}
2018-10-07 00:58:59 +03:00
p . RUnlock ( )
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { Reason : dnsfilter . FilteredSafeSearch } , err
2018-08-30 17:25:33 +03:00
}
2018-10-07 00:58:59 +03:00
p . RUnlock ( )
2018-08-30 17:25:33 +03:00
// is it in hosts?
2018-09-14 16:50:56 +03:00
if val , ok := p . hosts [ host ] ; ok {
2018-08-30 17:25:33 +03:00
// it is, if it's a loopback host, reply with NXDOMAIN
2018-09-25 18:34:01 +03:00
// TODO: research if it's better than 127.0.0.1
if false && val . IsLoopback ( ) {
2018-09-26 18:27:31 +03:00
rcode , err := p . writeNXdomain ( ctx , w , r )
2018-08-30 17:25:33 +03:00
if err != nil {
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { } , err
2018-08-30 17:25:33 +03:00
}
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { Reason : dnsfilter . FilteredInvalid } , err
2018-08-30 17:25:33 +03:00
}
// it's not a loopback host, replace it with value specified
2018-09-14 16:50:56 +03:00
rcode , err := p . replaceHostWithValAndReply ( ctx , w , r , host , val . String ( ) , question )
2018-08-30 17:25:33 +03:00
if err != nil {
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { } , err
2018-08-30 17:25:33 +03:00
}
2018-10-15 14:31:21 +03:00
// TODO: This must be handled in the dnsfilter and not here!
rule := val . String ( ) + " " + host
return rcode , dnsfilter . Result { Reason : dnsfilter . FilteredBlackList , Rule : rule } , err
2018-08-30 17:25:33 +03:00
}
// needs to be filtered instead
2018-10-07 00:58:59 +03:00
p . RLock ( )
2018-09-14 16:50:56 +03:00
result , err := p . d . CheckHost ( host )
2018-08-30 17:25:33 +03:00
if err != nil {
log . Printf ( "plugin/dnsfilter: %s\n" , err )
2018-10-07 00:58:59 +03:00
p . RUnlock ( )
2018-09-14 16:50:56 +03:00
return dns . RcodeServerFailure , dnsfilter . Result { } , fmt . Errorf ( "plugin/dnsfilter: %s" , err )
2018-08-30 17:25:33 +03:00
}
2018-10-07 00:58:59 +03:00
p . RUnlock ( )
2018-08-30 17:25:33 +03:00
2018-09-06 02:09:57 +03:00
if result . IsFiltered {
switch result . Reason {
case dnsfilter . FilteredSafeBrowsing :
// return cname safebrowsing.block.dns.adguard.com
2018-10-07 21:24:22 +03:00
val := p . settings . SafeBrowsingBlockHost
2018-09-14 16:50:56 +03:00
rcode , err := p . replaceHostWithValAndReply ( ctx , w , r , host , val , question )
2018-09-06 02:09:57 +03:00
if err != nil {
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { } , err
2018-09-06 02:09:57 +03:00
}
2018-09-14 16:50:56 +03:00
return rcode , result , err
2018-09-06 02:09:57 +03:00
case dnsfilter . FilteredParental :
// return cname family.block.dns.adguard.com
2018-10-07 21:24:22 +03:00
val := p . settings . ParentalBlockHost
2018-09-14 16:50:56 +03:00
rcode , err := p . replaceHostWithValAndReply ( ctx , w , r , host , val , question )
2018-09-06 02:09:57 +03:00
if err != nil {
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { } , err
2018-09-06 02:09:57 +03:00
}
2018-09-14 16:50:56 +03:00
return rcode , result , err
2018-09-06 02:09:57 +03:00
case dnsfilter . FilteredBlackList :
// return NXdomain
2018-09-26 18:27:31 +03:00
rcode , err := p . writeNXdomain ( ctx , w , r )
2018-09-06 02:09:57 +03:00
if err != nil {
2018-09-14 16:50:56 +03:00
return rcode , dnsfilter . Result { } , err
2018-09-06 02:09:57 +03:00
}
2018-09-14 16:50:56 +03:00
return rcode , result , err
2018-10-05 07:31:56 +03:00
case dnsfilter . FilteredInvalid :
// return NXdomain
rcode , err := p . writeNXdomain ( ctx , w , r )
if err != nil {
return rcode , dnsfilter . Result { } , err
}
return rcode , result , err
2018-09-06 02:09:57 +03:00
default :
2018-10-05 07:31:56 +03:00
log . Printf ( "SHOULD NOT HAPPEN -- got unknown reason for filtering host \"%s\": %v, %+v" , host , result . Reason , result )
2018-08-30 17:25:33 +03:00
}
2018-09-06 02:09:57 +03:00
} else {
switch result . Reason {
case dnsfilter . NotFilteredWhiteList :
2018-09-14 16:50:56 +03:00
rcode , err := plugin . NextOrFailure ( p . Name ( ) , p . Next , ctx , w , r )
return rcode , result , err
2018-09-06 02:09:57 +03:00
case dnsfilter . NotFilteredNotFound :
// do nothing, pass through to lower code
default :
2018-10-05 07:31:56 +03:00
log . Printf ( "SHOULD NOT HAPPEN -- got unknown reason for not filtering host \"%s\": %v, %+v" , host , result . Reason , result )
2018-08-30 17:25:33 +03:00
}
}
}
2018-09-14 16:50:56 +03:00
rcode , err := plugin . NextOrFailure ( p . Name ( ) , p . Next , ctx , w , r )
return rcode , dnsfilter . Result { } , err
2018-08-30 17:25:33 +03:00
}
2018-09-14 16:50:56 +03:00
// ServeDNS handles the DNS request and refuses if it's in filterlists
func ( p * plug ) ServeDNS ( ctx context . Context , w dns . ResponseWriter , r * dns . Msg ) ( int , error ) {
2018-08-30 17:25:33 +03:00
start := time . Now ( )
requests . Inc ( )
2018-08-31 19:59:04 +03:00
state := request . Request { W : w , Req : r }
ip := state . IP ( )
2018-08-30 17:25:33 +03:00
// capture the written answer
rrw := dnstest . NewRecorder ( w )
2018-09-14 16:50:56 +03:00
rcode , result , err := p . serveDNSInternal ( ctx , rrw , r )
2018-08-30 17:25:33 +03:00
if rcode > 0 {
// actually send the answer if we have one
answer := new ( dns . Msg )
answer . SetRcode ( r , rcode )
state . SizeAndDo ( answer )
2018-09-14 16:50:56 +03:00
err = w . WriteMsg ( answer )
if err != nil {
return dns . RcodeServerFailure , err
}
2018-08-30 17:25:33 +03:00
}
// increment counters
switch {
case err != nil :
errorsTotal . Inc ( )
case result . Reason == dnsfilter . FilteredBlackList :
filtered . Inc ( )
filteredLists . Inc ( )
case result . Reason == dnsfilter . FilteredSafeBrowsing :
filtered . Inc ( )
filteredSafebrowsing . Inc ( )
case result . Reason == dnsfilter . FilteredParental :
filtered . Inc ( )
filteredParental . Inc ( )
case result . Reason == dnsfilter . FilteredInvalid :
filtered . Inc ( )
filteredInvalid . Inc ( )
case result . Reason == dnsfilter . FilteredSafeSearch :
// the request was passsed through but not filtered, don't increment filtered
safesearch . Inc ( )
case result . Reason == dnsfilter . NotFilteredWhiteList :
whitelisted . Inc ( )
case result . Reason == dnsfilter . NotFilteredNotFound :
// do nothing
case result . Reason == dnsfilter . NotFilteredError :
text := "SHOULD NOT HAPPEN: got DNSFILTER_NOTFILTERED_ERROR without err != nil!"
log . Println ( text )
err = errors . New ( text )
rcode = dns . RcodeServerFailure
}
// log
2018-10-09 04:45:05 +03:00
elapsed := time . Since ( start )
elapsedTime . Observe ( elapsed . Seconds ( ) )
2018-10-07 21:24:22 +03:00
if p . settings . QueryLogEnabled {
2018-09-05 21:21:46 +03:00
logRequest ( r , rrw . Msg , result , time . Since ( start ) , ip )
2018-08-30 17:25:33 +03:00
}
return rcode , err
}
2018-09-14 16:50:56 +03:00
// Name returns name of the plugin as seen in Corefile and plugin.cfg
func ( p * plug ) Name ( ) string { return "dnsfilter" }
2018-08-30 17:25:33 +03:00
2018-10-11 20:52:23 +03:00
var onceHook sync . Once
2018-09-06 02:07:57 +03:00
var onceQueryLog sync . Once