| 
									
										
										
										
											2016-04-11 07:56:38 +01:00
										 |  |  | package test | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2018-04-22 08:34:35 +01:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2017-08-16 15:30:58 +01:00
										 |  |  | 	"sort" | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/miekg/dns" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | type sect int | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | 	// Answer is the answer section in an Msg. | 
					
						
							|  |  |  | 	Answer sect = iota | 
					
						
							| 
									
										
										
										
											2017-08-24 11:05:16 -04:00
										 |  |  | 	// Ns is the authoritative section in an Msg. | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 	Ns | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | 	// Extra is the additional section in an Msg. | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 	Extra | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // RRSet represents a list of RRs. | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | type RRSet []dns.RR | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p RRSet) Len() int           { return len(p) } | 
					
						
							|  |  |  | func (p RRSet) Swap(i, j int)      { p[i], p[j] = p[j], p[i] } | 
					
						
							|  |  |  | func (p RRSet) Less(i, j int) bool { return p[i].String() < p[j].String() } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // Case represents a test case that encapsulates various data from a query and response. | 
					
						
							|  |  |  | // Note that is the TTL of a record is 303 we don't compare it with the TTL. | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | type Case struct { | 
					
						
							|  |  |  | 	Qname  string | 
					
						
							|  |  |  | 	Qtype  uint16 | 
					
						
							|  |  |  | 	Rcode  int | 
					
						
							| 
									
										
										
										
											2016-03-28 21:18:16 +01:00
										 |  |  | 	Do     bool | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 	Answer []dns.RR | 
					
						
							|  |  |  | 	Ns     []dns.RR | 
					
						
							|  |  |  | 	Extra  []dns.RR | 
					
						
							| 
									
										
										
										
											2017-08-03 15:16:50 -04:00
										 |  |  | 	Error  error | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // Msg returns a *dns.Msg embedded in c. | 
					
						
							| 
									
										
										
										
											2016-03-28 21:18:16 +01:00
										 |  |  | func (c Case) Msg() *dns.Msg { | 
					
						
							|  |  |  | 	m := new(dns.Msg) | 
					
						
							|  |  |  | 	m.SetQuestion(dns.Fqdn(c.Qname), c.Qtype) | 
					
						
							|  |  |  | 	if c.Do { | 
					
						
							|  |  |  | 		o := new(dns.OPT) | 
					
						
							|  |  |  | 		o.Hdr.Name = "." | 
					
						
							|  |  |  | 		o.Hdr.Rrtype = dns.TypeOPT | 
					
						
							|  |  |  | 		o.SetDo() | 
					
						
							|  |  |  | 		o.SetUDPSize(4096) | 
					
						
							|  |  |  | 		m.Extra = []dns.RR{o} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return m | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // A returns an A record from rr. It panics on errors. | 
					
						
							|  |  |  | func A(rr string) *dns.A { r, _ := dns.NewRR(rr); return r.(*dns.A) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // AAAA returns an AAAA record from rr. It panics on errors. | 
					
						
							|  |  |  | func AAAA(rr string) *dns.AAAA { r, _ := dns.NewRR(rr); return r.(*dns.AAAA) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // CNAME returns a CNAME record from rr. It panics on errors. | 
					
						
							|  |  |  | func CNAME(rr string) *dns.CNAME { r, _ := dns.NewRR(rr); return r.(*dns.CNAME) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-26 17:37:06 +08:00
										 |  |  | // DNAME returns a DNAME record from rr. It panics on errors. | 
					
						
							|  |  |  | func DNAME(rr string) *dns.DNAME { r, _ := dns.NewRR(rr); return r.(*dns.DNAME) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // SRV returns a SRV record from rr. It panics on errors. | 
					
						
							|  |  |  | func SRV(rr string) *dns.SRV { r, _ := dns.NewRR(rr); return r.(*dns.SRV) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SOA returns a SOA record from rr. It panics on errors. | 
					
						
							|  |  |  | func SOA(rr string) *dns.SOA { r, _ := dns.NewRR(rr); return r.(*dns.SOA) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NS returns an NS record from rr. It panics on errors. | 
					
						
							|  |  |  | func NS(rr string) *dns.NS { r, _ := dns.NewRR(rr); return r.(*dns.NS) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // PTR returns a PTR record from rr. It panics on errors. | 
					
						
							|  |  |  | func PTR(rr string) *dns.PTR { r, _ := dns.NewRR(rr); return r.(*dns.PTR) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TXT returns a TXT record from rr. It panics on errors. | 
					
						
							|  |  |  | func TXT(rr string) *dns.TXT { r, _ := dns.NewRR(rr); return r.(*dns.TXT) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-06 13:40:46 -05:00
										 |  |  | // HINFO returns a HINFO record from rr. It panics on errors. | 
					
						
							|  |  |  | func HINFO(rr string) *dns.HINFO { r, _ := dns.NewRR(rr); return r.(*dns.HINFO) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // MX returns an MX record from rr. It panics on errors. | 
					
						
							|  |  |  | func MX(rr string) *dns.MX { r, _ := dns.NewRR(rr); return r.(*dns.MX) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RRSIG returns an RRSIG record from rr. It panics on errors. | 
					
						
							|  |  |  | func RRSIG(rr string) *dns.RRSIG { r, _ := dns.NewRR(rr); return r.(*dns.RRSIG) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NSEC returns an NSEC record from rr. It panics on errors. | 
					
						
							|  |  |  | func NSEC(rr string) *dns.NSEC { r, _ := dns.NewRR(rr); return r.(*dns.NSEC) } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // DNSKEY returns a DNSKEY record from rr. It panics on errors. | 
					
						
							| 
									
										
										
										
											2016-04-26 17:57:11 +01:00
										 |  |  | func DNSKEY(rr string) *dns.DNSKEY { r, _ := dns.NewRR(rr); return r.(*dns.DNSKEY) } | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-05 14:39:49 +00:00
										 |  |  | // DS returns a DS record from rr. It panics on errors. | 
					
						
							|  |  |  | func DS(rr string) *dns.DS { r, _ := dns.NewRR(rr); return r.(*dns.DS) } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // OPT returns an OPT record with UDP buffer size set to bufsize and the DO bit set to do. | 
					
						
							| 
									
										
										
										
											2016-04-09 16:17:53 +01:00
										 |  |  | func OPT(bufsize int, do bool) *dns.OPT { | 
					
						
							|  |  |  | 	o := new(dns.OPT) | 
					
						
							|  |  |  | 	o.Hdr.Name = "." | 
					
						
							|  |  |  | 	o.Hdr.Rrtype = dns.TypeOPT | 
					
						
							|  |  |  | 	o.SetVersion(0) | 
					
						
							|  |  |  | 	o.SetUDPSize(uint16(bufsize)) | 
					
						
							|  |  |  | 	if do { | 
					
						
							|  |  |  | 		o.SetDo() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return o | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // Header test if the header in resp matches the header as defined in tc. | 
					
						
							| 
									
										
										
										
											2016-04-02 16:56:16 +01:00
										 |  |  | func Header(t *testing.T, tc Case, resp *dns.Msg) bool { | 
					
						
							|  |  |  | 	if resp.Rcode != tc.Rcode { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 		t.Errorf("Rcode is %q, expected %q", dns.RcodeToString[resp.Rcode], dns.RcodeToString[tc.Rcode]) | 
					
						
							| 
									
										
										
										
											2016-04-02 16:56:16 +01:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(resp.Answer) != len(tc.Answer) { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 		t.Errorf("Answer for %q contained %d results, %d expected", tc.Qname, len(resp.Answer), len(tc.Answer)) | 
					
						
							| 
									
										
										
										
											2016-04-02 16:56:16 +01:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(resp.Ns) != len(tc.Ns) { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 		t.Errorf("Authority for %q contained %d results, %d expected", tc.Qname, len(resp.Ns), len(tc.Ns)) | 
					
						
							| 
									
										
										
										
											2016-04-02 16:56:16 +01:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(resp.Extra) != len(tc.Extra) { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 		t.Errorf("Additional for %q contained %d results, %d expected", tc.Qname, len(resp.Extra), len(tc.Extra)) | 
					
						
							| 
									
										
										
										
											2016-04-02 16:56:16 +01:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // Section tests if the the section in tc matches rr. | 
					
						
							|  |  |  | func Section(t *testing.T, tc Case, sec sect, rr []dns.RR) bool { | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 	section := []dns.RR{} | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | 	switch sec { | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 	case 0: | 
					
						
							|  |  |  | 		section = tc.Answer | 
					
						
							|  |  |  | 	case 1: | 
					
						
							|  |  |  | 		section = tc.Ns | 
					
						
							|  |  |  | 	case 2: | 
					
						
							|  |  |  | 		section = tc.Extra | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i, a := range rr { | 
					
						
							|  |  |  | 		if a.Header().Name != section[i].Header().Name { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 			t.Errorf("RR %d should have a Header Name of %q, but has %q", i, section[i].Header().Name, a.Header().Name) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 			return false | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// 303 signals: don't care what the ttl is. | 
					
						
							|  |  |  | 		if section[i].Header().Ttl != 303 && a.Header().Ttl != section[i].Header().Ttl { | 
					
						
							| 
									
										
										
										
											2016-04-12 21:30:08 +01:00
										 |  |  | 			if _, ok := section[i].(*dns.OPT); !ok { | 
					
						
							|  |  |  | 				// we check edns0 bufize on this one | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Header TTL of %d, but has %d", i, section[i].Header().Ttl, a.Header().Ttl) | 
					
						
							| 
									
										
										
										
											2016-04-12 21:30:08 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		if a.Header().Rrtype != section[i].Header().Rrtype { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 			t.Errorf("RR %d should have a header rr type of %d, but has %d", i, section[i].Header().Rrtype, a.Header().Rrtype) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 			return false | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		switch x := a.(type) { | 
					
						
							|  |  |  | 		case *dns.SRV: | 
					
						
							|  |  |  | 			if x.Priority != section[i].(*dns.SRV).Priority { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Priority of %d, but has %d", i, section[i].(*dns.SRV).Priority, x.Priority) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if x.Weight != section[i].(*dns.SRV).Weight { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Weight of %d, but has %d", i, section[i].(*dns.SRV).Weight, x.Weight) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if x.Port != section[i].(*dns.SRV).Port { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Port of %d, but has %d", i, section[i].(*dns.SRV).Port, x.Port) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if x.Target != section[i].(*dns.SRV).Target { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Target of %q, but has %q", i, section[i].(*dns.SRV).Target, x.Target) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-03-28 21:18:16 +01:00
										 |  |  | 		case *dns.RRSIG: | 
					
						
							|  |  |  | 			if x.TypeCovered != section[i].(*dns.RRSIG).TypeCovered { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a TypeCovered of %d, but has %d", i, section[i].(*dns.RRSIG).TypeCovered, x.TypeCovered) | 
					
						
							| 
									
										
										
										
											2016-03-28 21:18:16 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if x.Labels != section[i].(*dns.RRSIG).Labels { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Labels of %d, but has %d", i, section[i].(*dns.RRSIG).Labels, x.Labels) | 
					
						
							| 
									
										
										
										
											2016-03-28 21:18:16 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if x.SignerName != section[i].(*dns.RRSIG).SignerName { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a SignerName of %s, but has %s", i, section[i].(*dns.RRSIG).SignerName, x.SignerName) | 
					
						
							| 
									
										
										
										
											2016-03-28 21:18:16 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case *dns.NSEC: | 
					
						
							|  |  |  | 			if x.NextDomain != section[i].(*dns.NSEC).NextDomain { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a NextDomain of %s, but has %s", i, section[i].(*dns.NSEC).NextDomain, x.NextDomain) | 
					
						
							| 
									
										
										
										
											2016-03-28 21:18:16 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// TypeBitMap | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 		case *dns.A: | 
					
						
							|  |  |  | 			if x.A.String() != section[i].(*dns.A).A.String() { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Address of %q, but has %q", i, section[i].(*dns.A).A.String(), x.A.String()) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case *dns.AAAA: | 
					
						
							|  |  |  | 			if x.AAAA.String() != section[i].(*dns.AAAA).AAAA.String() { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Address of %q, but has %q", i, section[i].(*dns.AAAA).AAAA.String(), x.AAAA.String()) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case *dns.TXT: | 
					
						
							|  |  |  | 			for j, txt := range x.Txt { | 
					
						
							|  |  |  | 				if txt != section[i].(*dns.TXT).Txt[j] { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 					t.Errorf("RR %d should have a Txt of %q, but has %q", i, section[i].(*dns.TXT).Txt[j], txt) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 					return false | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-02-06 13:40:46 -05:00
										 |  |  | 		case *dns.HINFO: | 
					
						
							|  |  |  | 			if x.Cpu != section[i].(*dns.HINFO).Cpu { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Cpu of %s, but has %s", i, section[i].(*dns.HINFO).Cpu, x.Cpu) | 
					
						
							| 
									
										
										
										
											2018-02-06 13:40:46 -05:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if x.Os != section[i].(*dns.HINFO).Os { | 
					
						
							| 
									
										
										
										
											2018-06-02 19:48:39 +01:00
										 |  |  | 				t.Errorf("RR %d should have a Os of %s, but has %s", i, section[i].(*dns.HINFO).Os, x.Os) | 
					
						
							| 
									
										
										
										
											2018-02-06 13:40:46 -05:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 		case *dns.SOA: | 
					
						
							|  |  |  | 			tt := section[i].(*dns.SOA) | 
					
						
							|  |  |  | 			if x.Ns != tt.Ns { | 
					
						
							| 
									
										
										
										
											2017-08-18 14:45:20 +01:00
										 |  |  | 				t.Errorf("SOA nameserver should be %q, but is %q", tt.Ns, x.Ns) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case *dns.PTR: | 
					
						
							|  |  |  | 			tt := section[i].(*dns.PTR) | 
					
						
							|  |  |  | 			if x.Ptr != tt.Ptr { | 
					
						
							| 
									
										
										
										
											2017-08-18 14:45:20 +01:00
										 |  |  | 				t.Errorf("PTR ptr should be %q, but is %q", tt.Ptr, x.Ptr) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case *dns.CNAME: | 
					
						
							|  |  |  | 			tt := section[i].(*dns.CNAME) | 
					
						
							|  |  |  | 			if x.Target != tt.Target { | 
					
						
							| 
									
										
										
										
											2017-08-18 14:45:20 +01:00
										 |  |  | 				t.Errorf("CNAME target should be %q, but is %q", tt.Target, x.Target) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case *dns.MX: | 
					
						
							|  |  |  | 			tt := section[i].(*dns.MX) | 
					
						
							|  |  |  | 			if x.Mx != tt.Mx { | 
					
						
							| 
									
										
										
										
											2017-08-18 14:45:20 +01:00
										 |  |  | 				t.Errorf("MX Mx should be %q, but is %q", tt.Mx, x.Mx) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if x.Preference != tt.Preference { | 
					
						
							| 
									
										
										
										
											2017-08-18 14:45:20 +01:00
										 |  |  | 				t.Errorf("MX Preference should be %q, but is %q", tt.Preference, x.Preference) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case *dns.NS: | 
					
						
							|  |  |  | 			tt := section[i].(*dns.NS) | 
					
						
							|  |  |  | 			if x.Ns != tt.Ns { | 
					
						
							| 
									
										
										
										
											2017-08-18 14:45:20 +01:00
										 |  |  | 				t.Errorf("NS nameserver should be %q, but is %q", tt.Ns, x.Ns) | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-04-09 16:17:53 +01:00
										 |  |  | 		case *dns.OPT: | 
					
						
							|  |  |  | 			tt := section[i].(*dns.OPT) | 
					
						
							| 
									
										
										
										
											2016-04-12 21:30:08 +01:00
										 |  |  | 			if x.UDPSize() != tt.UDPSize() { | 
					
						
							|  |  |  | 				t.Errorf("OPT UDPSize should be %d, but is %d", tt.UDPSize(), x.UDPSize()) | 
					
						
							| 
									
										
										
										
											2016-04-09 16:17:53 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-04-12 21:30:08 +01:00
										 |  |  | 			if x.Do() != tt.Do() { | 
					
						
							|  |  |  | 				t.Errorf("OPT DO should be %t, but is %t", tt.Do(), x.Do()) | 
					
						
							| 
									
										
										
										
											2016-04-09 16:17:53 +01:00
										 |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-24 11:05:16 -04:00
										 |  |  | // CNAMEOrder makes sure that CNAMES do not appear after their target records | 
					
						
							|  |  |  | func CNAMEOrder(t *testing.T, res *dns.Msg) { | 
					
						
							|  |  |  | 	for i, c := range res.Answer { | 
					
						
							|  |  |  | 		if c.Header().Rrtype != dns.TypeCNAME { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for _, a := range res.Answer[:i] { | 
					
						
							|  |  |  | 			if a.Header().Name != c.(*dns.CNAME).Target { | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			t.Errorf("CNAME found after target record\n") | 
					
						
							|  |  |  | 			t.Logf("%v\n", res) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 15:30:58 +01:00
										 |  |  | // SortAndCheck sorts resp and the checks the header and three sections against the testcase in tc. | 
					
						
							|  |  |  | func SortAndCheck(t *testing.T, resp *dns.Msg, tc Case) { | 
					
						
							|  |  |  | 	sort.Sort(RRSet(resp.Answer)) | 
					
						
							|  |  |  | 	sort.Sort(RRSet(resp.Ns)) | 
					
						
							|  |  |  | 	sort.Sort(RRSet(resp.Extra)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !Header(t, tc, resp) { | 
					
						
							|  |  |  | 		t.Logf("%v\n", resp) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !Section(t, tc, Answer, resp.Answer) { | 
					
						
							|  |  |  | 		t.Logf("%v\n", resp) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !Section(t, tc, Ns, resp.Ns) { | 
					
						
							|  |  |  | 		t.Logf("%v\n", resp) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !Section(t, tc, Extra, resp.Extra) { | 
					
						
							|  |  |  | 		t.Logf("%v\n", resp) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | // ErrorHandler returns a Handler that returns ServerFailure error when called. | 
					
						
							| 
									
										
										
										
											2016-04-04 08:19:06 +01:00
										 |  |  | func ErrorHandler() Handler { | 
					
						
							|  |  |  | 	return HandlerFunc(func(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) { | 
					
						
							| 
									
										
										
										
											2016-03-28 10:49:28 +01:00
										 |  |  | 		m := new(dns.Msg) | 
					
						
							|  |  |  | 		m.SetRcode(r, dns.RcodeServerFailure) | 
					
						
							|  |  |  | 		w.WriteMsg(m) | 
					
						
							|  |  |  | 		return dns.RcodeServerFailure, nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-04 08:19:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-17 17:09:05 +01:00
										 |  |  | // NextHandler returns a Handler that returns rcode and err. | 
					
						
							|  |  |  | func NextHandler(rcode int, err error) Handler { | 
					
						
							|  |  |  | 	return HandlerFunc(func(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) { | 
					
						
							|  |  |  | 		return rcode, err | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Copied here to prevent an import cycle, so that we can define to above handlers. | 
					
						
							| 
									
										
										
										
											2016-09-23 09:14:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-04 08:19:06 +01:00
										 |  |  | type ( | 
					
						
							|  |  |  | 	// HandlerFunc is a convenience type like dns.HandlerFunc, except | 
					
						
							|  |  |  | 	// ServeDNS returns an rcode and an error. | 
					
						
							|  |  |  | 	HandlerFunc func(context.Context, dns.ResponseWriter, *dns.Msg) (int, error) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 09:36:06 +01:00
										 |  |  | 	// Handler interface defines a plugin. | 
					
						
							| 
									
										
										
										
											2016-04-04 08:19:06 +01:00
										 |  |  | 	Handler interface { | 
					
						
							|  |  |  | 		ServeDNS(context.Context, dns.ResponseWriter, *dns.Msg) (int, error) | 
					
						
							| 
									
										
										
										
											2016-10-26 10:01:52 +01:00
										 |  |  | 		Name() string | 
					
						
							| 
									
										
										
										
											2016-04-04 08:19:06 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ServeDNS implements the Handler interface. | 
					
						
							|  |  |  | func (f HandlerFunc) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) { | 
					
						
							|  |  |  | 	return f(ctx, w, r) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-10-26 10:01:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 11:48:37 +00:00
										 |  |  | // Name implements the Handler interface. | 
					
						
							| 
									
										
										
										
											2016-10-26 10:01:52 +01:00
										 |  |  | func (f HandlerFunc) Name() string { return "handlerfunc" } |