* plugin/trace: fix struct allignment A 64 bit entity needs to be the first in a struct to make it work on 32 bit systems. Fixes: #4111 Signed-off-by: Miek Gieben <miek@miek.nl> * plugin/erratic does makes the same mistake Signed-off-by: Miek Gieben <miek@miek.nl>
109 lines
2.1 KiB
Go
109 lines
2.1 KiB
Go
// Package erratic implements a plugin that returns erratic answers (delayed, dropped).
|
|
package erratic
|
|
|
|
import (
|
|
"context"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/coredns/coredns/request"
|
|
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
// Erratic is a plugin that returns erratic responses to each client.
|
|
type Erratic struct {
|
|
q uint64 // counter of queries
|
|
drop uint64
|
|
delay uint64
|
|
truncate uint64
|
|
|
|
duration time.Duration
|
|
large bool // undocumented feature; return large responses for A request (>512B, to test compression).
|
|
}
|
|
|
|
// ServeDNS implements the plugin.Handler interface.
|
|
func (e *Erratic) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
|
|
state := request.Request{W: w, Req: r}
|
|
drop := false
|
|
delay := false
|
|
trunc := false
|
|
|
|
queryNr := atomic.LoadUint64(&e.q)
|
|
atomic.AddUint64(&e.q, 1)
|
|
|
|
if e.drop > 0 && queryNr%e.drop == 0 {
|
|
drop = true
|
|
}
|
|
if e.delay > 0 && queryNr%e.delay == 0 {
|
|
delay = true
|
|
}
|
|
if e.truncate > 0 && queryNr&e.truncate == 0 {
|
|
trunc = true
|
|
}
|
|
|
|
m := new(dns.Msg)
|
|
m.SetReply(r)
|
|
m.Authoritative = true
|
|
if trunc {
|
|
m.Truncated = true
|
|
}
|
|
|
|
// small dance to copy rrA or rrAAAA into a non-pointer var that allows us to overwrite the ownername
|
|
// in a non-racy way.
|
|
switch state.QType() {
|
|
case dns.TypeA:
|
|
rr := *(rrA.(*dns.A))
|
|
rr.Header().Name = state.QName()
|
|
m.Answer = append(m.Answer, &rr)
|
|
if e.large {
|
|
for i := 0; i < 29; i++ {
|
|
m.Answer = append(m.Answer, &rr)
|
|
}
|
|
}
|
|
case dns.TypeAAAA:
|
|
rr := *(rrAAAA.(*dns.AAAA))
|
|
rr.Header().Name = state.QName()
|
|
m.Answer = append(m.Answer, &rr)
|
|
case dns.TypeAXFR:
|
|
if drop {
|
|
return 0, nil
|
|
}
|
|
if delay {
|
|
time.Sleep(e.duration)
|
|
}
|
|
|
|
xfr(state, trunc)
|
|
return 0, nil
|
|
|
|
default:
|
|
if drop {
|
|
return 0, nil
|
|
}
|
|
if delay {
|
|
time.Sleep(e.duration)
|
|
}
|
|
// coredns will return error.
|
|
return dns.RcodeServerFailure, nil
|
|
}
|
|
|
|
if drop {
|
|
return 0, nil
|
|
}
|
|
|
|
if delay {
|
|
time.Sleep(e.duration)
|
|
}
|
|
|
|
w.WriteMsg(m)
|
|
|
|
return 0, nil
|
|
}
|
|
|
|
// Name implements the Handler interface.
|
|
func (e *Erratic) Name() string { return "erratic" }
|
|
|
|
var (
|
|
rrA, _ = dns.NewRR(". IN 0 A 192.0.2.53")
|
|
rrAAAA, _ = dns.NewRR(". IN 0 AAAA 2001:DB8::53")
|
|
)
|