2016-09-07 11:10:16 +01:00
|
|
|
package replacer
|
2016-03-18 20:57:35 +00:00
|
|
|
|
2017-08-07 03:49:40 -07:00
|
|
|
import (
|
2018-11-13 14:20:49 -05:00
|
|
|
"context"
|
2019-07-17 02:57:46 -04:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
2017-08-07 03:49:40 -07:00
|
|
|
"testing"
|
|
|
|
|
2018-11-13 14:20:49 -05:00
|
|
|
"github.com/coredns/coredns/plugin/metadata"
|
2017-09-21 15:15:47 +01:00
|
|
|
"github.com/coredns/coredns/plugin/pkg/dnstest"
|
2017-09-14 09:36:06 +01:00
|
|
|
"github.com/coredns/coredns/plugin/test"
|
2019-02-12 07:38:49 +00:00
|
|
|
"github.com/coredns/coredns/request"
|
2017-08-07 03:49:40 -07:00
|
|
|
|
|
|
|
"github.com/miekg/dns"
|
|
|
|
)
|
|
|
|
|
2019-07-17 02:57:46 -04:00
|
|
|
// This is the default format used by the log package
|
|
|
|
const CommonLogFormat = `{remote}:{port} - {>id} "{type} {class} {name} {proto} {size} {>do} {>bufsize}" {rcode} {>rflags} {rsize} {duration}`
|
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
func TestReplacer(t *testing.T) {
|
2017-09-21 15:15:47 +01:00
|
|
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
2017-08-07 03:49:40 -07:00
|
|
|
r := new(dns.Msg)
|
|
|
|
r.SetQuestion("example.org.", dns.TypeHINFO)
|
|
|
|
r.MsgHdr.AuthenticatedData = true
|
2019-02-12 07:38:49 +00:00
|
|
|
state := request.Request{W: w, Req: r}
|
2017-08-07 03:49:40 -07:00
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
replacer := New()
|
2016-03-18 20:57:35 +00:00
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
if x := replacer.Replace(context.TODO(), state, nil, "{type}"); x != "HINFO" {
|
|
|
|
t.Errorf("Expected type to be HINFO, got %q", x)
|
|
|
|
}
|
|
|
|
if x := replacer.Replace(context.TODO(), state, nil, "{name}"); x != "example.org." {
|
|
|
|
t.Errorf("Expected request name to be example.org., got %q", x)
|
|
|
|
}
|
|
|
|
if x := replacer.Replace(context.TODO(), state, nil, "{size}"); x != "29" {
|
|
|
|
t.Errorf("Expected size to be 29, got %q", x)
|
|
|
|
}
|
|
|
|
}
|
2016-03-18 20:57:35 +00:00
|
|
|
|
2019-07-17 02:57:46 -04:00
|
|
|
func TestParseFormat(t *testing.T) {
|
|
|
|
type formatTest struct {
|
|
|
|
Format string
|
|
|
|
Expected replacer
|
|
|
|
}
|
|
|
|
tests := []formatTest{
|
|
|
|
{
|
|
|
|
Format: "",
|
|
|
|
Expected: replacer{},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: "A",
|
|
|
|
Expected: replacer{
|
|
|
|
{"A", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: "A {A}",
|
|
|
|
Expected: replacer{
|
|
|
|
{"A {A}", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: "{{remote}}",
|
|
|
|
Expected: replacer{
|
|
|
|
{"{", typeLiteral},
|
|
|
|
{"{remote}", typeLabel},
|
|
|
|
{"}", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: "{ A {remote} A }",
|
|
|
|
Expected: replacer{
|
|
|
|
{"{ A ", typeLiteral},
|
|
|
|
{"{remote}", typeLabel},
|
|
|
|
{" A }", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: "{remote}}",
|
|
|
|
Expected: replacer{
|
|
|
|
{"{remote}", typeLabel},
|
|
|
|
{"}", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: "{{remote}",
|
|
|
|
Expected: replacer{
|
|
|
|
{"{", typeLiteral},
|
|
|
|
{"{remote}", typeLabel},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: `Foo } {remote}`,
|
|
|
|
Expected: replacer{
|
|
|
|
// we don't do any optimizations to join adjacent literals
|
|
|
|
{"Foo }", typeLiteral},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{remote}", typeLabel},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: `{ Foo`,
|
|
|
|
Expected: replacer{
|
|
|
|
{"{ Foo", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: `} Foo`,
|
|
|
|
Expected: replacer{
|
|
|
|
{"}", typeLiteral},
|
|
|
|
{" Foo", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: "A { {remote} {type} {/meta1} } B",
|
|
|
|
Expected: replacer{
|
|
|
|
{"A { ", typeLiteral},
|
|
|
|
{"{remote}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{type}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"meta1", typeMetadata},
|
|
|
|
{" }", typeLiteral},
|
|
|
|
{" B", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Format: `LOG {remote}:{port} - {>id} "{type} {class} {name} {proto} ` +
|
|
|
|
`{size} {>do} {>bufsize}" {rcode} {>rflags} {rsize} {/meta1}-{/meta2} ` +
|
|
|
|
`{duration} END OF LINE`,
|
|
|
|
Expected: replacer{
|
|
|
|
{"LOG ", typeLiteral},
|
|
|
|
{"{remote}", typeLabel},
|
|
|
|
{":", typeLiteral},
|
|
|
|
{"{port}", typeLabel},
|
|
|
|
{" - ", typeLiteral},
|
|
|
|
{"{>id}", typeLabel},
|
|
|
|
{` "`, typeLiteral},
|
|
|
|
{"{type}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{class}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{name}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{proto}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{size}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{>do}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{>bufsize}", typeLabel},
|
|
|
|
{`" `, typeLiteral},
|
|
|
|
{"{rcode}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{>rflags}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{rsize}", typeLabel},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"meta1", typeMetadata},
|
|
|
|
{"-", typeLiteral},
|
|
|
|
{"meta2", typeMetadata},
|
|
|
|
{" ", typeLiteral},
|
|
|
|
{"{duration}", typeLabel},
|
|
|
|
{" END OF LINE", typeLiteral},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for i, x := range tests {
|
|
|
|
r := parseFormat(x.Format)
|
|
|
|
if !reflect.DeepEqual(r, x.Expected) {
|
|
|
|
t.Errorf("%d: Expected:\n\t%+v\nGot:\n\t%+v", i, x.Expected, r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParseFormatNodes(t *testing.T) {
|
|
|
|
// If we parse the format successfully the result of joining all the
|
|
|
|
// segments should match the original format.
|
|
|
|
formats := []string{
|
|
|
|
"",
|
|
|
|
"msg",
|
|
|
|
"{remote}",
|
|
|
|
"{remote}",
|
|
|
|
"{{remote}",
|
|
|
|
"{{remote}}",
|
|
|
|
"{{remote}} A",
|
|
|
|
CommonLogFormat,
|
|
|
|
CommonLogFormat + " FOO} {BAR}",
|
|
|
|
"A " + CommonLogFormat + " FOO} {BAR}",
|
|
|
|
"A " + CommonLogFormat + " {/meta}",
|
|
|
|
}
|
|
|
|
join := func(r replacer) string {
|
|
|
|
a := make([]string, len(r))
|
|
|
|
for i, n := range r {
|
|
|
|
if n.typ == typeMetadata {
|
|
|
|
a[i] = "{/" + n.value + "}"
|
|
|
|
} else {
|
|
|
|
a[i] = n.value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return strings.Join(a, "")
|
|
|
|
}
|
|
|
|
for _, format := range formats {
|
|
|
|
r := parseFormat(format)
|
|
|
|
s := join(r)
|
|
|
|
if s != format {
|
|
|
|
t.Errorf("Expected format to be: '%s' got: '%s'", format, s)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
func TestLabels(t *testing.T) {
|
|
|
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
|
|
|
r := new(dns.Msg)
|
|
|
|
r.SetQuestion("example.org.", dns.TypeHINFO)
|
|
|
|
r.Id = 1053
|
|
|
|
r.AuthenticatedData = true
|
|
|
|
r.CheckingDisabled = true
|
|
|
|
w.WriteMsg(r)
|
|
|
|
state := request.Request{W: w, Req: r}
|
|
|
|
|
|
|
|
replacer := New()
|
|
|
|
ctx := context.TODO()
|
|
|
|
|
|
|
|
// This couples the test very tightly to the code, but so be it.
|
|
|
|
expect := map[string]string{
|
|
|
|
"{type}": "HINFO",
|
|
|
|
"{name}": "example.org.",
|
|
|
|
"{class}": "IN",
|
|
|
|
"{proto}": "udp",
|
|
|
|
"{size}": "29",
|
|
|
|
"{remote}": "10.240.0.1",
|
|
|
|
"{port}": "40212",
|
|
|
|
"{local}": "127.0.0.1",
|
|
|
|
headerReplacer + "id}": "1053",
|
|
|
|
headerReplacer + "opcode}": "0",
|
|
|
|
headerReplacer + "do}": "false",
|
|
|
|
headerReplacer + "bufsize}": "512",
|
|
|
|
"{rcode}": "NOERROR",
|
|
|
|
"{rsize}": "29",
|
|
|
|
"{duration}": "0",
|
2019-02-21 07:13:05 +00:00
|
|
|
headerReplacer + "rflags}": "rd,ad,cd",
|
2019-02-12 07:38:49 +00:00
|
|
|
}
|
|
|
|
if len(expect) != len(labels) {
|
|
|
|
t.Fatalf("Expect %d labels, got %d", len(expect), len(labels))
|
|
|
|
}
|
|
|
|
|
2019-07-17 02:57:46 -04:00
|
|
|
for lbl := range labels {
|
2019-02-12 07:38:49 +00:00
|
|
|
repl := replacer.Replace(ctx, state, w, lbl)
|
|
|
|
if lbl == "{duration}" {
|
|
|
|
if repl[len(repl)-1] != 's' {
|
|
|
|
t.Errorf("Expected seconds, got %q", repl)
|
|
|
|
}
|
|
|
|
continue
|
2016-03-18 20:57:35 +00:00
|
|
|
}
|
2019-02-12 07:38:49 +00:00
|
|
|
if repl != expect[lbl] {
|
|
|
|
t.Errorf("Expected value %q, got %q", expect[lbl], repl)
|
2018-06-07 11:21:17 -04:00
|
|
|
}
|
2016-03-18 20:57:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
func BenchmarkReplacer(b *testing.B) {
|
2017-09-21 15:15:47 +01:00
|
|
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
2017-08-07 03:49:40 -07:00
|
|
|
r := new(dns.Msg)
|
|
|
|
r.SetQuestion("example.org.", dns.TypeHINFO)
|
|
|
|
r.MsgHdr.AuthenticatedData = true
|
2019-02-12 07:38:49 +00:00
|
|
|
state := request.Request{W: w, Req: r}
|
2016-03-18 20:57:35 +00:00
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
b.ResetTimer()
|
|
|
|
b.ReportAllocs()
|
2016-03-18 20:57:35 +00:00
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
replacer := New()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
replacer.Replace(context.TODO(), state, nil, "{type} {name} {size}")
|
2016-03-18 20:57:35 +00:00
|
|
|
}
|
|
|
|
}
|
2018-11-13 14:20:49 -05:00
|
|
|
|
2019-07-17 02:57:46 -04:00
|
|
|
func BenchmarkReplacer_CommonLogFormat(b *testing.B) {
|
|
|
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
|
|
|
r := new(dns.Msg)
|
|
|
|
r.SetQuestion("example.org.", dns.TypeHINFO)
|
|
|
|
r.Id = 1053
|
|
|
|
r.AuthenticatedData = true
|
|
|
|
r.CheckingDisabled = true
|
|
|
|
r.MsgHdr.AuthenticatedData = true
|
|
|
|
w.WriteMsg(r)
|
|
|
|
state := request.Request{W: w, Req: r}
|
|
|
|
|
|
|
|
replacer := New()
|
|
|
|
ctxt := context.TODO()
|
|
|
|
|
|
|
|
b.ResetTimer()
|
|
|
|
b.ReportAllocs()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
replacer.Replace(ctxt, state, w, CommonLogFormat)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkParseFormat(b *testing.B) {
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
parseFormat(CommonLogFormat)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-13 14:20:49 -05:00
|
|
|
type testProvider map[string]metadata.Func
|
|
|
|
|
|
|
|
func (tp testProvider) Metadata(ctx context.Context, state request.Request) context.Context {
|
|
|
|
for k, v := range tp {
|
|
|
|
metadata.SetValueFunc(ctx, k, v)
|
|
|
|
}
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
|
|
|
|
type testHandler struct{ ctx context.Context }
|
|
|
|
|
|
|
|
func (m *testHandler) Name() string { return "test" }
|
|
|
|
|
|
|
|
func (m *testHandler) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
|
|
|
|
m.ctx = ctx
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMetadataReplacement(t *testing.T) {
|
2019-02-12 07:38:49 +00:00
|
|
|
tests := []struct {
|
|
|
|
expr string
|
|
|
|
result string
|
|
|
|
}{
|
|
|
|
{"{/test/meta2}", "two"},
|
|
|
|
{"{/test/meta2} {/test/key4}", "two -"},
|
|
|
|
{"{/test/meta2} {/test/meta3}", "two three"},
|
|
|
|
}
|
|
|
|
|
|
|
|
next := &testHandler{}
|
|
|
|
m := metadata.Metadata{
|
|
|
|
Zones: []string{"."},
|
|
|
|
Providers: []metadata.Provider{
|
|
|
|
testProvider{"test/meta2": func() string { return "two" }},
|
|
|
|
testProvider{"test/meta3": func() string { return "three" }},
|
|
|
|
},
|
|
|
|
Next: next,
|
|
|
|
}
|
|
|
|
|
|
|
|
m.ServeDNS(context.TODO(), &test.ResponseWriter{}, new(dns.Msg))
|
|
|
|
ctx := next.ctx // important because the m.ServeDNS has only now populated the context
|
|
|
|
|
|
|
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
|
|
|
r := new(dns.Msg)
|
|
|
|
r.SetQuestion("example.org.", dns.TypeHINFO)
|
2018-11-13 14:20:49 -05:00
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
repl := New()
|
|
|
|
state := request.Request{W: w, Req: r}
|
|
|
|
|
|
|
|
for i, ts := range tests {
|
|
|
|
r := repl.Replace(ctx, state, nil, ts.expr)
|
|
|
|
if r != ts.result {
|
|
|
|
t.Errorf("Test %d - expr : %s, expected %q, got %q", i, ts.expr, ts.result, r)
|
|
|
|
}
|
2018-11-13 14:20:49 -05:00
|
|
|
}
|
2019-02-12 07:38:49 +00:00
|
|
|
}
|
2018-11-13 14:20:49 -05:00
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
func TestMetadataMalformed(t *testing.T) {
|
2018-11-13 14:20:49 -05:00
|
|
|
tests := []struct {
|
|
|
|
expr string
|
|
|
|
result string
|
|
|
|
}{
|
2019-02-12 07:38:49 +00:00
|
|
|
{"{/test/meta2", "{/test/meta2"},
|
|
|
|
{"{test/meta2} {/test/meta4}", "{test/meta2} -"},
|
|
|
|
{"{test}", "{test}"},
|
2018-11-13 14:20:49 -05:00
|
|
|
}
|
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
next := &testHandler{}
|
2018-11-13 14:20:49 -05:00
|
|
|
m := metadata.Metadata{
|
|
|
|
Zones: []string{"."},
|
2019-02-12 07:38:49 +00:00
|
|
|
Providers: []metadata.Provider{testProvider{"test/meta2": func() string { return "two" }}},
|
2018-11-13 14:20:49 -05:00
|
|
|
Next: next,
|
|
|
|
}
|
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
m.ServeDNS(context.TODO(), &test.ResponseWriter{}, new(dns.Msg))
|
|
|
|
ctx := next.ctx // important because the m.ServeDNS has only now populated the context
|
2018-11-13 14:20:49 -05:00
|
|
|
|
|
|
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
|
|
|
r := new(dns.Msg)
|
|
|
|
r.SetQuestion("example.org.", dns.TypeHINFO)
|
|
|
|
|
2019-02-12 07:38:49 +00:00
|
|
|
repl := New()
|
|
|
|
state := request.Request{W: w, Req: r}
|
2018-11-13 14:20:49 -05:00
|
|
|
|
|
|
|
for i, ts := range tests {
|
2019-02-12 07:38:49 +00:00
|
|
|
r := repl.Replace(ctx, state, nil, ts.expr)
|
2018-11-13 14:20:49 -05:00
|
|
|
if r != ts.result {
|
2019-02-12 07:38:49 +00:00
|
|
|
t.Errorf("Test %d - expr : %s, expected %q, got %q", i, ts.expr, ts.result, r)
|
2018-11-13 14:20:49 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-07-09 13:15:34 +02:00
|
|
|
|
|
|
|
func TestNoResponseWasWritten(t *testing.T) {
|
|
|
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
|
|
|
r := new(dns.Msg)
|
|
|
|
r.SetQuestion("example.org.", dns.TypeHINFO)
|
|
|
|
r.Id = 1053
|
|
|
|
r.AuthenticatedData = true
|
|
|
|
r.CheckingDisabled = true
|
|
|
|
state := request.Request{W: w, Req: r}
|
|
|
|
|
|
|
|
replacer := New()
|
|
|
|
ctx := context.TODO()
|
|
|
|
|
|
|
|
// This couples the test very tightly to the code, but so be it.
|
|
|
|
expect := map[string]string{
|
|
|
|
"{type}": "HINFO",
|
|
|
|
"{name}": "example.org.",
|
|
|
|
"{class}": "IN",
|
|
|
|
"{proto}": "udp",
|
|
|
|
"{size}": "29",
|
|
|
|
"{remote}": "10.240.0.1",
|
|
|
|
"{port}": "40212",
|
|
|
|
"{local}": "127.0.0.1",
|
|
|
|
headerReplacer + "id}": "1053",
|
|
|
|
headerReplacer + "opcode}": "0",
|
|
|
|
headerReplacer + "do}": "false",
|
|
|
|
headerReplacer + "bufsize}": "512",
|
|
|
|
"{rcode}": "-",
|
|
|
|
"{rsize}": "0",
|
|
|
|
"{duration}": "0",
|
|
|
|
headerReplacer + "rflags}": "-",
|
|
|
|
}
|
|
|
|
if len(expect) != len(labels) {
|
|
|
|
t.Fatalf("Expect %d labels, got %d", len(expect), len(labels))
|
|
|
|
}
|
|
|
|
|
|
|
|
for lbl := range labels {
|
|
|
|
repl := replacer.Replace(ctx, state, w, lbl)
|
|
|
|
if lbl == "{duration}" {
|
|
|
|
if repl[len(repl)-1] != 's' {
|
|
|
|
t.Errorf("Expected seconds, got %q", repl)
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if repl != expect[lbl] {
|
|
|
|
t.Errorf("Expected value %q, got %q", expect[lbl], repl)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|