mirror of
https://github.com/GNS3/gns3-server
synced 2025-02-17 18:42:00 +00:00
When exporting project raise error if export is not possible
This commit is contained in:
parent
08c35f5558
commit
f68c1f0bde
@ -439,6 +439,12 @@ class Project:
|
|||||||
:returns: ZipStream object
|
:returns: ZipStream object
|
||||||
"""
|
"""
|
||||||
z = zipstream.ZipFile()
|
z = zipstream.ZipFile()
|
||||||
|
|
||||||
|
# First we process the .gns3 in order to be sure we don't have an error
|
||||||
|
for file in os.listdir(self._path):
|
||||||
|
if file.endswith(".gns3"):
|
||||||
|
self._export_project_file(os.path.join(self._path, file), z, include_images)
|
||||||
|
|
||||||
for root, dirs, files in os.walk(self._path, topdown=True):
|
for root, dirs, files in os.walk(self._path, topdown=True):
|
||||||
# Remove snapshots and capture
|
# Remove snapshots and capture
|
||||||
if os.path.split(root)[-1:][0] == "project-files":
|
if os.path.split(root)[-1:][0] == "project-files":
|
||||||
@ -457,9 +463,8 @@ class Project:
|
|||||||
log.warn(msg)
|
log.warn(msg)
|
||||||
self.emit("log.warning", {"message": msg})
|
self.emit("log.warning", {"message": msg})
|
||||||
continue
|
continue
|
||||||
# We rename the .gns3 project.gns3 to avoid the task to the client to guess the file name
|
|
||||||
if file.endswith(".gns3"):
|
if file.endswith(".gns3"):
|
||||||
self._export_project_file(path, z, include_images)
|
pass
|
||||||
else:
|
else:
|
||||||
z.write(path, os.path.relpath(path, self._path), compress_type=zipfile.ZIP_DEFLATED)
|
z.write(path, os.path.relpath(path, self._path), compress_type=zipfile.ZIP_DEFLATED)
|
||||||
return z
|
return z
|
||||||
@ -468,6 +473,8 @@ class Project:
|
|||||||
"""
|
"""
|
||||||
Take a project file (.gns3) and patch it for the export
|
Take a project file (.gns3) and patch it for the export
|
||||||
|
|
||||||
|
We rename the .gns3 project.gns3 to avoid the task to the client to guess the file name
|
||||||
|
|
||||||
:param path: Path of the .gns3
|
:param path: Path of the .gns3
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -478,12 +485,15 @@ class Project:
|
|||||||
topology = json.load(f)
|
topology = json.load(f)
|
||||||
if "topology" in topology and "nodes" in topology["topology"]:
|
if "topology" in topology and "nodes" in topology["topology"]:
|
||||||
for node in topology["topology"]["nodes"]:
|
for node in topology["topology"]["nodes"]:
|
||||||
|
if node["node_type"] in ["virtualbox", "vmware", "cloud"]:
|
||||||
|
raise aiohttp.web.HTTPConflict(text="Topology with a {} could not be exported".format(node["node_type"]))
|
||||||
|
|
||||||
if "properties" in node and node["node_type"] != "Docker":
|
if "properties" in node and node["node_type"] != "Docker":
|
||||||
for prop, value in node["properties"].items():
|
for prop, value in node["properties"].items():
|
||||||
if prop.endswith("image"):
|
if prop.endswith("image"):
|
||||||
node["properties"][prop] = os.path.basename(value)
|
node["properties"][prop] = os.path.basename(value)
|
||||||
if include_images is True:
|
if include_images is True:
|
||||||
images.append(value)
|
images.add(value)
|
||||||
|
|
||||||
for image in images:
|
for image in images:
|
||||||
self._export_images(image, z)
|
self._export_images(image, z)
|
||||||
|
@ -231,14 +231,21 @@ class ProjectHandler:
|
|||||||
controller = Controller.instance()
|
controller = Controller.instance()
|
||||||
project = controller.get_project(request.match_info["project_id"])
|
project = controller.get_project(request.match_info["project_id"])
|
||||||
|
|
||||||
response.content_type = 'application/gns3project'
|
|
||||||
response.headers['CONTENT-DISPOSITION'] = 'attachment; filename="{}.gns3project"'.format(project.name)
|
started = False
|
||||||
response.enable_chunked_encoding()
|
|
||||||
# Very important: do not send a content length otherwise QT closes the connection (curl can consume the feed)
|
|
||||||
response.content_length = None
|
|
||||||
response.start(request)
|
|
||||||
|
|
||||||
for data in project.export(include_images=bool(request.GET.get("include_images", "0"))):
|
for data in project.export(include_images=bool(request.GET.get("include_images", "0"))):
|
||||||
|
# We need to do that now because export could failed and raise an HTTP error
|
||||||
|
# that why response start need to be the later possible
|
||||||
|
if not started:
|
||||||
|
response.content_type = 'application/gns3project'
|
||||||
|
response.headers['CONTENT-DISPOSITION'] = 'attachment; filename="{}.gns3project"'.format(project.name)
|
||||||
|
response.enable_chunked_encoding()
|
||||||
|
# Very important: do not send a content length otherwise QT closes the connection (curl can consume the feed)
|
||||||
|
response.content_length = None
|
||||||
|
response.start(request)
|
||||||
|
started = True
|
||||||
|
|
||||||
response.write(data)
|
response.write(data)
|
||||||
yield from response.drain()
|
yield from response.drain()
|
||||||
|
|
||||||
|
@ -351,6 +351,30 @@ def test_export(tmpdir, project):
|
|||||||
assert 'vm-1/dynamips/test_log.txt' not in myzip.namelist()
|
assert 'vm-1/dynamips/test_log.txt' not in myzip.namelist()
|
||||||
|
|
||||||
|
|
||||||
|
def test_export_disallow_some_type(tmpdir, project):
|
||||||
|
"""
|
||||||
|
Fix absolute image path
|
||||||
|
"""
|
||||||
|
|
||||||
|
path = project.path
|
||||||
|
|
||||||
|
topology = {
|
||||||
|
"topology": {
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"node_type": "virtualbox"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
with open(os.path.join(path, "test.gns3"), 'w+') as f:
|
||||||
|
json.dump(topology, f)
|
||||||
|
|
||||||
|
with pytest.raises(aiohttp.web.HTTPConflict):
|
||||||
|
z = project.export()
|
||||||
|
|
||||||
|
|
||||||
def test_export_fix_path(tmpdir, project):
|
def test_export_fix_path(tmpdir, project):
|
||||||
"""
|
"""
|
||||||
Fix absolute image path
|
Fix absolute image path
|
||||||
|
Loading…
Reference in New Issue
Block a user