mirror of
https://github.com/GNS3/gns3-server.git
synced 2025-01-11 15:32:39 +00:00
04022677bd
Handle packet captures for VPCS & Qemu nodes. Fixes #548.
142 lines
5.3 KiB
Python
142 lines
5.3 KiB
Python
#!/usr/bin/env python
|
||
#
|
||
# Copyright (C) 2016 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/>.
|
||
|
||
import asyncio
|
||
import aiohttp
|
||
|
||
|
||
from .link import Link
|
||
|
||
|
||
class UDPLink(Link):
|
||
|
||
def __init__(self, project, link_id=None):
|
||
super().__init__(project, link_id=link_id)
|
||
self._capture_node = None
|
||
|
||
@asyncio.coroutine
|
||
def create(self):
|
||
"""
|
||
Create the link on the nodes
|
||
"""
|
||
|
||
node1 = self._nodes[0]["node"]
|
||
adapter_number1 = self._nodes[0]["adapter_number"]
|
||
port_number1 = self._nodes[0]["port_number"]
|
||
node2 = self._nodes[1]["node"]
|
||
adapter_number2 = self._nodes[1]["adapter_number"]
|
||
port_number2 = self._nodes[1]["port_number"]
|
||
|
||
# Reserve a UDP port on both side
|
||
response = yield from node1.compute.post("/projects/{}/ports/udp".format(self._project.id))
|
||
self._node1_port = response.json["udp_port"]
|
||
response = yield from node2.compute.post("/projects/{}/ports/udp".format(self._project.id))
|
||
self._node2_port = response.json["udp_port"]
|
||
|
||
# Create the tunnel on both side
|
||
data = {
|
||
"lport": self._node1_port,
|
||
"rhost": node2.compute.host,
|
||
"rport": self._node2_port,
|
||
"type": "nio_udp"
|
||
}
|
||
yield from node1.post("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number1, port_number=port_number1), data=data)
|
||
|
||
data = {
|
||
"lport": self._node2_port,
|
||
"rhost": node1.compute.host,
|
||
"rport": self._node1_port,
|
||
"type": "nio_udp"
|
||
}
|
||
yield from node2.post("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number2, port_number=port_number2), data=data)
|
||
|
||
@asyncio.coroutine
|
||
def delete(self):
|
||
"""
|
||
Delete the link and free the resources
|
||
"""
|
||
try:
|
||
node1 = self._nodes[0]["node"]
|
||
adapter_number1 = self._nodes[0]["adapter_number"]
|
||
port_number1 = self._nodes[0]["port_number"]
|
||
except IndexError:
|
||
return
|
||
|
||
yield from node1.delete("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number1, port_number=port_number1))
|
||
|
||
try:
|
||
node2 = self._nodes[1]["node"]
|
||
adapter_number2 = self._nodes[1]["adapter_number"]
|
||
port_number2 = self._nodes[1]["port_number"]
|
||
except IndexError:
|
||
return
|
||
yield from node2.delete("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number2, port_number=port_number2))
|
||
|
||
@asyncio.coroutine
|
||
def start_capture(self, data_link_type="DLT_EN10MB", capture_file_name=None):
|
||
"""
|
||
Start capture on a link
|
||
"""
|
||
if not capture_file_name:
|
||
capture_file_name = self.default_capture_file_name()
|
||
self._capture_node = self._choose_capture_side()
|
||
data = {
|
||
"capture_file_name": capture_file_name,
|
||
"data_link_type": data_link_type
|
||
}
|
||
yield from self._capture_node["node"].post("/adapters/{adapter_number}/ports/{port_number}/start_capture".format(adapter_number=self._capture_node["adapter_number"], port_number=self._capture_node["port_number"]), data=data)
|
||
yield from super().start_capture(data_link_type=data_link_type, capture_file_name=capture_file_name)
|
||
|
||
@asyncio.coroutine
|
||
def stop_capture(self):
|
||
"""
|
||
Stop capture on a link
|
||
"""
|
||
if self._capture_node:
|
||
yield from self._capture_node["node"].post("/adapters/{adapter_number}/ports/{port_number}/stop_capture".format(adapter_number=self._capture_node["adapter_number"], port_number=self._capture_node["port_number"]))
|
||
self._capture_node = None
|
||
yield from super().stop_capture()
|
||
|
||
def _choose_capture_side(self):
|
||
"""
|
||
Run capture on the best candidate.
|
||
|
||
The ideal candidate is a node who support capture on controller server
|
||
|
||
:returns: Node where the capture should run
|
||
"""
|
||
|
||
# use the local node first to save bandwidth
|
||
for node in self._nodes:
|
||
if node["node"].compute.id == "local" and node["node"].node_type not in [""]: # FIXME
|
||
return node
|
||
|
||
for node in self._nodes:
|
||
if node["node"].node_type not in [""]: # FIXME
|
||
return node
|
||
|
||
raise aiohttp.web.HTTPConflict(text="Capture is not supported for this link")
|
||
|
||
@asyncio.coroutine
|
||
def read_pcap_from_source(self):
|
||
"""
|
||
Return a FileStream of the Pcap from the compute node
|
||
"""
|
||
if self._capture_node:
|
||
compute = self._capture_node["node"].compute
|
||
return compute.steam_file(self._project, "tmp/captures/" + self._capture_file_name)
|