Ranges for UDP, console, auxiliary console and hypervisor ports.

Dynamips UDP NIO auto back-end for UDP tunnel connections (excepting
stubs).
This commit is contained in:
grossmj 2014-04-30 20:44:13 -06:00
parent 97e19ac7a5
commit 93058f92d4
10 changed files with 420 additions and 242 deletions

View File

@ -0,0 +1,66 @@
# -*- coding: utf-8 -*-
#
# Copyright (C) 2014 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/>.
"""
Useful functions... in the attic ;)
"""
import socket
import errno
def find_unused_port(start_port, end_port, host='127.0.0.1', socket_type="TCP", ignore_ports=[]):
"""
Finds an unused port in a range.
:param start_port: first port in the range
:param end_port: last port in the range
:param host: host/address for bind()
:param socket_type: TCP (default) or UDP
:param ignore_ports: list of port to ignore within the range
"""
if end_port < start_port:
raise Exception("Invalid port range {}-{}".format(start_port, end_port))
if socket_type == "UDP":
socket_type = socket.SOCK_DGRAM
else:
socket_type = socket.SOCK_STREAM
for port in range(start_port, end_port + 1):
if port in ignore_ports:
continue
try:
if ":" in host:
# IPv6 address support
with socket.socket(socket.AF_INET6, socket_type) as s:
s.bind((host, port)) # the port is available if bind is a success
else:
with socket.socket(socket.AF_INET, socket_type) as s:
s.bind((host, port)) # the port is available if bind is a success
return port
except OSError as e:
if e.errno == errno.EADDRINUSE: # socket already in use
if port + 1 == end_port:
break
else:
continue
else:
raise Exception("Could not find an unused port: {}".format(e))
raise Exception("Could not find a free port between {0} and {1}".format(start_port, end_port))

View File

@ -326,18 +326,18 @@ class Dynamips(IModule):
:returns: a NIO object :returns: a NIO object
""" """
#TODO: JSON schema validation
nio = None nio = None
if request["nio"]["type"] == "nio_udp": if request["nio"]["type"] == "nio_udp":
lport = request["nio"]["lport"] lport = request["nio"]["lport"]
rhost = request["nio"]["rhost"] rhost = request["nio"]["rhost"]
rport = request["nio"]["rport"] rport = request["nio"]["rport"]
nio = NIO_UDP(node.hypervisor, lport, rhost, rport) # check if we have an allocated NIO UDP auto
# elif request["nio"] == "nio_udp_auto": nio = node.hypervisor.get_nio_udp_auto(lport)
# lhost = request["lhost"] if not nio:
# lport_start = request["lport_start"] # otherwise create an NIO UDP
# lport_end = request["lport_end"] nio = NIO_UDP(node.hypervisor, lport, rhost, rport)
# nio = NIO_UDP_auto(node.hypervisor, lhost, lport_start, lport_end) else:
nio.connect(rhost, rport)
elif request["nio"]["type"] == "nio_generic_ethernet": elif request["nio"]["type"] == "nio_generic_ethernet":
ethernet_device = request["nio"]["ethernet_device"] ethernet_device = request["nio"]["ethernet_device"]
nio = NIO_GenericEthernet(node.hypervisor, ethernet_device) nio = NIO_GenericEthernet(node.hypervisor, ethernet_device)
@ -379,20 +379,6 @@ class Dynamips(IModule):
return response return response
# def allocate_udp_port_auto(self, node, lport_start, lport_end):
# """
# Allocates a UDP port in order to create an UDP NIO Auto.
#
# :param node: the node that needs to allocate an UDP port
# """
#
# self._nio_udp_auto = NIO_UDP_auto(node.hypervisor, node.hypervisor.host, lport_start, lport_end)
#
# response = {"lport": self._nio_udp_auto.lport,
# "lhost": self._nio_udp_auto.lhost}
#
# return response
def set_ghost_ios(self, router): def set_ghost_ios(self, router):
""" """
Manages Ghost IOS support. Manages Ghost IOS support.

View File

@ -25,6 +25,8 @@ class DynamipsError(Exception):
def __init__(self, message, original_exception=None): def __init__(self, message, original_exception=None):
Exception.__init__(self, message) Exception.__init__(self, message)
if isinstance(message, Exception):
message = str(message)
self._message = message self._message = message
self._original_exception = original_exception self._original_exception = original_exception

View File

@ -21,10 +21,10 @@ http://github.com/GNS3/dynamips/blob/master/README.hypervisor#L46
""" """
import socket import socket
import errno
import re import re
import logging import logging
from .dynamips_error import DynamipsError from .dynamips_error import DynamipsError
from .nios.nio_udp_auto import NIO_UDP_auto
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
@ -53,10 +53,13 @@ class DynamipsHypervisor(object):
self._ghosts = {} self._ghosts = {}
self._jitsharing_groups = {} self._jitsharing_groups = {}
self._working_dir = working_dir self._working_dir = working_dir
self._baseconsole = 2000 self._console_start_port_range = 2001
self._baseaux = 2100 self._console_end_port_range = 2500
self._baseudp = 10000 self._aux_start_port_range = 2501
self._current_udp_port = self._baseudp self._aux_end_port_range = 3000
self._udp_start_port_range = 10001
self._udp_end_port_range = 20000
self._nio_udp_auto_instances = {}
self._version = "N/A" self._version = "N/A"
self._timeout = 30 self._timeout = 30
self._socket = None self._socket = None
@ -136,6 +139,7 @@ class DynamipsHypervisor(object):
self.send("hypervisor stop") self.send("hypervisor stop")
self._socket.close() self._socket.close()
self._socket = None self._socket = None
self._nio_udp_auto_instances.clear()
def reset(self): def reset(self):
""" """
@ -143,6 +147,7 @@ class DynamipsHypervisor(object):
""" """
self.send('hypervisor reset') self.send('hypervisor reset')
self._nio_udp_auto_instances.clear()
@property @property
def working_dir(self): def working_dir(self):
@ -220,68 +225,124 @@ class DynamipsHypervisor(object):
self._devices = devices self._devices = devices
@property @property
def baseconsole(self): def console_start_port_range(self):
""" """
Returns base console TCP port for this hypervisor. Returns the console start port range value
:returns: base console value (integer) :returns: console start port range value (integer)
""" """
return self._baseconsole return self._console_start_port_range
@baseconsole.setter @console_start_port_range.setter
def baseconsole(self, baseconsole): def console_start_port_range(self, console_start_port_range):
""" """
Sets the base console TCP port for this hypervisor. Set a new console start port range value
:param baseconsole: base console value (integer) :param console_start_port_range: console start port range value (integer)
""" """
self._baseconsole = baseconsole self._console_start_port_range = console_start_port_range
@property @property
def baseaux(self): def console_end_port_range(self):
""" """
Returns base auxiliary port for this hypervisor. Returns the console end port range value
:returns: base auxiliary port value (integer) :returns: console end port range value (integer)
""" """
return self._baseaux return self._console_end_port_range
@baseaux.setter @console_end_port_range.setter
def baseaux(self, baseaux): def console_end_port_range(self, console_end_port_range):
""" """
Sets the base auxiliary TCP port for this hypervisor. Set a new console end port range value
:param baseaux: base auxiliary port value (integer) :param console_end_port_range: console end port range value (integer)
""" """
self._baseaux = baseaux self._console_end_port_range = console_end_port_range
@property @property
def baseudp(self): def aux_start_port_range(self):
""" """
Returns the next available UDP port for UDP NIOs. Returns the auxiliary console start port range value
:returns: base UDP port value (integer) :returns: auxiliary console start port range value (integer)
""" """
return self._baseudp return self._aux_start_port_range
@baseudp.setter @aux_start_port_range.setter
def baseudp(self, baseudp): def aux_start_port_range(self, aux_start_port_range):
""" """
Sets the next open UDP port for NIOs for this hypervisor. Sets a new auxiliary console start port range value
:param baseudp: base UDP port value (integer) :param aux_start_port_range: auxiliary console start port range value (integer)
""" """
self._baseudp = baseudp self._aux_start_port_range = aux_start_port_range
self._current_udp_port = self._baseudp
#FIXME @property
log.info("hypervisor a new base UDP {}".format(self._baseudp)) def aux_end_port_range(self):
"""
Returns the auxiliary console end port range value
:returns: auxiliary console end port range value (integer)
"""
return self._aux_end_port_range
@aux_end_port_range.setter
def aux_end_port_range(self, aux_end_port_range):
"""
Sets a new auxiliary console end port range value
:param aux_end_port_range: auxiliary console end port range value (integer)
"""
self._aux_end_port_range = aux_end_port_range
@property
def udp_start_port_range(self):
"""
Returns the UDP start port range value
:returns: UDP start port range value (integer)
"""
return self._udp_start_port_range
@udp_start_port_range.setter
def udp_start_port_range(self, udp_start_port_range):
"""
Sets a new UDP start port range value
:param udp_start_port_range: UDP start port range value (integer)
"""
self._udp_start_port_range = udp_start_port_range
@property
def udp_end_port_range(self):
"""
Returns the UDP end port range value
:returns: UDP end port range value (integer)
"""
return self._udp_end_port_range
@udp_end_port_range.setter
def udp_end_port_range(self, udp_end_port_range):
"""
Sets an new UDP end port range value
:param udp_end_port_range: UDP end port range value (integer)
"""
self._udp_end_port_range = udp_end_port_range
@property @property
def ghosts(self): def ghosts(self):
@ -343,58 +404,29 @@ class DynamipsHypervisor(object):
return self._port return self._port
@staticmethod def get_nio_udp_auto(self, port):
def find_unused_port(start_port, end_port, host='127.0.0.1', socket_type="TCP"):
""" """
Finds an unused port in a range. Returns an allocated NIO UDP auto instance.
:param start_port: first port in the range :returns: NIO UDP auto instance
:param end_port: last port in the range
:param host: host/address for bind()
:param socket_type: TCP (default) or UDP
""" """
if socket_type == "UDP": if port in self._nio_udp_auto_instances:
socket_type = socket.SOCK_DGRAM return self._nio_udp_auto_instances.pop(port)
else: else:
socket_type = socket.SOCK_STREAM return None
for port in range(start_port, end_port): def allocate_udp_port(self):
try:
if ":" in host:
# IPv6 address support
with socket.socket(socket.AF_INET6, socket_type) as s:
s.bind((host, port)) # the port is available if bind is a success
else:
with socket.socket(socket.AF_INET, socket_type) as s:
s.bind((host, port)) # the port is available if bind is a success
return port
except OSError as e:
if e.errno == errno.EADDRINUSE: # socket already in use
if port + 1 == end_port:
raise DynamipsError("Could not find a free port between {0} and {1}".format(start_port, end_port))
else:
continue
else:
raise DynamipsError("Could not find an unused port: {}".format(e))
def allocate_udp_port(self, max_port=100):
""" """
Allocates a new UDP port for creating an UDP NIO. Allocates a new UDP port for creating an UDP NIO Auto.
:param max_port: maximum number of port to scan in
order to find one available for use.
:returns: port number (integer) :returns: port number (integer)
""" """
start_port = self._current_udp_port # use Dynamips's NIO UDP auto back-end.
end_port = start_port + max_port nio = NIO_UDP_auto(self, self._host, self._udp_start_port_range, self._udp_end_port_range)
allocated_port = DynamipsHypervisor.find_unused_port(start_port, end_port, self._host, socket_type="UDP") self._nio_udp_auto_instances[nio.lport] = nio
if allocated_port - self._current_udp_port > 1: allocated_port = nio.lport
self._current_udp_port += allocated_port - self._current_udp_port
else:
self._current_udp_port += 1
return allocated_port return allocated_port
def send_raw(self, string): def send_raw(self, string):

View File

@ -21,6 +21,7 @@ Manages Dynamips hypervisors (load-balancing etc.)
from .hypervisor import Hypervisor from .hypervisor import Hypervisor
from .dynamips_error import DynamipsError from .dynamips_error import DynamipsError
from ..attic import find_unused_port
from pkg_resources import parse_version from pkg_resources import parse_version
import os import os
@ -44,26 +45,20 @@ class HypervisorManager(object):
:param base_udp: base UDP port for UDP tunnels :param base_udp: base UDP port for UDP tunnels
""" """
def __init__(self, def __init__(self, path, working_dir, host='127.0.0.1'):
path,
working_dir,
host='127.0.0.1',
base_hypervisor_port=7200,
base_console_port=2000,
base_aux_port=3000,
base_udp_port=10000):
self._hypervisors = [] self._hypervisors = []
self._path = path self._path = path
self._working_dir = working_dir self._working_dir = working_dir
self._host = host self._host = host
self._base_hypervisor_port = base_hypervisor_port self._hypervisor_start_port_range = 7200
self._current_port = self._base_hypervisor_port self._hypervisor_end_port_range = 7700
self._base_console_port = base_console_port self._console_start_port_range = 2001
self._base_aux_port = base_aux_port self._console_end_port_range = 2500
self._base_udp_port = base_udp_port self._aux_start_port_range = 2501
self._current_base_udp_port = self._base_udp_port self._aux_end_port_range = 3000
self._udp_incrementation_per_hypervisor = 100 self._udp_start_port_range = 10001
self._udp_end_port_range = 20000
self._ghost_ios_support = True self._ghost_ios_support = True
self._mmap_support = True self._mmap_support = True
self._jit_sharing_support = False self._jit_sharing_support = False
@ -136,94 +131,204 @@ class HypervisorManager(object):
hypervisor.working_dir = self._working_dir hypervisor.working_dir = self._working_dir
@property @property
def base_hypervisor_port(self): def hypervisor_start_port_range(self):
""" """
Returns the base hypervisor port. Returns the hypervisor start port range value
:returns: base hypervisor port (integer) :returns: hypervisor start port range value (integer)
""" """
return self._base_hypervisor_port return self._hypervisor_start_port_range
@base_hypervisor_port.setter @hypervisor_start_port_range.setter
def base_hypervisor_port(self, base_hypervisor_port): def hypervisor_start_port_range(self, hypervisor_start_port_range):
""" """
Set a new base hypervisor port. Sets a new hypervisor start port range value
:param base_hypervisor_port: base hypervisor port (integer) :param hypervisor_start_port_range: hypervisor start port range value (integer)
""" """
if self._base_hypervisor_port != base_hypervisor_port: if self._hypervisor_start_port_range != hypervisor_start_port_range:
self._base_hypervisor_port = base_hypervisor_port self._hypervisor_start_port_range = hypervisor_start_port_range
self._current_port = self._base_hypervisor_port log.info("hypervisor start port range value set to {}".format(self._hypervisor_start_port_range))
log.info("base hypervisor port set to {}".format(self._base_hypervisor_port))
@property @property
def base_console_port(self): def hypervisor_end_port_range(self):
""" """
Returns the base console port. Returns the hypervisor end port range value
:returns: base console port (integer) :returns: hypervisor end port range value (integer)
""" """
return self._base_console_port return self._hypervisor_end_port_range
@base_console_port.setter @hypervisor_end_port_range.setter
def base_console_port(self, base_console_port): def hypervisor_end_port_range(self, hypervisor_end_port_range):
""" """
Set a new base console port. Sets a new hypervisor end port range value
:param base_console_port: base console port (integer) :param hypervisor_end_port_range: hypervisor end port range value (integer)
""" """
if self._base_console_port != base_console_port: if self._hypervisor_end_port_range != hypervisor_end_port_range:
self._base_console_port = base_console_port self._hypervisor_end_port_range = hypervisor_end_port_range
log.info("base console port set to {}".format(self._base_console_port)) log.info("hypervisor end port range value set to {}".format(self._hypervisor_end_port_range))
@property @property
def base_aux_port(self): def console_start_port_range(self):
""" """
Returns the base auxiliary console port. Returns the console start port range value
:returns: base auxiliary console port (integer) :returns: console start port range value (integer)
""" """
return self._base_aux_port return self._console_start_port_range
@base_aux_port.setter @console_start_port_range.setter
def base_aux_port(self, base_aux_port): def console_start_port_range(self, console_start_port_range):
""" """
Set a new base auxiliary console port. Sets a new console start port range value
:param base_aux_port: base auxiliary console port (integer) :param console_start_port_range: console start port range value (integer)
""" """
if self._base_aux_port != base_aux_port: if self._console_start_port_range != console_start_port_range:
self._base_aux_port = base_aux_port self._console_start_port_range = console_start_port_range
log.info("base aux port set to {}".format(self._base_aux_port)) log.info("console start port range value set to {}".format(self._console_start_port_range))
# update all existing hypervisors with the new value
for hypervisor in self._hypervisors:
hypervisor.console_start_port_range = console_start_port_range
@property @property
def base_udp_port(self): def console_end_port_range(self):
""" """
Returns the base UDP port. Returns the console end port range value
:returns: base UDP port (integer) :returns: console end port range value (integer)
""" """
return self._base_udp_port return self._console_end_port_range
@base_udp_port.setter @console_end_port_range.setter
def base_udp_port(self, base_udp_port): def console_end_port_range(self, console_end_port_range):
""" """
Set a new base UDP port. Sets a new console end port range value
:param base_udp_port: base UDP port (integer) :param console_end_port_range: console end port range value (integer)
""" """
if self._base_udp_port != base_udp_port: if self._console_end_port_range != console_end_port_range:
self._base_udp_port = base_udp_port self._console_end_port_range = console_end_port_range
self._current_base_udp_port = self._base_udp_port log.info("console end port range value set to {}".format(self._console_end_port_range))
log.info("base UDP port set to {}".format(self._base_udp_port))
# update all existing hypervisors with the new value
for hypervisor in self._hypervisors:
hypervisor.console_end_port_range = console_end_port_range
@property
def aux_start_port_range(self):
"""
Returns the auxiliary console start port range value
:returns: auxiliary console start port range value (integer)
"""
return self._aux_start_port_range
@aux_start_port_range.setter
def aux_start_port_range(self, aux_start_port_range):
"""
Sets a new auxiliary console start port range value
:param aux_start_port_range: auxiliary console start port range value (integer)
"""
if self._aux_start_port_range != aux_start_port_range:
self._aux_start_port_range = aux_start_port_range
log.info("auxiliary console start port range value set to {}".format(self._aux_start_port_range))
# update all existing hypervisors with the new value
for hypervisor in self._hypervisors:
hypervisor.aux_start_port_range = aux_start_port_range
@property
def aux_end_port_range(self):
"""
Returns the auxiliary console end port range value
:returns: auxiliary console end port range value (integer)
"""
return self._aux_end_port_range
@aux_end_port_range.setter
def aux_end_port_range(self, aux_end_port_range):
"""
Sets a new auxiliary console end port range value
:param aux_end_port_range: auxiliary console end port range value (integer)
"""
if self._aux_end_port_range != aux_end_port_range:
self._aux_end_port_range = aux_end_port_range
log.info("auxiliary console end port range value set to {}".format(self._aux_end_port_range))
# update all existing hypervisors with the new value
for hypervisor in self._hypervisors:
hypervisor.aux_end_port_range = aux_end_port_range
@property
def udp_start_port_range(self):
"""
Returns the UDP start port range value
:returns: UDP start port range value (integer)
"""
return self._udp_start_port_range
@udp_start_port_range.setter
def udp_start_port_range(self, udp_start_port_range):
"""
Sets a new UDP start port range value
:param udp_start_port_range: UDP start port range value (integer)
"""
if self._udp_start_port_range != udp_start_port_range:
self._udp_start_port_range = udp_start_port_range
log.info("UDP start port range value set to {}".format(self._udp_start_port_range))
# update all existing hypervisors with the new value
for hypervisor in self._hypervisors:
hypervisor.udp_start_port_range = udp_start_port_range
@property
def udp_end_port_range(self):
"""
Returns the UDP end port range value
:returns: UDP end port range value (integer)
"""
return self._udp_end_port_range
@udp_end_port_range.setter
def udp_end_port_range(self, udp_end_port_range):
"""
Sets a new UDP end port range value
:param udp_end_port_range: UDP end port range value (integer)
"""
if self._udp_end_port_range != udp_end_port_range:
self._udp_end_port_range = udp_end_port_range
log.info("UDP end port range value set to {}".format(self._udp_end_port_range))
# update all existing hypervisors with the new value
for hypervisor in self._hypervisors:
hypervisor.udp_end_port_range = udp_end_port_range
@property @property
def ghost_ios_support(self): def ghost_ios_support(self):
@ -436,25 +541,6 @@ class HypervisorManager(object):
else: else:
log.info("Dynamips server ready after {:.4f} seconds".format(time.time() - begin)) log.info("Dynamips server ready after {:.4f} seconds".format(time.time() - begin))
def allocate_tcp_port(self, max_port=100):
"""
Allocates a new TCP port for a Dynamips hypervisor.
:param max_port: maximum number of port to scan in
order to find one available for use.
:returns: port number (integer)
"""
start_port = self._current_port
end_port = start_port + max_port
allocated_port = Hypervisor.find_unused_port(start_port, end_port, self._host)
if allocated_port - self._current_port > 1:
self._current_port += allocated_port - self._current_port
else:
self._current_port += 1
return allocated_port
def start_new_hypervisor(self): def start_new_hypervisor(self):
""" """
Creates a new Dynamips process and start it. Creates a new Dynamips process and start it.
@ -462,7 +548,11 @@ class HypervisorManager(object):
:returns: the new hypervisor instance :returns: the new hypervisor instance
""" """
port = self.allocate_tcp_port() try:
port = find_unused_port(self._hypervisor_start_port_range, self._hypervisor_end_port_range, self._host)
except Exception as e:
raise DynamipsError(e)
hypervisor = Hypervisor(self._path, hypervisor = Hypervisor(self._path,
self._working_dir, self._working_dir,
self._host, self._host,
@ -477,10 +567,13 @@ class HypervisorManager(object):
hypervisor.connect() hypervisor.connect()
if parse_version(hypervisor.version) < parse_version('0.2.11'): if parse_version(hypervisor.version) < parse_version('0.2.11'):
raise DynamipsError("Dynamips version must be >= 0.2.11, detected version is {}".format(hypervisor.version)) raise DynamipsError("Dynamips version must be >= 0.2.11, detected version is {}".format(hypervisor.version))
hypervisor.baseconsole = self._base_console_port
hypervisor.baseaux = self._base_aux_port hypervisor.console_start_port_range = self._console_start_port_range
hypervisor.baseudp = self._current_base_udp_port hypervisor.console_end_port_range = self._console_end_port_range
self._current_base_udp_port += self._udp_incrementation_per_hypervisor hypervisor.aux_start_port_range = self._aux_start_port_range
hypervisor.aux_end_port_range = self._aux_end_port_range
hypervisor.udp_start_port_range = self._udp_start_port_range
hypervisor.udp_end_port_range = self._udp_end_port_range
self._hypervisors.append(hypervisor) self._hypervisors.append(hypervisor)
return hypervisor return hypervisor

View File

@ -20,8 +20,9 @@ Interface for Dynamips virtual Machine module ("vm")
http://github.com/GNS3/dynamips/blob/master/README.hypervisor#L77 http://github.com/GNS3/dynamips/blob/master/README.hypervisor#L77
""" """
from ..dynamips_hypervisor import DynamipsHypervisor
from ..dynamips_error import DynamipsError from ..dynamips_error import DynamipsError
from ...attic import find_unused_port
import time import time
import sys import sys
import os import os
@ -41,6 +42,8 @@ class Router(object):
""" """
_allocated_names = [] _allocated_names = []
_allocated_console_ports = []
_allocated_aux_ports = []
_instance_count = 1 _instance_count = 1
_status = {0: "inactive", _status = {0: "inactive",
1: "shutting down", 1: "shutting down",
@ -103,9 +106,29 @@ class Router(object):
platform=platform, platform=platform,
id=self._id)) id=self._id))
# get console and aux ports try:
self.console = (self._hypervisor.baseconsole - 1) + self._id # allocate a console port
self.aux = (self._hypervisor.baseaux - 1) + self._id self._console = find_unused_port(self._hypervisor.console_start_port_range,
self._hypervisor.console_end_port_range,
self._hypervisor.host,
ignore_ports=self._allocated_console_ports)
self._hypervisor.send("vm set_con_tcp_port {name} {console}".format(name=self._name,
console=self._console))
self._allocated_console_ports.append(self._console)
# allocate a auxiliary console port
self._aux = find_unused_port(self._hypervisor.aux_start_port_range,
self._hypervisor.aux_end_port_range,
self._hypervisor.host,
ignore_ports=self._allocated_aux_ports)
self._hypervisor.send("vm set_aux_tcp_port {name} {aux}".format(name=self._name,
aux=self._aux))
self._allocated_aux_ports.append(self._aux)
except Exception as e:
raise DynamipsError(e)
# get the default base MAC address # get the default base MAC address
self._mac_addr = self._hypervisor.send("{platform} get_mac_addr {name}".format(platform=self._platform, self._mac_addr = self._hypervisor.send("{platform} get_mac_addr {name}".format(platform=self._platform,
@ -124,6 +147,8 @@ class Router(object):
cls._instance_count = 1 cls._instance_count = 1
cls._allocated_names.clear() cls._allocated_names.clear()
cls._allocated_console_ports.clear()
cls._allocated_aux_ports.clear()
def defaults(self): def defaults(self):
""" """
@ -260,6 +285,10 @@ class Router(object):
log.info("router {name} [id={id}] has been deleted".format(name=self._name, id=self._id)) log.info("router {name} [id={id}] has been deleted".format(name=self._name, id=self._id))
self._allocated_names.remove(self.name) self._allocated_names.remove(self.name)
if self.console:
self._allocated_console_ports.remove(self.console)
if self.aux:
self._allocated_aux_ports.remove(self.aux)
def start(self): def start(self):
""" """
@ -285,20 +314,6 @@ class Router(object):
if elf_header_start != b'\x7fELF\x01\x02\x01': if elf_header_start != b'\x7fELF\x01\x02\x01':
raise DynamipsError("'{}' is not a valid IOU image".format(self._image)) raise DynamipsError("'{}' is not a valid IOU image".format(self._image))
if self.console and self.aux:
# check that console and aux ports are available
try:
#FIXME: use a defined range
DynamipsHypervisor.find_unused_port(self.console, self.console + 100, self._hypervisor.host)
except DynamipsError:
raise DynamipsError("console port {} is not available".format(self.console))
try:
#FIXME: use a defined range
DynamipsHypervisor.find_unused_port(self.aux, self.aux + 100, self._hypervisor.host)
except DynamipsError:
raise DynamipsError("aux port {} is not available".format(self.aux))
self._hypervisor.send("vm start {}".format(self._name)) self._hypervisor.send("vm start {}".format(self._name))
log.info("router {name} [id={id}] has been started".format(name=self._name, id=self._id)) log.info("router {name} [id={id}] has been started".format(name=self._name, id=self._id))
@ -1021,6 +1036,9 @@ class Router(object):
if console == self._console: if console == self._console:
return return
if console in self._allocated_console_ports:
raise DynamipsError("Console port {} is already used by another router".format(console))
self._hypervisor.send("vm set_con_tcp_port {name} {console}".format(name=self._name, self._hypervisor.send("vm set_con_tcp_port {name} {console}".format(name=self._name,
console=console)) console=console))
@ -1028,7 +1046,9 @@ class Router(object):
id=self._id, id=self._id,
old_console=self._console, old_console=self._console,
new_console=console)) new_console=console))
self._allocated_console_ports.remove(self._console)
self._console = console self._console = console
self._allocated_console_ports.append(self._console)
@property @property
def aux(self): def aux(self):
@ -1051,6 +1071,9 @@ class Router(object):
if aux == self._aux: if aux == self._aux:
return return
if aux in self._allocated_aux_ports:
raise DynamipsError("Auxiliary console port {} is already used by another router".format(aux))
self._hypervisor.send("vm set_aux_tcp_port {name} {aux}".format(name=self._name, self._hypervisor.send("vm set_aux_tcp_port {name} {aux}".format(name=self._name,
aux=aux)) aux=aux))
@ -1058,7 +1081,10 @@ class Router(object):
id=self._id, id=self._id,
old_aux=self._aux, old_aux=self._aux,
new_aux=aux)) new_aux=aux))
self._allocated_aux_ports.remove(self._aux)
self._aux = aux self._aux = aux
self._allocated_aux_ports.append(self._aux)
def get_cpu_info(self, cpu_id=0): def get_cpu_info(self, cpu_id=0):
""" """

View File

@ -30,12 +30,13 @@ import shutil
from gns3server.modules import IModule from gns3server.modules import IModule
from gns3server.config import Config from gns3server.config import Config
import gns3server.jsonrpc as jsonrpc
from .iou_device import IOUDevice from .iou_device import IOUDevice
from .iou_error import IOUError from .iou_error import IOUError
from .nios.nio_udp import NIO_UDP from .nios.nio_udp import NIO_UDP
from .nios.nio_tap import NIO_TAP from .nios.nio_tap import NIO_TAP
from .nios.nio_generic_ethernet import NIO_GenericEthernet from .nios.nio_generic_ethernet import NIO_GenericEthernet
import gns3server.jsonrpc as jsonrpc from ..attic import find_unused_port
from .schemas import IOU_CREATE_SCHEMA from .schemas import IOU_CREATE_SCHEMA
from .schemas import IOU_DELETE_SCHEMA from .schemas import IOU_DELETE_SCHEMA
@ -90,6 +91,7 @@ class IOU(IModule):
self._iou_instances = {} self._iou_instances = {}
self._console_start_port_range = 4001 self._console_start_port_range = 4001
self._console_end_port_range = 4512 self._console_end_port_range = 4512
self._allocated_console_ports = []
self._current_console_port = self._console_start_port_range self._current_console_port = self._console_start_port_range
self._udp_start_port_range = 30001 self._udp_start_port_range = 30001
self._udp_end_port_range = 40001 self._udp_end_port_range = 40001
@ -299,9 +301,12 @@ class IOU(IModule):
iou_instance = IOUDevice(iou_path, self._working_dir, host=self._host, name=name) iou_instance = IOUDevice(iou_path, self._working_dir, host=self._host, name=name)
# find a console port # find a console port
if self._current_console_port >= self._console_end_port_range: if self._current_console_port > self._console_end_port_range:
self._current_console_port = self._console_start_port_range self._current_console_port = self._console_start_port_range
iou_instance.console = IOUDevice.find_unused_port(self._current_console_port, self._console_end_port_range, self._host) try:
iou_instance.console = find_unused_port(self._current_console_port, self._console_end_port_range, self._host)
except Exception as e:
raise IOUError(e)
self._current_console_port += 1 self._current_console_port += 1
except IOUError as e: except IOUError as e:
self.send_custom_error(str(e)) self.send_custom_error(str(e))
@ -532,7 +537,10 @@ class IOU(IModule):
# find a UDP port # find a UDP port
if self._current_udp_port >= self._udp_end_port_range: if self._current_udp_port >= self._udp_end_port_range:
self._current_udp_port = self._udp_start_port_range self._current_udp_port = self._udp_start_port_range
port = IOUDevice.find_unused_port(self._current_udp_port, self._udp_end_port_range, host=self._host, socket_type="UDP") try:
port = find_unused_port(self._current_udp_port, self._udp_end_port_range, host=self._host, socket_type="UDP")
except Exception as e:
raise IOUError(e)
self._current_udp_port += 1 self._current_udp_port += 1
log.info("{} [id={}] has allocated UDP port {} with host {}".format(iou_instance.name, log.info("{} [id={}] has allocated UDP port {} with host {}".format(iou_instance.name,

View File

@ -22,8 +22,6 @@ order to run an IOU instance.
import os import os
import re import re
import socket
import errno
import signal import signal
import subprocess import subprocess
import argparse import argparse
@ -813,38 +811,3 @@ class IOUDevice(object):
adapters=len(self._serial_adapters))) adapters=len(self._serial_adapters)))
self._slots = self._ethernet_adapters + self._serial_adapters self._slots = self._ethernet_adapters + self._serial_adapters
@staticmethod
def find_unused_port(start_port, end_port, host='127.0.0.1', socket_type="TCP"):
"""
Finds an unused port in the specified range.
:param start_port: first port in the range
:param end_port: last port in the range
:param host: host/address for bind()
:param socket_type: TCP (default) or UDP
"""
if socket_type == "UDP":
socket_type = socket.SOCK_DGRAM
else:
socket_type = socket.SOCK_STREAM
for port in range(start_port, end_port):
try:
if ":" in host:
# IPv6 address support
with socket.socket(socket.AF_INET6, socket_type) as s:
s.bind((host, port)) # the port is available if bind is a success
else:
with socket.socket(socket.AF_INET, socket_type) as s:
s.bind((host, port)) # the port is available if bind is a success
return port
except OSError as e:
if e.errno == errno.EADDRINUSE: # socket already in use
if port + 1 == end_port:
raise IOUError("Could not find a free port between {0} and {1}".format(start_port, end_port))
else:
continue
else:
raise IOUError("Could not find an unused port: {}".format(e))

View File

@ -25,6 +25,8 @@ class IOUError(Exception):
def __init__(self, message, original_exception=None): def __init__(self, message, original_exception=None):
Exception.__init__(self, message) Exception.__init__(self, message)
if isinstance(message, Exception):
message = str(message)
self._message = message self._message = message
self._original_exception = original_exception self._original_exception = original_exception

View File

@ -23,5 +23,5 @@
# or negative for a release candidate or beta (after the base version # or negative for a release candidate or beta (after the base version
# number has been incremented) # number has been incremented)
__version__ = "1.0a3.dev2" __version__ = "1.0a3.dev3"
__version_info__ = (1, 0, 0, -99) __version_info__ = (1, 0, 0, -99)