#!/usr/bin/env python # # Copyright (C) 2020 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 os import uuid import json import pytest import socket from unittest.mock import MagicMock, patch from tests.utils import AsyncioMagicMock, asyncio_patch from gns3server.controller.compute import Compute from gns3server.controller.controller_error import ControllerError, ControllerNotFoundError from gns3server.version import __version__ def test_save(controller, controller_config_path): controller.save() assert os.path.exists(controller_config_path) with open(controller_config_path) as f: data = json.load(f) assert data["computes"] == [] assert data["version"] == __version__ assert data["iou_license"] == controller.iou_license assert data["gns3vm"] == controller.gns3vm.__json__() def test_load_controller_settings(controller, controller_config_path): controller.save() with open(controller_config_path) as f: data = json.load(f) data["computes"] = [ { "host": "localhost", "port": 8000, "protocol": "http", "user": "admin", "password": "root", "compute_id": "test1" } ] data["gns3vm"] = {"vmname": "Test VM"} with open(controller_config_path, "w+") as f: json.dump(data, f) assert len(controller._load_controller_settings()) == 1 assert controller.gns3vm.settings["vmname"] == "Test VM" def test_load_controller_settings_with_no_computes_section(controller, controller_config_path): controller.save() with open(controller_config_path) as f: data = json.load(f) del data['computes'] with open(controller_config_path, "w+") as f: json.dump(data, f) assert len(controller._load_controller_settings()) == 0 def test_import_computes_1_x(controller, controller_config_path): """ At first start the server should import the computes from the gns3_gui 1.X """ gns3_gui_conf = { "Servers": { "remote_servers": [ { "host": "127.0.0.1", "password": "", "port": 3081, "protocol": "http", "url": "http://127.0.0.1:3081", "user": "" } ] } } config_dir = os.path.dirname(controller_config_path) os.makedirs(config_dir, exist_ok=True) with open(os.path.join(config_dir, "gns3_gui.conf"), "w+") as f: json.dump(gns3_gui_conf, f) controller._load_controller_settings() for compute in controller.computes.values(): if compute.id != "local": assert len(compute.id) == 36 assert compute.host == "127.0.0.1" assert compute.port == 3081 assert compute.protocol == "http" assert compute.name == "http://127.0.0.1:3081" assert compute.user is None assert compute.password is None @pytest.mark.asyncio async def test_load_projects(controller, projects_dir): controller.save() os.makedirs(os.path.join(projects_dir, "project1")) with open(os.path.join(projects_dir, "project1", "project1.gns3"), "w+") as f: f.write("") with asyncio_patch("gns3server.controller.Controller.load_project") as mock_load_project: await controller.load_projects() mock_load_project.assert_called_with(os.path.join(projects_dir, "project1", "project1.gns3"), load=False) @pytest.mark.asyncio async def test_add_compute(controller): controller._notification = MagicMock() c = await controller.add_compute(compute_id="test1", connect=False) controller._notification.controller_emit.assert_called_with("compute.created", c.__json__()) assert len(controller.computes) == 1 await controller.add_compute(compute_id="test1", connect=False) controller._notification.controller_emit.assert_called_with("compute.updated", c.__json__()) assert len(controller.computes) == 1 await controller.add_compute(compute_id="test2", connect=False) assert len(controller.computes) == 2 @pytest.mark.asyncio async def test_addDuplicateCompute(controller): controller._notification = MagicMock() c = await controller.add_compute(compute_id="test1", name="Test", connect=False) assert len(controller.computes) == 1 with pytest.raises(ControllerError): await controller.add_compute(compute_id="test2", name="Test", connect=False) @pytest.mark.asyncio async def test_deleteCompute(controller, controller_config_path): c = await controller.add_compute(compute_id="test1", connect=False) assert len(controller.computes) == 1 controller._notification = MagicMock() c._connected = True await controller.delete_compute("test1") assert len(controller.computes) == 0 controller._notification.controller_emit.assert_called_with("compute.deleted", c.__json__()) with open(controller_config_path) as f: data = json.load(f) assert len(data["computes"]) == 0 assert c.connected is False @pytest.mark.asyncio async def test_deleteComputeProjectOpened(controller, controller_config_path): """ When you delete a compute the project using it are close """ c = await controller.add_compute(compute_id="test1", connect=False) c.post = AsyncioMagicMock() assert len(controller.computes) == 1 project1 = await controller.add_project(name="Test1") await project1.open() # We simulate that the project use this compute project1._project_created_on_compute.add(c) project2 = await controller.add_project(name="Test2") await project2.open() controller._notification = MagicMock() c._connected = True await controller.delete_compute("test1") assert len(controller.computes) == 0 controller._notification.controller_emit.assert_called_with("compute.deleted", c.__json__()) with open(controller_config_path) as f: data = json.load(f) assert len(data["computes"]) == 0 assert c.connected is False # Project 1 use this compute it should be close before deleting the compute assert project1.status == "closed" assert project2.status == "opened" @pytest.mark.asyncio async def test_addComputeConfigFile(controller, controller_config_path): await controller.add_compute(compute_id="test1", name="Test", connect=False) assert len(controller.computes) == 1 with open(controller_config_path) as f: data = json.load(f) assert data["computes"] == [ { 'compute_id': 'test1', 'name': 'Test', 'host': 'localhost', 'port': 3080, 'protocol': 'http', 'user': None, 'password': None } ] @pytest.mark.asyncio async def test_getCompute(controller): compute = await controller.add_compute(compute_id="test1", connect=False) assert controller.get_compute("test1") == compute with pytest.raises(ControllerNotFoundError): assert controller.get_compute("dsdssd") @pytest.mark.asyncio async def test_has_compute(controller): await controller.add_compute(compute_id="test1", connect=False) assert controller.has_compute("test1") assert not controller.has_compute("test2") @pytest.mark.asyncio async def test_add_project(controller): uuid1 = str(uuid.uuid4()) uuid2 = str(uuid.uuid4()) await controller.add_project(project_id=uuid1, name="Test") assert len(controller.projects) == 1 await controller.add_project(project_id=uuid1, name="Test") assert len(controller.projects) == 1 await controller.add_project(project_id=uuid2, name="Test 2") assert len(controller.projects) == 2 @pytest.mark.asyncio async def test_addDuplicateProject(controller): uuid1 = str(uuid.uuid4()) uuid2 = str(uuid.uuid4()) await controller.add_project(project_id=uuid1, name="Test") assert len(controller.projects) == 1 with pytest.raises(ControllerError): await controller.add_project(project_id=uuid2, name="Test") @pytest.mark.asyncio async def test_remove_project(controller): uuid1 = str(uuid.uuid4()) project1 = await controller.add_project(project_id=uuid1, name="Test") assert len(controller.projects) == 1 controller.remove_project(project1) assert len(controller.projects) == 0 @pytest.mark.asyncio async def test_addProject_with_compute(controller): uuid1 = str(uuid.uuid4()) compute = Compute("test1", controller=MagicMock()) compute.post = MagicMock() controller._computes = {"test1": compute} await controller.add_project(project_id=uuid1, name="Test") @pytest.mark.asyncio async def test_getProject(controller): uuid1 = str(uuid.uuid4()) project = await controller.add_project(project_id=uuid1, name="Test") assert controller.get_project(uuid1) == project with pytest.raises(ControllerNotFoundError): assert controller.get_project("dsdssd") @pytest.mark.asyncio async def test_start(controller): controller.gns3vm.settings = { "enable": False, "engine": "vmware", "vmname": "GNS3 VM" } with asyncio_patch("gns3server.controller.compute.Compute.connect") as mock: await controller.start() assert mock.called assert len(controller.computes) == 1 # Local compute is created assert controller.computes["local"].name == socket.gethostname() @pytest.mark.asyncio async def test_start_vm(controller): """ Start the controller with a GNS3 VM """ controller.gns3vm.settings = { "enable": True, "engine": "vmware", "vmname": "GNS3 VM" } with asyncio_patch("gns3server.controller.gns3vm.vmware_gns3_vm.VMwareGNS3VM.start") as mock: with asyncio_patch("gns3server.controller.gns3vm.GNS3VM._check_network"): with asyncio_patch("gns3server.controller.compute.Compute.connect"): await controller.start() assert mock.called assert "local" in controller.computes assert "vm" in controller.computes assert len(controller.computes) == 2 # Local compute and vm are created @pytest.mark.asyncio async def test_stop(controller): c = await controller.add_compute(compute_id="test1", connect=False) c._connected = True await controller.stop() assert c.connected is False @pytest.mark.asyncio async def test_stop_vm(controller): """ Stop GNS3 VM if configured """ controller.gns3vm.settings = { "enable": True, "engine": "vmware", "when_exit": "stop", "vmname": "GNS3 VM" } controller.gns3vm.current_engine().running = True with asyncio_patch("gns3server.controller.gns3vm.vmware_gns3_vm.VMwareGNS3VM.stop") as mock: await controller.stop() assert mock.called @pytest.mark.asyncio async def test_suspend_vm(controller): """ Suspend GNS3 VM if configured """ controller.gns3vm.settings = { "enable": True, "engine": "vmware", "when_exit": "suspend", "vmname": "GNS3 VM" } controller.gns3vm.current_engine().running = True with asyncio_patch("gns3server.controller.gns3vm.vmware_gns3_vm.VMwareGNS3VM.suspend") as mock: await controller.stop() assert mock.called @pytest.mark.asyncio async def test_keep_vm(controller): """ Keep GNS3 VM if configured """ controller.gns3vm.settings = { "enable": True, "engine": "vmware", "when_exit": "keep", "vmname": "GNS3 VM" } controller.gns3vm.current_engine().running = True with asyncio_patch("gns3server.controller.gns3vm.vmware_gns3_vm.VMwareGNS3VM.suspend") as mock: await controller.stop() assert not mock.called @pytest.mark.asyncio async def test_get_free_project_name(controller): await controller.add_project(project_id=str(uuid.uuid4()), name="Test") assert controller.get_free_project_name("Test") == "Test-1" await controller.add_project(project_id=str(uuid.uuid4()), name="Test-1") assert controller.get_free_project_name("Test") == "Test-2" assert controller.get_free_project_name("Hello") == "Hello" @pytest.mark.asyncio async def test_load_base_files(controller, config, tmpdir): config.set_section_config("Server", {"configs_path": str(tmpdir)}) with open(str(tmpdir / 'iou_l2_base_startup-config.txt'), 'w+') as f: f.write('test') controller.load_base_files() assert os.path.exists(str(tmpdir / 'iou_l3_base_startup-config.txt')) # Check is the file has not been overwrite with open(str(tmpdir / 'iou_l2_base_startup-config.txt')) as f: assert f.read() == 'test' def test_appliances(controller, tmpdir): my_appliance = { "name": "My Appliance", "status": "stable" } with open(str(tmpdir / "my_appliance.gns3a"), 'w+') as f: json.dump(my_appliance, f) # A broken appliance my_appliance = { "name": "Broken" } with open(str(tmpdir / "my_appliance2.gns3a"), 'w+') as f: json.dump(my_appliance, f) with patch("gns3server.config.Config.get_section_config", return_value={"appliances_path": str(tmpdir)}): controller.appliance_manager.load_appliances() assert len(controller.appliance_manager.appliances) > 0 for appliance in controller.appliance_manager.appliances.values(): assert appliance.__json__()["status"] != "broken" assert "Alpine Linux" in [c.__json__()["name"] for c in controller.appliance_manager.appliances.values()] assert "My Appliance" in [c.__json__()["name"] for c in controller.appliance_manager.appliances.values()] for c in controller.appliance_manager.appliances.values(): j = c.__json__() if j["name"] == "Alpine Linux": assert j["builtin"] elif j["name"] == "My Appliance": assert not j["builtin"] def test_load_templates(controller): controller._settings = {} controller.template_manager.load_templates() assert "Cloud" in [template.name for template in controller.template_manager.templates.values()] assert "VPCS" in [template.name for template in controller.template_manager.templates.values()] for template in controller.template_manager.templates.values(): if template.name == "VPCS": assert template._settings["properties"] == {"base_script_file": "vpcs_base_config.txt"} # UUID should not change when you run again the function for template in controller.template_manager.templates.values(): if template.name == "Test": qemu_uuid = template.id elif template.name == "Cloud": cloud_uuid = template.id controller.template_manager.load_templates() for template in controller.template_manager.templates.values(): if template.name == "Test": assert qemu_uuid == template.id elif template.name == "Cloud": assert cloud_uuid == template.id @pytest.mark.asyncio async def test_autoidlepc(controller): controller._computes["local"] = AsyncioMagicMock() node_mock = AsyncioMagicMock() with asyncio_patch("gns3server.controller.Project.add_node", return_value=node_mock): await controller.autoidlepc("local", "c7200", "test.bin", 512) assert node_mock.dynamips_auto_idlepc.called assert len(controller.projects) == 0