mirror of
https://github.com/GNS3/gns3-server
synced 2025-01-13 09:30:54 +00:00
Prevent capture on non running node
Fix https://github.com/GNS3/gns3-gui/issues/1656
This commit is contained in:
parent
31c04cc95b
commit
875051d05e
@ -410,7 +410,7 @@ class Compute:
|
|||||||
self._memory_usage_percent = event["memory_usage_percent"]
|
self._memory_usage_percent = event["memory_usage_percent"]
|
||||||
self._controller.notification.emit("compute.updated", self.__json__())
|
self._controller.notification.emit("compute.updated", self.__json__())
|
||||||
else:
|
else:
|
||||||
self._controller.notification.dispatch(action, event, compute_id=self.id)
|
yield from self._controller.notification.dispatch(action, event, compute_id=self.id)
|
||||||
if self._ws:
|
if self._ws:
|
||||||
yield from self._ws.close()
|
yield from self._ws.close()
|
||||||
|
|
||||||
|
@ -91,6 +91,8 @@ class Link:
|
|||||||
|
|
||||||
if len(self._nodes) == 2:
|
if len(self._nodes) == 2:
|
||||||
yield from self.create()
|
yield from self.create()
|
||||||
|
for n in self._nodes:
|
||||||
|
n["node"].add_link(self)
|
||||||
self._created = True
|
self._created = True
|
||||||
self._project.controller.notification.emit("link.created", self.__json__())
|
self._project.controller.notification.emit("link.created", self.__json__())
|
||||||
|
|
||||||
@ -121,8 +123,8 @@ class Link:
|
|||||||
"""
|
"""
|
||||||
Delete the link
|
Delete the link
|
||||||
"""
|
"""
|
||||||
|
for port in self._nodes:
|
||||||
raise NotImplementedError
|
port["node"].remove_link(self)
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def start_capture(self, data_link_type="DLT_EN10MB", capture_file_name=None):
|
def start_capture(self, data_link_type="DLT_EN10MB", capture_file_name=None):
|
||||||
@ -173,6 +175,13 @@ class Link:
|
|||||||
|
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def node_updated(self, node):
|
||||||
|
"""
|
||||||
|
Called when a node member of the link is updated
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
def default_capture_file_name(self):
|
def default_capture_file_name(self):
|
||||||
"""
|
"""
|
||||||
:returns: File name for a capture on this link
|
:returns: File name for a capture on this link
|
||||||
@ -214,6 +223,9 @@ class Link:
|
|||||||
return False
|
return False
|
||||||
return self.id == other.id
|
return self.id == other.id
|
||||||
|
|
||||||
|
def __hash__(self):
|
||||||
|
return hash(self._id)
|
||||||
|
|
||||||
def __json__(self, topology_dump=False):
|
def __json__(self, topology_dump=False):
|
||||||
"""
|
"""
|
||||||
:param topology_dump: Filter to keep only properties require for saving on disk
|
:param topology_dump: Filter to keep only properties require for saving on disk
|
||||||
|
@ -59,6 +59,7 @@ class Node:
|
|||||||
self._node_type = node_type
|
self._node_type = node_type
|
||||||
|
|
||||||
self._label = None
|
self._label = None
|
||||||
|
self._links = set()
|
||||||
self._name = None
|
self._name = None
|
||||||
self.name = name
|
self.name = name
|
||||||
self._console = None
|
self._console = None
|
||||||
@ -268,6 +269,22 @@ class Node:
|
|||||||
def first_port_name(self, val):
|
def first_port_name(self, val):
|
||||||
self._first_port_name = val
|
self._first_port_name = val
|
||||||
|
|
||||||
|
def add_link(self, link):
|
||||||
|
"""
|
||||||
|
A link is connected to the node
|
||||||
|
"""
|
||||||
|
self._links.add(link)
|
||||||
|
|
||||||
|
def remove_link(self, link):
|
||||||
|
"""
|
||||||
|
A link is connected to the node
|
||||||
|
"""
|
||||||
|
self._links.remove(link)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def link(self):
|
||||||
|
return self._links
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def create(self):
|
def create(self):
|
||||||
"""
|
"""
|
||||||
@ -291,7 +308,7 @@ class Node:
|
|||||||
else:
|
else:
|
||||||
raise e
|
raise e
|
||||||
else:
|
else:
|
||||||
self.parse_node_response(response.json)
|
yield from self.parse_node_response(response.json)
|
||||||
return True
|
return True
|
||||||
trial += 1
|
trial += 1
|
||||||
|
|
||||||
@ -328,9 +345,10 @@ class Node:
|
|||||||
if update_compute:
|
if update_compute:
|
||||||
data = self._node_data(properties=compute_properties)
|
data = self._node_data(properties=compute_properties)
|
||||||
response = yield from self.put(None, data=data)
|
response = yield from self.put(None, data=data)
|
||||||
self.parse_node_response(response.json)
|
yield from self.parse_node_response(response.json)
|
||||||
self.project.dump()
|
self.project.dump()
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def parse_node_response(self, response):
|
def parse_node_response(self, response):
|
||||||
"""
|
"""
|
||||||
Update the object with the remote node object
|
Update the object with the remote node object
|
||||||
@ -353,6 +371,8 @@ class Node:
|
|||||||
else:
|
else:
|
||||||
self._properties[key] = value
|
self._properties[key] = value
|
||||||
self._list_ports()
|
self._list_ports()
|
||||||
|
for link in self._links:
|
||||||
|
yield from link.node_updated(self)
|
||||||
|
|
||||||
def _node_data(self, properties=None):
|
def _node_data(self, properties=None):
|
||||||
"""
|
"""
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
import aiohttp
|
import aiohttp
|
||||||
|
import asyncio
|
||||||
from contextlib import contextmanager
|
from contextlib import contextmanager
|
||||||
|
|
||||||
from ..notification_queue import NotificationQueue
|
from ..notification_queue import NotificationQueue
|
||||||
@ -51,6 +52,7 @@ class Notification:
|
|||||||
"""
|
"""
|
||||||
return project.id in self._listeners and len(self._listeners[project.id]) > 0
|
return project.id in self._listeners and len(self._listeners[project.id]) > 0
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def dispatch(self, action, event, compute_id):
|
def dispatch(self, action, event, compute_id):
|
||||||
"""
|
"""
|
||||||
Notification received from compute node. Send it directly
|
Notification received from compute node. Send it directly
|
||||||
@ -65,7 +67,7 @@ class Notification:
|
|||||||
# Update controller node data and send the event node.updated
|
# Update controller node data and send the event node.updated
|
||||||
project = self._controller.get_project(event["project_id"])
|
project = self._controller.get_project(event["project_id"])
|
||||||
node = project.get_node(event["node_id"])
|
node = project.get_node(event["node_id"])
|
||||||
node.parse_node_response(event)
|
yield from node.parse_node_response(event)
|
||||||
|
|
||||||
self.emit("node.updated", node.__json__())
|
self.emit("node.updated", node.__json__())
|
||||||
except aiohttp.web.HTTPNotFound:
|
except aiohttp.web.HTTPNotFound:
|
||||||
|
@ -115,6 +115,7 @@ class UDPLink(Link):
|
|||||||
# If the node is already delete (user selected multiple element and delete all in the same time)
|
# If the node is already delete (user selected multiple element and delete all in the same time)
|
||||||
except aiohttp.web.HTTPNotFound:
|
except aiohttp.web.HTTPNotFound:
|
||||||
pass
|
pass
|
||||||
|
yield from super().delete()
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def start_capture(self, data_link_type="DLT_EN10MB", capture_file_name=None):
|
def start_capture(self, data_link_type="DLT_EN10MB", capture_file_name=None):
|
||||||
@ -145,21 +146,31 @@ class UDPLink(Link):
|
|||||||
"""
|
"""
|
||||||
Run capture on the best candidate.
|
Run capture on the best candidate.
|
||||||
|
|
||||||
The ideal candidate is a node who support capture on controller server
|
The ideal candidate is a node who on controller server and always
|
||||||
|
running (capture will not be cut off)
|
||||||
|
|
||||||
:returns: Node where the capture should run
|
:returns: Node where the capture should run
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# use the local node first to save bandwidth
|
ALWAYS_RUNNING_NODES_TYPE = ("cloud", "nat", "ethernet_switch", "ethernet_hub")
|
||||||
|
|
||||||
for node in self._nodes:
|
for node in self._nodes:
|
||||||
if node["node"].compute.id == "local" and node["node"].node_type not in [""]: # FIXME
|
if node["node"].compute.id == "local" and node["node"].node_type in ALWAYS_RUNNING_NODES_TYPE and node["node"].status == "started":
|
||||||
return node
|
return node
|
||||||
|
|
||||||
for node in self._nodes:
|
for node in self._nodes:
|
||||||
if node["node"].node_type not in [""]: # FIXME
|
if node["node"].node_type in ALWAYS_RUNNING_NODES_TYPE and node["node"].status == "started":
|
||||||
return node
|
return node
|
||||||
|
|
||||||
raise aiohttp.web.HTTPConflict(text="Capture is not supported for this link")
|
for node in self._nodes:
|
||||||
|
if node["node"].compute.id == "local" and node["node"].status == "started":
|
||||||
|
return node
|
||||||
|
|
||||||
|
for node in self._nodes:
|
||||||
|
if node["node"].node_type and node["node"].status == "started":
|
||||||
|
return node
|
||||||
|
|
||||||
|
raise aiohttp.web.HTTPConflict(text="Can not capture because no running device on this link")
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def read_pcap_from_source(self):
|
def read_pcap_from_source(self):
|
||||||
@ -169,3 +180,11 @@ class UDPLink(Link):
|
|||||||
if self._capture_node:
|
if self._capture_node:
|
||||||
compute = self._capture_node["node"].compute
|
compute = self._capture_node["node"].compute
|
||||||
return compute.stream_file(self._project, "tmp/captures/" + self._capture_file_name)
|
return compute.stream_file(self._project, "tmp/captures/" + self._capture_file_name)
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def node_updated(self, node):
|
||||||
|
"""
|
||||||
|
Called when a node member of the link is updated
|
||||||
|
"""
|
||||||
|
if self._capture_node and node == self._capture_node["node"] and node.status != "started":
|
||||||
|
yield from self.stop_capture()
|
||||||
|
@ -97,6 +97,7 @@ def test_add_node(async_run, project, compute):
|
|||||||
|
|
||||||
assert link.create.called
|
assert link.create.called
|
||||||
link._project.controller.notification.emit.assert_called_with("link.created", link.__json__())
|
link._project.controller.notification.emit.assert_called_with("link.created", link.__json__())
|
||||||
|
assert link in node2.link
|
||||||
|
|
||||||
|
|
||||||
def test_add_node_cloud(async_run, project, compute):
|
def test_add_node_cloud(async_run, project, compute):
|
||||||
@ -302,3 +303,23 @@ def test_stop_capture(link, async_run, tmpdir, project, controller):
|
|||||||
async_run(link.stop_capture())
|
async_run(link.stop_capture())
|
||||||
assert link._capturing is False
|
assert link._capturing is False
|
||||||
controller._notification.emit.assert_called_with("link.updated", link.__json__())
|
controller._notification.emit.assert_called_with("link.updated", link.__json__())
|
||||||
|
|
||||||
|
|
||||||
|
def test_delete(async_run, project, compute):
|
||||||
|
node1 = Node(project, compute, "node1", node_type="qemu")
|
||||||
|
node1._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
|
|
||||||
|
link = Link(project)
|
||||||
|
link.create = AsyncioMagicMock()
|
||||||
|
link._project.controller.notification.emit = MagicMock()
|
||||||
|
project.dump = AsyncioMagicMock()
|
||||||
|
async_run(link.add_node(node1, 0, 4))
|
||||||
|
|
||||||
|
node2 = Node(project, compute, "node2", node_type="qemu")
|
||||||
|
node2._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
|
async_run(link.add_node(node2, 0, 4))
|
||||||
|
|
||||||
|
assert link in node2.link
|
||||||
|
|
||||||
|
async_run(link.delete())
|
||||||
|
assert link not in node2.link
|
||||||
|
@ -474,3 +474,14 @@ def test_get_port(node):
|
|||||||
assert port.adapter_number == 1
|
assert port.adapter_number == 1
|
||||||
with pytest.raises(aiohttp.web.HTTPNotFound):
|
with pytest.raises(aiohttp.web.HTTPNotFound):
|
||||||
port = node.get_port(42, 0)
|
port = node.get_port(42, 0)
|
||||||
|
|
||||||
|
|
||||||
|
def test_parse_node_response(node, async_run):
|
||||||
|
"""
|
||||||
|
When a node is updated we notify the links connected to it
|
||||||
|
"""
|
||||||
|
link = MagicMock()
|
||||||
|
link.node_updated = AsyncioMagicMock()
|
||||||
|
node.add_link(link)
|
||||||
|
async_run(node.parse_node_response({"status": "started"}))
|
||||||
|
assert link.node_updated.called
|
||||||
|
@ -77,7 +77,7 @@ def test_dispatch(async_run, controller, project):
|
|||||||
with notif.queue(project) as queue:
|
with notif.queue(project) as queue:
|
||||||
assert len(notif._listeners[project.id]) == 1
|
assert len(notif._listeners[project.id]) == 1
|
||||||
async_run(queue.get(0.1)) # ping
|
async_run(queue.get(0.1)) # ping
|
||||||
notif.dispatch("test", {}, compute_id=1)
|
async_run(notif.dispatch("test", {}, compute_id=1))
|
||||||
msg = async_run(queue.get(5))
|
msg = async_run(queue.get(5))
|
||||||
assert msg == ('test', {}, {})
|
assert msg == ('test', {}, {})
|
||||||
|
|
||||||
@ -87,7 +87,7 @@ def test_dispatch_ping(async_run, controller, project):
|
|||||||
with notif.queue(project) as queue:
|
with notif.queue(project) as queue:
|
||||||
assert len(notif._listeners[project.id]) == 1
|
assert len(notif._listeners[project.id]) == 1
|
||||||
async_run(queue.get(0.1)) # ping
|
async_run(queue.get(0.1)) # ping
|
||||||
notif.dispatch("ping", {}, compute_id=12)
|
async_run(notif.dispatch("ping", {}, compute_id=12))
|
||||||
msg = async_run(queue.get(5))
|
msg = async_run(queue.get(5))
|
||||||
assert msg == ('ping', {'compute_id': 12}, {})
|
assert msg == ('ping', {'compute_id': 12}, {})
|
||||||
|
|
||||||
@ -102,13 +102,13 @@ def test_dispatch_node_updated(async_run, controller, node, project):
|
|||||||
with notif.queue(project) as queue:
|
with notif.queue(project) as queue:
|
||||||
assert len(notif._listeners[project.id]) == 1
|
assert len(notif._listeners[project.id]) == 1
|
||||||
async_run(queue.get(0.1)) # ping
|
async_run(queue.get(0.1)) # ping
|
||||||
notif.dispatch("node.updated", {
|
async_run(notif.dispatch("node.updated", {
|
||||||
"node_id": node.id,
|
"node_id": node.id,
|
||||||
"project_id": project.id,
|
"project_id": project.id,
|
||||||
"name": "hello",
|
"name": "hello",
|
||||||
"startup_config": "ip 192"
|
"startup_config": "ip 192"
|
||||||
},
|
},
|
||||||
compute_id=1)
|
compute_id=1))
|
||||||
assert node.name == "hello"
|
assert node.name == "hello"
|
||||||
action, event, _ = async_run(queue.get(5))
|
action, event, _ = async_run(queue.get(5))
|
||||||
assert action == "node.updated"
|
assert action == "node.updated"
|
||||||
|
@ -187,6 +187,37 @@ def test_choose_capture_side(async_run, project):
|
|||||||
compute2 = MagicMock()
|
compute2 = MagicMock()
|
||||||
compute2.id = "local"
|
compute2.id = "local"
|
||||||
|
|
||||||
|
# Capture should run on the local node
|
||||||
|
node_vpcs = Node(project, compute1, "node1", node_type="vpcs")
|
||||||
|
node_vpcs._status = "started"
|
||||||
|
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
|
node_iou = Node(project, compute2, "node2", node_type="iou")
|
||||||
|
node_iou._status = "started"
|
||||||
|
node_iou._ports = [EthernetPort("E0", 0, 3, 1)]
|
||||||
|
|
||||||
|
link = UDPLink(project)
|
||||||
|
link.create = AsyncioMagicMock()
|
||||||
|
async_run(link.add_node(node_vpcs, 0, 4))
|
||||||
|
async_run(link.add_node(node_iou, 3, 1))
|
||||||
|
|
||||||
|
assert link._choose_capture_side()["node"] == node_iou
|
||||||
|
|
||||||
|
# Capture should choose always running node
|
||||||
|
node_iou = Node(project, compute1, "node5", node_type="iou")
|
||||||
|
node_iou._status = "started"
|
||||||
|
node_iou._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
|
node_switch = Node(project, compute1, "node6", node_type="ethernet_switch")
|
||||||
|
node_switch._status = "started"
|
||||||
|
node_switch._ports = [EthernetPort("E0", 0, 3, 1)]
|
||||||
|
|
||||||
|
link = UDPLink(project)
|
||||||
|
link.create = AsyncioMagicMock()
|
||||||
|
async_run(link.add_node(node_iou, 0, 4))
|
||||||
|
async_run(link.add_node(node_switch, 3, 1))
|
||||||
|
|
||||||
|
assert link._choose_capture_side()["node"] == node_switch
|
||||||
|
|
||||||
|
# Capture should raise error if node are not started
|
||||||
node_vpcs = Node(project, compute1, "node1", node_type="vpcs")
|
node_vpcs = Node(project, compute1, "node1", node_type="vpcs")
|
||||||
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
node_iou = Node(project, compute2, "node2", node_type="iou")
|
node_iou = Node(project, compute2, "node2", node_type="iou")
|
||||||
@ -197,36 +228,18 @@ def test_choose_capture_side(async_run, project):
|
|||||||
async_run(link.add_node(node_vpcs, 0, 4))
|
async_run(link.add_node(node_vpcs, 0, 4))
|
||||||
async_run(link.add_node(node_iou, 3, 1))
|
async_run(link.add_node(node_iou, 3, 1))
|
||||||
|
|
||||||
assert link._choose_capture_side()["node"] == node_iou
|
with pytest.raises(aiohttp.web.HTTPConflict):
|
||||||
|
link._choose_capture_side()
|
||||||
node_vpcs = Node(project, compute1, "node3", node_type="vpcs")
|
# If you start a node you can capture on it
|
||||||
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
node_vpcs._status = "started"
|
||||||
node_vpcs2 = Node(project, compute1, "node4", node_type="vpcs")
|
assert link._choose_capture_side()["node"] == node_vpcs
|
||||||
node_vpcs2._ports = [EthernetPort("E0", 0, 3, 1)]
|
|
||||||
|
|
||||||
link = UDPLink(project)
|
|
||||||
link.create = AsyncioMagicMock()
|
|
||||||
async_run(link.add_node(node_vpcs, 0, 4))
|
|
||||||
async_run(link.add_node(node_vpcs2, 3, 1))
|
|
||||||
|
|
||||||
# Capture should run on the local node
|
|
||||||
node_iou = Node(project, compute1, "node5", node_type="iou")
|
|
||||||
node_iou._ports = [EthernetPort("E0", 0, 0, 4)]
|
|
||||||
node_iou2 = Node(project, compute2, "node6", node_type="iou")
|
|
||||||
node_iou2._ports = [EthernetPort("E0", 0, 3, 1)]
|
|
||||||
|
|
||||||
link = UDPLink(project)
|
|
||||||
link.create = AsyncioMagicMock()
|
|
||||||
async_run(link.add_node(node_iou, 0, 4))
|
|
||||||
async_run(link.add_node(node_iou2, 3, 1))
|
|
||||||
|
|
||||||
assert link._choose_capture_side()["node"] == node_iou2
|
|
||||||
|
|
||||||
|
|
||||||
def test_capture(async_run, project):
|
def test_capture(async_run, project):
|
||||||
compute1 = MagicMock()
|
compute1 = MagicMock()
|
||||||
|
|
||||||
node_vpcs = Node(project, compute1, "V1", node_type="vpcs")
|
node_vpcs = Node(project, compute1, "V1", node_type="vpcs")
|
||||||
|
node_vpcs._status = "started"
|
||||||
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
node_iou = Node(project, compute1, "I1", node_type="iou")
|
node_iou = Node(project, compute1, "I1", node_type="iou")
|
||||||
node_iou._ports = [EthernetPort("E0", 0, 3, 1)]
|
node_iou._ports = [EthernetPort("E0", 0, 3, 1)]
|
||||||
@ -254,6 +267,7 @@ def test_read_pcap_from_source(project, async_run):
|
|||||||
compute1 = MagicMock()
|
compute1 = MagicMock()
|
||||||
|
|
||||||
node_vpcs = Node(project, compute1, "V1", node_type="vpcs")
|
node_vpcs = Node(project, compute1, "V1", node_type="vpcs")
|
||||||
|
node_vpcs._status = "started"
|
||||||
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
node_vpcs._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
node_iou = Node(project, compute1, "I1", node_type="iou")
|
node_iou = Node(project, compute1, "I1", node_type="iou")
|
||||||
node_iou._ports = [EthernetPort("E0", 0, 3, 1)]
|
node_iou._ports = [EthernetPort("E0", 0, 3, 1)]
|
||||||
@ -268,3 +282,23 @@ def test_read_pcap_from_source(project, async_run):
|
|||||||
|
|
||||||
async_run(link.read_pcap_from_source())
|
async_run(link.read_pcap_from_source())
|
||||||
link._capture_node["node"].compute.stream_file.assert_called_with(project, "tmp/captures/" + link._capture_file_name)
|
link._capture_node["node"].compute.stream_file.assert_called_with(project, "tmp/captures/" + link._capture_file_name)
|
||||||
|
|
||||||
|
|
||||||
|
def test_node_updated(project, async_run):
|
||||||
|
"""
|
||||||
|
If a node stop when capturing we stop the capture
|
||||||
|
"""
|
||||||
|
compute1 = MagicMock()
|
||||||
|
node_vpcs = Node(project, compute1, "V1", node_type="vpcs")
|
||||||
|
node_vpcs._status = "started"
|
||||||
|
|
||||||
|
link = UDPLink(project)
|
||||||
|
link._capture_node = {"node": node_vpcs}
|
||||||
|
link.stop_capture = AsyncioMagicMock()
|
||||||
|
|
||||||
|
async_run(link.node_updated(node_vpcs))
|
||||||
|
assert not link.stop_capture.called
|
||||||
|
|
||||||
|
node_vpcs._status = "stopped"
|
||||||
|
async_run(link.node_updated(node_vpcs))
|
||||||
|
assert link.stop_capture.called
|
||||||
|
Loading…
Reference in New Issue
Block a user