mirror of
https://github.com/GNS3/gns3-server
synced 2024-12-26 16:58:28 +00:00
So we have running code for a qemu module
Now the handlers. The telnet code is not yet async
This commit is contained in:
parent
25bcbfb073
commit
b03b9226ff
41
gns3server/modules/qemu/__init__.py
Normal file
41
gns3server/modules/qemu/__init__.py
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
# -*- 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/>.
|
||||||
|
|
||||||
|
"""
|
||||||
|
Qemu server module.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
from ..base_manager import BaseManager
|
||||||
|
from .qemu_error import QemuError
|
||||||
|
from .qemu_vm import QemuVM
|
||||||
|
|
||||||
|
|
||||||
|
class Qemu(BaseManager):
|
||||||
|
_VM_CLASS = QemuVM
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_legacy_vm_workdir_name(legacy_vm_id):
|
||||||
|
"""
|
||||||
|
Returns the name of the legacy working directory name for a VM.
|
||||||
|
|
||||||
|
:param legacy_vm_id: legacy VM identifier (integer)
|
||||||
|
:returns: working directory name
|
||||||
|
"""
|
||||||
|
|
||||||
|
return "pc-{}".format(legacy_vm_id)
|
27
gns3server/modules/qemu/qemu_error.py
Normal file
27
gns3server/modules/qemu/qemu_error.py
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright (C) 2013 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/>.
|
||||||
|
|
||||||
|
"""
|
||||||
|
Custom exceptions for Qemu module.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from ..vm_error import VMError
|
||||||
|
|
||||||
|
|
||||||
|
class QemuError(VMError):
|
||||||
|
|
||||||
|
pass
|
@ -29,6 +29,7 @@ import ntpath
|
|||||||
import telnetlib
|
import telnetlib
|
||||||
import time
|
import time
|
||||||
import re
|
import re
|
||||||
|
import asyncio
|
||||||
|
|
||||||
from gns3server.config import Config
|
from gns3server.config import Config
|
||||||
|
|
||||||
@ -66,10 +67,6 @@ class QemuVM(BaseVM):
|
|||||||
:param monitor_end_port_range: TCP monitor port range end
|
:param monitor_end_port_range: TCP monitor port range end
|
||||||
"""
|
"""
|
||||||
|
|
||||||
_instances = []
|
|
||||||
_allocated_console_ports = []
|
|
||||||
_allocated_monitor_ports = []
|
|
||||||
|
|
||||||
def __init__(self,
|
def __init__(self,
|
||||||
name,
|
name,
|
||||||
vm_id,
|
vm_id,
|
||||||
@ -102,7 +99,7 @@ class QemuVM(BaseVM):
|
|||||||
self._monitor_end_port_range = monitor_end_port_range
|
self._monitor_end_port_range = monitor_end_port_range
|
||||||
|
|
||||||
# QEMU settings
|
# QEMU settings
|
||||||
self._qemu_path = qemu_path
|
self.qemu_path = qemu_path
|
||||||
self._hda_disk_image = ""
|
self._hda_disk_image = ""
|
||||||
self._hdb_disk_image = ""
|
self._hdb_disk_image = ""
|
||||||
self._options = ""
|
self._options = ""
|
||||||
@ -127,136 +124,6 @@ class QemuVM(BaseVM):
|
|||||||
log.info("QEMU VM {name} [id={id}] has been created".format(name=self._name,
|
log.info("QEMU VM {name} [id={id}] has been created".format(name=self._name,
|
||||||
id=self._id))
|
id=self._id))
|
||||||
|
|
||||||
def defaults(self):
|
|
||||||
"""
|
|
||||||
Returns all the default attribute values for this QEMU VM.
|
|
||||||
|
|
||||||
:returns: default values (dictionary)
|
|
||||||
"""
|
|
||||||
|
|
||||||
qemu_defaults = {"name": self._name,
|
|
||||||
"qemu_path": self._qemu_path,
|
|
||||||
"ram": self._ram,
|
|
||||||
"hda_disk_image": self._hda_disk_image,
|
|
||||||
"hdb_disk_image": self._hdb_disk_image,
|
|
||||||
"options": self._options,
|
|
||||||
"adapters": self.adapters,
|
|
||||||
"adapter_type": self._adapter_type,
|
|
||||||
"console": self._console,
|
|
||||||
"monitor": self._monitor,
|
|
||||||
"initrd": self._initrd,
|
|
||||||
"kernel_image": self._kernel_image,
|
|
||||||
"kernel_command_line": self._kernel_command_line,
|
|
||||||
"legacy_networking": self._legacy_networking,
|
|
||||||
"cpu_throttling": self._cpu_throttling,
|
|
||||||
"process_priority": self._process_priority
|
|
||||||
}
|
|
||||||
|
|
||||||
return qemu_defaults
|
|
||||||
|
|
||||||
@property
|
|
||||||
def id(self):
|
|
||||||
"""
|
|
||||||
Returns the unique ID for this QEMU VM.
|
|
||||||
|
|
||||||
:returns: id (integer)
|
|
||||||
"""
|
|
||||||
|
|
||||||
return self._id
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def reset(cls):
|
|
||||||
"""
|
|
||||||
Resets allocated instance list.
|
|
||||||
"""
|
|
||||||
|
|
||||||
cls._instances.clear()
|
|
||||||
cls._allocated_console_ports.clear()
|
|
||||||
cls._allocated_monitor_ports.clear()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self):
|
|
||||||
"""
|
|
||||||
Returns the name of this QEMU VM.
|
|
||||||
|
|
||||||
:returns: name
|
|
||||||
"""
|
|
||||||
|
|
||||||
return self._name
|
|
||||||
|
|
||||||
@name.setter
|
|
||||||
def name(self, new_name):
|
|
||||||
"""
|
|
||||||
Sets the name of this QEMU VM.
|
|
||||||
|
|
||||||
:param new_name: name
|
|
||||||
"""
|
|
||||||
|
|
||||||
log.info("QEMU VM {name} [id={id}]: renamed to {new_name}".format(name=self._name,
|
|
||||||
id=self._id,
|
|
||||||
new_name=new_name))
|
|
||||||
|
|
||||||
self._name = new_name
|
|
||||||
|
|
||||||
@property
|
|
||||||
def working_dir(self):
|
|
||||||
"""
|
|
||||||
Returns current working directory
|
|
||||||
|
|
||||||
:returns: path to the working directory
|
|
||||||
"""
|
|
||||||
|
|
||||||
return self._working_dir
|
|
||||||
|
|
||||||
@working_dir.setter
|
|
||||||
def working_dir(self, working_dir):
|
|
||||||
"""
|
|
||||||
Sets the working directory this QEMU VM.
|
|
||||||
|
|
||||||
:param working_dir: path to the working directory
|
|
||||||
"""
|
|
||||||
|
|
||||||
try:
|
|
||||||
os.makedirs(working_dir)
|
|
||||||
except FileExistsError:
|
|
||||||
pass
|
|
||||||
except OSError as e:
|
|
||||||
raise QemuError("Could not create working directory {}: {}".format(working_dir, e))
|
|
||||||
|
|
||||||
self._working_dir = working_dir
|
|
||||||
log.info("QEMU VM {name} [id={id}]: working directory changed to {wd}".format(name=self._name,
|
|
||||||
id=self._id,
|
|
||||||
wd=self._working_dir))
|
|
||||||
|
|
||||||
@property
|
|
||||||
def console(self):
|
|
||||||
"""
|
|
||||||
Returns the TCP console port.
|
|
||||||
|
|
||||||
:returns: console port (integer)
|
|
||||||
"""
|
|
||||||
|
|
||||||
return self._console
|
|
||||||
|
|
||||||
@console.setter
|
|
||||||
def console(self, console):
|
|
||||||
"""
|
|
||||||
Sets the TCP console port.
|
|
||||||
|
|
||||||
:param console: console port (integer)
|
|
||||||
"""
|
|
||||||
|
|
||||||
if console in self._allocated_console_ports:
|
|
||||||
raise QemuError("Console port {} is already used by another QEMU VM".format(console))
|
|
||||||
|
|
||||||
self._allocated_console_ports.remove(self._console)
|
|
||||||
self._console = console
|
|
||||||
self._allocated_console_ports.append(self._console)
|
|
||||||
|
|
||||||
log.info("QEMU VM {name} [id={id}]: console port set to {port}".format(name=self._name,
|
|
||||||
id=self._id,
|
|
||||||
port=console))
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def monitor(self):
|
def monitor(self):
|
||||||
"""
|
"""
|
||||||
@ -275,15 +142,15 @@ class QemuVM(BaseVM):
|
|||||||
:param monitor: monitor port (integer)
|
:param monitor: monitor port (integer)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if monitor in self._allocated_monitor_ports:
|
if monitor == self._monitor:
|
||||||
raise QemuError("Monitor port {} is already used by another QEMU VM".format(monitor))
|
return
|
||||||
|
if self._monitor:
|
||||||
self._allocated_monitor_ports.remove(self._monitor)
|
self._manager.port_manager.release_monitor_port(self._monitor)
|
||||||
self._monitor = monitor
|
self._monitor = self._manager.port_manager.reserve_monitor_port(monitor)
|
||||||
self._allocated_monitor_ports.append(self._monitor)
|
log.info("{module}: '{name}' [{id}]: monitor port set to {port}".format(
|
||||||
|
module=self.manager.module_name,
|
||||||
log.info("QEMU VM {name} [id={id}]: monitor port set to {port}".format(name=self._name,
|
name=self.name,
|
||||||
id=self._id,
|
id=self.id,
|
||||||
port=monitor))
|
port=monitor))
|
||||||
|
|
||||||
def delete(self):
|
def delete(self):
|
||||||
@ -304,53 +171,6 @@ class QemuVM(BaseVM):
|
|||||||
log.info("QEMU VM {name} [id={id}] has been deleted".format(name=self._name,
|
log.info("QEMU VM {name} [id={id}] has been deleted".format(name=self._name,
|
||||||
id=self._id))
|
id=self._id))
|
||||||
|
|
||||||
def clean_delete(self):
|
|
||||||
"""
|
|
||||||
Deletes this QEMU VM & all files.
|
|
||||||
"""
|
|
||||||
|
|
||||||
self.stop()
|
|
||||||
if self._id in self._instances:
|
|
||||||
self._instances.remove(self._id)
|
|
||||||
|
|
||||||
if self._console:
|
|
||||||
self._allocated_console_ports.remove(self._console)
|
|
||||||
|
|
||||||
if self._monitor:
|
|
||||||
self._allocated_monitor_ports.remove(self._monitor)
|
|
||||||
|
|
||||||
try:
|
|
||||||
shutil.rmtree(self._working_dir)
|
|
||||||
except OSError as e:
|
|
||||||
log.error("could not delete QEMU VM {name} [id={id}]: {error}".format(name=self._name,
|
|
||||||
id=self._id,
|
|
||||||
error=e))
|
|
||||||
return
|
|
||||||
|
|
||||||
log.info("QEMU VM {name} [id={id}] has been deleted (including associated files)".format(name=self._name,
|
|
||||||
id=self._id))
|
|
||||||
|
|
||||||
@property
|
|
||||||
def cloud_path(self):
|
|
||||||
"""
|
|
||||||
Returns the cloud path where images can be downloaded from.
|
|
||||||
|
|
||||||
:returns: cloud path
|
|
||||||
"""
|
|
||||||
|
|
||||||
return self._cloud_path
|
|
||||||
|
|
||||||
@cloud_path.setter
|
|
||||||
def cloud_path(self, cloud_path):
|
|
||||||
"""
|
|
||||||
Sets the cloud path where images can be downloaded from.
|
|
||||||
|
|
||||||
:param cloud_path:
|
|
||||||
:return:
|
|
||||||
"""
|
|
||||||
|
|
||||||
self._cloud_path = cloud_path
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def qemu_path(self):
|
def qemu_path(self):
|
||||||
"""
|
"""
|
||||||
@ -369,10 +189,20 @@ class QemuVM(BaseVM):
|
|||||||
:param qemu_path: QEMU path
|
:param qemu_path: QEMU path
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
if qemu_path and os.pathsep not in qemu_path:
|
||||||
|
qemu_path = shutil.which(qemu_path)
|
||||||
|
|
||||||
|
if qemu_path is None:
|
||||||
|
raise QemuError("QEMU binary path is not set or not found in the path")
|
||||||
|
if not os.path.exists(qemu_path):
|
||||||
|
raise QemuError("QEMU binary '{}' is not accessible".format(qemu_path))
|
||||||
|
if not os.access(qemu_path, os.X_OK):
|
||||||
|
raise QemuError("QEMU binary '{}' is not executable".format(qemu_path))
|
||||||
|
|
||||||
|
self._qemu_path = qemu_path
|
||||||
log.info("QEMU VM {name} [id={id}] has set the QEMU path to {qemu_path}".format(name=self._name,
|
log.info("QEMU VM {name} [id={id}] has set the QEMU path to {qemu_path}".format(name=self._name,
|
||||||
id=self._id,
|
id=self._id,
|
||||||
qemu_path=qemu_path))
|
qemu_path=qemu_path))
|
||||||
self._qemu_path = qemu_path
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def hda_disk_image(self):
|
def hda_disk_image(self):
|
||||||
@ -658,6 +488,7 @@ class QemuVM(BaseVM):
|
|||||||
kernel_command_line=kernel_command_line))
|
kernel_command_line=kernel_command_line))
|
||||||
self._kernel_command_line = kernel_command_line
|
self._kernel_command_line = kernel_command_line
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def _set_process_priority(self):
|
def _set_process_priority(self):
|
||||||
"""
|
"""
|
||||||
Changes the process priority
|
Changes the process priority
|
||||||
@ -700,7 +531,8 @@ class QemuVM(BaseVM):
|
|||||||
else:
|
else:
|
||||||
priority = 0
|
priority = 0
|
||||||
try:
|
try:
|
||||||
subprocess.call(['renice', '-n', str(priority), '-p', str(self._process.pid)])
|
process = yield from asyncio.create_subprocess_exec('renice', '-n', str(priority), '-p', str(self._process.pid))
|
||||||
|
yield from process.wait()
|
||||||
except (OSError, subprocess.SubprocessError) as e:
|
except (OSError, subprocess.SubprocessError) as e:
|
||||||
log.error("could not change process priority for QEMU VM {}: {}".format(self._name, e))
|
log.error("could not change process priority for QEMU VM {}: {}".format(self._name, e))
|
||||||
|
|
||||||
@ -729,13 +561,14 @@ class QemuVM(BaseVM):
|
|||||||
cpulimit_exec = os.path.join(os.path.dirname(os.path.abspath(sys.executable)), "cpulimit", "cpulimit.exe")
|
cpulimit_exec = os.path.join(os.path.dirname(os.path.abspath(sys.executable)), "cpulimit", "cpulimit.exe")
|
||||||
else:
|
else:
|
||||||
cpulimit_exec = "cpulimit"
|
cpulimit_exec = "cpulimit"
|
||||||
subprocess.Popen([cpulimit_exec, "--lazy", "--pid={}".format(self._process.pid), "--limit={}".format(self._cpu_throttling)], cwd=self._working_dir)
|
subprocess.Popen([cpulimit_exec, "--lazy", "--pid={}".format(self._process.pid), "--limit={}".format(self._cpu_throttling)], cwd=self.working_dir)
|
||||||
log.info("CPU throttled to {}%".format(self._cpu_throttling))
|
log.info("CPU throttled to {}%".format(self._cpu_throttling))
|
||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
raise QemuError("cpulimit could not be found, please install it or deactivate CPU throttling")
|
raise QemuError("cpulimit could not be found, please install it or deactivate CPU throttling")
|
||||||
except (OSError, subprocess.SubprocessError) as e:
|
except (OSError, subprocess.SubprocessError) as e:
|
||||||
raise QemuError("Could not throttle CPU: {}".format(e))
|
raise QemuError("Could not throttle CPU: {}".format(e))
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def start(self):
|
def start(self):
|
||||||
"""
|
"""
|
||||||
Starts this QEMU VM.
|
Starts this QEMU VM.
|
||||||
@ -748,92 +581,28 @@ class QemuVM(BaseVM):
|
|||||||
return
|
return
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
self._command = yield from self._build_command()
|
||||||
if not os.path.isfile(self._qemu_path) or not os.path.exists(self._qemu_path):
|
|
||||||
found = False
|
|
||||||
paths = [os.getcwd()] + os.environ["PATH"].split(os.pathsep)
|
|
||||||
# look for the qemu binary in the current working directory and $PATH
|
|
||||||
for path in paths:
|
|
||||||
try:
|
|
||||||
if self._qemu_path in os.listdir(path) and os.access(os.path.join(path, self._qemu_path), os.X_OK):
|
|
||||||
self._qemu_path = os.path.join(path, self._qemu_path)
|
|
||||||
found = True
|
|
||||||
break
|
|
||||||
except OSError:
|
|
||||||
continue
|
|
||||||
|
|
||||||
if not found:
|
|
||||||
raise QemuError("QEMU binary '{}' is not accessible".format(self._qemu_path))
|
|
||||||
|
|
||||||
if self.cloud_path is not None:
|
|
||||||
# Download from Cloud Files
|
|
||||||
if self.hda_disk_image != "":
|
|
||||||
_, filename = ntpath.split(self.hda_disk_image)
|
|
||||||
src = '{}/{}'.format(self.cloud_path, filename)
|
|
||||||
dst = os.path.join(self.working_dir, filename)
|
|
||||||
if not os.path.isfile(dst):
|
|
||||||
cloud_settings = Config.instance().cloud_settings()
|
|
||||||
provider = get_provider(cloud_settings)
|
|
||||||
log.debug("Downloading file from {} to {}...".format(src, dst))
|
|
||||||
provider.download_file(src, dst)
|
|
||||||
log.debug("Download of {} complete.".format(src))
|
|
||||||
self.hda_disk_image = dst
|
|
||||||
if self.hdb_disk_image != "":
|
|
||||||
_, filename = ntpath.split(self.hdb_disk_image)
|
|
||||||
src = '{}/{}'.format(self.cloud_path, filename)
|
|
||||||
dst = os.path.join(self.working_dir, filename)
|
|
||||||
if not os.path.isfile(dst):
|
|
||||||
cloud_settings = Config.instance().cloud_settings()
|
|
||||||
provider = get_provider(cloud_settings)
|
|
||||||
log.debug("Downloading file from {} to {}...".format(src, dst))
|
|
||||||
provider.download_file(src, dst)
|
|
||||||
log.debug("Download of {} complete.".format(src))
|
|
||||||
self.hdb_disk_image = dst
|
|
||||||
|
|
||||||
if self.initrd != "":
|
|
||||||
_, filename = ntpath.split(self.initrd)
|
|
||||||
src = '{}/{}'.format(self.cloud_path, filename)
|
|
||||||
dst = os.path.join(self.working_dir, filename)
|
|
||||||
if not os.path.isfile(dst):
|
|
||||||
cloud_settings = Config.instance().cloud_settings()
|
|
||||||
provider = get_provider(cloud_settings)
|
|
||||||
log.debug("Downloading file from {} to {}...".format(src, dst))
|
|
||||||
provider.download_file(src, dst)
|
|
||||||
log.debug("Download of {} complete.".format(src))
|
|
||||||
self.initrd = dst
|
|
||||||
if self.kernel_image != "":
|
|
||||||
_, filename = ntpath.split(self.kernel_image)
|
|
||||||
src = '{}/{}'.format(self.cloud_path, filename)
|
|
||||||
dst = os.path.join(self.working_dir, filename)
|
|
||||||
if not os.path.isfile(dst):
|
|
||||||
cloud_settings = Config.instance().cloud_settings()
|
|
||||||
provider = get_provider(cloud_settings)
|
|
||||||
log.debug("Downloading file from {} to {}...".format(src, dst))
|
|
||||||
provider.download_file(src, dst)
|
|
||||||
log.debug("Download of {} complete.".format(src))
|
|
||||||
self.kernel_image = dst
|
|
||||||
|
|
||||||
self._command = self._build_command()
|
|
||||||
try:
|
try:
|
||||||
log.info("starting QEMU: {}".format(self._command))
|
log.info("starting QEMU: {}".format(self._command))
|
||||||
self._stdout_file = os.path.join(self._working_dir, "qemu.log")
|
self._stdout_file = os.path.join(self.working_dir, "qemu.log")
|
||||||
log.info("logging to {}".format(self._stdout_file))
|
log.info("logging to {}".format(self._stdout_file))
|
||||||
with open(self._stdout_file, "w") as fd:
|
with open(self._stdout_file, "w") as fd:
|
||||||
self._process = subprocess.Popen(self._command,
|
self._process = yield from asyncio.create_subprocess_exec(*self._command,
|
||||||
stdout=fd,
|
stdout=fd,
|
||||||
stderr=subprocess.STDOUT,
|
stderr=subprocess.STDOUT,
|
||||||
cwd=self._working_dir)
|
cwd=self.working_dir)
|
||||||
log.info("QEMU VM instance {} started PID={}".format(self._id, self._process.pid))
|
log.info("QEMU VM instance {} started PID={}".format(self._id, self._process.pid))
|
||||||
self._started = True
|
self._started = True
|
||||||
except (OSError, subprocess.SubprocessError) as e:
|
except (OSError, subprocess.SubprocessError) as e:
|
||||||
stdout = self.read_stdout()
|
stdout = self.read_stdout()
|
||||||
log.error("could not start QEMU {}: {}\n{}".format(self._qemu_path, e, stdout))
|
log.error("could not start QEMU {}: {}\n{}".format(self.qemu_path, e, stdout))
|
||||||
raise QemuError("could not start QEMU {}: {}\n{}".format(self._qemu_path, e, stdout))
|
raise QemuError("could not start QEMU {}: {}\n{}".format(self.qemu_path, e, stdout))
|
||||||
|
|
||||||
self._set_process_priority()
|
self._set_process_priority()
|
||||||
if self._cpu_throttling:
|
if self._cpu_throttling:
|
||||||
self._set_cpu_throttling()
|
self._set_cpu_throttling()
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def stop(self):
|
def stop(self):
|
||||||
"""
|
"""
|
||||||
Stops this QEMU VM.
|
Stops this QEMU VM.
|
||||||
@ -854,6 +623,7 @@ class QemuVM(BaseVM):
|
|||||||
self._started = False
|
self._started = False
|
||||||
self._stop_cpulimit()
|
self._stop_cpulimit()
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def _control_vm(self, command, expected=None, timeout=30):
|
def _control_vm(self, command, expected=None, timeout=30):
|
||||||
"""
|
"""
|
||||||
Executes a command with QEMU monitor when this VM is running.
|
Executes a command with QEMU monitor when this VM is running.
|
||||||
@ -889,6 +659,18 @@ class QemuVM(BaseVM):
|
|||||||
tn.close()
|
tn.close()
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def close(self):
|
||||||
|
|
||||||
|
yield from self.stop()
|
||||||
|
if self._console:
|
||||||
|
self._manager.port_manager.release_console_port(self._console)
|
||||||
|
self._console = None
|
||||||
|
if self._monitor:
|
||||||
|
self._manager.port_manager.release_console_port(self._monitor)
|
||||||
|
self._monitor = None
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def _get_vm_status(self):
|
def _get_vm_status(self):
|
||||||
"""
|
"""
|
||||||
Returns this VM suspend status (running|paused)
|
Returns this VM suspend status (running|paused)
|
||||||
@ -898,43 +680,47 @@ class QemuVM(BaseVM):
|
|||||||
|
|
||||||
result = None
|
result = None
|
||||||
|
|
||||||
match = self._control_vm("info status", [b"running", b"paused"])
|
match = yield from self._control_vm("info status", [b"running", b"paused"])
|
||||||
if match:
|
if match:
|
||||||
result = match.group(0).decode('ascii')
|
result = match.group(0).decode('ascii')
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def suspend(self):
|
def suspend(self):
|
||||||
"""
|
"""
|
||||||
Suspends this QEMU VM.
|
Suspends this QEMU VM.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
vm_status = self._get_vm_status()
|
vm_status = yield from self._get_vm_status()
|
||||||
if vm_status == "running":
|
if vm_status == "running":
|
||||||
self._control_vm("stop")
|
yield from self._control_vm("stop")
|
||||||
log.debug("QEMU VM has been suspended")
|
log.debug("QEMU VM has been suspended")
|
||||||
else:
|
else:
|
||||||
log.info("QEMU VM is not running to be suspended, current status is {}".format(vm_status))
|
log.info("QEMU VM is not running to be suspended, current status is {}".format(vm_status))
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def reload(self):
|
def reload(self):
|
||||||
"""
|
"""
|
||||||
Reloads this QEMU VM.
|
Reloads this QEMU VM.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
self._control_vm("system_reset")
|
yield from self._control_vm("system_reset")
|
||||||
log.debug("QEMU VM has been reset")
|
log.debug("QEMU VM has been reset")
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def resume(self):
|
def resume(self):
|
||||||
"""
|
"""
|
||||||
Resumes this QEMU VM.
|
Resumes this QEMU VM.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
vm_status = self._get_vm_status()
|
vm_status = yield from self._get_vm_status()
|
||||||
if vm_status == "paused":
|
if vm_status == "paused":
|
||||||
self._control_vm("cont")
|
yield from self._control_vm("cont")
|
||||||
log.debug("QEMU VM has been resumed")
|
log.debug("QEMU VM has been resumed")
|
||||||
else:
|
else:
|
||||||
log.info("QEMU VM is not paused to be resumed, current status is {}".format(vm_status))
|
log.info("QEMU VM is not paused to be resumed, current status is {}".format(vm_status))
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def port_add_nio_binding(self, adapter_id, nio):
|
def port_add_nio_binding(self, adapter_id, nio):
|
||||||
"""
|
"""
|
||||||
Adds a port NIO binding.
|
Adds a port NIO binding.
|
||||||
@ -953,15 +739,15 @@ class QemuVM(BaseVM):
|
|||||||
# dynamically configure an UDP tunnel on the QEMU VM adapter
|
# dynamically configure an UDP tunnel on the QEMU VM adapter
|
||||||
if nio and isinstance(nio, NIO_UDP):
|
if nio and isinstance(nio, NIO_UDP):
|
||||||
if self._legacy_networking:
|
if self._legacy_networking:
|
||||||
self._control_vm("host_net_remove {} gns3-{}".format(adapter_id, adapter_id))
|
yield from self._control_vm("host_net_remove {} gns3-{}".format(adapter_id, adapter_id))
|
||||||
self._control_vm("host_net_add udp vlan={},name=gns3-{},sport={},dport={},daddr={}".format(adapter_id,
|
yield from self._control_vm("host_net_add udp vlan={},name=gns3-{},sport={},dport={},daddr={}".format(adapter_id,
|
||||||
adapter_id,
|
adapter_id,
|
||||||
nio.lport,
|
nio.lport,
|
||||||
nio.rport,
|
nio.rport,
|
||||||
nio.rhost))
|
nio.rhost))
|
||||||
else:
|
else:
|
||||||
self._control_vm("host_net_remove {} gns3-{}".format(adapter_id, adapter_id))
|
yield from self._control_vm("host_net_remove {} gns3-{}".format(adapter_id, adapter_id))
|
||||||
self._control_vm("host_net_add socket vlan={},name=gns3-{},udp={}:{},localaddr={}:{}".format(adapter_id,
|
yield from self._control_vm("host_net_add socket vlan={},name=gns3-{},udp={}:{},localaddr={}:{}".format(adapter_id,
|
||||||
adapter_id,
|
adapter_id,
|
||||||
nio.rhost,
|
nio.rhost,
|
||||||
nio.rport,
|
nio.rport,
|
||||||
@ -974,6 +760,7 @@ class QemuVM(BaseVM):
|
|||||||
nio=nio,
|
nio=nio,
|
||||||
adapter_id=adapter_id))
|
adapter_id=adapter_id))
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def port_remove_nio_binding(self, adapter_id):
|
def port_remove_nio_binding(self, adapter_id):
|
||||||
"""
|
"""
|
||||||
Removes a port NIO binding.
|
Removes a port NIO binding.
|
||||||
@ -991,8 +778,8 @@ class QemuVM(BaseVM):
|
|||||||
|
|
||||||
if self.is_running():
|
if self.is_running():
|
||||||
# dynamically disable the QEMU VM adapter
|
# dynamically disable the QEMU VM adapter
|
||||||
self._control_vm("host_net_remove {} gns3-{}".format(adapter_id, adapter_id))
|
yield from self._control_vm("host_net_remove {} gns3-{}".format(adapter_id, adapter_id))
|
||||||
self._control_vm("host_net_add user vlan={},name=gns3-{}".format(adapter_id, adapter_id))
|
yield from self._control_vm("host_net_add user vlan={},name=gns3-{}".format(adapter_id, adapter_id))
|
||||||
|
|
||||||
nio = adapter.get_nio(0)
|
nio = adapter.get_nio(0)
|
||||||
adapter.remove_nio(0)
|
adapter.remove_nio(0)
|
||||||
@ -1034,7 +821,7 @@ class QemuVM(BaseVM):
|
|||||||
:returns: True or False
|
:returns: True or False
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if self._process and self._process.poll() is None:
|
if self._process:
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@ -1061,11 +848,12 @@ class QemuVM(BaseVM):
|
|||||||
else:
|
else:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def _disk_options(self):
|
def _disk_options(self):
|
||||||
|
|
||||||
options = []
|
options = []
|
||||||
qemu_img_path = ""
|
qemu_img_path = ""
|
||||||
qemu_path_dir = os.path.dirname(self._qemu_path)
|
qemu_path_dir = os.path.dirname(self.qemu_path)
|
||||||
try:
|
try:
|
||||||
for f in os.listdir(qemu_path_dir):
|
for f in os.listdir(qemu_path_dir):
|
||||||
if f.startswith("qemu-img"):
|
if f.startswith("qemu-img"):
|
||||||
@ -1083,17 +871,19 @@ class QemuVM(BaseVM):
|
|||||||
raise QemuError("hda disk image '{}' linked to '{}' is not accessible".format(self._hda_disk_image, os.path.realpath(self._hda_disk_image)))
|
raise QemuError("hda disk image '{}' linked to '{}' is not accessible".format(self._hda_disk_image, os.path.realpath(self._hda_disk_image)))
|
||||||
else:
|
else:
|
||||||
raise QemuError("hda disk image '{}' is not accessible".format(self._hda_disk_image))
|
raise QemuError("hda disk image '{}' is not accessible".format(self._hda_disk_image))
|
||||||
hda_disk = os.path.join(self._working_dir, "hda_disk.qcow2")
|
hda_disk = os.path.join(self.working_dir, "hda_disk.qcow2")
|
||||||
if not os.path.exists(hda_disk):
|
if not os.path.exists(hda_disk):
|
||||||
retcode = subprocess.call([qemu_img_path, "create", "-o",
|
process = yield from asyncio.create_subprocess_exec(qemu_img_path, "create", "-o",
|
||||||
"backing_file={}".format(self._hda_disk_image),
|
"backing_file={}".format(self._hda_disk_image),
|
||||||
"-f", "qcow2", hda_disk])
|
"-f", "qcow2", hda_disk)
|
||||||
|
retcode = yield from process.wait()
|
||||||
log.info("{} returned with {}".format(qemu_img_path, retcode))
|
log.info("{} returned with {}".format(qemu_img_path, retcode))
|
||||||
else:
|
else:
|
||||||
# create a "FLASH" with 256MB if no disk image has been specified
|
# create a "FLASH" with 256MB if no disk image has been specified
|
||||||
hda_disk = os.path.join(self._working_dir, "flash.qcow2")
|
hda_disk = os.path.join(self.working_dir, "flash.qcow2")
|
||||||
if not os.path.exists(hda_disk):
|
if not os.path.exists(hda_disk):
|
||||||
retcode = subprocess.call([qemu_img_path, "create", "-f", "qcow2", hda_disk, "128M"])
|
process = yield from asyncio.create_subprocess_exec(qemu_img_path, "create", "-f", "qcow2", hda_disk, "128M")
|
||||||
|
retcode = yield from process.wait()
|
||||||
log.info("{} returned with {}".format(qemu_img_path, retcode))
|
log.info("{} returned with {}".format(qemu_img_path, retcode))
|
||||||
|
|
||||||
except (OSError, subprocess.SubprocessError) as e:
|
except (OSError, subprocess.SubprocessError) as e:
|
||||||
@ -1106,12 +896,13 @@ class QemuVM(BaseVM):
|
|||||||
raise QemuError("hdb disk image '{}' linked to '{}' is not accessible".format(self._hdb_disk_image, os.path.realpath(self._hdb_disk_image)))
|
raise QemuError("hdb disk image '{}' linked to '{}' is not accessible".format(self._hdb_disk_image, os.path.realpath(self._hdb_disk_image)))
|
||||||
else:
|
else:
|
||||||
raise QemuError("hdb disk image '{}' is not accessible".format(self._hdb_disk_image))
|
raise QemuError("hdb disk image '{}' is not accessible".format(self._hdb_disk_image))
|
||||||
hdb_disk = os.path.join(self._working_dir, "hdb_disk.qcow2")
|
hdb_disk = os.path.join(self.working_dir, "hdb_disk.qcow2")
|
||||||
if not os.path.exists(hdb_disk):
|
if not os.path.exists(hdb_disk):
|
||||||
try:
|
try:
|
||||||
retcode = subprocess.call([qemu_img_path, "create", "-o",
|
process = yield from asyncio.create_subprocess_exec(qemu_img_path, "create", "-o",
|
||||||
"backing_file={}".format(self._hdb_disk_image),
|
"backing_file={}".format(self._hdb_disk_image),
|
||||||
"-f", "qcow2", hdb_disk])
|
"-f", "qcow2", hdb_disk)
|
||||||
|
retcode = yield from process.wait()
|
||||||
log.info("{} returned with {}".format(qemu_img_path, retcode))
|
log.info("{} returned with {}".format(qemu_img_path, retcode))
|
||||||
except (OSError, subprocess.SubprocessError) as e:
|
except (OSError, subprocess.SubprocessError) as e:
|
||||||
raise QemuError("Could not create disk image {}".format(e))
|
raise QemuError("Could not create disk image {}".format(e))
|
||||||
@ -1170,16 +961,18 @@ class QemuVM(BaseVM):
|
|||||||
|
|
||||||
return network_options
|
return network_options
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
def _build_command(self):
|
def _build_command(self):
|
||||||
"""
|
"""
|
||||||
Command to start the QEMU process.
|
Command to start the QEMU process.
|
||||||
(to be passed to subprocess.Popen())
|
(to be passed to subprocess.Popen())
|
||||||
"""
|
"""
|
||||||
|
|
||||||
command = [self._qemu_path]
|
command = [self.qemu_path]
|
||||||
command.extend(["-name", self._name])
|
command.extend(["-name", self._name])
|
||||||
command.extend(["-m", str(self._ram)])
|
command.extend(["-m", str(self._ram)])
|
||||||
command.extend(self._disk_options())
|
disk_options = yield from self._disk_options()
|
||||||
|
command.extend(disk_options)
|
||||||
command.extend(self._linux_boot_options())
|
command.extend(self._linux_boot_options())
|
||||||
command.extend(self._serial_options())
|
command.extend(self._serial_options())
|
||||||
command.extend(self._monitor_options())
|
command.extend(self._monitor_options())
|
||||||
|
188
tests/modules/qemu/test_qemu_vm.py
Normal file
188
tests/modules/qemu/test_qemu_vm.py
Normal file
@ -0,0 +1,188 @@
|
|||||||
|
# -*- 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
|
||||||
|
import stat
|
||||||
|
from tests.utils import asyncio_patch
|
||||||
|
|
||||||
|
|
||||||
|
from unittest.mock import patch, MagicMock
|
||||||
|
from gns3server.modules.qemu.qemu_vm import QemuVM
|
||||||
|
from gns3server.modules.qemu.qemu_error import QemuError
|
||||||
|
from gns3server.modules.qemu import Qemu
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(scope="module")
|
||||||
|
def manager(port_manager):
|
||||||
|
m = Qemu.instance()
|
||||||
|
m.port_manager = port_manager
|
||||||
|
return m
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def fake_qemu_img_binary():
|
||||||
|
|
||||||
|
bin_path = os.path.join(os.environ["PATH"], "qemu-img")
|
||||||
|
with open(bin_path, "w+") as f:
|
||||||
|
f.write("1")
|
||||||
|
os.chmod(bin_path, stat.S_IRUSR | stat.S_IWUSR | stat.S_IXUSR)
|
||||||
|
return bin_path
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def fake_qemu_binary():
|
||||||
|
|
||||||
|
bin_path = os.path.join(os.environ["PATH"], "qemu_x42")
|
||||||
|
with open(bin_path, "w+") as f:
|
||||||
|
f.write("1")
|
||||||
|
os.chmod(bin_path, stat.S_IRUSR | stat.S_IWUSR | stat.S_IXUSR)
|
||||||
|
return bin_path
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(scope="function")
|
||||||
|
def vm(project, manager, fake_qemu_binary, fake_qemu_img_binary):
|
||||||
|
return QemuVM("test", "00010203-0405-0607-0809-0a0b0c0d0e0f", project, manager, qemu_path=fake_qemu_binary)
|
||||||
|
|
||||||
|
|
||||||
|
def test_vm(project, manager, fake_qemu_binary):
|
||||||
|
vm = QemuVM("test", "00010203-0405-0607-0809-0a0b0c0d0e0f", project, manager, qemu_path=fake_qemu_binary)
|
||||||
|
assert vm.name == "test"
|
||||||
|
assert vm.id == "00010203-0405-0607-0809-0a0b0c0d0e0f"
|
||||||
|
|
||||||
|
|
||||||
|
def test_start(loop, vm):
|
||||||
|
with asyncio_patch("asyncio.create_subprocess_exec", return_value=MagicMock()):
|
||||||
|
loop.run_until_complete(asyncio.async(vm.start()))
|
||||||
|
assert vm.is_running()
|
||||||
|
|
||||||
|
|
||||||
|
def test_stop(loop, vm):
|
||||||
|
process = MagicMock()
|
||||||
|
|
||||||
|
# Wait process kill success
|
||||||
|
future = asyncio.Future()
|
||||||
|
future.set_result(True)
|
||||||
|
process.wait.return_value = future
|
||||||
|
|
||||||
|
with asyncio_patch("asyncio.create_subprocess_exec", return_value=process):
|
||||||
|
nio = Qemu.instance().create_nio(vm.qemu_path, {"type": "nio_udp", "lport": 4242, "rport": 4243, "rhost": "127.0.0.1"})
|
||||||
|
vm.port_add_nio_binding(0, nio)
|
||||||
|
loop.run_until_complete(asyncio.async(vm.start()))
|
||||||
|
assert vm.is_running()
|
||||||
|
loop.run_until_complete(asyncio.async(vm.stop()))
|
||||||
|
assert vm.is_running() is False
|
||||||
|
process.terminate.assert_called_with()
|
||||||
|
|
||||||
|
|
||||||
|
def test_reload(loop, vm):
|
||||||
|
|
||||||
|
with asyncio_patch("gns3server.modules.qemu.QemuVM._control_vm") as mock:
|
||||||
|
loop.run_until_complete(asyncio.async(vm.reload()))
|
||||||
|
assert mock.called_with("system_reset")
|
||||||
|
|
||||||
|
|
||||||
|
def test_add_nio_binding_udp(vm, loop):
|
||||||
|
nio = Qemu.instance().create_nio(vm.qemu_path, {"type": "nio_udp", "lport": 4242, "rport": 4243, "rhost": "127.0.0.1"})
|
||||||
|
loop.run_until_complete(asyncio.async(vm.port_add_nio_binding(0, nio)))
|
||||||
|
assert nio.lport == 4242
|
||||||
|
|
||||||
|
|
||||||
|
def test_add_nio_binding_tap(vm, loop):
|
||||||
|
with patch("gns3server.modules.base_manager.BaseManager._has_privileged_access", return_value=True):
|
||||||
|
nio = Qemu.instance().create_nio(vm.qemu_path, {"type": "nio_tap", "tap_device": "test"})
|
||||||
|
loop.run_until_complete(asyncio.async(vm.port_add_nio_binding(0, nio)))
|
||||||
|
assert nio.tap_device == "test"
|
||||||
|
|
||||||
|
|
||||||
|
def test_port_remove_nio_binding(vm, loop):
|
||||||
|
nio = Qemu.instance().create_nio(vm.qemu_path, {"type": "nio_udp", "lport": 4242, "rport": 4243, "rhost": "127.0.0.1"})
|
||||||
|
loop.run_until_complete(asyncio.async(vm.port_add_nio_binding(0, nio)))
|
||||||
|
loop.run_until_complete(asyncio.async(vm.port_remove_nio_binding(0)))
|
||||||
|
assert vm._ethernet_adapters[0].ports[0] is None
|
||||||
|
|
||||||
|
|
||||||
|
def test_close(vm, port_manager, loop):
|
||||||
|
with asyncio_patch("asyncio.create_subprocess_exec", return_value=MagicMock()):
|
||||||
|
loop.run_until_complete(asyncio.async(vm.start()))
|
||||||
|
|
||||||
|
console_port = vm.console
|
||||||
|
monitor_port = vm.monitor
|
||||||
|
|
||||||
|
loop.run_until_complete(asyncio.async(vm.close()))
|
||||||
|
|
||||||
|
# Raise an exception if the port is not free
|
||||||
|
port_manager.reserve_console_port(console_port)
|
||||||
|
# Raise an exception if the port is not free
|
||||||
|
port_manager.reserve_console_port(monitor_port)
|
||||||
|
|
||||||
|
assert vm.is_running() is False
|
||||||
|
|
||||||
|
|
||||||
|
def test_set_qemu_path(vm, tmpdir, fake_qemu_binary):
|
||||||
|
|
||||||
|
# Raise because none
|
||||||
|
with pytest.raises(QemuError):
|
||||||
|
vm.qemu_path = None
|
||||||
|
|
||||||
|
path = str(tmpdir / "bla")
|
||||||
|
|
||||||
|
# Raise because file doesn't exists
|
||||||
|
with pytest.raises(QemuError):
|
||||||
|
vm.qemu_path = path
|
||||||
|
|
||||||
|
with open(path, "w+") as f:
|
||||||
|
f.write("1")
|
||||||
|
|
||||||
|
# Raise because file is not executable
|
||||||
|
with pytest.raises(QemuError):
|
||||||
|
vm.qemu_path = path
|
||||||
|
|
||||||
|
os.chmod(path, stat.S_IRUSR | stat.S_IWUSR | stat.S_IXUSR)
|
||||||
|
|
||||||
|
vm.qemu_path = path
|
||||||
|
assert vm.qemu_path == path
|
||||||
|
|
||||||
|
|
||||||
|
def test_set_qemu_path_environ(vm, tmpdir, fake_qemu_binary):
|
||||||
|
|
||||||
|
# It should find the binary in the path
|
||||||
|
vm.qemu_path = "qemu_x42"
|
||||||
|
|
||||||
|
assert vm.qemu_path == fake_qemu_binary
|
||||||
|
|
||||||
|
|
||||||
|
def test_disk_options(vm, loop, fake_qemu_img_binary):
|
||||||
|
|
||||||
|
with asyncio_patch("asyncio.create_subprocess_exec", return_value=MagicMock()) as process:
|
||||||
|
loop.run_until_complete(asyncio.async(vm._disk_options()))
|
||||||
|
assert process.called
|
||||||
|
args, kwargs = process.call_args
|
||||||
|
assert args == (fake_qemu_img_binary, "create", "-f", "qcow2", os.path.join(vm.working_dir, "flash.qcow2"), "128M")
|
||||||
|
|
||||||
|
|
||||||
|
def test_set_process_priority(vm, loop, fake_qemu_img_binary):
|
||||||
|
|
||||||
|
with asyncio_patch("asyncio.create_subprocess_exec", return_value=MagicMock()) as process:
|
||||||
|
vm._process = MagicMock()
|
||||||
|
vm._process.pid = 42
|
||||||
|
loop.run_until_complete(asyncio.async(vm._set_process_priority()))
|
||||||
|
assert process.called
|
||||||
|
args, kwargs = process.call_args
|
||||||
|
assert args == ("renice", "-n", "5", "-p", "42")
|
Loading…
Reference in New Issue
Block a user