mirror of
https://github.com/GNS3/gns3-server.git
synced 2025-01-02 19:16:42 +00:00
Support uploading iourc
This commit is contained in:
parent
964ea0f577
commit
66cdf39ea2
@ -61,7 +61,8 @@ class IOUHandler:
|
|||||||
ram=request.json.get("ram"),
|
ram=request.json.get("ram"),
|
||||||
nvram=request.json.get("nvram"),
|
nvram=request.json.get("nvram"),
|
||||||
l1_keepalives=request.json.get("l1_keepalives"),
|
l1_keepalives=request.json.get("l1_keepalives"),
|
||||||
initial_config=request.json.get("initial_config_content")
|
initial_config=request.json.get("initial_config_content"),
|
||||||
|
iourc_content=request.json.get("iourc_content")
|
||||||
)
|
)
|
||||||
vm.path = request.json.get("path", vm.path)
|
vm.path = request.json.get("path", vm.path)
|
||||||
response.set_status(201)
|
response.set_status(201)
|
||||||
|
@ -20,6 +20,7 @@ import logging
|
|||||||
import aiohttp
|
import aiohttp
|
||||||
import shutil
|
import shutil
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import tempfile
|
||||||
|
|
||||||
from ..utils.asyncio import wait_run_in_executor
|
from ..utils.asyncio import wait_run_in_executor
|
||||||
|
|
||||||
@ -45,6 +46,7 @@ class BaseVM:
|
|||||||
self._project = project
|
self._project = project
|
||||||
self._manager = manager
|
self._manager = manager
|
||||||
self._console = console
|
self._console = console
|
||||||
|
self._temporary_directory = None
|
||||||
|
|
||||||
if self._console is not None:
|
if self._console is not None:
|
||||||
self._console = self._manager.port_manager.reserve_tcp_port(self._console)
|
self._console = self._manager.port_manager.reserve_tcp_port(self._console)
|
||||||
@ -61,6 +63,9 @@ class BaseVM:
|
|||||||
def __del__(self):
|
def __del__(self):
|
||||||
|
|
||||||
self.close()
|
self.close()
|
||||||
|
if self._temporary_directory is not None:
|
||||||
|
if os.path.exists(self._temporary_directory):
|
||||||
|
shutil.rmtree(self._temporary_directory)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def project(self):
|
def project(self):
|
||||||
@ -124,6 +129,12 @@ class BaseVM:
|
|||||||
|
|
||||||
return self._project.vm_working_directory(self)
|
return self._project.vm_working_directory(self)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def temporary_directory(self):
|
||||||
|
if self._temporary_directory is None:
|
||||||
|
self._temporary_directory = tempfile.mkdtemp()
|
||||||
|
return self._temporary_directory
|
||||||
|
|
||||||
def create(self):
|
def create(self):
|
||||||
"""
|
"""
|
||||||
Creates the VM.
|
Creates the VM.
|
||||||
|
@ -67,6 +67,7 @@ class IOUVM(BaseVM):
|
|||||||
:params nvram: Nvram KB
|
:params nvram: Nvram KB
|
||||||
:params l1_keepalives: Always up ethernet interface:
|
:params l1_keepalives: Always up ethernet interface:
|
||||||
:params initial_config: Content of the initial configuration file
|
:params initial_config: Content of the initial configuration file
|
||||||
|
:params iourc_content: Content of the iourc file if no licence is installed on server
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, name, vm_id, project, manager,
|
def __init__(self, name, vm_id, project, manager,
|
||||||
@ -76,7 +77,8 @@ class IOUVM(BaseVM):
|
|||||||
ethernet_adapters=None,
|
ethernet_adapters=None,
|
||||||
serial_adapters=None,
|
serial_adapters=None,
|
||||||
l1_keepalives=None,
|
l1_keepalives=None,
|
||||||
initial_config=None):
|
initial_config=None,
|
||||||
|
iourc_content=None):
|
||||||
|
|
||||||
super().__init__(name, vm_id, project, manager, console=console)
|
super().__init__(name, vm_id, project, manager, console=console)
|
||||||
|
|
||||||
@ -99,6 +101,7 @@ class IOUVM(BaseVM):
|
|||||||
self._ram = 256 if ram is None else ram # Megabytes
|
self._ram = 256 if ram is None else ram # Megabytes
|
||||||
self._l1_keepalives = False if l1_keepalives is None else l1_keepalives # used to overcome the always-up Ethernet interfaces (not supported by all IOSes).
|
self._l1_keepalives = False if l1_keepalives is None else l1_keepalives # used to overcome the always-up Ethernet interfaces (not supported by all IOSes).
|
||||||
|
|
||||||
|
self.iourc_content = iourc_content
|
||||||
if initial_config is not None:
|
if initial_config is not None:
|
||||||
self.initial_config = initial_config
|
self.initial_config = initial_config
|
||||||
|
|
||||||
@ -209,7 +212,8 @@ class IOUVM(BaseVM):
|
|||||||
"nvram": self._nvram,
|
"nvram": self._nvram,
|
||||||
"l1_keepalives": self._l1_keepalives,
|
"l1_keepalives": self._l1_keepalives,
|
||||||
"initial_config": self.relative_initial_config_file,
|
"initial_config": self.relative_initial_config_file,
|
||||||
"use_default_iou_values": self._use_default_iou_values}
|
"use_default_iou_values": self._use_default_iou_values,
|
||||||
|
"iourc_path": self.iourc_path}
|
||||||
|
|
||||||
# return the relative path if the IOU image is in the images_path directory
|
# return the relative path if the IOU image is in the images_path directory
|
||||||
server_config = self.manager.config.get_section_config("Server")
|
server_config = self.manager.config.get_section_config("Server")
|
||||||
@ -250,6 +254,10 @@ class IOUVM(BaseVM):
|
|||||||
path = os.path.join(self.working_dir, "iourc")
|
path = os.path.join(self.working_dir, "iourc")
|
||||||
if os.path.exists(path):
|
if os.path.exists(path):
|
||||||
return path
|
return path
|
||||||
|
# look for the iourc file in the temporary dir.
|
||||||
|
path = os.path.join(self.temporary_directory, "iourc")
|
||||||
|
if os.path.exists(path):
|
||||||
|
return path
|
||||||
return iourc_path
|
return iourc_path
|
||||||
|
|
||||||
@property
|
@property
|
||||||
@ -322,6 +330,17 @@ class IOUVM(BaseVM):
|
|||||||
def application_id(self):
|
def application_id(self):
|
||||||
return self._manager.get_application_id(self.id)
|
return self._manager.get_application_id(self.id)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def iourc_content(self):
|
||||||
|
with open(os.path.join(self.temporary_directory, "iourc")) as f:
|
||||||
|
return f.read()
|
||||||
|
|
||||||
|
@iourc_content.setter
|
||||||
|
def iourc_content(self, value):
|
||||||
|
if value is not None:
|
||||||
|
with open(os.path.join(self.temporary_directory, "iourc"), "w+") as f:
|
||||||
|
f.write(value)
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def _library_check(self):
|
def _library_check(self):
|
||||||
"""
|
"""
|
||||||
|
@ -73,6 +73,10 @@ IOU_CREATE_SCHEMA = {
|
|||||||
"initial_config_content": {
|
"initial_config_content": {
|
||||||
"description": "Initial configuration of the IOU",
|
"description": "Initial configuration of the IOU",
|
||||||
"type": ["string", "null"]
|
"type": ["string", "null"]
|
||||||
|
},
|
||||||
|
"iourc_content": {
|
||||||
|
"description": "Content of the iourc file, if a file exist on servers this variable is ignored. It's mostly for compatibility with < 1.3 releases",
|
||||||
|
"type": ["string", "null"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"additionalProperties": False,
|
"additionalProperties": False,
|
||||||
@ -192,6 +196,10 @@ IOU_OBJECT_SCHEMA = {
|
|||||||
"use_default_iou_values": {
|
"use_default_iou_values": {
|
||||||
"description": "Use default IOU values",
|
"description": "Use default IOU values",
|
||||||
"type": ["boolean", "null"]
|
"type": ["boolean", "null"]
|
||||||
|
},
|
||||||
|
"iourc_path": {
|
||||||
|
"description": "Path of the iourc file used by remote servers",
|
||||||
|
"type": ["string", "null"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"additionalProperties": False,
|
"additionalProperties": False,
|
||||||
|
@ -75,6 +75,7 @@ def test_iou_create_with_params(server, project, base_params):
|
|||||||
params["l1_keepalives"] = True
|
params["l1_keepalives"] = True
|
||||||
params["initial_config_content"] = "hostname test"
|
params["initial_config_content"] = "hostname test"
|
||||||
params["use_default_iou_values"] = True
|
params["use_default_iou_values"] = True
|
||||||
|
params["iourc_content"] = "test"
|
||||||
|
|
||||||
response = server.post("/projects/{project_id}/iou/vms".format(project_id=project.id), params, example=True)
|
response = server.post("/projects/{project_id}/iou/vms".format(project_id=project.id), params, example=True)
|
||||||
assert response.status == 201
|
assert response.status == 201
|
||||||
@ -92,6 +93,8 @@ def test_iou_create_with_params(server, project, base_params):
|
|||||||
with open(initial_config_file(project, response.json)) as f:
|
with open(initial_config_file(project, response.json)) as f:
|
||||||
assert f.read() == params["initial_config_content"]
|
assert f.read() == params["initial_config_content"]
|
||||||
|
|
||||||
|
assert "iourc" in response.json["iourc_path"]
|
||||||
|
|
||||||
|
|
||||||
def test_iou_get(server, project, vm):
|
def test_iou_get(server, project, vm):
|
||||||
response = server.get("/projects/{project_id}/iou/vms/{vm_id}".format(project_id=vm["project_id"], vm_id=vm["vm_id"]), example=True)
|
response = server.get("/projects/{project_id}/iou/vms/{vm_id}".format(project_id=vm["project_id"], vm_id=vm["vm_id"]), example=True)
|
||||||
|
@ -367,3 +367,11 @@ def test_invalid_iou_file(loop, vm, iourc_file):
|
|||||||
with pytest.raises(IOUError):
|
with pytest.raises(IOUError):
|
||||||
os.remove(iourc_file)
|
os.remove(iourc_file)
|
||||||
loop.run_until_complete(asyncio.async(vm._check_iou_licence()))
|
loop.run_until_complete(asyncio.async(vm._check_iou_licence()))
|
||||||
|
|
||||||
|
|
||||||
|
def test_iourc_content(vm):
|
||||||
|
|
||||||
|
vm.iourc_content = "test"
|
||||||
|
|
||||||
|
with open(os.path.join(vm.temporary_directory, "iourc")) as f:
|
||||||
|
assert f.read() == "test"
|
||||||
|
45
tests/modules/test_base_vm.py
Normal file
45
tests/modules/test_base_vm.py
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# 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 pytest
|
||||||
|
import aiohttp
|
||||||
|
import asyncio
|
||||||
|
import os
|
||||||
|
from tests.utils import asyncio_patch
|
||||||
|
|
||||||
|
|
||||||
|
from unittest.mock import patch, MagicMock
|
||||||
|
from gns3server.modules.vpcs.vpcs_vm import VPCSVM
|
||||||
|
from gns3server.modules.vpcs.vpcs_error import VPCSError
|
||||||
|
from gns3server.modules.vpcs import VPCS
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(scope="module")
|
||||||
|
def manager(port_manager):
|
||||||
|
m = VPCS.instance()
|
||||||
|
m.port_manager = port_manager
|
||||||
|
return m
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(scope="function")
|
||||||
|
def vm(project, manager):
|
||||||
|
return VPCSVM("test", "00010203-0405-0607-0809-0a0b0c0d0e0f", project, manager)
|
||||||
|
|
||||||
|
|
||||||
|
def test_temporary_directory(project, manager):
|
||||||
|
vm = VPCSVM("test", "00010203-0405-0607-0809-0a0b0c0d0e0f", project, manager)
|
||||||
|
assert isinstance(vm.temporary_directory, str)
|
Loading…
Reference in New Issue
Block a user