2015-02-14 05:01:18 +02: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 18:13:23 +03:00
from gns3server . utils import parse_version
2017-05-26 18:20:18 +03:00
from gns3server . utils . asyncio . embed_shell import EmbedShell , create_telnet_shell
2015-02-14 05:01:18 +02:00
from . device import Device
2015-02-24 04:00:34 +02:00
from . . nios . nio_udp import NIOUDP
2015-02-14 05:01:18 +02:00
from . . dynamips_error import DynamipsError
2016-09-29 17:01:18 +03:00
from . . . error import NodeError
2015-02-14 05:01:18 +02:00
import logging
log = logging . getLogger ( __name__ )
2017-05-26 18:20:18 +03:00
class EthernetSwitchConsole ( EmbedShell ) :
"""
Console for the ethernet switch
"""
def __init__ ( self , node ) :
2017-06-22 01:23:33 +03:00
super ( ) . __init__ ( welcome_message = " Welcome to GNS3 builtin Ethernet switch. \n \n Type help for available commands \n " )
2017-05-26 18:20:18 +03:00
self . _node = node
@asyncio.coroutine
2018-01-14 18:10:26 +02:00
def mac ( self ) :
2017-05-26 18:20:18 +03:00
"""
2018-01-14 18:10:26 +02:00
Show MAC address table
2017-05-26 18:20:18 +03:00
"""
2017-06-22 01:23:33 +03:00
res = ' Port Mac VLAN \n '
2017-05-26 18:20:18 +03:00
result = ( yield from self . _node . _hypervisor . send ( ' ethsw show_mac_addr_table {} ' . format ( self . _node . name ) ) )
for line in result :
2017-06-22 01:23:33 +03:00
mac , vlan , nio = line . replace ( ' ' , ' ' ) . split ( ' ' )
2017-05-26 18:20:18 +03:00
mac = mac . replace ( ' . ' , ' ' )
mac = " {} : {} : {} : {} : {} : {} " . format (
mac [ 0 : 2 ] ,
mac [ 2 : 4 ] ,
mac [ 4 : 6 ] ,
mac [ 6 : 8 ] ,
mac [ 8 : 10 ] ,
mac [ 10 : 12 ] )
2017-06-22 01:23:33 +03:00
for port_number , switch_nio in self . _node . nios . items ( ) :
if switch_nio . name == nio :
res + = ' Ethernet ' + str ( port_number ) + ' ' + mac + ' ' + vlan + ' \n '
break
2017-05-26 18:20:18 +03:00
return res
2015-02-14 05:01:18 +02:00
class EthernetSwitch ( Device ) :
2015-02-23 18:21:39 +02:00
2015-02-14 05:01:18 +02:00
"""
Dynamips Ethernet switch .
: param name : name for this switch
2016-05-20 07:45:04 +03:00
: param node_id : Node instance identifier
2015-02-14 05:01:18 +02:00
: param project : Project instance
: param manager : Parent VM Manager
2016-05-20 07:45:04 +03:00
: param ports : initial switch ports
2015-02-14 05:01:18 +02:00
: param hypervisor : Dynamips hypervisor instance
"""
2016-05-20 07:45:04 +03:00
def __init__ ( self , name , node_id , project , manager , ports = None , hypervisor = None ) :
2015-02-14 05:01:18 +02:00
2016-05-20 07:45:04 +03:00
super ( ) . __init__ ( name , node_id , project , manager , hypervisor )
2015-02-14 05:01:18 +02:00
self . _nios = { }
2015-02-15 21:18:12 +02:00
self . _mappings = { }
2017-05-26 18:20:18 +03:00
self . _telnet_console = None
self . _telnet_shell = None
2018-09-02 11:32:33 +03:00
self . _telnet_server = None
2017-05-26 18:20:18 +03:00
self . _console = self . _manager . port_manager . get_free_tcp_port ( self . _project )
2016-05-20 07:45:04 +03:00
if ports is None :
# create 8 ports by default
self . _ports = [ ]
2016-09-29 15:59:11 +03:00
for port_number in range ( 0 , 8 ) :
2016-05-20 07:45:04 +03:00
self . _ports . append ( { " port_number " : port_number ,
" name " : " Ethernet {} " . format ( port_number ) ,
" type " : " access " ,
" vlan " : 1 } )
else :
self . _ports = ports
2015-02-15 21:18:12 +02:00
def __json__ ( self ) :
ethernet_switch_info = { " name " : self . name ,
2017-05-26 18:20:18 +03:00
" console " : self . console ,
" console_type " : " telnet " ,
2016-05-20 07:45:04 +03:00
" node_id " : self . id ,
" project_id " : self . project . id ,
2016-09-13 10:47:22 +03:00
" ports_mapping " : self . _ports ,
2016-05-20 07:45:04 +03:00
" status " : " started " }
return ethernet_switch_info
2017-05-26 18:20:18 +03:00
@property
def console ( self ) :
return self . _console
@console.setter
def console ( self , val ) :
self . _console = val
2016-05-20 07:45:04 +03:00
@property
2016-09-29 17:01:18 +03:00
def ports_mapping ( self ) :
2016-05-20 07:45:04 +03:00
"""
Ports on this switch
2015-02-15 21:18:12 +02:00
2016-05-20 07:45:04 +03:00
: returns : ports info
"""
2015-02-15 21:18:12 +02:00
2016-05-20 07:45:04 +03:00
return self . _ports
2016-09-29 17:01:18 +03:00
@ports_mapping.setter
def ports_mapping ( self , ports ) :
2016-05-20 07:45:04 +03:00
"""
Set the ports on this switch
: param ports : ports info
"""
2016-09-29 17:01:18 +03:00
if ports != self . _ports :
2017-03-29 14:22:24 +03:00
if len ( self . _nios ) > 0 and len ( ports ) != len ( self . _ports ) :
2016-09-29 17:01:18 +03:00
raise NodeError ( " Can ' t modify a switch already connected. " )
2016-05-20 07:45:04 +03:00
2016-09-29 17:01:18 +03: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-14 05:01:18 +02:00
2017-03-29 14:22:24 +03:00
@asyncio.coroutine
def update_port_settings ( self ) :
for port_settings in self . _ports :
port_number = port_settings [ " port_number " ]
if port_number in self . _nios and self . _nios [ port_number ] is not None :
yield from self . set_port_settings ( port_number , port_settings )
2015-02-14 05:01:18 +02:00
@asyncio.coroutine
def create ( self ) :
if self . _hypervisor is None :
2015-02-16 07:13:24 +02: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-14 05:01:18 +02: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 ) )
2017-05-26 18:20:18 +03:00
self . _telnet_shell = EthernetSwitchConsole ( self )
self . _telnet_shell . prompt = self . _name + ' > '
2017-11-20 14:51:07 +02:00
self . _telnet = create_telnet_shell ( self . _telnet_shell )
2018-04-16 10:30:06 +03:00
try :
self . _telnet_server = ( yield from asyncio . start_server ( self . _telnet . run , self . _manager . port_manager . console_host , self . console ) )
except OSError as e :
self . project . emit ( " log.warning " , { " message " : " Could not start Telnet server on socket {} : {} : {} " . format ( self . _manager . port_manager . console_host , self . console , e ) } )
2015-02-14 05:01:18 +02:00
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 21:18:12 +02:00
def mappings ( self ) :
2015-02-14 05:01:18 +02:00
"""
2015-02-15 21:18:12 +02:00
Returns port mappings
2015-02-14 05:01:18 +02:00
2015-02-15 21:18:12 +02:00
: returns : mappings list
2015-02-14 05:01:18 +02:00
"""
2015-02-15 21:18:12 +02:00
return self . _mappings
2015-02-14 05:01:18 +02:00
@asyncio.coroutine
def delete ( self ) :
2016-12-12 20:17:06 +02:00
return ( yield from self . close ( ) )
@asyncio.coroutine
def close ( self ) :
2015-02-14 05:01:18 +02:00
"""
Deletes this Ethernet switch .
"""
2017-11-20 14:51:07 +02:00
yield from self . _telnet . close ( )
2018-09-02 11:32:33 +03:00
if self . _telnet_server :
self . _telnet_server . close ( )
2015-02-24 04:00:34 +02:00
for nio in self . _nios . values ( ) :
2017-07-10 21:38:28 +03:00
if nio :
yield from nio . close ( )
2017-05-26 18:20:18 +03:00
self . manager . port_manager . release_tcp_port ( self . _console , self . _project )
2016-12-12 20:17:06 +02: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 20:52:52 +02:00
if self . _hypervisor and self in self . _hypervisor . devices :
self . _hypervisor . devices . remove ( self )
2015-02-15 21:18:12 +02:00
if self . _hypervisor and not self . _hypervisor . devices :
yield from self . hypervisor . stop ( )
2016-12-12 20:17:06 +02:00
self . _hypervisor = None
return True
2015-02-14 05:01:18 +02: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-20 07:45:04 +03: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-14 05:01:18 +02: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-24 04:00:34 +02:00
if isinstance ( nio , NIOUDP ) :
2015-03-22 01:19:12 +02:00
self . manager . port_manager . release_udp_port ( nio . lport , self . _project )
2017-02-20 13:19:38 +02:00
if self . _hypervisor :
yield from self . _hypervisor . send ( ' ethsw remove_nio " {name} " {nio} ' . format ( name = self . _name , nio = nio ) )
2015-02-14 05:01:18 +02: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 21:18:12 +02:00
if port_number in self . _mappings :
del self . _mappings [ port_number ]
2015-02-14 05:01:18 +02:00
return nio
2015-02-15 21:18:12 +02: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 " :
2017-05-04 11:33:02 +03:00
yield from self . set_qinq_port ( port_number , settings [ " vlan " ] , settings . get ( " ethertype " ) )
2015-02-15 21:18:12 +02:00
2015-02-14 05:01:18 +02: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 21:18:12 +02:00
self . _mappings [ port_number ] = ( " access " , vlan_id )
2015-02-14 05:01:18 +02: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 21:18:12 +02:00
self . _mappings [ port_number ] = ( " dot1q " , native_vlan )
2015-02-14 05:01:18 +02:00
@asyncio.coroutine
2015-08-20 08:45:30 +03:00
def set_qinq_port ( self , port_number , outer_vlan , ethertype ) :
2015-02-14 05:01:18 +02: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 12:03:11 +03: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 12:03:11 +03:00
nio = nio ,
outer_vlan = outer_vlan ,
ethertype = ethertype if ethertype != " 0x8100 " else " " ) )
2015-02-14 05:01:18 +02: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 12:03:11 +03: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-14 05:01:18 +02: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-02 00:42:17 +03:00
if not nio :
raise DynamipsError ( " Port {} is not connected " . format ( port_number ) )
2015-02-14 05:01:18 +02: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-25 08:12:09 +02:00
yield from nio . setup_filter ( " both " , ' {} " {} " ' . format ( data_link_type , output_file ) )
2015-02-14 05:01:18 +02:00
2015-02-16 02:45:53 +02: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-14 05:01:18 +02: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-02 00:42:17 +03:00
if not nio :
raise DynamipsError ( " Port {} is not connected " . format ( port_number ) )
2015-02-14 05:01:18 +02:00
yield from nio . unbind_filter ( " both " )
2015-02-16 02:45:53 +02:00
log . info ( ' Ethernet switch " {name} " [ {id} ]: stopping packet capture on port {port} ' . format ( name = self . _name ,
id = self . _id ,
port = port_number ) )