mirror of
https://github.com/GNS3/gns3-server.git
synced 2025-01-18 15:33:49 +02:00
OVA file support
This commit is contained in:
parent
034ac392b7
commit
3d7d8682ee
@ -357,7 +357,7 @@ class QEMUHandler:
|
|||||||
response.json(vms)
|
response.json(vms)
|
||||||
|
|
||||||
@Route.post(
|
@Route.post(
|
||||||
r"/qemu/vms/{path}",
|
r"/qemu/vms/{path:.+}",
|
||||||
status_codes={
|
status_codes={
|
||||||
204: "Image uploaded",
|
204: "Image uploaded",
|
||||||
},
|
},
|
||||||
|
@ -31,6 +31,7 @@ from uuid import UUID, uuid4
|
|||||||
from gns3server.utils.interfaces import is_interface_up
|
from gns3server.utils.interfaces import is_interface_up
|
||||||
from ..config import Config
|
from ..config import Config
|
||||||
from ..utils.asyncio import wait_run_in_executor
|
from ..utils.asyncio import wait_run_in_executor
|
||||||
|
from ..utils import force_unix_path
|
||||||
from .project_manager import ProjectManager
|
from .project_manager import ProjectManager
|
||||||
|
|
||||||
from .nios.nio_udp import NIOUDP
|
from .nios.nio_udp import NIOUDP
|
||||||
@ -408,10 +409,10 @@ class BaseManager:
|
|||||||
if not os.path.exists(path):
|
if not os.path.exists(path):
|
||||||
old_path = os.path.normpath(os.path.join(img_directory, '..', *s))
|
old_path = os.path.normpath(os.path.join(img_directory, '..', *s))
|
||||||
if os.path.exists(old_path):
|
if os.path.exists(old_path):
|
||||||
return old_path
|
return force_unix_path(old_path)
|
||||||
|
|
||||||
return path
|
return force_unix_path(path)
|
||||||
return path
|
return force_unix_path(path)
|
||||||
|
|
||||||
def get_relative_image_path(self, path):
|
def get_relative_image_path(self, path):
|
||||||
"""
|
"""
|
||||||
@ -427,8 +428,8 @@ class BaseManager:
|
|||||||
return ""
|
return ""
|
||||||
img_directory = self.get_images_directory()
|
img_directory = self.get_images_directory()
|
||||||
path = self.get_abs_image_path(path)
|
path = self.get_abs_image_path(path)
|
||||||
if os.path.dirname(path) == img_directory:
|
if os.path.commonprefix([img_directory, path]) == img_directory:
|
||||||
return os.path.basename(path)
|
return os.path.relpath(path, img_directory)
|
||||||
return path
|
return path
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
@ -460,11 +461,13 @@ class BaseManager:
|
|||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def write_image(self, filename, stream):
|
def write_image(self, filename, stream):
|
||||||
directory = self.get_images_directory()
|
directory = self.get_images_directory()
|
||||||
path = os.path.join(directory, os.path.basename(filename))
|
path = os.path.abspath(os.path.join(directory, *os.path.split(filename)))
|
||||||
|
if os.path.commonprefix([directory, path]) != directory:
|
||||||
|
raise aiohttp.web.HTTPForbidden(text="Could not write image: {}, {} is forbiden".format(filename, path))
|
||||||
log.info("Writting image file %s", path)
|
log.info("Writting image file %s", path)
|
||||||
try:
|
try:
|
||||||
remove_checksum(path)
|
remove_checksum(path)
|
||||||
os.makedirs(directory, exist_ok=True)
|
os.makedirs(os.path.dirname(path), exist_ok=True)
|
||||||
with open(path, 'wb+') as f:
|
with open(path, 'wb+') as f:
|
||||||
while True:
|
while True:
|
||||||
packet = yield from stream.read(512)
|
packet = yield from stream.read(512)
|
||||||
@ -474,4 +477,4 @@ class BaseManager:
|
|||||||
os.chmod(path, stat.S_IWRITE | stat.S_IREAD | stat.S_IEXEC)
|
os.chmod(path, stat.S_IWRITE | stat.S_IREAD | stat.S_IEXEC)
|
||||||
md5sum(path)
|
md5sum(path)
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
raise aiohttp.web.HTTPConflict(text="Could not write image: {} to {}".format(filename, e))
|
raise aiohttp.web.HTTPConflict(text="Could not write image: {} because {}".format(filename, e))
|
||||||
|
@ -0,0 +1,28 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# Copyright (C) 2015 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 posixpath
|
||||||
|
|
||||||
|
|
||||||
|
def force_unix_path(path):
|
||||||
|
"""
|
||||||
|
:param path: Path to convert
|
||||||
|
"""
|
||||||
|
|
||||||
|
path = path.replace("\\", "/")
|
||||||
|
return posixpath.normpath(path)
|
@ -232,6 +232,25 @@ def test_upload_vm(server, tmpdir):
|
|||||||
assert checksum == "033bd94b1168d7e4f0d644c3c95e35bf"
|
assert checksum == "033bd94b1168d7e4f0d644c3c95e35bf"
|
||||||
|
|
||||||
|
|
||||||
|
def test_upload_vm_ova(server, tmpdir):
|
||||||
|
with patch("gns3server.modules.Qemu.get_images_directory", return_value=str(tmpdir),):
|
||||||
|
response = server.post("/qemu/vms/test2.ova/test2.vmdk", body="TEST", raw=True)
|
||||||
|
assert response.status == 204
|
||||||
|
|
||||||
|
with open(str(tmpdir / "test2.ova" / "test2.vmdk")) as f:
|
||||||
|
assert f.read() == "TEST"
|
||||||
|
|
||||||
|
with open(str(tmpdir / "test2.ova" / "test2.vmdk.md5sum")) as f:
|
||||||
|
checksum = f.read()
|
||||||
|
assert checksum == "033bd94b1168d7e4f0d644c3c95e35bf"
|
||||||
|
|
||||||
|
|
||||||
|
def test_upload_vm_forbiden_location(server, tmpdir):
|
||||||
|
with patch("gns3server.modules.Qemu.get_images_directory", return_value=str(tmpdir),):
|
||||||
|
response = server.post("/qemu/vms/../../test2", body="TEST", raw=True)
|
||||||
|
assert response.status == 403
|
||||||
|
|
||||||
|
|
||||||
def test_upload_vm_permission_denied(server, tmpdir):
|
def test_upload_vm_permission_denied(server, tmpdir):
|
||||||
with open(str(tmpdir / "test2"), "w+") as f:
|
with open(str(tmpdir / "test2"), "w+") as f:
|
||||||
f.write("")
|
f.write("")
|
||||||
|
@ -396,6 +396,13 @@ def test_hda_disk_image(vm, tmpdir):
|
|||||||
assert vm.hda_disk_image == str(tmpdir / "QEMU" / "test")
|
assert vm.hda_disk_image == str(tmpdir / "QEMU" / "test")
|
||||||
|
|
||||||
|
|
||||||
|
def test_hda_disk_image_ova(vm, tmpdir):
|
||||||
|
|
||||||
|
with patch("gns3server.config.Config.get_section_config", return_value={"images_path": str(tmpdir)}):
|
||||||
|
vm.hda_disk_image = "test.ovf/test.vmdk"
|
||||||
|
assert vm.hda_disk_image == str(tmpdir / "QEMU" / "test.ovf" / "test.vmdk")
|
||||||
|
|
||||||
|
|
||||||
def test_hdb_disk_image(vm, tmpdir):
|
def test_hdb_disk_image(vm, tmpdir):
|
||||||
|
|
||||||
with patch("gns3server.config.Config.get_section_config", return_value={"images_path": str(tmpdir)}):
|
with patch("gns3server.config.Config.get_section_config", return_value={"images_path": str(tmpdir)}):
|
||||||
|
@ -123,6 +123,16 @@ def test_get_relative_image_path(qemu, tmpdir):
|
|||||||
assert qemu.get_relative_image_path("../test1.bin") == path1
|
assert qemu.get_relative_image_path("../test1.bin") == path1
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_relative_image_path_ova(qemu, tmpdir):
|
||||||
|
os.makedirs(str(tmpdir / "QEMU" / "test.ovf"))
|
||||||
|
path = str(tmpdir / "QEMU" / "test.ovf" / "test.bin")
|
||||||
|
open(path, 'w+').close()
|
||||||
|
|
||||||
|
with patch("gns3server.config.Config.get_section_config", return_value={"images_path": str(tmpdir)}):
|
||||||
|
assert qemu.get_relative_image_path(path) == os.path.join("test.ovf", "test.bin")
|
||||||
|
assert qemu.get_relative_image_path(os.path.join("test.ovf", "test.bin")) == os.path.join("test.ovf", "test.bin")
|
||||||
|
|
||||||
|
|
||||||
def test_list_images(loop, qemu, tmpdir):
|
def test_list_images(loop, qemu, tmpdir):
|
||||||
|
|
||||||
fake_images = ["a.bin", "b.bin", ".blu.bin", "a.bin.md5sum"]
|
fake_images = ["a.bin", "b.bin", ".blu.bin", "a.bin.md5sum"]
|
||||||
|
25
tests/test_utils.py
Normal file
25
tests/test_utils.py
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# Copyright (C) 2015 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/>.
|
||||||
|
|
||||||
|
|
||||||
|
from gns3server.utils import force_unix_path
|
||||||
|
|
||||||
|
|
||||||
|
def test_force_unix_path():
|
||||||
|
assert force_unix_path("a/b") == "a/b"
|
||||||
|
assert force_unix_path("a\\b") == "a/b"
|
||||||
|
assert force_unix_path("a\\b\\..\\c") == "a/c"
|
Loading…
Reference in New Issue
Block a user