3cb782f0a2
While there, modernize the way in which we run tests to avoid depending on the fake files scattered around the tree and to use some well defined mock structures instead. Part of https://github.com/ooni/probe/issues/1591
182 lines
4.1 KiB
Go
182 lines
4.1 KiB
Go
package dnsx
|
|
|
|
import (
|
|
"net"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
func TestDecoderUnpackError(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(dns.TypeA, nil)
|
|
if err == nil {
|
|
t.Fatal("expected an error here")
|
|
}
|
|
if data != nil {
|
|
t.Fatal("expected nil data here")
|
|
}
|
|
}
|
|
|
|
func TestDecoderNXDOMAIN(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(dns.TypeA, genReplyError(t, dns.RcodeNameError))
|
|
if err == nil || !strings.HasSuffix(err.Error(), "no such host") {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if data != nil {
|
|
t.Fatal("expected nil data here")
|
|
}
|
|
}
|
|
|
|
func TestDecoderOtherError(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(dns.TypeA, genReplyError(t, dns.RcodeRefused))
|
|
if err == nil || !strings.HasSuffix(err.Error(), "query failed") {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if data != nil {
|
|
t.Fatal("expected nil data here")
|
|
}
|
|
}
|
|
|
|
func TestDecoderNoAddress(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(dns.TypeA, genReplySuccess(t, dns.TypeA))
|
|
if err == nil || !strings.HasSuffix(err.Error(), "no response returned") {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if data != nil {
|
|
t.Fatal("expected nil data here")
|
|
}
|
|
}
|
|
|
|
func TestDecoderDecodeA(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(
|
|
dns.TypeA, genReplySuccess(t, dns.TypeA, "1.1.1.1", "8.8.8.8"))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(data) != 2 {
|
|
t.Fatal("expected two entries here")
|
|
}
|
|
if data[0] != "1.1.1.1" {
|
|
t.Fatal("invalid first IPv4 entry")
|
|
}
|
|
if data[1] != "8.8.8.8" {
|
|
t.Fatal("invalid second IPv4 entry")
|
|
}
|
|
}
|
|
|
|
func TestDecoderDecodeAAAA(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(
|
|
dns.TypeAAAA, genReplySuccess(t, dns.TypeAAAA, "::1", "fe80::1"))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(data) != 2 {
|
|
t.Fatal("expected two entries here")
|
|
}
|
|
if data[0] != "::1" {
|
|
t.Fatal("invalid first IPv6 entry")
|
|
}
|
|
if data[1] != "fe80::1" {
|
|
t.Fatal("invalid second IPv6 entry")
|
|
}
|
|
}
|
|
|
|
func TestDecoderUnexpectedAReply(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(
|
|
dns.TypeA, genReplySuccess(t, dns.TypeAAAA, "::1", "fe80::1"))
|
|
if err == nil || !strings.HasSuffix(err.Error(), "no response returned") {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if data != nil {
|
|
t.Fatal("expected nil data here")
|
|
}
|
|
}
|
|
|
|
func TestDecoderUnexpectedAAAAReply(t *testing.T) {
|
|
d := &MiekgDecoder{}
|
|
data, err := d.Decode(
|
|
dns.TypeAAAA, genReplySuccess(t, dns.TypeA, "1.1.1.1", "8.8.4.4."))
|
|
if err == nil || !strings.HasSuffix(err.Error(), "no response returned") {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if data != nil {
|
|
t.Fatal("expected nil data here")
|
|
}
|
|
}
|
|
|
|
func genReplyError(t *testing.T, code int) []byte {
|
|
question := dns.Question{
|
|
Name: dns.Fqdn("x.org"),
|
|
Qtype: dns.TypeA,
|
|
Qclass: dns.ClassINET,
|
|
}
|
|
query := new(dns.Msg)
|
|
query.Id = dns.Id()
|
|
query.RecursionDesired = true
|
|
query.Question = make([]dns.Question, 1)
|
|
query.Question[0] = question
|
|
reply := new(dns.Msg)
|
|
reply.Compress = true
|
|
reply.MsgHdr.RecursionAvailable = true
|
|
reply.SetRcode(query, code)
|
|
data, err := reply.Pack()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return data
|
|
}
|
|
|
|
func genReplySuccess(t *testing.T, qtype uint16, ips ...string) []byte {
|
|
question := dns.Question{
|
|
Name: dns.Fqdn("x.org"),
|
|
Qtype: qtype,
|
|
Qclass: dns.ClassINET,
|
|
}
|
|
query := new(dns.Msg)
|
|
query.Id = dns.Id()
|
|
query.RecursionDesired = true
|
|
query.Question = make([]dns.Question, 1)
|
|
query.Question[0] = question
|
|
reply := new(dns.Msg)
|
|
reply.Compress = true
|
|
reply.MsgHdr.RecursionAvailable = true
|
|
reply.SetReply(query)
|
|
for _, ip := range ips {
|
|
switch qtype {
|
|
case dns.TypeA:
|
|
reply.Answer = append(reply.Answer, &dns.A{
|
|
Hdr: dns.RR_Header{
|
|
Name: dns.Fqdn("x.org"),
|
|
Rrtype: qtype,
|
|
Class: dns.ClassINET,
|
|
Ttl: 0,
|
|
},
|
|
A: net.ParseIP(ip),
|
|
})
|
|
case dns.TypeAAAA:
|
|
reply.Answer = append(reply.Answer, &dns.AAAA{
|
|
Hdr: dns.RR_Header{
|
|
Name: dns.Fqdn("x.org"),
|
|
Rrtype: qtype,
|
|
Class: dns.ClassINET,
|
|
Ttl: 0,
|
|
},
|
|
AAAA: net.ParseIP(ip),
|
|
})
|
|
}
|
|
}
|
|
data, err := reply.Pack()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return data
|
|
}
|