Cloud schema and remove use_ubridge for VMware VM.

This commit is contained in:
grossmj 2016-05-28 21:46:48 -06:00
parent df654b40ff
commit 39a3f2fae2
6 changed files with 306 additions and 74 deletions

View File

@ -19,6 +19,9 @@ import asyncio
from ...node_error import NodeError from ...node_error import NodeError
from ...base_node import BaseNode from ...base_node import BaseNode
from ...nios.nio_udp import NIOUDP
from gns3server.utils.interfaces import interfaces
import logging import logging
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
@ -35,17 +38,52 @@ class Cloud(BaseNode):
:param manager: Parent VM Manager :param manager: Parent VM Manager
""" """
def __init__(self, name, node_id, project, manager): def __init__(self, name, node_id, project, manager, ports=None):
super().__init__(name, node_id, project, manager) super().__init__(name, node_id, project, manager)
self._nios = {}
self._ports = []
if ports:
self._ports = ports
def __json__(self): def __json__(self):
host_interfaces = []
network_interfaces = interfaces()
for interface in network_interfaces:
interface_type = "ethernet"
if interface["name"].startswith("tap"):
# found no way to reliably detect a TAP interface
interface_type = "tap"
host_interfaces.append({"name": interface["name"],
"type": interface_type})
return {"name": self.name, return {"name": self.name,
"node_id": self.id, "node_id": self.id,
"project_id": self.project.id} "project_id": self.project.id,
"ports": self._ports,
"interfaces": host_interfaces}
@property
def ports(self):
"""
Ports on this cloud.
:returns: ports info
"""
return self._ports
@ports.setter
def ports(self, ports):
"""
Set the ports on this cloud.
:param ports: ports info
"""
self._ports = ports
@asyncio.coroutine
def create(self): def create(self):
""" """
Creates this cloud. Creates this cloud.
@ -54,13 +92,17 @@ class Cloud(BaseNode):
super().create() super().create()
log.info('Cloud "{name}" [{id}] has been created'.format(name=self._name, id=self._id)) log.info('Cloud "{name}" [{id}] has been created'.format(name=self._name, id=self._id))
@asyncio.coroutine
def delete(self): def delete(self):
""" """
Deletes this cloud. Deletes this cloud.
""" """
raise NotImplementedError() for nio in self._nios.values():
if nio and isinstance(nio, NIOUDP):
self.manager.port_manager.release_udp_port(nio.lport, self._project)
super().delete()
log.info('Cloud "{name}" [{id}] has been deleted'.format(name=self._name, id=self._id))
@asyncio.coroutine @asyncio.coroutine
def add_nio(self, nio, port_number): def add_nio(self, nio, port_number):
@ -71,7 +113,19 @@ class Cloud(BaseNode):
:param port_number: port to allocate for the NIO :param port_number: port to allocate for the NIO
""" """
raise NotImplementedError() if port_number in self._nios:
raise NodeError("Port {} isn't free".format(port_number))
log.info('Cloud "{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
for port_settings in self._ports:
if port_settings["port_number"] == port_number:
#yield from self.set_port_settings(port_number, port_settings)
break
@asyncio.coroutine @asyncio.coroutine
def remove_nio(self, port_number): def remove_nio(self, port_number):
@ -83,7 +137,20 @@ class Cloud(BaseNode):
:returns: the NIO that was bound to the allocated port :returns: the NIO that was bound to the allocated port
""" """
raise NotImplementedError() if port_number not in self._nios:
raise NodeError("Port {} is not allocated".format(port_number))
nio = self._nios[port_number]
if isinstance(nio, NIOUDP):
self.manager.port_manager.release_udp_port(nio.lport, self._project)
log.info('Cloud "{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]
return nio
@asyncio.coroutine @asyncio.coroutine
def start_capture(self, port_number, output_file, data_link_type="DLT_EN10MB"): def start_capture(self, port_number, output_file, data_link_type="DLT_EN10MB"):

View File

@ -71,7 +71,7 @@ class VMwareVM(BaseNode):
self._adapters = 0 self._adapters = 0
self._ethernet_adapters = {} self._ethernet_adapters = {}
self._adapter_type = "e1000" self._adapter_type = "e1000"
self._use_ubridge = True self._use_ubridge = True # TODO: clean this (old ubridge code)
self._use_any_adapter = False self._use_any_adapter = False
if not os.path.exists(vmx_path): if not os.path.exists(vmx_path):
@ -89,7 +89,6 @@ class VMwareVM(BaseNode):
"enable_remote_console": self.enable_remote_console, "enable_remote_console": self.enable_remote_console,
"adapters": self._adapters, "adapters": self._adapters,
"adapter_type": self.adapter_type, "adapter_type": self.adapter_type,
"use_ubridge": self.use_ubridge,
"use_any_adapter": self.use_any_adapter, "use_any_adapter": self.use_any_adapter,
"status": self.status, "status": self.status,
"node_directory": self.working_dir} "node_directory": self.working_dir}
@ -740,30 +739,6 @@ class VMwareVM(BaseNode):
id=self.id, id=self.id,
adapter_type=adapter_type)) adapter_type=adapter_type))
@property
def use_ubridge(self):
"""
Returns either GNS3 can use uBridge for network connections.
:returns: boolean
"""
return self._use_ubridge
@use_ubridge.setter
def use_ubridge(self, use_ubridge):
"""
Allows GNS3 to use uBridge for network connections.
:param use_ubridge: boolean
"""
if use_ubridge:
log.info("VMware VM '{name}' [{id}] will use uBridge for network connections".format(name=self.name, id=self.id))
else:
log.info("VMware VM '{name}' [{id}] will not use uBridge for network connections".format(name=self.name, id=self.id))
self._use_ubridge = use_ubridge
@property @property
def use_any_adapter(self): def use_any_adapter(self):
""" """

View File

@ -51,7 +51,8 @@ class CloudHandler:
node = yield from builtin_manager.create_node(request.json.pop("name"), node = yield from builtin_manager.create_node(request.json.pop("name"),
request.match_info["project_id"], request.match_info["project_id"],
request.json.get("node_id"), request.json.get("node_id"),
node_type="cloud") node_type="cloud",
ports=request.json.get("ports"))
response.set_status(201) response.set_status(201)
response.json(node) response.json(node)
@ -93,6 +94,9 @@ class CloudHandler:
builtin_manager = Builtin.instance() builtin_manager = Builtin.instance()
node = builtin_manager.get_node(request.match_info["node_id"], project_id=request.match_info["project_id"]) node = builtin_manager.get_node(request.match_info["node_id"], project_id=request.match_info["project_id"])
for name, value in request.json.items():
if hasattr(node, name) and getattr(node, name) != value:
setattr(node, name, value)
node.updated() node.updated()
response.json(node) response.json(node)

View File

@ -16,13 +16,14 @@
# along with this program. If not, see <http://www.gnu.org/licenses/>. # along with this program. If not, see <http://www.gnu.org/licenses/>.
CLOUD_CREATE_SCHEMA = { CLOUD_CREATE_SCHEMA = {
"$schema": "http://json-schema.org/draft-04/schema#", "$schema": "http://json-schema.org/draft-04/schema#",
"description": "Request validation to create a new cloud instance", "description": "Request validation to create a new cloud instance",
"type": "object", "type": "object",
"definitions": { "definitions": {
"EthernetHubPort": { "EthernetInterfacePort": {
"description": "Ethernet port", "description": "Ethernet interface port",
"properties": { "properties": {
"name": { "name": {
"description": "Port name", "description": "Port name",
@ -34,8 +35,96 @@ CLOUD_CREATE_SCHEMA = {
"type": "integer", "type": "integer",
"minimum": 1 "minimum": 1
}, },
"type": {
"description": "Port type",
"enum": ["ethernet"]
},
"interface": {
"description": "Ethernet interface name e.g. eth0",
"type": "string",
"minLength": 1
},
}, },
"required": ["name", "port_number"], "required": ["name", "port_number", "type", "interface"],
"additionalProperties": False
},
"TAPInterfacePort": {
"description": "TAP interface port",
"properties": {
"name": {
"description": "Port name",
"type": "string",
"minLength": 1,
},
"port_number": {
"description": "Port number",
"type": "integer",
"minimum": 1
},
"type": {
"description": "Port type",
"enum": ["tap"]
},
"interface": {
"description": "TAP interface name e.g. tap0",
"type": "string",
"minLength": 1
},
},
"required": ["name", "port_number", "type", "interface"],
"additionalProperties": False
},
"UDPTunnelPort": {
"description": "UDP tunnel port",
"properties": {
"name": {
"description": "Port name",
"type": "string",
"minLength": 1,
},
"port_number": {
"description": "Port number",
"type": "integer",
"minimum": 1
},
"type": {
"description": "Port type",
"enum": ["udp"]
},
"lport": {
"description": "Local UDP tunnel port",
"type": "integer",
"minimum": 1,
"maximum": 65535
},
"rhost": {
"description": "Remote UDP tunnel host",
"type": "string",
"minLength": 1
},
"rport": {
"description": "Remote UDP tunnel port",
"type": "integer",
"minimum": 1,
"maximum": 65535
}
},
"required": ["name", "port_number", "type", "lport", "rhost", "rport"],
"additionalProperties": False
},
"HostInterfaces": {
"description": "Interfaces on this host",
"properties": {
"name": {
"description": "Interface name",
"type": "string",
"minLength": 1,
},
"type": {
"enum": ["Ethernet", "TAP"]
},
},
"required": ["name", "type"],
"additionalProperties": False "additionalProperties": False
}, },
}, },
@ -54,15 +143,26 @@ CLOUD_CREATE_SCHEMA = {
"pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$"} "pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$"}
] ]
}, },
# "ports": { "ports": {
# "type": "array", "type": "array",
# "items": [ "items": [
# {"type": "object", {"type": "object",
# "oneOf": [ "oneOf": [
# {"$ref": "#/definitions/EthernetHubPort"} {"$ref": "#/definitions/EthernetInterfacePort"},
# ]}, {"$ref": "#/definitions/TAPInterfacePort"},
# ] {"$ref": "#/definitions/UDPTunnelPort"}
# }, ]},
]
},
"interfaces": {
"type": "array",
"items": [
{"type": "object",
"oneOf": [
{"$ref": "#/definitions/HostInterfaces"}
]},
]
},
}, },
"additionalProperties": False, "additionalProperties": False,
"required": ["name"] "required": ["name"]
@ -73,8 +173,8 @@ CLOUD_OBJECT_SCHEMA = {
"description": "Cloud instance", "description": "Cloud instance",
"type": "object", "type": "object",
"definitions": { "definitions": {
"EthernetHubPort": { "EthernetInterfacePort": {
"description": "Ethernet port", "description": "Ethernet interface port",
"properties": { "properties": {
"name": { "name": {
"description": "Port name", "description": "Port name",
@ -86,8 +186,96 @@ CLOUD_OBJECT_SCHEMA = {
"type": "integer", "type": "integer",
"minimum": 1 "minimum": 1
}, },
"type": {
"description": "Port type",
"enum": ["ethernet"]
},
"interface": {
"description": "Ethernet interface name e.g. eth0",
"type": "string",
"minLength": 1
},
}, },
"required": ["name", "port_number"], "required": ["name", "port_number", "type", "interface"],
"additionalProperties": False
},
"TAPInterfacePort": {
"description": "TAP interface port",
"properties": {
"name": {
"description": "Port name",
"type": "string",
"minLength": 1,
},
"port_number": {
"description": "Port number",
"type": "integer",
"minimum": 1
},
"type": {
"description": "Port type",
"enum": ["tap"]
},
"interface": {
"description": "TAP interface name e.g. tap0",
"type": "string",
"minLength": 1
},
},
"required": ["name", "port_number", "type", "interface"],
"additionalProperties": False
},
"UDPTunnelPort": {
"description": "UDP tunnel port",
"properties": {
"name": {
"description": "Port name",
"type": "string",
"minLength": 1,
},
"port_number": {
"description": "Port number",
"type": "integer",
"minimum": 1
},
"type": {
"description": "Port type",
"enum": ["udp"]
},
"lport": {
"description": "Local UDP tunnel port",
"type": "integer",
"minimum": 1,
"maximum": 65535
},
"rhost": {
"description": "Remote UDP tunnel host",
"type": "string",
"minLength": 1
},
"rport": {
"description": "Remote UDP tunnel port",
"type": "integer",
"minimum": 1,
"maximum": 65535
}
},
"required": ["name", "port_number", "type", "lport", "rhost", "rport"],
"additionalProperties": False
},
"HostInterfaces": {
"description": "Interfaces on this host",
"properties": {
"name": {
"description": "Interface name",
"type": "string",
"minLength": 1,
},
"type": {
"enum": ["ethernet", "tap"]
},
},
"required": ["name", "type"],
"additionalProperties": False "additionalProperties": False
}, },
}, },
@ -111,22 +299,33 @@ CLOUD_OBJECT_SCHEMA = {
"maxLength": 36, "maxLength": 36,
"pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$" "pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$"
}, },
# "ports": { "ports": {
# "type": "array", "type": "array",
# "items": [ "items": [
# {"type": "object", {"type": "object",
# "oneOf": [ "oneOf": [
# {"$ref": "#/definitions/EthernetHubPort"} {"$ref": "#/definitions/EthernetInterfacePort"},
# ]}, {"$ref": "#/definitions/TAPInterfacePort"},
# ] {"$ref": "#/definitions/UDPTunnelPort"}
# }, ]},
]
},
"interfaces": {
"type": "array",
"items": [
{"type": "object",
"oneOf": [
{"$ref": "#/definitions/HostInterfaces"}
]},
]
},
"status": { "status": {
"description": "Node status", "description": "Node status",
"enum": ["started", "stopped", "suspended"] "enum": ["started", "stopped", "suspended"]
}, },
}, },
"additionalProperties": False, "additionalProperties": False,
"required": ["name", "node_id", "project_id"] #, "ports"] "required": ["name", "node_id", "project_id", "ports"]
} }
CLOUD_UPDATE_SCHEMA = CLOUD_OBJECT_SCHEMA CLOUD_UPDATE_SCHEMA = CLOUD_OBJECT_SCHEMA

View File

@ -71,10 +71,6 @@ VMWARE_CREATE_SCHEMA = {
"type": "string", "type": "string",
"minLength": 1, "minLength": 1,
}, },
"use_ubridge": {
"description": "Use uBridge for network connections",
"type": "boolean",
},
"use_any_adapter": { "use_any_adapter": {
"description": "Allow GNS3 to use any VMware adapter", "description": "Allow GNS3 to use any VMware adapter",
"type": "boolean", "type": "boolean",
@ -128,10 +124,6 @@ VMWARE_UPDATE_SCHEMA = {
"type": "string", "type": "string",
"minLength": 1, "minLength": 1,
}, },
"use_ubridge": {
"description": "Use uBridge for network connections",
"type": "boolean",
},
"use_any_adapter": { "use_any_adapter": {
"description": "Allow GNS3 to use any VMware adapter", "description": "Allow GNS3 to use any VMware adapter",
"type": "boolean", "type": "boolean",
@ -201,10 +193,6 @@ VMWARE_OBJECT_SCHEMA = {
"type": "string", "type": "string",
"minLength": 1, "minLength": 1,
}, },
"use_ubridge": {
"description": "Use uBridge for network connections",
"type": "boolean",
},
"use_any_adapter": { "use_any_adapter": {
"description": "Allow GNS3 to use any VMware adapter", "description": "Allow GNS3 to use any VMware adapter",
"type": "boolean", "type": "boolean",

View File

@ -21,9 +21,9 @@ import asyncio
class Pool(): class Pool():
""" """
Limit concurrency for running parallel task Limit concurrency for running parallel tasks
""" """
def __init__(self, concurrency=2): def __init__(self, concurrency=5):
self._tasks = [] self._tasks = []
self._concurrency = concurrency self._concurrency = concurrency
@ -41,7 +41,6 @@ class Pool():
task, args, kwargs = self._tasks.pop(0) task, args, kwargs = self._tasks.pop(0)
pending.add(task(*args, **kwargs)) pending.add(task(*args, **kwargs))
(done, pending) = yield from asyncio.wait(pending, return_when=asyncio.FIRST_COMPLETED) (done, pending) = yield from asyncio.wait(pending, return_when=asyncio.FIRST_COMPLETED)
print(done)
def main(): def main():