Adds the dnstap I/O routines and should fix some issues (#1083)
* adds the dnstap I/O thread and should fix a lot of mistakes * docs * -race test * oops * docs
This commit is contained in:
parent
2a32cd4159
commit
daf8ef0da8
7 changed files with 164 additions and 22 deletions
69
plugin/dnstap/dnstapio/io.go
Normal file
69
plugin/dnstap/dnstapio/io.go
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
package dnstapio
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
tap "github.com/dnstap/golang-dnstap"
|
||||||
|
"github.com/golang/protobuf/proto"
|
||||||
|
)
|
||||||
|
|
||||||
|
// DnstapIO wraps the dnstap I/O routine.
|
||||||
|
type DnstapIO struct {
|
||||||
|
writer io.WriteCloser
|
||||||
|
queue chan tap.Dnstap
|
||||||
|
stop chan bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// Protocol is either `out.TCP` or `out.Socket`.
|
||||||
|
type Protocol interface {
|
||||||
|
// Write takes a single frame at once.
|
||||||
|
Write([]byte) (int, error)
|
||||||
|
|
||||||
|
Close() error
|
||||||
|
}
|
||||||
|
|
||||||
|
// New dnstap I/O routine from Protocol.
|
||||||
|
func New(w Protocol) *DnstapIO {
|
||||||
|
dio := DnstapIO{}
|
||||||
|
dio.writer = w
|
||||||
|
dio.queue = make(chan tap.Dnstap, 10)
|
||||||
|
dio.stop = make(chan bool)
|
||||||
|
go dio.serve()
|
||||||
|
return &dio
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dnstap enqueues the payload for log.
|
||||||
|
func (dio *DnstapIO) Dnstap(payload tap.Dnstap) {
|
||||||
|
select {
|
||||||
|
case dio.queue <- payload:
|
||||||
|
default:
|
||||||
|
fmt.Println("[WARN] Dnstap payload dropped.")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dio *DnstapIO) serve() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case payload := <-dio.queue:
|
||||||
|
frame, err := proto.Marshal(&payload)
|
||||||
|
if err == nil {
|
||||||
|
dio.writer.Write(frame)
|
||||||
|
} else {
|
||||||
|
fmt.Println("[ERROR] Invalid dnstap payload dropped.")
|
||||||
|
}
|
||||||
|
case <-dio.stop:
|
||||||
|
close(dio.queue)
|
||||||
|
dio.stop <- true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close waits until the I/O routine is finished to return.
|
||||||
|
func (dio DnstapIO) Close() error {
|
||||||
|
dio.stop <- true
|
||||||
|
<-dio.stop
|
||||||
|
close(dio.stop)
|
||||||
|
return dio.writer.Close()
|
||||||
|
}
|
71
plugin/dnstap/dnstapio/io_test.go
Normal file
71
plugin/dnstap/dnstapio/io_test.go
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
package dnstapio
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"sync"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
tap "github.com/dnstap/golang-dnstap"
|
||||||
|
)
|
||||||
|
|
||||||
|
type buf struct {
|
||||||
|
*bytes.Buffer
|
||||||
|
cost time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b buf) Write(frame []byte) (int, error) {
|
||||||
|
time.Sleep(b.cost)
|
||||||
|
return b.Buffer.Write(frame)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b buf) Close() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRace(t *testing.T) {
|
||||||
|
b := buf{&bytes.Buffer{}, 100 * time.Millisecond}
|
||||||
|
dio := New(b)
|
||||||
|
wg := &sync.WaitGroup{}
|
||||||
|
wg.Add(10)
|
||||||
|
for i := 0; i < 10; i++ {
|
||||||
|
timeout := time.After(time.Second)
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-timeout:
|
||||||
|
wg.Done()
|
||||||
|
return
|
||||||
|
default:
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
dio.Dnstap(tap.Dnstap{})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
wg.Wait()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestClose(t *testing.T) {
|
||||||
|
done := make(chan bool)
|
||||||
|
var dio *DnstapIO
|
||||||
|
go func() {
|
||||||
|
b := buf{&bytes.Buffer{}, 0}
|
||||||
|
dio = New(b)
|
||||||
|
dio.Close()
|
||||||
|
close(done)
|
||||||
|
}()
|
||||||
|
select {
|
||||||
|
case <-done:
|
||||||
|
case <-time.After(time.Second):
|
||||||
|
t.Fatal("Not closing.")
|
||||||
|
}
|
||||||
|
func() {
|
||||||
|
defer func() {
|
||||||
|
if err := recover(); err == nil {
|
||||||
|
t.Fatal("Send on closed channel.")
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
dio.Dnstap(tap.Dnstap{})
|
||||||
|
}()
|
||||||
|
}
|
|
@ -16,11 +16,15 @@ import (
|
||||||
// Dnstap is the dnstap handler.
|
// Dnstap is the dnstap handler.
|
||||||
type Dnstap struct {
|
type Dnstap struct {
|
||||||
Next plugin.Handler
|
Next plugin.Handler
|
||||||
Out io.Writer
|
IO IORoutine
|
||||||
Pack bool
|
Pack bool
|
||||||
}
|
}
|
||||||
|
|
||||||
type (
|
type (
|
||||||
|
// IORoutine is the dnstap I/O thread as defined by: <http://dnstap.info/Architecture>.
|
||||||
|
IORoutine interface {
|
||||||
|
Dnstap(tap.Dnstap)
|
||||||
|
}
|
||||||
// Tapper is implemented by the Context passed by the dnstap handler.
|
// Tapper is implemented by the Context passed by the dnstap handler.
|
||||||
Tapper interface {
|
Tapper interface {
|
||||||
TapMessage(*tap.Message) error
|
TapMessage(*tap.Message) error
|
||||||
|
@ -49,7 +53,8 @@ func tapMessageTo(w io.Writer, m *tap.Message) error {
|
||||||
|
|
||||||
// TapMessage implements Tapper.
|
// TapMessage implements Tapper.
|
||||||
func (h Dnstap) TapMessage(m *tap.Message) error {
|
func (h Dnstap) TapMessage(m *tap.Message) error {
|
||||||
return tapMessageTo(h.Out, m)
|
h.IO.Dnstap(msg.Wrap(m))
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// TapBuilder implements Tapper.
|
// TapBuilder implements Tapper.
|
||||||
|
|
|
@ -1,21 +1,18 @@
|
||||||
package dnstap
|
package dnstap
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/coredns/coredns/plugin/dnstap/test"
|
"github.com/coredns/coredns/plugin/dnstap/test"
|
||||||
mwtest "github.com/coredns/coredns/plugin/test"
|
mwtest "github.com/coredns/coredns/plugin/test"
|
||||||
|
|
||||||
tap "github.com/dnstap/golang-dnstap"
|
tap "github.com/dnstap/golang-dnstap"
|
||||||
"github.com/golang/protobuf/proto"
|
|
||||||
"github.com/miekg/dns"
|
"github.com/miekg/dns"
|
||||||
"golang.org/x/net/context"
|
"golang.org/x/net/context"
|
||||||
)
|
)
|
||||||
|
|
||||||
func testCase(t *testing.T, tapq, tapr *tap.Message, q, r *dns.Msg) {
|
func testCase(t *testing.T, tapq, tapr *tap.Message, q, r *dns.Msg) {
|
||||||
w := writer{}
|
w := writer{t: t}
|
||||||
w.queue = append(w.queue, tapq, tapr)
|
w.queue = append(w.queue, tapq, tapr)
|
||||||
h := Dnstap{
|
h := Dnstap{
|
||||||
Next: mwtest.HandlerFunc(func(_ context.Context,
|
Next: mwtest.HandlerFunc(func(_ context.Context,
|
||||||
|
@ -23,7 +20,7 @@ func testCase(t *testing.T, tapq, tapr *tap.Message, q, r *dns.Msg) {
|
||||||
|
|
||||||
return 0, w.WriteMsg(r)
|
return 0, w.WriteMsg(r)
|
||||||
}),
|
}),
|
||||||
Out: &w,
|
IO: &w,
|
||||||
Pack: false,
|
Pack: false,
|
||||||
}
|
}
|
||||||
_, err := h.ServeDNS(context.TODO(), &mwtest.ResponseWriter{}, q)
|
_, err := h.ServeDNS(context.TODO(), &mwtest.ResponseWriter{}, q)
|
||||||
|
@ -33,22 +30,18 @@ func testCase(t *testing.T, tapq, tapr *tap.Message, q, r *dns.Msg) {
|
||||||
}
|
}
|
||||||
|
|
||||||
type writer struct {
|
type writer struct {
|
||||||
|
t *testing.T
|
||||||
queue []*tap.Message
|
queue []*tap.Message
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *writer) Write(b []byte) (int, error) {
|
func (w *writer) Dnstap(e tap.Dnstap) {
|
||||||
e := tap.Dnstap{}
|
|
||||||
if err := proto.Unmarshal(b, &e); err != nil {
|
|
||||||
return 0, err
|
|
||||||
}
|
|
||||||
if len(w.queue) == 0 {
|
if len(w.queue) == 0 {
|
||||||
return 0, errors.New("message not expected")
|
w.t.Error("Message not expected.")
|
||||||
}
|
}
|
||||||
if !test.MsgEqual(w.queue[0], e.Message) {
|
if !test.MsgEqual(w.queue[0], e.Message) {
|
||||||
return 0, fmt.Errorf("want: %v, have: %v", w.queue[0], e.Message)
|
w.t.Errorf("want: %v, have: %v", w.queue[0], e.Message)
|
||||||
}
|
}
|
||||||
w.queue = w.queue[1:]
|
w.queue = w.queue[1:]
|
||||||
return len(b), nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDnstap(t *testing.T) {
|
func TestDnstap(t *testing.T) {
|
||||||
|
|
|
@ -7,9 +7,10 @@ import (
|
||||||
"github.com/golang/protobuf/proto"
|
"github.com/golang/protobuf/proto"
|
||||||
)
|
)
|
||||||
|
|
||||||
func wrap(m *lib.Message) *lib.Dnstap {
|
// Wrap a dnstap message in the top-level dnstap type.
|
||||||
|
func Wrap(m *lib.Message) lib.Dnstap {
|
||||||
t := lib.Dnstap_MESSAGE
|
t := lib.Dnstap_MESSAGE
|
||||||
return &lib.Dnstap{
|
return lib.Dnstap{
|
||||||
Type: &t,
|
Type: &t,
|
||||||
Message: m,
|
Message: m,
|
||||||
}
|
}
|
||||||
|
@ -17,7 +18,8 @@ func wrap(m *lib.Message) *lib.Dnstap {
|
||||||
|
|
||||||
// Marshal encodes the message to a binary dnstap payload.
|
// Marshal encodes the message to a binary dnstap payload.
|
||||||
func Marshal(m *lib.Message) (data []byte, err error) {
|
func Marshal(m *lib.Message) (data []byte, err error) {
|
||||||
data, err = proto.Marshal(wrap(m))
|
payload := Wrap(m)
|
||||||
|
data, err = proto.Marshal(&payload)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = fmt.Errorf("proto: %s", err)
|
err = fmt.Errorf("proto: %s", err)
|
||||||
return
|
return
|
||||||
|
|
|
@ -32,7 +32,7 @@ func (s *TCP) Write(frame []byte) (n int, err error) {
|
||||||
// Flush the remaining frames.
|
// Flush the remaining frames.
|
||||||
func (s *TCP) Flush() error {
|
func (s *TCP) Flush() error {
|
||||||
defer func() {
|
defer func() {
|
||||||
s.frames = s.frames[0:]
|
s.frames = s.frames[:0]
|
||||||
}()
|
}()
|
||||||
c, err := net.DialTimeout("tcp", s.address, time.Second)
|
c, err := net.DialTimeout("tcp", s.address, time.Second)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -8,6 +8,7 @@ import (
|
||||||
|
|
||||||
"github.com/coredns/coredns/core/dnsserver"
|
"github.com/coredns/coredns/core/dnsserver"
|
||||||
"github.com/coredns/coredns/plugin"
|
"github.com/coredns/coredns/plugin"
|
||||||
|
"github.com/coredns/coredns/plugin/dnstap/dnstapio"
|
||||||
"github.com/coredns/coredns/plugin/dnstap/out"
|
"github.com/coredns/coredns/plugin/dnstap/out"
|
||||||
"github.com/coredns/coredns/plugin/pkg/dnsutil"
|
"github.com/coredns/coredns/plugin/pkg/dnsutil"
|
||||||
|
|
||||||
|
@ -79,11 +80,12 @@ func setup(c *caddy.Controller) error {
|
||||||
} else {
|
} else {
|
||||||
o = out.NewTCP(conf.target)
|
o = out.NewTCP(conf.target)
|
||||||
}
|
}
|
||||||
dnstap.Out = o
|
dio := dnstapio.New(o)
|
||||||
|
dnstap.IO = dio
|
||||||
|
|
||||||
c.OnShutdown(func() error {
|
c.OnShutdown(func() error {
|
||||||
if err := o.Close(); err != nil {
|
if err := dio.Close(); err != nil {
|
||||||
return fmt.Errorf("output: %s", err)
|
return fmt.Errorf("dnstap io routine: %s", err)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
Loading…
Add table
Reference in a new issue