#!/usr/bin/env python # # Copyright (C) 2016 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 . import sys import copy import asyncio from .vmware_gns3_vm import VMwareGNS3VM from .virtualbox_gns3_vm import VirtualBoxGNS3VM import logging log = logging.getLogger(__name__) class GNS3VM: """ Proxy between the controller and the GNS3 VM engine """ def __init__(self, controller, settings={}): self._controller = controller # Keep instance of the loaded engines self._engines = {} self._settings = { "vmname": None, "auto_stop": True, "headless": False, "enable": False, "engine": "vmware" } self.settings = settings def engine_list(self): """ :returns: Return list of engines supported by GNS3 for the GNS3VM """ virtualbox_informations = { "engine_id": "virtualbox", "name": "VirtualBox", "description": "VirtualBox doesn't support nested virtualization, this means running Qemu based VM could be very slow." } vmware_informations = { "engine_id": "vmware", "description": "VMware is the recommended choice for best performances." } if sys.platform.startswith("darwin"): vmware_informations["name"] = "VMware Fusion" else: vmware_informations["name"] = "VMware Workstation / Player" return [ vmware_informations, virtualbox_informations ] def _current_engine(self): return self._get_engine(self._settings["engine"]) @property def ip_address(self): """ Returns the GNS3 VM IP address. :returns: VM IP address """ return self._current_engine().ip_address @property def running(self): """ Returns if the GNS3 VM is running. :returns: Boolean """ return self._current_engine().running @property def user(self): """ Returns the GNS3 VM user. :returns: VM user """ return self._current_engine().user @property def password(self): """ Returns the GNS3 VM password. :returns: VM password """ return self._current_engine().password @property def port(self): """ Returns the GNS3 VM port. :returns: VM port """ return self._current_engine().port @property def protocol(self): """ Returns the GNS3 VM protocol. :returns: VM protocol """ return self._current_engine().protocol @property def enable(self): """ The GNSVM is activated """ return self._settings.get("enable", False) @property def auto_stop(self): """ The GNSVM should auto stop """ return self._settings["auto_stop"] @property def settings(self): return self._settings @settings.setter def settings(self, val): self._settings.update(val) @asyncio.coroutine def update_settings(self, settings): """ Update settings and will restart the VM if require """ new_settings = copy.copy(self._settings) new_settings.update(settings) if self.settings != new_settings: yield from self._stop() self._settings = settings self._controller.save() yield from self.auto_start_vm() def _get_engine(self, engine): """ Load an engine """ if engine in self._engines: return self._engines[engine] if engine == "vmware": self._engines["vmware"] = VMwareGNS3VM() return self._engines["vmware"] elif engine == "virtualbox": self._engines["virtualbox"] = VirtualBoxGNS3VM() return self._engines["virtualbox"] raise NotImplementedError("The engine {} for the GNS3 VM is not supported".format(engine)) def __json__(self): return self._settings @asyncio.coroutine def list(self, engine): """ List VMS for an engine """ engine = self._get_engine(engine) vms = [] for vm in (yield from engine.list()): vms.append({"vmname": vm["vmname"]}) return vms @asyncio.coroutine def auto_start_vm(self): """ Auto start the GNS3 VM if require """ if self.enable: yield from self._start() @asyncio.coroutine def auto_stop_vm(self): if self.enable and self.auto_stop: yield from self._stop() @asyncio.coroutine def _start(self): """ Start the GNS3 VM """ engine = self._current_engine() if not engine.running: log.info("Start the GNS3 VM") engine.vmname = self._settings["vmname"] yield from engine.start() yield from self._controller.add_compute(compute_id="vm", name="GNS3 VM", protocol=self.protocol, host=self.ip_address, port=self.port, user=self.user, password=self.password, force=True) @asyncio.coroutine def _stop(self): """ Stop the GNS3 VM """ engine = self._current_engine() if "vm" in self._controller.computes: yield from self._controller.delete_compute("vm") if engine.running: log.info("Stop the GNS3 VM") yield from engine.stop()