| """ |
| Copyright 2016 Open Networking Foundation ( ONF ) |
| |
| Please refer questions to either the onos test mailing list at <onos-test@onosproject.org>, |
| the System Testing Plans and Results wiki page at <https://wiki.onosproject.org/x/voMg>, |
| or the System Testing Guide page at <https://wiki.onosproject.org/x/WYQg> |
| |
| TestON is free software: you can redistribute it and/or modify |
| it under the terms of the GNU General Public License as published by |
| the Free Software Foundation, either version 2 of the License, or |
| ( at your option ) any later version. |
| |
| TestON is distributed in the hope that it will be useful, |
| but WITHOUT ANY WARRANTY; without even the implied warranty of |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| GNU General Public License for more details. |
| |
| You should have received a copy of the GNU General Public License |
| along with TestON. If not, see <http://www.gnu.org/licenses/>. |
| """ |
| # CASE1: Startup |
| # CASE2: Load vpls topology and configurations from demo script |
| # CASE3: Test CLI commands |
| |
| |
| class VPLSBasic: |
| |
| def __init__( self ): |
| self.default = '' |
| |
| def CASE1( self, main ): |
| """ |
| CASE1 is to compile ONOS and push it to the test machines |
| |
| Startup sequence: |
| cell <name> |
| onos-verify-cell |
| NOTE: temporary - onos-remove-raft-logs |
| onos-uninstall |
| start mininet |
| git pull |
| mvn clean install |
| onos-package |
| onos-install -f |
| onos-wait-for-start |
| start cli sessions |
| start tcpdump |
| """ |
| import imp |
| import time |
| import json |
| |
| try: |
| from tests.dependencies.ONOSSetup import ONOSSetup |
| main.testSetUp = ONOSSetup() |
| except ImportError: |
| main.log.error( "ONOSSetup not found. exiting the test" ) |
| main.cleanAndExit() |
| main.testSetUp.envSetupDescription() |
| stepResult = main.FALSE |
| try: |
| # load some variables from the params file |
| main.cellName = main.params[ 'ENV' ][ 'cellName' ] |
| main.sleep = int( main.params[ 'SLEEP' ][ 'netcfg' ] ) |
| main.apps = main.params[ 'ENV' ][ 'cellApps' ] |
| |
| ofPort = main.params[ 'CTRL' ][ 'port' ] |
| stepResult = main.testSetUp.envSetup() |
| except Exception as e: |
| main.testSetUp.envSetupException( e ) |
| main.testSetUp.envSetupConclusion( stepResult ) |
| |
| main.testSetUp.ONOSSetUp( main.Cluster, |
| cellName=main.cellName ) |
| |
| main.step( "Starting Mininet" ) |
| # scp topo file to mininet |
| # TODO: move to params? |
| topoName = "vpls" |
| topoFile = "vpls.py" |
| filePath = main.ONOSbench.home + "/tools/test/topos/" |
| main.ONOSbench.scp( main.Mininet1, |
| filePath + topoFile, |
| main.Mininet1.home, |
| direction="to" ) |
| topo = " --custom " + main.Mininet1.home + topoFile + " --topo " + topoName |
| args = " --switch ovs,protocols=OpenFlow13" |
| for ctrl in main.Cluster.active(): |
| args += " --controller=remote,ip=" + ctrl.ipAddress |
| mnCmd = "sudo mn" + topo + args |
| mnResult = main.Mininet1.startNet( mnCmd=mnCmd ) |
| utilities.assert_equals( expect=main.TRUE, actual=mnResult, |
| onpass="Mininet Started", |
| onfail="Error starting Mininet" ) |
| |
| main.step( "Activate apps defined in the params file" ) |
| # get data from the params |
| apps = main.params.get( 'apps' ) |
| if apps: |
| apps = apps.split( ',' ) |
| main.log.warn( apps ) |
| activateResult = True |
| for app in apps: |
| main.Cluster.active( 0 ).CLI.app( app, "Activate" ) |
| # TODO: check this worked |
| time.sleep( main.sleep ) # wait for apps to activate |
| for app in apps: |
| state = main.Cluster.active( 0 ).CLI.appStatus( app ) |
| if state == "ACTIVE": |
| activateResult = activateResult and True |
| else: |
| main.log.error( "{} is in {} state".format( app, state ) ) |
| activateResult = False |
| utilities.assert_equals( expect=True, |
| actual=activateResult, |
| onpass="Successfully activated apps", |
| onfail="Failed to activate apps" ) |
| else: |
| main.log.warn( "No apps were specified to be loaded after startup" ) |
| |
| main.step( "Set ONOS configurations" ) |
| config = main.params.get( 'ONOS_Configuration' ) |
| if config: |
| main.log.debug( config ) |
| checkResult = main.TRUE |
| for component in config: |
| for setting in config[ component ]: |
| value = config[ component ][ setting ] |
| check = main.Cluster.active( 0 ).CLI.setCfg( component, setting, value ) |
| main.log.info( "Value was changed? {}".format( main.TRUE == check ) ) |
| checkResult = check and checkResult |
| utilities.assert_equals( expect=main.TRUE, |
| actual=checkResult, |
| onpass="Successfully set config", |
| onfail="Failed to set config" ) |
| else: |
| main.log.warn( "No configurations were specified to be changed after startup" ) |
| |
| main.step( "App Ids check" ) |
| appCheck = main.Cluster.command( "appToIDCheck", returnBool=True ) |
| if not appCheck: |
| main.log.warn( main.Cluster.active( 0 ).CLI.apps() ) |
| main.log.warn( main.Cluster.active( 0 ).CLI.appIDs() ) |
| utilities.assert_equals( expect=True, actual=appCheck, |
| onpass="App Ids seem to be correct", |
| onfail="Something is wrong with app Ids" ) |
| |
| def CASE2( self, main ): |
| """ |
| Load and test vpls configurations from json configuration file |
| """ |
| import os.path |
| from tests.USECASE.VPLS.dependencies import vpls |
| |
| main.vpls = vpls |
| pprint = main.Cluster.active( 0 ).REST.pprint |
| hosts = int( main.params[ 'vpls' ][ 'hosts' ] ) |
| main.step( "Discover hosts using pings" ) |
| for i in range( 1, hosts + 1 ): |
| src = "h" + str( i ) |
| for j in range( 1, hosts + 1 ): |
| if j == i: |
| continue |
| dst = "h" + str( j ) |
| pingResult = main.Mininet1.pingHost( SRC=src, TARGET=dst ) |
| |
| main.step( "Load VPLS configurations" ) |
| fileName = main.params[ 'DEPENDENCY' ][ 'topology' ] |
| app = main.params[ 'vpls' ][ 'name' ] |
| |
| loadVPLSResult = main.ONOSbench.onosNetCfg( main.Cluster.active( 0 ).ipAddress, "", fileName ) |
| utilities.assert_equals( expect=main.TRUE, |
| actual=loadVPLSResult, |
| onpass="Loaded vpls configuration.", |
| onfail="Failed to load vpls configuration." ) |
| |
| # Time for netcfg to load data |
| time.sleep( main.sleep ) |
| # 'Master' copy of test configuration |
| |
| try: |
| with open( os.path.expanduser( fileName ) ) as dataFile: |
| originalCfg = json.load( dataFile ) |
| main.vplsConfig = originalCfg[ 'apps' ].get( app ).get( 'vpls' ).get( 'vplsList' ) |
| except Exception as e: |
| main.log.error( "Error loading config file: {}".format( e ) ) |
| if main.vplsConfig: |
| result = True |
| else: |
| result = False |
| utilities.assert_equals( expect=True, |
| actual=result, |
| onpass="Loaded vpls configuration", |
| onfail="Failed to load vpls configuration" ) |
| |
| main.step( "Check interface configurations" ) |
| result = False |
| getPorts = utilities.retry( f=main.Cluster.active( 0 ).REST.getNetCfg, |
| retValue=False, |
| kwargs={ "subjectClass": "ports" }, |
| sleep=main.sleep ) |
| onosCfg = pprint( getPorts ) |
| sentCfg = pprint( originalCfg.get( "ports" ) ) |
| |
| if onosCfg == sentCfg: |
| main.log.info( "ONOS interfaces NetCfg matches what was sent" ) |
| result = True |
| else: |
| main.log.error( "ONOS interfaces NetCfg doesn't match what was sent" ) |
| main.log.debug( "ONOS config: {}".format( onosCfg ) ) |
| main.log.debug( "Sent config: {}".format( sentCfg ) ) |
| utilities.assert_equals( expect=True, |
| actual=result, |
| onpass="Net Cfg added for interfaces", |
| onfail="Net Cfg not added for interfaces" ) |
| |
| # Run a bunch of checks to verify functionality based on configs |
| vpls.verify( main ) |
| |
| main.step( "Loading vpls configuration in case any configuration was missed." ) |
| loadVPLSResult = main.ONOSbench.onosNetCfg( main.Cluster.active( 0 ).ipAddress, "", fileName ) |
| utilities.assert_equals( expect=main.TRUE, |
| actual=loadVPLSResult, |
| onpass="Loaded vpls configuration.", |
| onfail="Failed to load vpls configuration." ) |
| |
| def CASE10( self, main ): |
| """ |
| Remove an interface from a vpls network |
| """ |
| main.step( "Remove an interface from a vpls network" ) |
| main.Cluster.active( 0 ).CLI.vplsRemIface( 'VPLS1', 'h1' ) |
| time.sleep( main.sleep ) |
| # update master config json |
| for network in main.vplsConfig: |
| if network.get( 'name' ) == 'VPLS1': |
| ifaces = network.get( 'interfaces' ) |
| ifaces.remove( 'h1' ) |
| main.vpls.verify( main ) |
| |
| def CASE11( self, main ): |
| """ |
| Clean all VPLS configurations |
| """ |
| main.step( "Clean all vpls configurations" ) |
| main.Cluster.active( 0 ).CLI.vplsClean() |
| time.sleep( main.sleep ) |
| main.vplsConfig = [] |
| main.vpls.verify( main ) |
| |
| def CASE12( self, main ): |
| """ |
| Create a new VPLS network. |
| """ |
| main.step( "Create a new vpls network" ) |
| name = "Network1" |
| main.Cluster.active( 0 ).CLI.vplsCreate( name ) |
| time.sleep( main.sleep ) |
| network1 = { 'name': name, 'interfaces': [], 'encapsulation': 'NONE' } |
| main.vplsConfig.append( network1 ) |
| main.vpls.verify( main ) |
| |
| def CASE13( self, main ): |
| """ |
| Add interfaces to the new VPLS network. |
| """ |
| main.step( "Add interfaces to the network" ) |
| utilities.retry( f=main.Cluster.active( 0 ).CLI.vplsAddIface, |
| retValue=main.FALSE, |
| args=( name, "h1" ), |
| sleep=main.sleep ) |
| utilities.retry( f=main.Cluster.active( 0 ).CLI.vplsAddIface, |
| retValue=main.FALSE, |
| args=( name, "h5" ), |
| sleep=main.sleep ) |
| utilities.retry( f=main.Cluster.active( 0 ).CLI.vplsAddIface, |
| retValue=main.FALSE, |
| args=( name, "h4" ), |
| sleep=main.sleep ) |
| time.sleep( main.sleep ) |
| for network in main.vplsConfig: |
| if network.get( 'name' ) == name: |
| ifaces = network.get( 'interfaces' ) |
| ifaces.append( 'h1' ) |
| ifaces.append( 'h4' ) |
| ifaces.append( 'h5' ) |
| network[ 'interfaces' ] = ifaces |
| main.vpls.verify( main ) |
| |
| def CASE14( self, main ): |
| """ |
| Add MPLS encapsulation. |
| """ |
| main.step( "Add MPLS encapsulation to a vpls network" ) |
| encapType = "MPLS" |
| main.Cluster.active( 0 ).CLI.vplsSetEncap( name, encapType ) |
| for network in main.vplsConfig: |
| if network.get( 'name' ) == name: |
| network[ 'encapsulation' ] = encapType |
| time.sleep( main.sleep ) |
| main.vpls.verify( main ) |
| |
| def CASE15( self, main ): |
| """ |
| Change an encapsulation type. |
| """ |
| main.step( "Change an encapsulation type" ) |
| encapType = "VLAN" |
| main.Cluster.active( 0 ).CLI.vplsSetEncap( name, encapType ) |
| for network in main.vplsConfig: |
| if network.get( 'name' ) == name: |
| network[ 'encapsulation' ] = encapType |
| time.sleep( main.sleep ) |
| main.vpls.verify( main ) |
| |
| def CASE16( self, main ): |
| """ |
| Remove encapsulation. |
| """ |
| main.step( "Remove encapsulation" ) |
| encapType = "NONE" |
| main.Cluster.active( 0 ).CLI.vplsSetEncap( name, encapType ) |
| for network in main.vplsConfig: |
| if network.get( 'name' ) == name: |
| network[ 'encapsulation' ] = encapType |
| time.sleep( main.sleep ) |
| main.vpls.verify( main ) |