2015-02-13 20:01:18 -07:00
# -*- coding: utf-8 -*-
#
# Copyright (C) 2013 GNS3 Technologies Inc.
#
# This program 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 3 of the License, or
# (at your option) any later version.
#
# This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
"""
Interface for Dynamips virtual Ethernet switch module ( " ethsw " ) .
http : / / github . com / GNS3 / dynamips / blob / master / README . hypervisor #L558
"""
import asyncio
2016-05-02 17:13:23 +02:00
from gns3server . utils import parse_version
2015-02-13 20:01:18 -07:00
from . device import Device
2015-02-23 19:00:34 -07:00
from . . nios . nio_udp import NIOUDP
2015-02-13 20:01:18 -07:00
from . . dynamips_error import DynamipsError
2016-09-29 16:01:18 +02:00
from . . . error import NodeError
2015-02-13 20:01:18 -07:00
import logging
log = logging . getLogger ( __name__ )
class EthernetSwitch ( Device ) :
2015-02-23 17:21:39 +01:00
2015-02-13 20:01:18 -07:00
"""
Dynamips Ethernet switch .
: param name : name for this switch
2016-05-19 22:45:04 -06:00
: param node_id : Node instance identifier
2015-02-13 20:01:18 -07:00
: param project : Project instance
: param manager : Parent VM Manager
2016-05-19 22:45:04 -06:00
: param ports : initial switch ports
2015-02-13 20:01:18 -07:00
: param hypervisor : Dynamips hypervisor instance
"""
2016-05-19 22:45:04 -06:00
def __init__ ( self , name , node_id , project , manager , ports = None , hypervisor = None ) :
2015-02-13 20:01:18 -07:00
2016-05-19 22:45:04 -06:00
super ( ) . __init__ ( name , node_id , project , manager , hypervisor )
2015-02-13 20:01:18 -07:00
self . _nios = { }
2015-02-15 12:18:12 -07:00
self . _mappings = { }
2016-05-19 22:45:04 -06:00
if ports is None :
# create 8 ports by default
self . _ports = [ ]
2016-09-29 14:59:11 +02:00
for port_number in range ( 0 , 8 ) :
2016-05-19 22:45:04 -06:00
self . _ports . append ( { " port_number " : port_number ,
" name " : " Ethernet {} " . format ( port_number ) ,
" type " : " access " ,
" vlan " : 1 } )
else :
self . _ports = ports
2015-02-15 12:18:12 -07:00
def __json__ ( self ) :
ethernet_switch_info = { " name " : self . name ,
2016-05-19 22:45:04 -06:00
" node_id " : self . id ,
" project_id " : self . project . id ,
2016-09-13 09:47:22 +02:00
" ports_mapping " : self . _ports ,
2016-05-19 22:45:04 -06:00
" status " : " started " }
return ethernet_switch_info
@property
2016-09-29 16:01:18 +02:00
def ports_mapping ( self ) :
2016-05-19 22:45:04 -06:00
"""
Ports on this switch
2015-02-15 12:18:12 -07:00
2016-05-19 22:45:04 -06:00
: returns : ports info
"""
2015-02-15 12:18:12 -07:00
2016-05-19 22:45:04 -06:00
return self . _ports
2016-09-29 16:01:18 +02:00
@ports_mapping.setter
def ports_mapping ( self , ports ) :
2016-05-19 22:45:04 -06:00
"""
Set the ports on this switch
: param ports : ports info
"""
2016-09-29 16:01:18 +02:00
if ports != self . _ports :
if len ( self . _nios ) > 0 :
raise NodeError ( " Can ' t modify a switch already connected. " )
2016-05-19 22:45:04 -06:00
2016-09-29 16:01:18 +02:00
port_number = 0
for port in ports :
port [ " name " ] = " Ethernet {} " . format ( port_number )
port [ " port_number " ] = port_number
port_number + = 1
self . _ports = ports
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def create ( self ) :
if self . _hypervisor is None :
2015-02-15 22:13:24 -07:00
module_workdir = self . project . module_working_directory ( self . manager . module_name . lower ( ) )
self . _hypervisor = yield from self . manager . start_new_hypervisor ( working_dir = module_workdir )
2015-02-13 20:01:18 -07:00
yield from self . _hypervisor . send ( ' ethsw create " {} " ' . format ( self . _name ) )
log . info ( ' Ethernet switch " {name} " [ {id} ] has been created ' . format ( name = self . _name , id = self . _id ) )
self . _hypervisor . devices . append ( self )
@asyncio.coroutine
def set_name ( self , new_name ) :
"""
Renames this Ethernet switch .
: param new_name : New name for this switch
"""
yield from self . _hypervisor . send ( ' ethsw rename " {name} " " {new_name} " ' . format ( name = self . _name , new_name = new_name ) )
log . info ( ' Ethernet switch " {name} " [ {id} ]: renamed to " {new_name} " ' . format ( name = self . _name ,
id = self . _id ,
new_name = new_name ) )
self . _name = new_name
@property
def nios ( self ) :
"""
Returns all the NIOs member of this Ethernet switch .
: returns : nio list
"""
return self . _nios
@property
2015-02-15 12:18:12 -07:00
def mappings ( self ) :
2015-02-13 20:01:18 -07:00
"""
2015-02-15 12:18:12 -07:00
Returns port mappings
2015-02-13 20:01:18 -07:00
2015-02-15 12:18:12 -07:00
: returns : mappings list
2015-02-13 20:01:18 -07:00
"""
2015-02-15 12:18:12 -07:00
return self . _mappings
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def delete ( self ) :
2016-12-12 19:17:06 +01:00
return ( yield from self . close ( ) )
@asyncio.coroutine
def close ( self ) :
2015-02-13 20:01:18 -07:00
"""
Deletes this Ethernet switch .
"""
2015-02-23 19:00:34 -07:00
for nio in self . _nios . values ( ) :
if nio and isinstance ( nio , NIOUDP ) :
2015-03-21 17:19:12 -06:00
self . manager . port_manager . release_udp_port ( nio . lport , self . _project )
2015-02-23 19:00:34 -07:00
2016-12-12 19:17:06 +01:00
if self . _hypervisor :
try :
yield from self . _hypervisor . send ( ' ethsw delete " {} " ' . format ( self . _name ) )
log . info ( ' Ethernet switch " {name} " [ {id} ] has been deleted ' . format ( name = self . _name , id = self . _id ) )
except DynamipsError :
log . debug ( " Could not properly delete Ethernet switch {} " . format ( self . _name ) )
2015-02-25 11:52:52 -07:00
if self . _hypervisor and self in self . _hypervisor . devices :
self . _hypervisor . devices . remove ( self )
2015-02-15 12:18:12 -07:00
if self . _hypervisor and not self . _hypervisor . devices :
yield from self . hypervisor . stop ( )
2016-12-12 19:17:06 +01:00
self . _hypervisor = None
return True
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def add_nio ( self , nio , port_number ) :
"""
Adds a NIO as new port on Ethernet switch .
: param nio : NIO instance to add
: param port_number : port to allocate for the NIO
"""
if port_number in self . _nios :
raise DynamipsError ( " Port {} isn ' t free " . format ( port_number ) )
yield from self . _hypervisor . send ( ' ethsw add_nio " {name} " {nio} ' . format ( name = self . _name , nio = nio ) )
log . info ( ' Ethernet switch " {name} " [ {id} ]: NIO {nio} bound to port {port} ' . format ( name = self . _name ,
id = self . _id ,
nio = nio ,
port = port_number ) )
self . _nios [ port_number ] = nio
2016-05-19 22:45:04 -06:00
for port_settings in self . _ports :
if port_settings [ " port_number " ] == port_number :
yield from self . set_port_settings ( port_number , port_settings )
break
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def remove_nio ( self , port_number ) :
"""
Removes the specified NIO as member of this Ethernet switch .
: param port_number : allocated port number
: returns : the NIO that was bound to the port
"""
if port_number not in self . _nios :
raise DynamipsError ( " Port {} is not allocated " . format ( port_number ) )
nio = self . _nios [ port_number ]
2015-02-23 19:00:34 -07:00
if isinstance ( nio , NIOUDP ) :
2015-03-21 17:19:12 -06:00
self . manager . port_manager . release_udp_port ( nio . lport , self . _project )
2017-02-20 12:19:38 +01:00
if self . _hypervisor :
yield from self . _hypervisor . send ( ' ethsw remove_nio " {name} " {nio} ' . format ( name = self . _name , nio = nio ) )
2015-02-13 20:01:18 -07:00
log . info ( ' Ethernet switch " {name} " [ {id} ]: NIO {nio} removed from port {port} ' . format ( name = self . _name ,
id = self . _id ,
nio = nio ,
port = port_number ) )
del self . _nios [ port_number ]
2015-02-15 12:18:12 -07:00
if port_number in self . _mappings :
del self . _mappings [ port_number ]
2015-02-13 20:01:18 -07:00
return nio
2015-02-15 12:18:12 -07:00
@asyncio.coroutine
def set_port_settings ( self , port_number , settings ) :
"""
Applies port settings to a specific port .
: param port_number : port number to set the settings
: param settings : port settings
"""
if settings [ " type " ] == " access " :
yield from self . set_access_port ( port_number , settings [ " vlan " ] )
elif settings [ " type " ] == " dot1q " :
yield from self . set_dot1q_port ( port_number , settings [ " vlan " ] )
elif settings [ " type " ] == " qinq " :
2015-11-09 12:28:00 +01:00
yield from self . set_qinq_port ( port_number , settings [ " vlan " ] , settings [ " ethertype " ] )
2015-02-15 12:18:12 -07:00
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def set_access_port ( self , port_number , vlan_id ) :
"""
Sets the specified port as an ACCESS port .
: param port_number : allocated port number
: param vlan_id : VLAN number membership
"""
if port_number not in self . _nios :
raise DynamipsError ( " Port {} is not allocated " . format ( port_number ) )
nio = self . _nios [ port_number ]
yield from self . _hypervisor . send ( ' ethsw set_access_port " {name} " {nio} {vlan_id} ' . format ( name = self . _name ,
nio = nio ,
vlan_id = vlan_id ) )
log . info ( ' Ethernet switch " {name} " [ {id} ]: port {port} set as an access port in VLAN {vlan_id} ' . format ( name = self . _name ,
id = self . _id ,
port = port_number ,
vlan_id = vlan_id ) )
2015-02-15 12:18:12 -07:00
self . _mappings [ port_number ] = ( " access " , vlan_id )
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def set_dot1q_port ( self , port_number , native_vlan ) :
"""
Sets the specified port as a 802.1 Q trunk port .
: param port_number : allocated port number
: param native_vlan : native VLAN for this trunk port
"""
if port_number not in self . _nios :
raise DynamipsError ( " Port {} is not allocated " . format ( port_number ) )
nio = self . _nios [ port_number ]
yield from self . _hypervisor . send ( ' ethsw set_dot1q_port " {name} " {nio} {native_vlan} ' . format ( name = self . _name ,
nio = nio ,
native_vlan = native_vlan ) )
log . info ( ' Ethernet switch " {name} " [ {id} ]: port {port} set as a 802.1Q port with native VLAN {vlan_id} ' . format ( name = self . _name ,
id = self . _id ,
port = port_number ,
vlan_id = native_vlan ) )
2015-02-15 12:18:12 -07:00
self . _mappings [ port_number ] = ( " dot1q " , native_vlan )
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
2015-08-20 08:45:30 +03:00
def set_qinq_port ( self , port_number , outer_vlan , ethertype ) :
2015-02-13 20:01:18 -07:00
"""
Sets the specified port as a trunk ( QinQ ) port .
: param port_number : allocated port number
: param outer_vlan : outer VLAN ( transport VLAN ) for this QinQ port
"""
if port_number not in self . _nios :
raise DynamipsError ( " Port {} is not allocated " . format ( port_number ) )
nio = self . _nios [ port_number ]
2015-09-08 03:03:11 -06:00
if ethertype != " 0x8100 " and parse_version ( self . hypervisor . version ) < parse_version ( ' 0.2.16 ' ) :
raise DynamipsError ( " Dynamips version required is >= 0.2.16 to change the default QinQ Ethernet type, detected version is {} " . format ( self . hypervisor . version ) )
2015-08-20 08:45:30 +03:00
yield from self . _hypervisor . send ( ' ethsw set_qinq_port " {name} " {nio} {outer_vlan} {ethertype} ' . format ( name = self . _name ,
2015-09-08 03:03:11 -06:00
nio = nio ,
outer_vlan = outer_vlan ,
ethertype = ethertype if ethertype != " 0x8100 " else " " ) )
2015-02-13 20:01:18 -07:00
2015-08-20 08:45:30 +03:00
log . info ( ' Ethernet switch " {name} " [ {id} ]: port {port} set as a QinQ ( {ethertype} ) port with outer VLAN {vlan_id} ' . format ( name = self . _name ,
2015-09-08 03:03:11 -06:00
id = self . _id ,
port = port_number ,
vlan_id = outer_vlan ,
ethertype = ethertype ) )
2015-08-20 08:45:30 +03:00
self . _mappings [ port_number ] = ( " qinq " , outer_vlan , ethertype )
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def get_mac_addr_table ( self ) :
"""
Returns the MAC address table for this Ethernet switch .
: returns : list of entries ( Ethernet address , VLAN , NIO )
"""
mac_addr_table = yield from self . _hypervisor . send ( ' ethsw show_mac_addr_table " {} " ' . format ( self . _name ) )
return mac_addr_table
@asyncio.coroutine
def clear_mac_addr_table ( self ) :
"""
Clears the MAC address table for this Ethernet switch .
"""
yield from self . _hypervisor . send ( ' ethsw clear_mac_addr_table " {} " ' . format ( self . _name ) )
@asyncio.coroutine
def start_capture ( self , port_number , output_file , data_link_type = " DLT_EN10MB " ) :
"""
Starts a packet capture .
: param port_number : allocated port number
: param output_file : PCAP destination file for the capture
: param data_link_type : PCAP data link type ( DLT_ * ) , default is DLT_EN10MB
"""
if port_number not in self . _nios :
raise DynamipsError ( " Port {} is not allocated " . format ( port_number ) )
nio = self . _nios [ port_number ]
2015-06-01 15:42:17 -06:00
if not nio :
raise DynamipsError ( " Port {} is not connected " . format ( port_number ) )
2015-02-13 20:01:18 -07:00
data_link_type = data_link_type . lower ( )
if data_link_type . startswith ( " dlt_ " ) :
data_link_type = data_link_type [ 4 : ]
if nio . input_filter [ 0 ] is not None and nio . output_filter [ 0 ] is not None :
raise DynamipsError ( " Port {} has already a filter applied " . format ( port_number ) )
yield from nio . bind_filter ( " both " , " capture " )
2015-02-24 23:12:09 -07:00
yield from nio . setup_filter ( " both " , ' {} " {} " ' . format ( data_link_type , output_file ) )
2015-02-13 20:01:18 -07:00
2015-02-15 17:45:53 -07:00
log . info ( ' Ethernet switch " {name} " [ {id} ]: starting packet capture on port {port} ' . format ( name = self . _name ,
id = self . _id ,
port = port_number ) )
2015-02-13 20:01:18 -07:00
@asyncio.coroutine
def stop_capture ( self , port_number ) :
"""
Stops a packet capture .
: param port_number : allocated port number
"""
if port_number not in self . _nios :
raise DynamipsError ( " Port {} is not allocated " . format ( port_number ) )
nio = self . _nios [ port_number ]
2015-06-01 15:42:17 -06:00
if not nio :
raise DynamipsError ( " Port {} is not connected " . format ( port_number ) )
2015-02-13 20:01:18 -07:00
yield from nio . unbind_filter ( " both " )
2015-02-15 17:45:53 -07:00
log . info ( ' Ethernet switch " {name} " [ {id} ]: stopping packet capture on port {port} ' . format ( name = self . _name ,
id = self . _id ,
port = port_number ) )