mirror of
https://github.com/coredns/coredns.git
synced 2025-10-28 08:44:17 -04:00
Dnstap plugin refactoring (#1257)
This commit is contained in:
committed by
Miek Gieben
parent
06006fac56
commit
6d6e1357b9
@@ -2,67 +2,100 @@ package dnstapio
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"log"
|
"log"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
|
||||||
tap "github.com/dnstap/golang-dnstap"
|
tap "github.com/dnstap/golang-dnstap"
|
||||||
|
fs "github.com/farsightsec/golang-framestream"
|
||||||
"github.com/golang/protobuf/proto"
|
"github.com/golang/protobuf/proto"
|
||||||
)
|
)
|
||||||
|
|
||||||
// DnstapIO wraps the dnstap I/O routine.
|
const (
|
||||||
type DnstapIO struct {
|
tcpTimeout = 4 * time.Second
|
||||||
protocol Protocol
|
flushTimeout = 1 * time.Second
|
||||||
|
queueSize = 1000
|
||||||
|
)
|
||||||
|
|
||||||
|
type dnstapIO struct {
|
||||||
|
enc *fs.Encoder
|
||||||
|
conn net.Conn
|
||||||
queue chan tap.Dnstap
|
queue chan tap.Dnstap
|
||||||
stop chan bool
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Protocol is either `out.TCP` or `out.Socket`.
|
// New returns a new and initialized DnstapIO.
|
||||||
type Protocol interface {
|
func New() DnstapIO {
|
||||||
// Write takes a single frame at once.
|
return &dnstapIO{queue: make(chan tap.Dnstap, queueSize)}
|
||||||
Write([]byte) (int, error)
|
|
||||||
|
|
||||||
Close() error
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// New dnstap I/O routine from Protocol.
|
// DnstapIO interface
|
||||||
func New(w Protocol) *DnstapIO {
|
type DnstapIO interface {
|
||||||
dio := DnstapIO{}
|
Connect(endpoint string, socket bool) error
|
||||||
dio.protocol = w
|
Dnstap(payload tap.Dnstap)
|
||||||
dio.queue = make(chan tap.Dnstap, 10)
|
Close()
|
||||||
dio.stop = make(chan bool)
|
}
|
||||||
|
|
||||||
|
// Connect connects to the dnstop endpoint.
|
||||||
|
func (dio *dnstapIO) Connect(endpoint string, socket bool) error {
|
||||||
|
var err error
|
||||||
|
if socket {
|
||||||
|
dio.conn, err = net.Dial("unix", endpoint)
|
||||||
|
} else {
|
||||||
|
dio.conn, err = net.DialTimeout("tcp", endpoint, tcpTimeout)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
dio.enc, err = fs.NewEncoder(dio.conn, &fs.EncoderOptions{
|
||||||
|
ContentType: []byte("protobuf:dnstap.Dnstap"),
|
||||||
|
Bidirectional: true,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
go dio.serve()
|
go dio.serve()
|
||||||
return &dio
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Dnstap enqueues the payload for log.
|
// Dnstap enqueues the payload for log.
|
||||||
func (dio *DnstapIO) Dnstap(payload tap.Dnstap) {
|
func (dio *dnstapIO) Dnstap(payload tap.Dnstap) {
|
||||||
select {
|
select {
|
||||||
case dio.queue <- payload:
|
case dio.queue <- payload:
|
||||||
default:
|
default:
|
||||||
log.Println("[WARN] Dnstap payload dropped.")
|
log.Printf("[ERROR] Dnstap payload dropped")
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (dio *DnstapIO) serve() {
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case payload := <-dio.queue:
|
|
||||||
frame, err := proto.Marshal(&payload)
|
|
||||||
if err == nil {
|
|
||||||
dio.protocol.Write(frame)
|
|
||||||
} else {
|
|
||||||
log.Printf("[ERROR] Invalid dnstap payload dropped: %s\n", err)
|
|
||||||
}
|
|
||||||
case <-dio.stop:
|
|
||||||
close(dio.queue)
|
|
||||||
dio.stop <- true
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close waits until the I/O routine is finished to return.
|
// Close waits until the I/O routine is finished to return.
|
||||||
func (dio DnstapIO) Close() error {
|
func (dio *dnstapIO) Close() {
|
||||||
dio.stop <- true
|
close(dio.queue)
|
||||||
<-dio.stop
|
}
|
||||||
close(dio.stop)
|
|
||||||
return dio.protocol.Close()
|
func (dio *dnstapIO) serve() {
|
||||||
|
timeout := time.After(flushTimeout)
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case payload, ok := <-dio.queue:
|
||||||
|
if !ok {
|
||||||
|
dio.enc.Close()
|
||||||
|
dio.conn.Close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
frame, err := proto.Marshal(&payload)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERROR] Invalid dnstap payload dropped: %s", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
_, err = dio.enc.Write(frame)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERROR] Cannot write dnstap payload: %s", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
case <-timeout:
|
||||||
|
err := dio.enc.Flush()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERROR] Cannot flush dnstap payloads: %s", err)
|
||||||
|
}
|
||||||
|
timeout = time.After(flushTimeout)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,78 +1,155 @@
|
|||||||
package dnstapio
|
package dnstapio
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"net"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
tap "github.com/dnstap/golang-dnstap"
|
tap "github.com/dnstap/golang-dnstap"
|
||||||
|
fs "github.com/farsightsec/golang-framestream"
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
func accept(t *testing.T, l net.Listener, count int) {
|
||||||
log.SetOutput(ioutil.Discard)
|
server, err := l.Accept()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("server accept: %s", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
type buf struct {
|
dec, err := fs.NewDecoder(server, &fs.DecoderOptions{
|
||||||
*bytes.Buffer
|
ContentType: []byte("protobuf:dnstap.Dnstap"),
|
||||||
cost time.Duration
|
Bidirectional: true,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("server decoder: %s", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b buf) Write(frame []byte) (int, error) {
|
for i := 0; i < count; i++ {
|
||||||
time.Sleep(b.cost)
|
if _, err := dec.Decode(); err != nil {
|
||||||
return b.Buffer.Write(frame)
|
t.Errorf("server decode: %s", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b buf) Close() error {
|
if err := server.Close(); err != nil {
|
||||||
return nil
|
t.Error(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const endpointTCP = "localhost:0"
|
||||||
|
|
||||||
|
func TestTCP(t *testing.T) {
|
||||||
|
dio := New()
|
||||||
|
|
||||||
|
err := dio.Connect(endpointTCP, false)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Not listening but no error")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start TCP listener
|
||||||
|
l, err := net.Listen("tcp", endpointTCP)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Cannot start listener: %s", err)
|
||||||
|
}
|
||||||
|
defer l.Close()
|
||||||
|
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
wg.Add(1)
|
||||||
|
go func() {
|
||||||
|
accept(t, l, 1)
|
||||||
|
wg.Done()
|
||||||
|
}()
|
||||||
|
|
||||||
|
err = dio.Connect(l.Addr().String(), false)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Cannot connect to listener: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
msg := tap.Dnstap_MESSAGE
|
||||||
|
dio.Dnstap(tap.Dnstap{Type: &msg})
|
||||||
|
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
|
dio.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
const endpointSocket = "dnstap.sock"
|
||||||
|
|
||||||
|
func TestSocket(t *testing.T) {
|
||||||
|
dio := New()
|
||||||
|
|
||||||
|
err := dio.Connect(endpointSocket, true)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Not listening but no error")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start Socket listener
|
||||||
|
l, err := net.Listen("unix", endpointSocket)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Cannot start listener: %s", err)
|
||||||
|
}
|
||||||
|
defer l.Close()
|
||||||
|
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
wg.Add(1)
|
||||||
|
go func() {
|
||||||
|
accept(t, l, 1)
|
||||||
|
wg.Done()
|
||||||
|
}()
|
||||||
|
|
||||||
|
err = dio.Connect(endpointSocket, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Cannot connect to listener: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
msg := tap.Dnstap_MESSAGE
|
||||||
|
dio.Dnstap(tap.Dnstap{Type: &msg})
|
||||||
|
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
|
dio.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRace(t *testing.T) {
|
func TestRace(t *testing.T) {
|
||||||
b := buf{&bytes.Buffer{}, 100 * time.Millisecond}
|
count := 10
|
||||||
dio := New(b)
|
dio := New()
|
||||||
wg := &sync.WaitGroup{}
|
|
||||||
wg.Add(10)
|
err := dio.Connect(endpointTCP, false)
|
||||||
for i := 0; i < 10; i++ {
|
if err == nil {
|
||||||
timeout := time.After(time.Second)
|
t.Fatal("Not listening but no error")
|
||||||
go func() {
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case <-timeout:
|
|
||||||
wg.Done()
|
|
||||||
return
|
|
||||||
default:
|
|
||||||
time.Sleep(50 * time.Millisecond)
|
|
||||||
t := tap.Dnstap_MESSAGE
|
|
||||||
dio.Dnstap(tap.Dnstap{Type: &t})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
|
||||||
wg.Wait()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestClose(t *testing.T) {
|
// Start TCP listener
|
||||||
done := make(chan bool)
|
l, err := net.Listen("tcp", endpointTCP)
|
||||||
var dio *DnstapIO
|
if err != nil {
|
||||||
|
t.Fatalf("Cannot start listener: %s", err)
|
||||||
|
}
|
||||||
|
defer l.Close()
|
||||||
|
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
wg.Add(1)
|
||||||
go func() {
|
go func() {
|
||||||
b := buf{&bytes.Buffer{}, 0}
|
accept(t, l, count)
|
||||||
dio = New(b)
|
wg.Done()
|
||||||
|
}()
|
||||||
|
|
||||||
|
err = dio.Connect(l.Addr().String(), false)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Cannot connect to listener: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
msg := tap.Dnstap_MESSAGE
|
||||||
|
wg.Add(count)
|
||||||
|
for i := 0; i < count; i++ {
|
||||||
|
go func(i byte) {
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
dio.Dnstap(tap.Dnstap{Type: &msg, Extra: []byte{i}})
|
||||||
|
wg.Done()
|
||||||
|
}(byte(i))
|
||||||
|
}
|
||||||
|
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
dio.Close()
|
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{})
|
|
||||||
}()
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,82 +0,0 @@
|
|||||||
package out
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
|
|
||||||
fs "github.com/farsightsec/golang-framestream"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Socket is a Frame Streams encoder over a UNIX socket.
|
|
||||||
type Socket struct {
|
|
||||||
path string
|
|
||||||
enc *fs.Encoder
|
|
||||||
conn net.Conn
|
|
||||||
err error
|
|
||||||
}
|
|
||||||
|
|
||||||
func openSocket(s *Socket) error {
|
|
||||||
conn, err := net.Dial("unix", s.path)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
s.conn = conn
|
|
||||||
|
|
||||||
enc, err := fs.NewEncoder(conn, &fs.EncoderOptions{
|
|
||||||
ContentType: []byte("protobuf:dnstap.Dnstap"),
|
|
||||||
Bidirectional: true,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
s.enc = enc
|
|
||||||
|
|
||||||
s.err = nil
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewSocket will always return a new Socket.
|
|
||||||
// err if nothing is listening to it, it will attempt to reconnect on the next Write.
|
|
||||||
func NewSocket(path string) (s *Socket, err error) {
|
|
||||||
s = &Socket{path: path}
|
|
||||||
if err = openSocket(s); err != nil {
|
|
||||||
err = fmt.Errorf("open socket: %s", err)
|
|
||||||
s.err = err
|
|
||||||
return
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write a single Frame Streams frame.
|
|
||||||
func (s *Socket) Write(frame []byte) (int, error) {
|
|
||||||
if s.err != nil {
|
|
||||||
// is the dnstap tool listening?
|
|
||||||
if err := openSocket(s); err != nil {
|
|
||||||
return 0, fmt.Errorf("open socket: %s", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
n, err := s.enc.Write(frame)
|
|
||||||
if err != nil {
|
|
||||||
// the dnstap command line tool is down
|
|
||||||
s.conn.Close()
|
|
||||||
s.err = err
|
|
||||||
return 0, err
|
|
||||||
}
|
|
||||||
return n, nil
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close the socket and flush the remaining frames.
|
|
||||||
func (s *Socket) Close() error {
|
|
||||||
if s.err != nil {
|
|
||||||
// nothing to close
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
defer s.conn.Close()
|
|
||||||
|
|
||||||
if err := s.enc.Flush(); err != nil {
|
|
||||||
return fmt.Errorf("flush: %s", err)
|
|
||||||
}
|
|
||||||
return s.enc.Close()
|
|
||||||
}
|
|
||||||
@@ -1,94 +0,0 @@
|
|||||||
package out
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
fs "github.com/farsightsec/golang-framestream"
|
|
||||||
)
|
|
||||||
|
|
||||||
func acceptOne(t *testing.T, l net.Listener) {
|
|
||||||
server, err := l.Accept()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("server accept: %s", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
dec, err := fs.NewDecoder(server, &fs.DecoderOptions{
|
|
||||||
ContentType: []byte("protobuf:dnstap.Dnstap"),
|
|
||||||
Bidirectional: true,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("server decoder: %s", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if _, err := dec.Decode(); err != nil {
|
|
||||||
t.Errorf("server decode: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := server.Close(); err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
func sendOne(socket *Socket) error {
|
|
||||||
if _, err := socket.Write([]byte("frame")); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := socket.enc.Flush(); err != nil {
|
|
||||||
// Would happen during Write in real life.
|
|
||||||
socket.conn.Close()
|
|
||||||
socket.err = err
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
func TestSocket(t *testing.T) {
|
|
||||||
socket, err := NewSocket("dnstap.sock")
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("new socket: not listening but no error")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := sendOne(socket); err == nil {
|
|
||||||
t.Fatal("not listening but no error")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
l, err := net.Listen("unix", "dnstap.sock")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
wait := make(chan bool)
|
|
||||||
go func() {
|
|
||||||
acceptOne(t, l)
|
|
||||||
wait <- true
|
|
||||||
}()
|
|
||||||
|
|
||||||
if err := sendOne(socket); err != nil {
|
|
||||||
t.Fatalf("send one: %s", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
<-wait
|
|
||||||
if err := sendOne(socket); err == nil {
|
|
||||||
panic("must fail")
|
|
||||||
}
|
|
||||||
|
|
||||||
go func() {
|
|
||||||
acceptOne(t, l)
|
|
||||||
wait <- true
|
|
||||||
}()
|
|
||||||
|
|
||||||
if err := sendOne(socket); err != nil {
|
|
||||||
t.Fatalf("send one: %s", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
<-wait
|
|
||||||
if err := l.Close(); err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
package out
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
fs "github.com/farsightsec/golang-framestream"
|
|
||||||
)
|
|
||||||
|
|
||||||
// TCP is a Frame Streams encoder over TCP.
|
|
||||||
type TCP struct {
|
|
||||||
address string
|
|
||||||
frames [][]byte
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewTCP returns a TCP writer.
|
|
||||||
func NewTCP(address string) *TCP {
|
|
||||||
s := &TCP{address: address}
|
|
||||||
s.frames = make([][]byte, 0, 13) // 13 messages buffer
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write a single Frame Streams frame.
|
|
||||||
func (s *TCP) Write(frame []byte) (n int, err error) {
|
|
||||||
s.frames = append(s.frames, frame)
|
|
||||||
if len(s.frames) == cap(s.frames) {
|
|
||||||
return len(frame), s.Flush()
|
|
||||||
}
|
|
||||||
return len(frame), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Flush the remaining frames.
|
|
||||||
func (s *TCP) Flush() error {
|
|
||||||
defer func() {
|
|
||||||
s.frames = s.frames[:0]
|
|
||||||
}()
|
|
||||||
c, err := net.DialTimeout("tcp", s.address, time.Second)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
enc, err := fs.NewEncoder(c, &fs.EncoderOptions{
|
|
||||||
ContentType: []byte("protobuf:dnstap.Dnstap"),
|
|
||||||
Bidirectional: true,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
for _, frame := range s.frames {
|
|
||||||
if _, err = enc.Write(frame); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return enc.Flush()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close is an alias to Flush to satisfy io.WriteCloser similarly to type Socket.
|
|
||||||
func (s *TCP) Close() error {
|
|
||||||
return s.Flush()
|
|
||||||
}
|
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
package out
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net"
|
|
||||||
"testing"
|
|
||||||
)
|
|
||||||
|
|
||||||
func sendOneTCP(tcp *TCP) error {
|
|
||||||
if _, err := tcp.Write([]byte("frame")); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return tcp.Flush()
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTCP(t *testing.T) {
|
|
||||||
tcp := NewTCP("localhost:14000")
|
|
||||||
|
|
||||||
if err := sendOneTCP(tcp); err == nil {
|
|
||||||
t.Fatal("Not listening but no error.")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
l, err := net.Listen("tcp", "localhost:14000")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
wait := make(chan bool)
|
|
||||||
go func() {
|
|
||||||
acceptOne(t, l)
|
|
||||||
wait <- true
|
|
||||||
}()
|
|
||||||
|
|
||||||
if err := sendOneTCP(tcp); err != nil {
|
|
||||||
t.Fatalf("send one: %s", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
<-wait
|
|
||||||
|
|
||||||
// TODO: When the server isn't responding according to the framestream protocol
|
|
||||||
// the thread is blocked.
|
|
||||||
/*
|
|
||||||
if err := sendOneTCP(tcp); err == nil {
|
|
||||||
panic("must fail")
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
go func() {
|
|
||||||
acceptOne(t, l)
|
|
||||||
wait <- true
|
|
||||||
}()
|
|
||||||
|
|
||||||
if err := sendOneTCP(tcp); err != nil {
|
|
||||||
t.Fatalf("send one: %s", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
<-wait
|
|
||||||
if err := l.Close(); err != nil {
|
|
||||||
t.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,15 +1,11 @@
|
|||||||
package dnstap
|
package dnstap
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"log"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"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/dnstapio"
|
||||||
"github.com/coredns/coredns/plugin/dnstap/out"
|
|
||||||
"github.com/coredns/coredns/plugin/pkg/dnsutil"
|
"github.com/coredns/coredns/plugin/pkg/dnsutil"
|
||||||
|
|
||||||
"github.com/mholt/caddy"
|
"github.com/mholt/caddy"
|
||||||
@@ -69,24 +65,24 @@ func setup(c *caddy.Controller) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
dnstap := Dnstap{Pack: conf.full}
|
dio := dnstapio.New()
|
||||||
|
dnstap := Dnstap{IO: dio, Pack: conf.full}
|
||||||
|
|
||||||
var o io.WriteCloser
|
c.OnStartup(func() error {
|
||||||
if conf.socket {
|
err := dio.Connect(conf.target, conf.socket)
|
||||||
o, err = out.NewSocket(conf.target)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("[WARN] Can't connect to %s at the moment: %s", conf.target, err)
|
return plugin.Error("dnstap", err)
|
||||||
}
|
}
|
||||||
} else {
|
return nil
|
||||||
o = out.NewTCP(conf.target)
|
})
|
||||||
}
|
|
||||||
dio := dnstapio.New(o)
|
|
||||||
dnstap.IO = dio
|
|
||||||
|
|
||||||
c.OnShutdown(func() error {
|
c.OnRestart(func() error {
|
||||||
if err := dio.Close(); err != nil {
|
dio.Close()
|
||||||
return fmt.Errorf("dnstap io routine: %s", err)
|
return nil
|
||||||
}
|
})
|
||||||
|
|
||||||
|
c.OnFinalShutdown(func() error {
|
||||||
|
dio.Close()
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user