summaryrefslogtreecommitdiffhomepage
path: root/test
diff options
context:
space:
mode:
Diffstat (limited to 'test')
-rw-r--r--test/lib/base.py6
-rw-r--r--test/lib/gobgp.py3
-rw-r--r--test/scenario_test/bgp_malformed_msg_handling_test.py108
3 files changed, 115 insertions, 2 deletions
diff --git a/test/lib/base.py b/test/lib/base.py
index 5b1e4349..9dc44e82 100644
--- a/test/lib/base.py
+++ b/test/lib/base.py
@@ -326,7 +326,8 @@ class BGPContainer(Container):
flowspec=False, bridge='', reload_config=True, as2=False,
graceful_restart=None, local_as=None, prefix_limit=None,
v6=False, llgr=None, vrf='', interface='', allow_as_in=0,
- remove_private_as=None, replace_peer_as=False, addpath=False):
+ remove_private_as=None, replace_peer_as=False, addpath=False,
+ treat_as_withdraw=False):
neigh_addr = ''
local_addr = ''
it = itertools.product(self.ip_addrs, peer.ip_addrs)
@@ -371,7 +372,8 @@ class BGPContainer(Container):
'allow_as_in': allow_as_in,
'remove_private_as': remove_private_as,
'replace_peer_as': replace_peer_as,
- 'addpath': addpath}
+ 'addpath': addpath,
+ 'treat_as_withdraw': treat_as_withdraw}
if self.is_running and reload_config:
self.create_config()
self.reload_config()
diff --git a/test/lib/gobgp.py b/test/lib/gobgp.py
index dfdc6808..7fa90291 100644
--- a/test/lib/gobgp.py
+++ b/test/lib/gobgp.py
@@ -426,6 +426,9 @@ class GoBGPContainer(BGPContainer):
for typ, d in info.get('default-policy', {}).iteritems():
n['apply-policy']['config']['default-{0}-policy'.format(typ)] = _f(d)
+ if info['treat_as_withdraw']:
+ n['error-handling'] = {'config': {'treat-as-withdraw': True}}
+
config['neighbors'].append(n)
config['defined-sets'] = {}
diff --git a/test/scenario_test/bgp_malformed_msg_handling_test.py b/test/scenario_test/bgp_malformed_msg_handling_test.py
new file mode 100644
index 00000000..feaac264
--- /dev/null
+++ b/test/scenario_test/bgp_malformed_msg_handling_test.py
@@ -0,0 +1,108 @@
+# Copyright (C) 2017 Nippon Telegraph and Telephone Corporation.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+# implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+from __future__ import absolute_import
+
+import sys
+import time
+import unittest
+
+from fabric.api import local
+import nose
+
+from lib.noseplugin import OptionParser, parser_option
+
+from lib import base
+from lib.base import BGP_FSM_ESTABLISHED
+from lib.gobgp import GoBGPContainer
+from lib.exabgp import ExaBGPContainer
+
+
+class GoBGPTestBase(unittest.TestCase):
+
+ @classmethod
+ def setUpClass(cls):
+ gobgp_ctn_image_name = parser_option.gobgp_image
+ base.TEST_PREFIX = parser_option.test_prefix
+
+ g1 = GoBGPContainer(name='g1', asn=65000, router_id='192.168.0.1',
+ ctn_image_name=gobgp_ctn_image_name,
+ log_level=parser_option.gobgp_log_level)
+ e1 = ExaBGPContainer(name='e1', asn=65001, router_id='192.168.0.2')
+
+ ctns = [g1, e1]
+ initial_wait_time = max(ctn.run() for ctn in ctns)
+ time.sleep(initial_wait_time)
+
+ g1.add_peer(e1, treat_as_withdraw=True)
+ e1.add_peer(g1)
+
+ cls.g1 = g1
+ cls.e1 = e1
+
+ # test each neighbor state is turned establish
+ def test_01_neighbor_established(self):
+ self.g1.wait_for(expected_state=BGP_FSM_ESTABLISHED, peer=self.e1)
+
+ def test_02_attribute_discard(self):
+ # Malformed attribute 'AGGREGATOR' should be discard, but the session should not be disconnected.
+ self.e1.add_route('10.0.0.0/24', attribute='0x07 0xc0 0x0000006400')
+
+ # Confirm the session is not disconnected
+ for _ in range(5):
+ state = self.g1.get_neighbor_state(self.e1)
+ self.assertTrue(BGP_FSM_ESTABLISHED, state)
+ time.sleep(1)
+
+ # Confirm the path is added
+ dests = self.g1.get_global_rib()
+ self.assertTrue(len(dests) == 1)
+ routes = dests[0]['paths']
+ self.assertTrue(len(routes) == 1)
+
+ # Confirm the attribute 'AGGREGATOR(type=7)' is discarded
+ for d in routes[0]['attrs']:
+ self.assertFalse(d['type'] == 7)
+
+ self.e1.del_route('10.0.0.0/24')
+
+ def test_03_treat_as_withdraw(self):
+ # Malformed attribute 'MULTI_EXIT_DESC' should be treated as withdraw,
+ # but the session should not be disconnected.
+ self.e1.add_route('20.0.0.0/24', attribute='0x04 0x80 0x00000064')
+ self.e1.add_route('30.0.0.0/24', attribute='0x04 0x80 0x00000064')
+ # Malformed
+ self.e1.add_route('30.0.0.0/24', attribute='0x04 0x80 0x0000000064')
+
+ # Confirm the session is not disconnected
+ for _ in range(5):
+ state = self.g1.get_neighbor_state(self.e1)
+ self.assertTrue(BGP_FSM_ESTABLISHED, state)
+ time.sleep(1)
+
+ # Confirm the number of path in RIB is only one
+ dests = self.g1.get_global_rib()
+ self.assertTrue(len(dests) == 1)
+ self.assertTrue(dests[0]['paths'][0]['nlri']['prefix'] == '20.0.0.0/24')
+
+
+if __name__ == '__main__':
+ output = local("which docker 2>&1 > /dev/null ; echo $?", capture=True)
+ if int(output) is not 0:
+ print "docker not found"
+ sys.exit(1)
+
+ nose.main(argv=sys.argv, addplugins=[OptionParser()],
+ defaultTest=sys.argv[0])