mirror of
https://github.com/coredns/coredns.git
synced 2025-10-28 08:44:17 -04:00
182 lines
5.1 KiB
Go
182 lines
5.1 KiB
Go
package grpc
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/coredns/coredns/pb"
|
|
"github.com/coredns/coredns/plugin/pkg/dnstest"
|
|
"github.com/coredns/coredns/plugin/pkg/fall"
|
|
"github.com/coredns/coredns/plugin/test"
|
|
|
|
"github.com/miekg/dns"
|
|
ot "github.com/opentracing/opentracing-go"
|
|
"github.com/opentracing/opentracing-go/mocktracer"
|
|
grpcgo "google.golang.org/grpc"
|
|
)
|
|
|
|
func TestGRPC(t *testing.T) {
|
|
m := &dns.Msg{}
|
|
msg, err := m.Pack()
|
|
if err != nil {
|
|
t.Fatalf("Error packing response: %s", err.Error())
|
|
}
|
|
dnsPacket := &pb.DnsPacket{Msg: msg}
|
|
tests := map[string]struct {
|
|
proxies []*Proxy
|
|
wantErr bool
|
|
}{
|
|
"single_proxy_ok": {
|
|
proxies: []*Proxy{
|
|
{client: &testServiceClient{dnsPacket: dnsPacket, err: nil}},
|
|
},
|
|
wantErr: false,
|
|
},
|
|
"multiple_proxies_ok": {
|
|
proxies: []*Proxy{
|
|
{client: &testServiceClient{dnsPacket: dnsPacket, err: nil}},
|
|
{client: &testServiceClient{dnsPacket: dnsPacket, err: nil}},
|
|
{client: &testServiceClient{dnsPacket: dnsPacket, err: nil}},
|
|
},
|
|
wantErr: false,
|
|
},
|
|
"single_proxy_ko": {
|
|
proxies: []*Proxy{
|
|
{client: &testServiceClient{dnsPacket: nil, err: errors.New("")}},
|
|
},
|
|
wantErr: true,
|
|
},
|
|
"multiple_proxies_one_ko": {
|
|
proxies: []*Proxy{
|
|
{client: &testServiceClient{dnsPacket: dnsPacket, err: nil}},
|
|
{client: &testServiceClient{dnsPacket: nil, err: errors.New("")}},
|
|
{client: &testServiceClient{dnsPacket: dnsPacket, err: nil}},
|
|
},
|
|
wantErr: false,
|
|
},
|
|
"multiple_proxies_ko": {
|
|
proxies: []*Proxy{
|
|
{client: &testServiceClient{dnsPacket: nil, err: errors.New("")}},
|
|
{client: &testServiceClient{dnsPacket: nil, err: errors.New("")}},
|
|
{client: &testServiceClient{dnsPacket: nil, err: errors.New("")}},
|
|
},
|
|
wantErr: true,
|
|
},
|
|
}
|
|
|
|
for name, tt := range tests {
|
|
t.Run(name, func(t *testing.T) {
|
|
g := newGRPC()
|
|
g.from = "."
|
|
g.proxies = tt.proxies
|
|
rec := dnstest.NewRecorder(&test.ResponseWriter{})
|
|
if _, err := g.ServeDNS(context.TODO(), rec, m); err != nil && !tt.wantErr {
|
|
t.Fatal("Expected to receive reply, but didn't")
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
// Test that fallthrough works correctly when there's no next plugin
|
|
func TestGRPCFallthroughNoNext(t *testing.T) {
|
|
g := newGRPC() // Use the constructor to properly initialize
|
|
g.Fall = fall.Root // Enable fallthrough for all zones
|
|
g.Next = nil // No next plugin
|
|
g.from = "."
|
|
|
|
// Create a test request
|
|
r := new(dns.Msg)
|
|
r.SetQuestion("test.example.org.", dns.TypeA)
|
|
|
|
w := &test.ResponseWriter{}
|
|
|
|
// Should return SERVFAIL since no backends are configured and no next plugin
|
|
rcode, err := g.ServeDNS(context.Background(), w, r)
|
|
|
|
// Should not return the "no next plugin found" error
|
|
if err != nil && strings.Contains(err.Error(), "no next plugin found") {
|
|
t.Errorf("Expected no 'no next plugin found' error, got: %v", err)
|
|
}
|
|
|
|
// Should return SERVFAIL
|
|
if rcode != dns.RcodeServerFailure {
|
|
t.Errorf("Expected SERVFAIL when no backends and no next plugin, got: %d", rcode)
|
|
}
|
|
}
|
|
|
|
// deadlineCheckingClient records whether a deadline was attached to ctx.
|
|
type deadlineCheckingClient struct {
|
|
sawDeadline bool
|
|
lastDeadline time.Time
|
|
dnsPacket *pb.DnsPacket
|
|
err error
|
|
}
|
|
|
|
func (c *deadlineCheckingClient) Query(ctx context.Context, in *pb.DnsPacket, opts ...grpcgo.CallOption) (*pb.DnsPacket, error) {
|
|
if dl, ok := ctx.Deadline(); ok {
|
|
c.sawDeadline = true
|
|
c.lastDeadline = dl
|
|
}
|
|
return c.dnsPacket, c.err
|
|
}
|
|
|
|
// Test that on error paths we still finish child spans, and that we set a per-call deadline.
|
|
func TestGRPC_SpansOnErrorPath(t *testing.T) {
|
|
m := &dns.Msg{}
|
|
msgBytes, err := m.Pack()
|
|
if err != nil {
|
|
t.Fatalf("Error packing response: %s", err)
|
|
}
|
|
dnsPacket := &pb.DnsPacket{Msg: msgBytes}
|
|
|
|
// Proxy 1: returns error, we should still finish its child span and have a deadline
|
|
p1 := &deadlineCheckingClient{dnsPacket: nil, err: errors.New("kaboom")}
|
|
// Proxy 2: returns success
|
|
p2 := &deadlineCheckingClient{dnsPacket: dnsPacket, err: nil}
|
|
|
|
g := newGRPC()
|
|
g.from = "."
|
|
g.proxies = []*Proxy{{client: p1}, {client: p2}}
|
|
|
|
// Ensure deterministic order of the retries: try p1 then p2
|
|
g.p = new(sequential)
|
|
|
|
// Set a parent span in context so ServeDNS creates child spans per attempt
|
|
tracer := mocktracer.New()
|
|
prev := ot.GlobalTracer()
|
|
ot.SetGlobalTracer(tracer)
|
|
defer ot.SetGlobalTracer(prev)
|
|
|
|
parent := tracer.StartSpan("parent")
|
|
ctx := ot.ContextWithSpan(t.Context(), parent)
|
|
|
|
rec := dnstest.NewRecorder(&test.ResponseWriter{})
|
|
if _, err := g.ServeDNS(ctx, rec, m); err != nil {
|
|
t.Fatalf("ServeDNS returned error: %v", err)
|
|
}
|
|
|
|
// Assert both attempts finished child spans with retries
|
|
// (2 query spans: error + success)
|
|
finished := tracer.FinishedSpans()
|
|
var finishedQueries int
|
|
for _, s := range finished {
|
|
if s.OperationName == "query" {
|
|
finishedQueries++
|
|
}
|
|
}
|
|
if finishedQueries != 2 {
|
|
t.Fatalf("expected 2 finished 'query' spans, got %d (finished: %v)", finishedQueries, finished)
|
|
}
|
|
|
|
// Assert we set a deadline on the call contexts
|
|
if !p1.sawDeadline {
|
|
t.Fatalf("expected deadline to be set on first proxy call context")
|
|
}
|
|
if !p2.sawDeadline {
|
|
t.Fatalf("expected deadline to be set on second proxy call context")
|
|
}
|
|
}
|