|
@@ -66,6 +66,30 @@ example_soa_question = Question(TEST_ZONE_NAME, TEST_RRCLASS, RRType.SOA())
|
|
|
default_questions = [example_axfr_question]
|
|
|
default_answers = [soa_rrset]
|
|
|
|
|
|
+def check_diffs(assert_fn, expected, actual):
|
|
|
+ '''A helper function checking the differences made in the XFR session.
|
|
|
+
|
|
|
+ This is expected called from some subclass of unittest.TestCase and
|
|
|
+ assert_fn is generally expected to be 'self.assertEqual' of that class.
|
|
|
+
|
|
|
+ '''
|
|
|
+ assert_fn(len(expected), len(actual))
|
|
|
+ for (diffs_exp, diffs_actual) in zip(expected, actual):
|
|
|
+ assert_fn(len(diffs_exp), len(diffs_actual))
|
|
|
+ for (diff_exp, diff_actual) in zip(diffs_exp, diffs_actual):
|
|
|
+
|
|
|
+ assert_fn(diff_exp[0], diff_actual[0])
|
|
|
+
|
|
|
+
|
|
|
+ assert_fn(diff_exp[1].get_name(), diff_actual[1].get_name())
|
|
|
+ assert_fn(diff_exp[1].get_type(), diff_actual[1].get_type())
|
|
|
+ assert_fn(diff_exp[1].get_class(), diff_actual[1].get_class())
|
|
|
+ assert_fn(diff_exp[1].get_rdata_count(),
|
|
|
+ diff_actual[1].get_rdata_count())
|
|
|
+ assert_fn(1, diff_exp[1].get_rdata_count())
|
|
|
+ assert_fn(diff_exp[1].get_rdata()[0],
|
|
|
+ diff_actual[1].get_rdata()[0])
|
|
|
+
|
|
|
class XfrinTestException(Exception):
|
|
|
pass
|
|
|
|
|
@@ -274,8 +298,12 @@ class TestXfrinState(unittest.TestCase):
|
|
|
RRTTL(3600))
|
|
|
self.ns_rrset.add_rdata(Rdata(RRType.NS(), TEST_RRCLASS,
|
|
|
'ns.example.com'))
|
|
|
+ self.a_rrset = RRset(TEST_ZONE_NAME, TEST_RRCLASS, RRType.A(),
|
|
|
+ RRTTL(3600))
|
|
|
+ self.a_rrset.add_rdata(Rdata(RRType.A(), TEST_RRCLASS, '192.0.2.1'))
|
|
|
+
|
|
|
self.conn._datasrc_client = MockDataSourceClient()
|
|
|
- self.conn._diff = Diff(MockDataSourceClient(), TEST_ZONE_NAME)
|
|
|
+ self.conn._diff = Diff(self.conn._datasrc_client, TEST_ZONE_NAME)
|
|
|
|
|
|
class TestXfrinStateBase(TestXfrinState):
|
|
|
def setUp(self):
|
|
@@ -312,6 +340,7 @@ class TestXfrinFirstData(TestXfrinState):
|
|
|
self.state = XfrinFirstData()
|
|
|
self.conn._request_type = RRType.IXFR()
|
|
|
self.conn._request_serial = 1230
|
|
|
+ self.conn._diff = None
|
|
|
|
|
|
def test_handle_ixfr_begin_soa(self):
|
|
|
self.conn._request_type = RRType.IXFR()
|
|
@@ -331,6 +360,8 @@ class TestXfrinFirstData(TestXfrinState):
|
|
|
|
|
|
self.assertFalse(self.state.handle_rr(self.conn, self.ns_rrset))
|
|
|
self.assertEqual(type(XfrinAXFR()), type(self.conn.get_xfrstate()))
|
|
|
+
|
|
|
+ self.assertNotEqual(None, self.conn._diff)
|
|
|
|
|
|
def test_handle_ixfr_to_axfr_by_different_soa(self):
|
|
|
|
|
@@ -338,6 +369,7 @@ class TestXfrinFirstData(TestXfrinState):
|
|
|
|
|
|
self.assertFalse(self.state.handle_rr(self.conn, soa_rrset))
|
|
|
self.assertEqual(type(XfrinAXFR()), type(self.conn.get_xfrstate()))
|
|
|
+ self.assertNotEqual(None, self.conn._diff)
|
|
|
|
|
|
def test_finish_message(self):
|
|
|
self.assertTrue(self.state.finish_message(self.conn))
|
|
@@ -475,12 +507,51 @@ class TestXfrinAXFR(TestXfrinState):
|
|
|
self.state = XfrinAXFR()
|
|
|
|
|
|
def test_handle_rr(self):
|
|
|
- self.assertRaises(XfrinException, self.state.handle_rr, self.conn,
|
|
|
- soa_rrset)
|
|
|
+ """
|
|
|
+ Test we can put data inside.
|
|
|
+ """
|
|
|
+
|
|
|
+ self.assertTrue(self.state.handle_rr(self.conn, self.a_rrset))
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ self.assertEqual([('add', self.a_rrset)], self.conn._diff.get_buffer())
|
|
|
+
|
|
|
+ self.assertTrue(self.state.handle_rr(self.conn, soa_rrset))
|
|
|
+
|
|
|
+ self.assertEqual(type(XfrinAXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
+
|
|
|
+ self.assertEqual([('add', self.a_rrset), ('add', soa_rrset)],
|
|
|
+ self.conn._diff.get_buffer())
|
|
|
|
|
|
def test_finish_message(self):
|
|
|
+ """
|
|
|
+ Check normal end of message.
|
|
|
+ """
|
|
|
+
|
|
|
self.assertTrue(self.state.finish_message(self.conn))
|
|
|
|
|
|
+class TestXfrinAXFREnd(TestXfrinState):
|
|
|
+ def setUp(self):
|
|
|
+ super().setUp()
|
|
|
+ self.state = XfrinAXFREnd()
|
|
|
+
|
|
|
+ def test_handle_rr(self):
|
|
|
+ self.assertRaises(XfrinProtocolError, self.state.handle_rr, self.conn,
|
|
|
+ self.ns_rrset)
|
|
|
+
|
|
|
+ def test_finish_message(self):
|
|
|
+ self.conn._diff.add_data(self.a_rrset)
|
|
|
+ self.conn._diff.add_data(soa_rrset)
|
|
|
+ self.assertFalse(self.state.finish_message(self.conn))
|
|
|
+
|
|
|
+
|
|
|
+ self.assertEqual([], self.conn._diff.get_buffer())
|
|
|
+ check_diffs(self.assertEqual, [[('add', self.a_rrset),
|
|
|
+ ('add', soa_rrset)]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
+ self.assertRaises(ValueError, self.conn._diff.commit)
|
|
|
+
|
|
|
class TestXfrinConnection(unittest.TestCase):
|
|
|
'''Convenient parent class for XFR-protocol tests.
|
|
|
|
|
@@ -569,30 +640,6 @@ class TestXfrinConnection(unittest.TestCase):
|
|
|
self.conn.reply_data = struct.pack('H', socket.htons(len(bogus_data)))
|
|
|
self.conn.reply_data += bogus_data
|
|
|
|
|
|
- def check_diffs(self, expected, actual):
|
|
|
- '''A helper method checking the differences made in the IXFR session.
|
|
|
-
|
|
|
- '''
|
|
|
- self.assertEqual(len(expected), len(actual))
|
|
|
- for (diffs_exp, diffs_actual) in zip(expected, actual):
|
|
|
- self.assertEqual(len(diffs_exp), len(diffs_actual))
|
|
|
- for (diff_exp, diff_actual) in zip(diffs_exp, diffs_actual):
|
|
|
-
|
|
|
- self.assertEqual(diff_exp[0], diff_actual[0])
|
|
|
-
|
|
|
-
|
|
|
- self.assertEqual(diff_exp[1].get_name(),
|
|
|
- diff_actual[1].get_name())
|
|
|
- self.assertEqual(diff_exp[1].get_type(),
|
|
|
- diff_actual[1].get_type())
|
|
|
- self.assertEqual(diff_exp[1].get_class(),
|
|
|
- diff_actual[1].get_class())
|
|
|
- self.assertEqual(diff_exp[1].get_rdata_count(),
|
|
|
- diff_actual[1].get_rdata_count())
|
|
|
- self.assertEqual(1, diff_exp[1].get_rdata_count())
|
|
|
- self.assertEqual(diff_exp[1].get_rdata()[0],
|
|
|
- diff_actual[1].get_rdata()[0])
|
|
|
-
|
|
|
def _create_a(self, address):
|
|
|
rrset = RRset(Name('a.example.com'), TEST_RRCLASS, RRType.A(),
|
|
|
RRTTL(3600))
|
|
@@ -606,6 +653,11 @@ class TestXfrinConnection(unittest.TestCase):
|
|
|
rrset.add_rdata(Rdata(RRType.SOA(), TEST_RRCLASS, rdata_str))
|
|
|
return rrset
|
|
|
|
|
|
+ def _create_ns(self, nsname='ns.'+TEST_ZONE_NAME_STR):
|
|
|
+ rrset = RRset(TEST_ZONE_NAME, TEST_RRCLASS, RRType.NS(), RRTTL(3600))
|
|
|
+ rrset.add_rdata(Rdata(RRType.NS(), TEST_RRCLASS, nsname))
|
|
|
+ return rrset
|
|
|
+
|
|
|
class TestAXFR(TestXfrinConnection):
|
|
|
def setUp(self):
|
|
|
super().setUp()
|
|
@@ -1081,8 +1133,9 @@ class TestIXFRResponse(TestXfrinConnection):
|
|
|
self.conn._handle_xfrin_responses()
|
|
|
self.assertEqual(type(XfrinIXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
self.assertEqual([], self.conn._datasrc_client.diffs)
|
|
|
- self.check_diffs([[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
- self.conn._datasrc_client.committed_diffs)
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
|
|
|
def test_ixfr_response_multi_sequences(self):
|
|
|
'''Similar to the previous case, but with multiple diff seqs.
|
|
@@ -1107,19 +1160,20 @@ class TestIXFRResponse(TestXfrinConnection):
|
|
|
self.conn._handle_xfrin_responses()
|
|
|
self.assertEqual(type(XfrinIXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
self.assertEqual([], self.conn._datasrc_client.diffs)
|
|
|
- self.check_diffs([[('delete', begin_soa_rrset),
|
|
|
- ('delete', self._create_a('192.0.2.1')),
|
|
|
- ('add', self._create_soa('1231')),
|
|
|
- ('add', self._create_a('192.0.2.2'))],
|
|
|
- [('delete', self._create_soa('1231')),
|
|
|
- ('delete', self._create_a('192.0.2.3')),
|
|
|
- ('add', self._create_soa('1232')),
|
|
|
- ('add', self._create_a('192.0.2.4'))],
|
|
|
- [('delete', self._create_soa('1232')),
|
|
|
- ('delete', self._create_a('192.0.2.5')),
|
|
|
- ('add', soa_rrset),
|
|
|
- ('add', self._create_a('192.0.2.6'))]],
|
|
|
- self.conn._datasrc_client.committed_diffs)
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [[('delete', begin_soa_rrset),
|
|
|
+ ('delete', self._create_a('192.0.2.1')),
|
|
|
+ ('add', self._create_soa('1231')),
|
|
|
+ ('add', self._create_a('192.0.2.2'))],
|
|
|
+ [('delete', self._create_soa('1231')),
|
|
|
+ ('delete', self._create_a('192.0.2.3')),
|
|
|
+ ('add', self._create_soa('1232')),
|
|
|
+ ('add', self._create_a('192.0.2.4'))],
|
|
|
+ [('delete', self._create_soa('1232')),
|
|
|
+ ('delete', self._create_a('192.0.2.5')),
|
|
|
+ ('add', soa_rrset),
|
|
|
+ ('add', self._create_a('192.0.2.6'))]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
|
|
|
def test_ixfr_response_multi_messages(self):
|
|
|
'''Similar to the first case, but RRs span over multiple messages.
|
|
@@ -1133,8 +1187,9 @@ class TestIXFRResponse(TestXfrinConnection):
|
|
|
answers=[soa_rrset])
|
|
|
self.conn._handle_xfrin_responses()
|
|
|
self.assertEqual(type(XfrinIXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
- self.check_diffs([[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
- self.conn._datasrc_client.committed_diffs)
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
|
|
|
def test_ixfr_response_broken(self):
|
|
|
'''Test with a broken response.
|
|
@@ -1148,7 +1203,8 @@ class TestIXFRResponse(TestXfrinConnection):
|
|
|
self.assertRaises(XfrinProtocolError,
|
|
|
self.conn._handle_xfrin_responses)
|
|
|
|
|
|
- self.check_diffs([], self.conn._datasrc_client.committed_diffs)
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [], self.conn._datasrc_client.committed_diffs)
|
|
|
|
|
|
def test_ixfr_response_extra(self):
|
|
|
'''Test with an extra RR after the end of IXFR diff sequences.
|
|
@@ -1163,8 +1219,64 @@ class TestIXFRResponse(TestXfrinConnection):
|
|
|
self._create_a('192.0.2.1')])
|
|
|
self.assertRaises(XfrinProtocolError,
|
|
|
self.conn._handle_xfrin_responses)
|
|
|
- self.check_diffs([[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
- self.conn._datasrc_client.committed_diffs)
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
+
|
|
|
+ def test_ixfr_to_axfr_response(self):
|
|
|
+ '''AXFR-style IXFR response.
|
|
|
+
|
|
|
+ It simply updates the zone's SOA one time.
|
|
|
+
|
|
|
+ '''
|
|
|
+ ns_rr = self._create_ns()
|
|
|
+ a_rr = self._create_a('192.0.2.1')
|
|
|
+ self.conn.reply_data = self.conn.create_response_data(
|
|
|
+ questions=[Question(TEST_ZONE_NAME, TEST_RRCLASS, RRType.IXFR())],
|
|
|
+ answers=[soa_rrset, ns_rr, a_rr, soa_rrset])
|
|
|
+ self.conn._handle_xfrin_responses()
|
|
|
+ self.assertEqual(type(XfrinAXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
+ self.assertEqual([], self.conn._datasrc_client.diffs)
|
|
|
+
|
|
|
+
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [[('add', ns_rr), ('add', a_rr), ('add', soa_rrset)]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
+
|
|
|
+ def test_ixfr_to_axfr_response_mismatch_soa(self):
|
|
|
+ '''AXFR-style IXFR response, but the two SOA are not the same.
|
|
|
+
|
|
|
+ In the current implementation, we accept it and use the second SOA.
|
|
|
+
|
|
|
+ '''
|
|
|
+ ns_rr = self._create_ns()
|
|
|
+ a_rr = self._create_a('192.0.2.1')
|
|
|
+ self.conn.reply_data = self.conn.create_response_data(
|
|
|
+ questions=[Question(TEST_ZONE_NAME, TEST_RRCLASS, RRType.IXFR())],
|
|
|
+ answers=[soa_rrset, ns_rr, a_rr, begin_soa_rrset])
|
|
|
+ self.conn._handle_xfrin_responses()
|
|
|
+ self.assertEqual(type(XfrinAXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
+ self.assertEqual([], self.conn._datasrc_client.diffs)
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [[('add', ns_rr), ('add', a_rr),
|
|
|
+ ('add', begin_soa_rrset)]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
+
|
|
|
+ def test_ixfr_to_axfr_response_extra(self):
|
|
|
+ '''Test with an extra RR after the end of AXFR-style IXFR session.
|
|
|
+
|
|
|
+ The session should be rejected, and nothing should be committed.
|
|
|
+
|
|
|
+ '''
|
|
|
+ ns_rr = self._create_ns()
|
|
|
+ a_rr = self._create_a('192.0.2.1')
|
|
|
+ self.conn.reply_data = self.conn.create_response_data(
|
|
|
+ questions=[Question(TEST_ZONE_NAME, TEST_RRCLASS, RRType.IXFR())],
|
|
|
+ answers=[soa_rrset, ns_rr, a_rr, soa_rrset, a_rr])
|
|
|
+ self.assertRaises(XfrinProtocolError,
|
|
|
+ self.conn._handle_xfrin_responses)
|
|
|
+ self.assertEqual(type(XfrinAXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
+ self.assertEqual([], self.conn._datasrc_client.committed_diffs)
|
|
|
|
|
|
class TestIXFRSession(TestXfrinConnection):
|
|
|
'''Tests for a full IXFR session (query and response).
|
|
@@ -1187,8 +1299,9 @@ class TestIXFRSession(TestXfrinConnection):
|
|
|
|
|
|
|
|
|
self.assertEqual(type(XfrinIXFREnd()), type(self.conn.get_xfrstate()))
|
|
|
- self.check_diffs([[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
- self.conn._datasrc_client.committed_diffs)
|
|
|
+ check_diffs(self.assertEqual,
|
|
|
+ [[('delete', begin_soa_rrset), ('add', soa_rrset)]],
|
|
|
+ self.conn._datasrc_client.committed_diffs)
|
|
|
|
|
|
|
|
|
qdata = self.conn.query_data[2:]
|
|
@@ -1249,6 +1362,12 @@ class TestIXFRSessionWithSQLite3(TestXfrinConnection):
|
|
|
self.assertEqual(1, soa.get_rdata_count())
|
|
|
return get_soa_serial(soa.get_rdata()[0])
|
|
|
|
|
|
+ def record_exist(self, name, type):
|
|
|
+ result, finder = self.conn._datasrc_client.find_zone(TEST_ZONE_NAME)
|
|
|
+ self.assertEqual(DataSourceClient.SUCCESS, result)
|
|
|
+ result, soa = finder.find(name, type, None, ZoneFinder.FIND_DEFAULT)
|
|
|
+ return result == ZoneFinder.SUCCESS
|
|
|
+
|
|
|
def test_do_xfrin_sqlite3(self):
|
|
|
def create_ixfr_response():
|
|
|
self.conn.reply_data = self.conn.create_response_data(
|
|
@@ -1280,6 +1399,47 @@ class TestIXFRSessionWithSQLite3(TestXfrinConnection):
|
|
|
self.assertEqual(XFRIN_FAIL, self.conn.do_xfrin(False, RRType.IXFR()))
|
|
|
self.assertEqual(1230, self.get_zone_serial())
|
|
|
|
|
|
+ def test_do_xfrin_axfr_sqlite3(self):
|
|
|
+ '''AXFR-style IXFR.
|
|
|
+
|
|
|
+ '''
|
|
|
+ def create_ixfr_response():
|
|
|
+ self.conn.reply_data = self.conn.create_response_data(
|
|
|
+ questions=[Question(TEST_ZONE_NAME, TEST_RRCLASS,
|
|
|
+ RRType.IXFR())],
|
|
|
+ answers=[soa_rrset, self._create_ns(), soa_rrset])
|
|
|
+ self.conn.response_generator = create_ixfr_response
|
|
|
+
|
|
|
+
|
|
|
+ self.assertEqual(1230, self.get_zone_serial())
|
|
|
+ self.assertTrue(self.record_exist(Name('dns01.example.com'),
|
|
|
+ RRType.A()))
|
|
|
+ self.assertEqual(XFRIN_OK, self.conn.do_xfrin(False, RRType.IXFR()))
|
|
|
+ self.assertEqual(1234, self.get_zone_serial())
|
|
|
+ self.assertFalse(self.record_exist(Name('dns01.example.com'),
|
|
|
+ RRType.A()))
|
|
|
+
|
|
|
+ def test_do_xfrin_axfr_sqlite3_fail(self):
|
|
|
+ '''Similar to the previous test, but xfrin fails due to error.
|
|
|
+
|
|
|
+ Check the DB is not changed.
|
|
|
+
|
|
|
+ '''
|
|
|
+ def create_ixfr_response():
|
|
|
+ self.conn.reply_data = self.conn.create_response_data(
|
|
|
+ questions=[Question(TEST_ZONE_NAME, TEST_RRCLASS,
|
|
|
+ RRType.IXFR())],
|
|
|
+ answers=[soa_rrset, self._create_ns(), soa_rrset, soa_rrset])
|
|
|
+ self.conn.response_generator = create_ixfr_response
|
|
|
+
|
|
|
+ self.assertEqual(1230, self.get_zone_serial())
|
|
|
+ self.assertTrue(self.record_exist(Name('dns01.example.com'),
|
|
|
+ RRType.A()))
|
|
|
+ self.assertEqual(XFRIN_FAIL, self.conn.do_xfrin(False, RRType.IXFR()))
|
|
|
+ self.assertEqual(1230, self.get_zone_serial())
|
|
|
+ self.assertTrue(self.record_exist(Name('dns01.example.com'),
|
|
|
+ RRType.A()))
|
|
|
+
|
|
|
class TestXfrinRecorder(unittest.TestCase):
|
|
|
def setUp(self):
|
|
|
self.recorder = XfrinRecorder()
|