diff options
author | Wataru Ishida <ishida.wataru@lab.ntt.co.jp> | 2016-10-09 07:18:13 -0700 |
---|---|---|
committer | FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp> | 2016-10-09 07:18:13 -0700 |
commit | 614746ca1159fe421047df04c5af6f07c38b2e65 (patch) | |
tree | 77e93a40ec38318a96ab3a195fb1baa68372acd2 /test | |
parent | 6b6f6974fcea37dc006f90dbd2f8d65495048725 (diff) |
*: support long lived graceful restart
Signed-off-by: Wataru Ishida <ishida.wataru@lab.ntt.co.jp>
Diffstat (limited to 'test')
-rw-r--r-- | test/lib/base.py | 5 | ||||
-rw-r--r-- | test/lib/gobgp.py | 6 | ||||
-rw-r--r-- | test/scenario_test/long_lived_graceful_restart_test.py | 170 |
3 files changed, 179 insertions, 2 deletions
diff --git a/test/lib/base.py b/test/lib/base.py index 5ccf952e..3d14cf4c 100644 --- a/test/lib/base.py +++ b/test/lib/base.py @@ -280,7 +280,7 @@ class BGPContainer(Container): is_rr_client=False, cluster_id=None, flowspec=False, bridge='', reload_config=True, as2=False, graceful_restart=None, local_as=None, prefix_limit=None, - v6=False): + v6=False, llgr=None): neigh_addr = '' local_addr = '' it = itertools.product(self.ip_addrs, peer.ip_addrs) @@ -317,7 +317,8 @@ class BGPContainer(Container): 'as2': as2, 'graceful_restart': graceful_restart, 'local_as': local_as, - 'prefix_limit': prefix_limit} + 'prefix_limit': prefix_limit, + 'llgr': llgr} 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 b8e669dd..77060893 100644 --- a/test/lib/gobgp.py +++ b/test/lib/gobgp.py @@ -306,6 +306,12 @@ class GoBGPContainer(BGPContainer): for afi_safi in afi_safi_list: afi_safi['mp-graceful-restart'] = {'config': {'enabled': True}} + if info['llgr'] is not None: + n['graceful-restart']['config']['restart-time'] = 1 + n['graceful-restart']['config']['long-lived-enabled'] = True + for afi_safi in afi_safi_list: + afi_safi['long-lived-graceful-restart'] = {'config': {'enabled': True, 'restart-time': 30}} + if info['is_rr_client']: clusterId = self.router_id if 'cluster_id' in info and info['cluster_id'] is not None: diff --git a/test/scenario_test/long_lived_graceful_restart_test.py b/test/scenario_test/long_lived_graceful_restart_test.py new file mode 100644 index 00000000..e0c51586 --- /dev/null +++ b/test/scenario_test/long_lived_graceful_restart_test.py @@ -0,0 +1,170 @@ +# Copyright (C) 2016 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. + +import unittest +from fabric.api import local +from lib import base +from lib.gobgp import * +from lib.quagga import * +import sys +import os +import time +import nose +from noseplugin import OptionParser, parser_option +from itertools import chain + + +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) + g2 = GoBGPContainer(name='g2', asn=65001, router_id='192.168.0.2', + ctn_image_name=gobgp_ctn_image_name, + log_level=parser_option.gobgp_log_level) + g3 = GoBGPContainer(name='g3', asn=65002, router_id='192.168.0.3', + ctn_image_name=gobgp_ctn_image_name, + log_level=parser_option.gobgp_log_level) + g4 = GoBGPContainer(name='g4', asn=65003, router_id='192.168.0.4', + ctn_image_name=gobgp_ctn_image_name, + log_level=parser_option.gobgp_log_level) + ctns = [g1, g2, g3, g4] + + initial_wait_time = max(ctn.run() for ctn in ctns) + + time.sleep(initial_wait_time) + + g1.add_peer(g2, graceful_restart=True, llgr=True) + g2.add_peer(g1, graceful_restart=True, llgr=True) + g1.add_peer(g3, graceful_restart=True, llgr=True) + g3.add_peer(g1, graceful_restart=True, llgr=True) + g1.add_peer(g4, graceful_restart=True) + g4.add_peer(g1, graceful_restart=True) + + cls.bgpds = {'g1': g1, 'g2': g2, 'g3': g3, 'g4': g4} + + # test each neighbor state is turned establish + def test_01_neighbor_established(self): + g1 = self.bgpds['g1'] + g2 = self.bgpds['g2'] + g3 = self.bgpds['g3'] + g4 = self.bgpds['g4'] + g1.wait_for(expected_state=BGP_FSM_ESTABLISHED, peer=g2) + g1.wait_for(expected_state=BGP_FSM_ESTABLISHED, peer=g3) + g1.wait_for(expected_state=BGP_FSM_ESTABLISHED, peer=g4) + + def test_02_graceful_restart(self): + g1 = self.bgpds['g1'] + g2 = self.bgpds['g2'] + g3 = self.bgpds['g3'] + g4 = self.bgpds['g4'] + + g2.local('gobgp global rib add 10.0.0.0/24') + g2.local('gobgp global rib add 10.10.0.0/24 community no-llgr') + + time.sleep(1) + + g2.graceful_restart() + g1.wait_for(expected_state=BGP_FSM_ACTIVE, peer=g2) + + time.sleep(1) + + self.assertTrue(len(g1.get_global_rib('10.0.0.0/24')) == 1) + # 10.10.0.0/24 is announced with no-llgr community + # must not exist in the rib + self.assertTrue(len(g1.get_global_rib('10.10.0.0/24')) == 0) + for d in g1.get_global_rib(): + for p in d['paths']: + self.assertTrue(p['stale']) + + self.assertTrue(len(g3.get_global_rib('10.0.0.0/24')) == 1) + # check llgr-stale community is added to 10.0.0.0/24 + r = g3.get_global_rib('10.0.0.0/24')[0]['paths'][0] + comms = list(chain.from_iterable([ attr['communities'] for attr in r['attrs'] if attr['type'] == 8])) + self.assertTrue(0xffff0006 in comms) + # g4 is not llgr capable, llgr-stale route must be + # withdrawn + self.assertTrue(len(g4.get_global_rib('10.0.0.0/24')) == 0) + + g2._start_gobgp(graceful_restart=True) + time.sleep(2) + g2.local('gobgp global rib add 10.0.0.0/24') + g2.local('gobgp global rib add 10.10.0.0/24') + + def test_03_neighbor_established(self): + g1 = self.bgpds['g1'] + g2 = self.bgpds['g2'] + g3 = self.bgpds['g3'] + g4 = self.bgpds['g4'] + g1.wait_for(expected_state=BGP_FSM_ESTABLISHED, peer=g2) + time.sleep(1) + self.assertTrue(len(g1.get_global_rib('10.0.0.0/24')) == 1) + self.assertTrue(len(g1.get_global_rib('10.10.0.0/24')) == 1) + for d in g1.get_global_rib(): + for p in d['paths']: + self.assertFalse(p.get('stale', False)) + + def test_04_llgr_stale_route_depreferenced(self): + g1 = self.bgpds['g1'] + g2 = self.bgpds['g2'] + g3 = self.bgpds['g3'] + g4 = self.bgpds['g4'] + g4.local('gobgp global rib add 10.0.0.0/24 med 100') + time.sleep(1) + # check g2's path is chosen as best and advertised + rib = g3.get_global_rib('10.0.0.0/24') + self.assertTrue(len(rib) == 1) + self.assertTrue(g2.asn in rib[0]['paths'][0]['aspath']) + + g2.graceful_restart() + g1.wait_for(expected_state=BGP_FSM_ACTIVE, peer=g2) + + time.sleep(1) + + # llgr_stale route depreference must happend + # check g4's path is chosen as best and advertised + rib = g3.get_global_rib('10.0.0.0/24') + self.assertTrue(len(rib) == 1) + self.assertTrue(g4.asn in rib[0]['paths'][0]['aspath']) + + # if no candidate exists, llgr_stale route will be chosen as best + rib = g3.get_global_rib('10.10.0.0/24') + self.assertTrue(len(rib) == 1) + self.assertTrue(g2.asn in rib[0]['paths'][0]['aspath']) + + + def test_05_llgr_restart_timer_expire(self): + time.sleep(35) + g3 = self.bgpds['g3'] + rib = g3.get_global_rib('10.10.0.0/24') + self.assertTrue(len(rib) == 0) + + +if __name__ == '__main__': + if os.geteuid() is not 0: + print "you are not root." + sys.exit(1) + 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]) |