mirror of
https://github.com/GNS3/gns3-server
synced 2024-11-24 17:28:08 +00:00
commit
90cdf14c2a
@ -122,6 +122,7 @@ class Link:
|
|||||||
self._streaming_pcap = None
|
self._streaming_pcap = None
|
||||||
self._created = False
|
self._created = False
|
||||||
self._link_type = "ethernet"
|
self._link_type = "ethernet"
|
||||||
|
self._suspend = False
|
||||||
self._filters = {}
|
self._filters = {}
|
||||||
|
|
||||||
@property
|
@property
|
||||||
@ -131,6 +132,15 @@ class Link:
|
|||||||
"""
|
"""
|
||||||
return self._filters
|
return self._filters
|
||||||
|
|
||||||
|
def get_active_filters(self):
|
||||||
|
"""
|
||||||
|
Return the active filters.
|
||||||
|
Filters are overridden if the link is suspended.
|
||||||
|
"""
|
||||||
|
if self._suspend:
|
||||||
|
return {"frequency_drop": [-1]}
|
||||||
|
return self._filters
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def update_filters(self, filters):
|
def update_filters(self, filters):
|
||||||
"""
|
"""
|
||||||
@ -155,6 +165,12 @@ class Link:
|
|||||||
if self._created:
|
if self._created:
|
||||||
yield from self.update()
|
yield from self.update()
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def update_suspend(self, value):
|
||||||
|
if value != self._suspend:
|
||||||
|
self._suspend = value
|
||||||
|
yield from self.update()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def created(self):
|
def created(self):
|
||||||
"""
|
"""
|
||||||
@ -401,7 +417,8 @@ class Link:
|
|||||||
return {
|
return {
|
||||||
"nodes": res,
|
"nodes": res,
|
||||||
"link_id": self._id,
|
"link_id": self._id,
|
||||||
"filters": self._filters
|
"filters": self._filters,
|
||||||
|
"suspend": self._suspend
|
||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
"nodes": res,
|
"nodes": res,
|
||||||
@ -411,5 +428,6 @@ class Link:
|
|||||||
"capture_file_name": self._capture_file_name,
|
"capture_file_name": self._capture_file_name,
|
||||||
"capture_file_path": self.capture_file_path,
|
"capture_file_path": self.capture_file_path,
|
||||||
"link_type": self._link_type,
|
"link_type": self._link_type,
|
||||||
"filters": self._filters
|
"filters": self._filters,
|
||||||
|
"suspend": self._suspend
|
||||||
}
|
}
|
||||||
|
@ -66,9 +66,9 @@ class UDPLink(Link):
|
|||||||
node2_filters = {}
|
node2_filters = {}
|
||||||
filter_node = self._get_filter_node()
|
filter_node = self._get_filter_node()
|
||||||
if filter_node == node1:
|
if filter_node == node1:
|
||||||
node1_filters = self._filters
|
node1_filters = self.get_active_filters()
|
||||||
elif filter_node == node2:
|
elif filter_node == node2:
|
||||||
node2_filters = self._filters
|
node2_filters = self.get_active_filters()
|
||||||
|
|
||||||
# Create the tunnel on both side
|
# Create the tunnel on both side
|
||||||
self._link_data.append({
|
self._link_data.append({
|
||||||
@ -106,12 +106,12 @@ class UDPLink(Link):
|
|||||||
if node1 == filter_node:
|
if node1 == filter_node:
|
||||||
adapter_number1 = self._nodes[0]["adapter_number"]
|
adapter_number1 = self._nodes[0]["adapter_number"]
|
||||||
port_number1 = self._nodes[0]["port_number"]
|
port_number1 = self._nodes[0]["port_number"]
|
||||||
self._link_data[0]["filters"] = self._filters
|
self._link_data[0]["filters"] = self.get_active_filters()
|
||||||
yield from node1.put("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number1, port_number=port_number1), data=self._link_data[0], timeout=120)
|
yield from node1.put("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number1, port_number=port_number1), data=self._link_data[0], timeout=120)
|
||||||
elif node2 == filter_node:
|
elif node2 == filter_node:
|
||||||
adapter_number2 = self._nodes[1]["adapter_number"]
|
adapter_number2 = self._nodes[1]["adapter_number"]
|
||||||
port_number2 = self._nodes[1]["port_number"]
|
port_number2 = self._nodes[1]["port_number"]
|
||||||
self._link_data[1]["filters"] = self._filters
|
self._link_data[1]["filters"] = self.get_active_filters()
|
||||||
yield from node2.put("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number2, port_number=port_number2), data=self._link_data[1], timeout=221)
|
yield from node2.put("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number2, port_number=port_number2), data=self._link_data[1], timeout=221)
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
@ -199,7 +199,7 @@ class UDPLink(Link):
|
|||||||
if node["node"].node_type and node["node"].status == "started":
|
if node["node"].node_type and node["node"].status == "started":
|
||||||
return node
|
return node
|
||||||
|
|
||||||
raise aiohttp.web.HTTPConflict(text="Can not capture because no running device on this link")
|
raise aiohttp.web.HTTPConflict(text="Cannot capture because there is no running device on this link")
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def read_pcap_from_source(self):
|
def read_pcap_from_source(self):
|
||||||
|
@ -62,7 +62,10 @@ class LinkHandler:
|
|||||||
|
|
||||||
project = yield from Controller.instance().get_loaded_project(request.match_info["project_id"])
|
project = yield from Controller.instance().get_loaded_project(request.match_info["project_id"])
|
||||||
link = yield from project.add_link()
|
link = yield from project.add_link()
|
||||||
yield from link.update_filters(request.json.get("filters", {}))
|
if "filters" in request.json:
|
||||||
|
yield from link.update_filters(request.json["filters"])
|
||||||
|
if "suspend" in request.json:
|
||||||
|
yield from link.update_suspend(request.json["suspend"])
|
||||||
try:
|
try:
|
||||||
for node in request.json["nodes"]:
|
for node in request.json["nodes"]:
|
||||||
yield from link.add_node(project.get_node(node["node_id"]),
|
yield from link.add_node(project.get_node(node["node_id"]),
|
||||||
@ -110,7 +113,10 @@ class LinkHandler:
|
|||||||
|
|
||||||
project = yield from Controller.instance().get_loaded_project(request.match_info["project_id"])
|
project = yield from Controller.instance().get_loaded_project(request.match_info["project_id"])
|
||||||
link = project.get_link(request.match_info["link_id"])
|
link = project.get_link(request.match_info["link_id"])
|
||||||
yield from link.update_filters(request.json.get("filters", {}))
|
if "filters" in request.json:
|
||||||
|
yield from link.update_filters(request.json["filters"])
|
||||||
|
if "suspend" in request.json:
|
||||||
|
yield from link.update_suspend(request.json["suspend"])
|
||||||
if "nodes" in request.json:
|
if "nodes" in request.json:
|
||||||
yield from link.update_nodes(request.json["nodes"])
|
yield from link.update_nodes(request.json["nodes"])
|
||||||
response.set_status(201)
|
response.set_status(201)
|
||||||
|
@ -64,6 +64,10 @@ LINK_OBJECT_SCHEMA = {
|
|||||||
"additionalProperties": False
|
"additionalProperties": False
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"suspend": {
|
||||||
|
"type": "boolean",
|
||||||
|
"description": "Suspend the link"
|
||||||
|
},
|
||||||
"filters": FILTER_OBJECT_SCHEMA,
|
"filters": FILTER_OBJECT_SCHEMA,
|
||||||
"capturing": {
|
"capturing": {
|
||||||
"description": "Read only property. True if a capture running on the link",
|
"description": "Read only property. True if a capture running on the link",
|
||||||
|
@ -231,6 +231,7 @@ def test_json(async_run, project, compute, link):
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"filters": {},
|
"filters": {},
|
||||||
|
"suspend": False,
|
||||||
"link_type": "ethernet",
|
"link_type": "ethernet",
|
||||||
"capturing": False,
|
"capturing": False,
|
||||||
"capture_file_name": None,
|
"capture_file_name": None,
|
||||||
@ -264,7 +265,8 @@ def test_json(async_run, project, compute, link):
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"filters": {}
|
"filters": {},
|
||||||
|
"suspend": False
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -384,3 +384,68 @@ def test_update(async_run, project):
|
|||||||
"bpf": ["icmp[icmptype] == 8"]
|
"bpf": ["icmp[icmptype] == 8"]
|
||||||
}
|
}
|
||||||
}, timeout=120)
|
}, timeout=120)
|
||||||
|
|
||||||
|
|
||||||
|
def test_update_suspend(async_run, project):
|
||||||
|
compute1 = MagicMock()
|
||||||
|
compute2 = MagicMock()
|
||||||
|
|
||||||
|
node1 = Node(project, compute1, "node1", node_type="vpcs")
|
||||||
|
node1._ports = [EthernetPort("E0", 0, 0, 4)]
|
||||||
|
node2 = Node(project, compute2, "node2", node_type="vpcs")
|
||||||
|
node2._ports = [EthernetPort("E0", 0, 3, 1)]
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def subnet_callback(compute2):
|
||||||
|
"""
|
||||||
|
Fake subnet callback
|
||||||
|
"""
|
||||||
|
return ("192.168.1.1", "192.168.1.2")
|
||||||
|
|
||||||
|
compute1.get_ip_on_same_subnet.side_effect = subnet_callback
|
||||||
|
|
||||||
|
link = UDPLink(project)
|
||||||
|
async_run(link.add_node(node1, 0, 4))
|
||||||
|
async_run(link.update_filters({"latency": [10]}))
|
||||||
|
async_run(link.update_suspend(True))
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def compute1_callback(path, data={}, **kwargs):
|
||||||
|
"""
|
||||||
|
Fake server
|
||||||
|
"""
|
||||||
|
if "/ports/udp" in path:
|
||||||
|
response = MagicMock()
|
||||||
|
response.json = {"udp_port": 1024}
|
||||||
|
return response
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def compute2_callback(path, data={}, **kwargs):
|
||||||
|
"""
|
||||||
|
Fake server
|
||||||
|
"""
|
||||||
|
if "/ports/udp" in path:
|
||||||
|
response = MagicMock()
|
||||||
|
response.json = {"udp_port": 2048}
|
||||||
|
return response
|
||||||
|
|
||||||
|
compute1.post.side_effect = compute1_callback
|
||||||
|
compute1.host = "example.com"
|
||||||
|
compute2.post.side_effect = compute2_callback
|
||||||
|
compute2.host = "example.org"
|
||||||
|
async_run(link.add_node(node2, 3, 1))
|
||||||
|
|
||||||
|
compute1.post.assert_any_call("/projects/{}/vpcs/nodes/{}/adapters/0/ports/4/nio".format(project.id, node1.id), data={
|
||||||
|
"lport": 1024,
|
||||||
|
"rhost": "192.168.1.2",
|
||||||
|
"rport": 2048,
|
||||||
|
"type": "nio_udp",
|
||||||
|
"filters": {"frequency_drop": [-1]}
|
||||||
|
}, timeout=120)
|
||||||
|
compute2.post.assert_any_call("/projects/{}/vpcs/nodes/{}/adapters/3/ports/1/nio".format(project.id, node2.id), data={
|
||||||
|
"lport": 2048,
|
||||||
|
"rhost": "192.168.1.1",
|
||||||
|
"rport": 1024,
|
||||||
|
"type": "nio_udp",
|
||||||
|
"filters": {}
|
||||||
|
}, timeout=120)
|
||||||
|
@ -128,6 +128,64 @@ def test_create_link_failure(http_controller, tmpdir, project, compute, async_ru
|
|||||||
assert len(project.links) == 0
|
assert len(project.links) == 0
|
||||||
|
|
||||||
|
|
||||||
|
def test_update_link_suspend(http_controller, tmpdir, project, compute, async_run):
|
||||||
|
response = MagicMock()
|
||||||
|
response.json = {"console": 2048}
|
||||||
|
compute.post = AsyncioMagicMock(return_value=response)
|
||||||
|
|
||||||
|
node1 = async_run(project.add_node(compute, "node1", None, node_type="qemu"))
|
||||||
|
node1._ports = [EthernetPort("E0", 0, 0, 3)]
|
||||||
|
node2 = async_run(project.add_node(compute, "node2", None, node_type="qemu"))
|
||||||
|
node2._ports = [EthernetPort("E0", 0, 2, 4)]
|
||||||
|
|
||||||
|
with asyncio_patch("gns3server.controller.udp_link.UDPLink.create"):
|
||||||
|
response = http_controller.post("/projects/{}/links".format(project.id), {
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"node_id": node1.id,
|
||||||
|
"adapter_number": 0,
|
||||||
|
"port_number": 3,
|
||||||
|
"label": {
|
||||||
|
"text": "Text",
|
||||||
|
"x": 42,
|
||||||
|
"y": 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"node_id": node2.id,
|
||||||
|
"adapter_number": 2,
|
||||||
|
"port_number": 4
|
||||||
|
}
|
||||||
|
]
|
||||||
|
})
|
||||||
|
link_id = response.json["link_id"]
|
||||||
|
assert response.json["nodes"][0]["label"]["x"] == 42
|
||||||
|
response = http_controller.put("/projects/{}/links/{}".format(project.id, link_id), {
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"node_id": node1.id,
|
||||||
|
"adapter_number": 0,
|
||||||
|
"port_number": 3,
|
||||||
|
"label": {
|
||||||
|
"text": "Hello",
|
||||||
|
"x": 64,
|
||||||
|
"y": 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"node_id": node2.id,
|
||||||
|
"adapter_number": 2,
|
||||||
|
"port_number": 4
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"suspend": True
|
||||||
|
})
|
||||||
|
assert response.status == 201
|
||||||
|
assert response.json["nodes"][0]["label"]["x"] == 64
|
||||||
|
assert response.json["suspend"]
|
||||||
|
assert response.json["filters"] == {}
|
||||||
|
|
||||||
|
|
||||||
def test_update_link(http_controller, tmpdir, project, compute, async_run):
|
def test_update_link(http_controller, tmpdir, project, compute, async_run):
|
||||||
response = MagicMock()
|
response = MagicMock()
|
||||||
response.json = {"console": 2048}
|
response.json = {"console": 2048}
|
||||||
|
Loading…
Reference in New Issue
Block a user