class DNSTest(TestCase):
def setUp(self):
- global server, server_ip, lp, creds
super(DNSTest, self).setUp()
- self.server = server_name
- self.server_ip = server_ip
- self.lp = lp
- self.creds = creds
+ self.timeout = None
def errstr(self, errcode):
"Return a readable error code"
"NXRRSET",
"NOTAUTH",
"NOTZONE",
+ "0x0B",
+ "0x0C",
+ "0x0D",
+ "0x0E",
+ "0x0F",
+ "BADSIG",
+ "BADKEY"
]
return string_codes[errcode]
+ def assert_rcode_equals(self, rcode, expected):
+ "Helper function to check return code"
+ self.assertEquals(rcode, expected, "Expected RCODE %s, got %s" %
+ (self.errstr(expected), self.errstr(rcode)))
def assert_dns_rcode_equals(self, packet, rcode):
"Helper function to check return code"
p_errcode = packet.operation & 0x000F
self.assertEquals(p_errcode, rcode, "Expected RCODE %s, got %s" %
- (self.errstr(rcode), self.errstr(p_errcode)))
+ (self.errstr(rcode), self.errstr(p_errcode)))
def assert_dns_opcode_equals(self, packet, opcode):
"Helper function to check opcode"
p_opcode = packet.operation & 0x7800
self.assertEquals(p_opcode, opcode, "Expected OPCODE %s, got %s" %
- (opcode, p_opcode))
+ (opcode, p_opcode))
def make_name_packet(self, opcode, qid=None):
"Helper creating a dns.name_packet"
p = dns.name_packet()
if qid is None:
- p.id = random.randint(0x0, 0xffff)
+ p.id = random.randint(0x0, 0xff00)
p.operation = opcode
p.questions = []
+ p.additional = []
return p
def finish_name_packet(self, packet, questions):
"Helper to get dns domain"
return self.creds.get_realm().lower()
- def dns_transaction_udp(self, packet, host=server_ip,
- dump=False, timeout=timeout):
+ def dns_transaction_udp(self, packet, host,
+ dump=False, timeout=None):
"send a DNS query and read the reply"
s = None
+ if timeout is None:
+ timeout = self.timeout
try:
send_packet = ndr.ndr_pack(packet)
if dump:
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, 0)
s.settimeout(timeout)
s.connect((host, 53))
- s.send(send_packet, 0)
+ s.sendall(send_packet, 0)
recv_packet = s.recv(2048, 0)
if dump:
print self.hexdump(recv_packet)
- return ndr.ndr_unpack(dns.name_packet, recv_packet)
+ response = ndr.ndr_unpack(dns.name_packet, recv_packet)
+ return (response, recv_packet)
finally:
if s is not None:
s.close()
- def dns_transaction_tcp(self, packet, host=server_ip,
- dump=False, timeout=timeout):
- "send a DNS query and read the reply"
+ def dns_transaction_tcp(self, packet, host,
+ dump=False, timeout=None):
+ "send a DNS query and read the reply, also return the raw packet"
s = None
+ if timeout is None:
+ timeout = self.timeout
try:
send_packet = ndr.ndr_pack(packet)
if dump:
s.connect((host, 53))
tcp_packet = struct.pack('!H', len(send_packet))
tcp_packet += send_packet
- s.send(tcp_packet, 0)
+ s.sendall(tcp_packet)
+
recv_packet = s.recv(0xffff + 2, 0)
if dump:
print self.hexdump(recv_packet)
- return ndr.ndr_unpack(dns.name_packet, recv_packet[2:])
+ response = ndr.ndr_unpack(dns.name_packet, recv_packet[2:])
+
finally:
- if s is not None:
- s.close()
+ if s is not None:
+ s.close()
+
+ # unpacking and packing again should produce same bytestream
+ my_packet = ndr.ndr_pack(response)
+ self.assertEquals(my_packet, recv_packet[2:])
+
+ return (response, recv_packet[2:])
def make_txt_update(self, prefix, txt_array):
p = self.make_name_packet(dns.DNS_OPCODE_UPDATE)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=self.server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assertEquals(response.ancount, 1)
self.assertEquals(response.answers[0].rdata.txt.str, txt_array)
+
class TestSimpleQueries(DNSTest):
+ def setUp(self):
+ super(TestSimpleQueries, self).setUp()
+ global server, server_ip, lp, creds, timeout
+ self.server = server_name
+ self.server_ip = server_ip
+ self.lp = lp
+ self.creds = creds
+ self.timeout = timeout
def test_one_a_query(self):
"create a query packet containing one query record"
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 1)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 1)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_tcp(p)
+ (response, response_packet) = self.dns_transaction_tcp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 1)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 0)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NXDOMAIN)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 0)
self.finish_name_packet(p, questions)
try:
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_FORMERR)
except socket.timeout:
# Windows chooses not to respond to incorrectly formatted queries.
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
num_answers = 1
dc_ipv6 = os.getenv('SERVER_IPV6')
self.finish_name_packet(p, questions)
try:
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NOTIMP)
except socket.timeout:
# Windows chooses not to respond to incorrectly formatted queries.
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
# We don't get SOA records for single hosts
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 1)
class TestDNSUpdates(DNSTest):
+ def setUp(self):
+ super(TestDNSUpdates, self).setUp()
+ global server, server_ip, lp, creds, timeout
+ self.server = server_name
+ self.server_ip = server_ip
+ self.lp = lp
+ self.creds = creds
+ self.timeout = timeout
def test_two_updates(self):
"create two update requests"
self.finish_name_packet(p, updates)
try:
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_FORMERR)
except socket.timeout:
# Windows chooses not to respond to incorrectly formatted queries.
updates.append(u)
self.finish_name_packet(p, updates)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NOTIMP)
def test_update_prereq_with_non_null_ttl(self):
p.answers = prereqs
try:
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_FORMERR)
except socket.timeout:
# Windows chooses not to respond to incorrectly formatted queries.
p.ancount = len(prereqs)
p.answers = prereqs
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NXRRSET)
def test_update_prereq_nonexisting_name(self):
p.ancount = len(prereqs)
p.answers = prereqs
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NXRRSET)
def test_update_add_txt_record(self):
"test adding records works"
prefix, txt = 'textrec', ['"This is a test"']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
p.nscount = len(updates)
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
# Now check the record is around
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
# Now delete the record
p.nscount = len(updates)
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
# And finally check it's gone
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NXDOMAIN)
def test_readd_record(self):
p.nscount = len(updates)
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
# Now check the record is around
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
# Now delete the record
p.nscount = len(updates)
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
# check it's gone
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NXDOMAIN)
# recreate the record
p.nscount = len(updates)
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
# Now check the record is around
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
def test_update_add_mx_record(self):
p.nscount = len(updates)
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
p = self.make_name_packet(dns.DNS_OPCODE_QUERY)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assertEqual(response.ancount, 1)
ans = response.answers[0]
updates = [r]
p.nscount = 1
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
def setUp(self):
super(TestComplexQueries, self).setUp()
- def tearDown(self):
- super(TestComplexQueries, self).tearDown()
+ global server, server_ip, lp, creds, timeout
+ self.server = server_name
+ self.server_ip = server_ip
+ self.lp = lp
+ self.creds = creds
+ self.timeout = timeout
def test_one_a_query(self):
"create a query packet containing one query record"
- name = "cname_test.%s" % self.get_dns_domain()
- rdata = "%s.%s" % (self.server, self.get_dns_domain())
- self.make_dns_update(name, rdata, dns.DNS_QTYPE_CNAME)
-
try:
# Create the record
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=self.server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 2)
p.nscount = len(updates)
p.nsrecs = updates
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=self.server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
def test_cname_two_chain(self):
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 3)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.answers[1].rdata, name0)
class TestInvalidQueries(DNSTest):
+ def setUp(self):
+ super(TestInvalidQueries, self).setUp()
+ global server, server_ip, lp, creds, timeout
+ self.server = server_name
+ self.server_ip = server_ip
+ self.lp = lp
+ self.creds = creds
+ self.timeout = timeout
def test_one_a_query(self):
"send 0 bytes follows by create a query packet containing one query record"
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=self.server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 1)
class TestZones(DNSTest):
def setUp(self):
super(TestZones, self).setUp()
+ global server, server_ip, lp, creds, timeout
+ self.server = server_name
+ self.server_ip = server_ip
+ self.lp = lp
+ self.creds = creds
+ self.timeout = timeout
+
self.zone = "test.lan"
self.rpc_conn = dnsserver.dnsserver("ncacn_ip_tcp:%s[sign]" % (self.server_ip),
self.lp, self.creds)
questions.append(q)
self.finish_name_packet(p, questions)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
# Windows returns OK while BIND logically seems to return NXDOMAIN
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NXDOMAIN)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 0)
self.create_zone(zone)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 1)
self.assertEquals(response.answers[0].rr_type, dns.DNS_QTYPE_SOA)
self.delete_zone(zone)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_NXDOMAIN)
self.assert_dns_opcode_equals(response, dns.DNS_OPCODE_QUERY)
self.assertEquals(response.ancount, 0)
class TestRPCRoundtrip(DNSTest):
def setUp(self):
super(TestRPCRoundtrip, self).setUp()
+ global server, server_ip, lp, creds
+ self.server = server_name
+ self.server_ip = server_ip
+ self.lp = lp
+ self.creds = creds
self.rpc_conn = dnsserver.dnsserver("ncacn_ip_tcp:%s[sign]" % (self.server_ip),
self.lp, self.creds)
"test adding records works"
prefix, txt = 'pad1textrec', ['"This is a test"', '', '']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
prefix, txt = 'pad2textrec', ['"This is a test"', '', '', 'more text']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
prefix, txt = 'pad3textrec', ['', '', '"This is a test"']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
"test adding records works"
prefix, txt = 'nulltextrec', ['NULL\x00BYTE']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, ['NULL'])
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
prefix, txt = 'nulltextrec2', ['NULL\x00BYTE', 'NULL\x00BYTE']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, ['NULL', 'NULL'])
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
"test adding records works"
prefix, txt = 'hextextrec', ['HIGH\xFFBYTE']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
"test adding records works"
prefix, txt = 'slashtextrec', ['Th\\=is=is a test']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
prefix, txt = 'textrec2', ['"This is a test"',
'"and this is a test, too"']
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,
"test adding two txt records works"
prefix, txt = 'emptytextrec', []
p = self.make_txt_update(prefix, txt)
- response = self.dns_transaction_udp(p)
+ (response, response_packet) = self.dns_transaction_udp(p, host=server_ip)
self.assert_dns_rcode_equals(response, dns.DNS_RCODE_OK)
self.check_query_txt(prefix, txt)
self.assertIsNotNone(dns_record_match(self.rpc_conn, self.server_ip,