2016-03-10 11:32:07 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- 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/>.
|
|
|
|
|
2016-04-26 15:34:49 +03:00
|
|
|
import os
|
2016-09-28 11:27:30 +03:00
|
|
|
import sys
|
2016-03-11 17:51:35 +02:00
|
|
|
import pytest
|
|
|
|
import aiohttp
|
2017-06-19 11:50:45 +03:00
|
|
|
import zipstream
|
2016-03-10 22:51:29 +02:00
|
|
|
from unittest.mock import MagicMock
|
2016-08-16 16:45:06 +03:00
|
|
|
from tests.utils import AsyncioMagicMock, asyncio_patch
|
2016-04-26 15:34:49 +03:00
|
|
|
from unittest.mock import patch
|
|
|
|
from uuid import uuid4
|
2016-03-10 11:32:07 +02:00
|
|
|
|
|
|
|
from gns3server.controller.project import Project
|
2017-04-12 15:35:49 +03:00
|
|
|
from gns3server.controller.appliance import Appliance
|
2016-09-15 15:51:40 +03:00
|
|
|
from gns3server.controller.ports.ethernet_port import EthernetPort
|
2016-04-26 15:34:49 +03:00
|
|
|
from gns3server.config import Config
|
2016-03-10 11:32:07 +02:00
|
|
|
|
2016-05-24 18:54:08 +03:00
|
|
|
|
2016-05-18 19:37:18 +03:00
|
|
|
@pytest.fixture
|
|
|
|
def project(controller):
|
2016-07-11 16:36:52 +03:00
|
|
|
return Project(controller=controller, name="Test")
|
2016-05-18 19:37:18 +03:00
|
|
|
|
2016-03-10 11:32:07 +02:00
|
|
|
|
2016-07-21 10:45:02 +03:00
|
|
|
@pytest.fixture
|
|
|
|
def node(controller, project, async_run):
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
node = async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
return node
|
|
|
|
|
|
|
|
|
2016-03-10 11:32:07 +02:00
|
|
|
def test_affect_uuid():
|
2016-07-11 16:36:52 +03:00
|
|
|
p = Project(name="Test")
|
2016-03-10 11:32:07 +02:00
|
|
|
assert len(p.id) == 36
|
|
|
|
|
2016-07-11 16:36:52 +03:00
|
|
|
p = Project(project_id='00010203-0405-0607-0809-0a0b0c0d0e0f', name="Test 2")
|
2016-03-10 11:32:07 +02:00
|
|
|
assert p.id == '00010203-0405-0607-0809-0a0b0c0d0e0f'
|
|
|
|
|
|
|
|
|
|
|
|
def test_json(tmpdir):
|
2016-07-11 16:36:52 +03:00
|
|
|
p = Project(name="Test")
|
2016-08-15 21:51:59 +03:00
|
|
|
assert p.__json__() == {
|
|
|
|
"name": "Test",
|
|
|
|
"project_id": p.id,
|
|
|
|
"path": p.path,
|
|
|
|
"status": "opened",
|
|
|
|
"filename": "Test.gns3",
|
|
|
|
"auto_start": False,
|
2016-08-17 12:05:16 +03:00
|
|
|
"auto_close": True,
|
2016-09-20 13:46:39 +03:00
|
|
|
"auto_open": False,
|
|
|
|
"scene_width": 2000,
|
2017-07-03 11:01:22 +03:00
|
|
|
"scene_height": 1000,
|
|
|
|
"zoom": 100,
|
|
|
|
"show_grid": False,
|
|
|
|
"show_interface_labels": False,
|
|
|
|
"show_layers": False,
|
|
|
|
"snap_to_grid": False
|
2016-08-15 21:51:59 +03:00
|
|
|
}
|
2016-04-26 15:34:49 +03:00
|
|
|
|
|
|
|
|
2016-08-15 17:44:09 +03:00
|
|
|
def test_update(controller, async_run):
|
|
|
|
project = Project(controller=controller, name="Hello")
|
|
|
|
controller._notification = MagicMock()
|
|
|
|
|
|
|
|
assert project.name == "Hello"
|
|
|
|
async_run(project.update(name="World"))
|
|
|
|
assert project.name == "World"
|
|
|
|
controller.notification.emit.assert_any_call("project.updated", project.__json__())
|
|
|
|
|
|
|
|
|
2016-04-26 15:34:49 +03:00
|
|
|
def test_path(tmpdir):
|
|
|
|
|
2016-05-11 19:42:55 +03:00
|
|
|
directory = Config.instance().get_section_config("Server").get("projects_path")
|
2016-04-26 15:34:49 +03:00
|
|
|
|
2016-05-11 19:42:55 +03:00
|
|
|
with patch("gns3server.utils.path.get_default_project_directory", return_value=directory):
|
2016-07-11 16:36:52 +03:00
|
|
|
p = Project(project_id=str(uuid4()), name="Test")
|
2016-04-26 15:34:49 +03:00
|
|
|
assert p.path == os.path.join(directory, p.id)
|
|
|
|
assert os.path.exists(os.path.join(directory, p.id))
|
|
|
|
|
|
|
|
|
2016-07-18 22:30:30 +03:00
|
|
|
def test_path_exist(tmpdir):
|
|
|
|
"""
|
|
|
|
Should raise an error when you try to owerwrite
|
|
|
|
an existing project
|
|
|
|
"""
|
|
|
|
os.makedirs(str(tmpdir / "demo"))
|
|
|
|
with pytest.raises(aiohttp.web.HTTPForbidden):
|
|
|
|
p = Project(name="Test", path=str(tmpdir / "demo"))
|
|
|
|
|
|
|
|
|
2016-04-26 15:34:49 +03:00
|
|
|
def test_init_path(tmpdir):
|
|
|
|
|
2016-07-11 16:36:52 +03:00
|
|
|
p = Project(path=str(tmpdir), project_id=str(uuid4()), name="Test")
|
2016-04-26 15:34:49 +03:00
|
|
|
assert p.path == str(tmpdir)
|
|
|
|
|
2016-10-04 11:56:38 +03:00
|
|
|
|
2016-09-28 11:27:30 +03:00
|
|
|
@pytest.mark.skipif(sys.platform.startswith("win"), reason="Not supported on Windows")
|
2016-04-26 15:34:49 +03:00
|
|
|
def test_changing_path_with_quote_not_allowed(tmpdir):
|
|
|
|
with pytest.raises(aiohttp.web.HTTPForbidden):
|
2016-07-11 16:36:52 +03:00
|
|
|
p = Project(project_id=str(uuid4()), name="Test")
|
2016-04-26 15:34:49 +03:00
|
|
|
p.path = str(tmpdir / "project\"53")
|
2016-03-10 22:51:29 +02:00
|
|
|
|
|
|
|
|
2016-04-26 18:10:33 +03:00
|
|
|
def test_captures_directory(tmpdir):
|
2016-07-18 22:30:30 +03:00
|
|
|
p = Project(path=str(tmpdir / "capturestest"), name="Test")
|
|
|
|
assert p.captures_directory == str(tmpdir / "capturestest" / "project-files" / "captures")
|
2016-04-26 18:10:33 +03:00
|
|
|
assert os.path.exists(p.captures_directory)
|
|
|
|
|
2016-05-16 19:17:24 +03:00
|
|
|
|
2016-05-18 19:37:18 +03:00
|
|
|
def test_add_node_local(async_run, controller):
|
2016-05-16 19:17:24 +03:00
|
|
|
"""
|
|
|
|
For a local server we send the project path
|
|
|
|
"""
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
2016-07-11 16:36:52 +03:00
|
|
|
project = Project(controller=controller, name="Test")
|
2016-05-18 19:37:18 +03:00
|
|
|
controller._notification = MagicMock()
|
2016-05-16 19:17:24 +03:00
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
2017-02-02 20:13:47 +02:00
|
|
|
node = async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_script": "test.cfg"}))
|
2016-05-17 14:11:43 +03:00
|
|
|
assert node.id in project._nodes
|
2016-05-16 19:17:24 +03:00
|
|
|
|
2016-05-16 22:12:32 +03:00
|
|
|
compute.post.assert_any_call('/projects', data={
|
2016-05-16 19:17:24 +03:00
|
|
|
"name": project._name,
|
|
|
|
"project_id": project._id,
|
|
|
|
"path": project._path
|
|
|
|
})
|
|
|
|
compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id),
|
|
|
|
data={'node_id': node.id,
|
2017-02-02 20:13:47 +02:00
|
|
|
'startup_script': 'test.cfg',
|
2016-10-04 11:56:38 +03:00
|
|
|
'name': 'test'},
|
|
|
|
timeout=120)
|
2016-05-16 19:17:24 +03:00
|
|
|
assert compute in project._project_created_on_compute
|
2016-05-18 19:37:18 +03:00
|
|
|
controller.notification.emit.assert_any_call("node.created", node.__json__())
|
2016-05-16 19:17:24 +03:00
|
|
|
|
|
|
|
|
2016-05-18 19:37:18 +03:00
|
|
|
def test_add_node_non_local(async_run, controller):
|
2016-05-16 19:17:24 +03:00
|
|
|
"""
|
|
|
|
For a non local server we do not send the project path
|
|
|
|
"""
|
2016-04-15 18:57:06 +03:00
|
|
|
compute = MagicMock()
|
2016-05-16 19:17:24 +03:00
|
|
|
compute.id = "remote"
|
2016-07-11 16:36:52 +03:00
|
|
|
project = Project(controller=controller, name="Test")
|
2016-05-18 19:37:18 +03:00
|
|
|
controller._notification = MagicMock()
|
2016-03-15 11:45:05 +02:00
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
2016-04-15 18:57:06 +03:00
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
2016-03-15 11:45:05 +02:00
|
|
|
|
2017-02-02 20:13:47 +02:00
|
|
|
node = async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_script": "test.cfg"}))
|
2016-03-15 11:45:05 +02:00
|
|
|
|
2016-05-16 22:12:32 +03:00
|
|
|
compute.post.assert_any_call('/projects', data={
|
2016-05-16 19:17:24 +03:00
|
|
|
"name": project._name,
|
2016-05-24 18:54:08 +03:00
|
|
|
"project_id": project._id
|
2016-05-16 19:17:24 +03:00
|
|
|
})
|
2016-05-11 20:41:45 +03:00
|
|
|
compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id),
|
|
|
|
data={'node_id': node.id,
|
2017-02-02 20:13:47 +02:00
|
|
|
'startup_script': 'test.cfg',
|
2016-10-04 11:56:38 +03:00
|
|
|
'name': 'test'},
|
|
|
|
timeout=120)
|
2016-05-11 11:56:43 +03:00
|
|
|
assert compute in project._project_created_on_compute
|
2016-05-18 19:37:18 +03:00
|
|
|
controller.notification.emit.assert_any_call("node.created", node.__json__())
|
2016-03-11 17:51:35 +02:00
|
|
|
|
|
|
|
|
2017-04-12 15:35:49 +03:00
|
|
|
def test_add_node_from_appliance(async_run, controller):
|
|
|
|
"""
|
|
|
|
For a local server we send the project path
|
|
|
|
"""
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
project = Project(controller=controller, name="Test")
|
|
|
|
controller._notification = MagicMock()
|
|
|
|
controller._appliances["fakeid"] = Appliance("fakeid", {
|
|
|
|
"server": "local",
|
|
|
|
"name": "Test",
|
|
|
|
"default_name_format": "{name}-{0}",
|
2017-06-28 10:37:19 +03:00
|
|
|
"node_type": "vpcs",
|
|
|
|
"properties": {
|
|
|
|
"a": 1
|
|
|
|
}
|
2017-04-12 15:35:49 +03:00
|
|
|
|
|
|
|
})
|
|
|
|
controller._computes["local"] = compute
|
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
node = async_run(project.add_node_from_appliance("fakeid", x=23, y=12))
|
|
|
|
|
|
|
|
compute.post.assert_any_call('/projects', data={
|
|
|
|
"name": project._name,
|
|
|
|
"project_id": project._id,
|
|
|
|
"path": project._path
|
|
|
|
})
|
|
|
|
compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id),
|
|
|
|
data={'node_id': node.id,
|
2017-06-28 10:37:19 +03:00
|
|
|
'name': 'Test-1',
|
|
|
|
'a': 1,
|
|
|
|
},
|
2017-04-12 15:35:49 +03:00
|
|
|
timeout=120)
|
|
|
|
assert compute in project._project_created_on_compute
|
|
|
|
controller.notification.emit.assert_any_call("node.created", node.__json__())
|
|
|
|
|
2017-06-28 10:37:19 +03:00
|
|
|
# Make sure we can call twice the node creation
|
|
|
|
node = async_run(project.add_node_from_appliance("fakeid", x=13, y=12))
|
|
|
|
compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id),
|
|
|
|
data={'node_id': node.id,
|
|
|
|
'name': 'Test-2',
|
|
|
|
'a': 1
|
|
|
|
},
|
|
|
|
timeout=120)
|
|
|
|
|
2017-04-12 15:35:49 +03:00
|
|
|
|
2016-05-18 19:37:18 +03:00
|
|
|
def test_delete_node(async_run, controller):
|
2016-05-17 14:11:43 +03:00
|
|
|
"""
|
|
|
|
For a local server we send the project path
|
|
|
|
"""
|
|
|
|
compute = MagicMock()
|
2016-07-11 16:36:52 +03:00
|
|
|
project = Project(controller=controller, name="Test")
|
2016-05-18 19:37:18 +03:00
|
|
|
controller._notification = MagicMock()
|
2016-05-17 14:11:43 +03:00
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
2016-06-12 02:31:30 +03:00
|
|
|
node = async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
2016-05-17 14:11:43 +03:00
|
|
|
assert node.id in project._nodes
|
|
|
|
async_run(project.delete_node(node.id))
|
|
|
|
assert node.id not in project._nodes
|
|
|
|
|
|
|
|
compute.delete.assert_any_call('/projects/{}/vpcs/nodes/{}'.format(project.id, node.id))
|
2016-05-18 19:37:18 +03:00
|
|
|
controller.notification.emit.assert_any_call("node.deleted", node.__json__())
|
2016-05-17 14:11:43 +03:00
|
|
|
|
|
|
|
|
2016-07-05 17:07:05 +03:00
|
|
|
def test_delete_node_delete_link(async_run, controller):
|
|
|
|
"""
|
|
|
|
Delete a node delete all the node connected
|
|
|
|
"""
|
|
|
|
compute = MagicMock()
|
2016-07-11 16:36:52 +03:00
|
|
|
project = Project(controller=controller, name="Test")
|
2016-07-05 17:07:05 +03:00
|
|
|
controller._notification = MagicMock()
|
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
node = async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
|
|
|
|
link = async_run(project.add_link())
|
|
|
|
async_run(link.add_node(node, 0, 0))
|
|
|
|
|
|
|
|
async_run(project.delete_node(node.id))
|
|
|
|
assert node.id not in project._nodes
|
|
|
|
assert link.id not in project._links
|
|
|
|
|
|
|
|
compute.delete.assert_any_call('/projects/{}/vpcs/nodes/{}'.format(project.id, node.id))
|
|
|
|
controller.notification.emit.assert_any_call("node.deleted", node.__json__())
|
|
|
|
controller.notification.emit.assert_any_call("link.deleted", link.__json__())
|
|
|
|
|
|
|
|
|
2016-07-25 15:47:37 +03:00
|
|
|
def test_get_node(async_run, controller):
|
2016-04-15 18:57:06 +03:00
|
|
|
compute = MagicMock()
|
2016-07-11 16:36:52 +03:00
|
|
|
project = Project(controller=controller, name="Test")
|
2016-03-15 11:45:05 +02:00
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
2016-04-15 18:57:06 +03:00
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
2016-03-15 11:45:05 +02:00
|
|
|
|
2016-06-12 02:31:30 +03:00
|
|
|
vm = async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
2016-05-11 20:35:36 +03:00
|
|
|
assert project.get_node(vm.id) == vm
|
2016-03-11 17:51:35 +02:00
|
|
|
|
|
|
|
with pytest.raises(aiohttp.web_exceptions.HTTPNotFound):
|
2016-05-11 20:35:36 +03:00
|
|
|
project.get_node("test")
|
2016-03-11 17:51:35 +02:00
|
|
|
|
2016-07-25 15:47:37 +03:00
|
|
|
# Raise an error if the project is not opened
|
|
|
|
async_run(project.close())
|
|
|
|
with pytest.raises(aiohttp.web.HTTPForbidden):
|
|
|
|
project.get_node(vm.id)
|
|
|
|
|
2016-03-11 17:51:35 +02:00
|
|
|
|
2017-06-27 11:09:21 +03:00
|
|
|
def test_add_link(async_run, project, controller):
|
2016-04-15 18:57:06 +03:00
|
|
|
compute = MagicMock()
|
2016-03-15 11:45:05 +02:00
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
2016-04-15 18:57:06 +03:00
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
2016-03-15 11:45:05 +02:00
|
|
|
|
2016-06-12 02:31:30 +03:00
|
|
|
vm1 = async_run(project.add_node(compute, "test1", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
2016-09-15 15:51:40 +03:00
|
|
|
vm1._ports = [EthernetPort("E0", 0, 3, 1)]
|
2016-06-12 02:31:30 +03:00
|
|
|
vm2 = async_run(project.add_node(compute, "test2", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
2016-09-15 15:51:40 +03:00
|
|
|
vm2._ports = [EthernetPort("E0", 0, 4, 2)]
|
2016-05-18 19:37:18 +03:00
|
|
|
controller._notification = MagicMock()
|
2016-05-11 20:35:36 +03:00
|
|
|
link = async_run(project.add_link())
|
|
|
|
async_run(link.add_node(vm1, 3, 1))
|
2016-09-02 15:39:38 +03:00
|
|
|
with asyncio_patch("gns3server.controller.udp_link.UDPLink.create") as mock_udp_create:
|
|
|
|
async_run(link.add_node(vm2, 4, 2))
|
|
|
|
assert mock_udp_create.called
|
2016-05-11 20:35:36 +03:00
|
|
|
assert len(link._nodes) == 2
|
2016-05-18 19:37:18 +03:00
|
|
|
controller.notification.emit.assert_any_call("link.created", link.__json__())
|
2016-03-11 17:51:35 +02:00
|
|
|
|
|
|
|
|
2017-06-27 11:09:21 +03:00
|
|
|
def test_get_link(async_run, project):
|
2016-04-15 18:57:06 +03:00
|
|
|
compute = MagicMock()
|
2016-03-15 11:45:05 +02:00
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
2016-04-15 18:57:06 +03:00
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
2016-03-15 11:45:05 +02:00
|
|
|
|
2016-05-11 20:35:36 +03:00
|
|
|
link = async_run(project.add_link())
|
|
|
|
assert project.get_link(link.id) == link
|
2016-03-11 17:51:35 +02:00
|
|
|
|
|
|
|
with pytest.raises(aiohttp.web_exceptions.HTTPNotFound):
|
2016-05-11 20:35:36 +03:00
|
|
|
project.get_link("test")
|
2016-05-18 19:37:18 +03:00
|
|
|
|
|
|
|
|
2017-06-27 11:09:21 +03:00
|
|
|
def test_delete_link(async_run, project, controller):
|
2016-05-18 19:37:18 +03:00
|
|
|
compute = MagicMock()
|
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
assert len(project._links) == 0
|
|
|
|
link = async_run(project.add_link())
|
|
|
|
assert len(project._links) == 1
|
|
|
|
controller._notification = MagicMock()
|
|
|
|
async_run(project.delete_link(link.id))
|
|
|
|
controller.notification.emit.assert_any_call("link.deleted", link.__json__())
|
|
|
|
assert len(project._links) == 0
|
2016-05-26 14:43:35 +03:00
|
|
|
|
|
|
|
|
2017-06-27 11:09:21 +03:00
|
|
|
def test_add_drawing(async_run, project, controller):
|
2016-06-20 19:45:31 +03:00
|
|
|
controller.notification.emit = MagicMock()
|
|
|
|
|
2016-06-23 12:17:23 +03:00
|
|
|
drawing = async_run(project.add_drawing(None, svg="<svg></svg>"))
|
|
|
|
assert len(project._drawings) == 1
|
|
|
|
controller.notification.emit.assert_any_call("drawing.created", drawing.__json__())
|
2016-06-20 19:45:31 +03:00
|
|
|
|
|
|
|
|
2017-06-27 11:09:21 +03:00
|
|
|
def test_get_drawing(async_run, project):
|
2016-06-23 12:17:23 +03:00
|
|
|
drawing = async_run(project.add_drawing(None))
|
|
|
|
assert project.get_drawing(drawing.id) == drawing
|
2016-06-20 19:45:31 +03:00
|
|
|
|
|
|
|
with pytest.raises(aiohttp.web_exceptions.HTTPNotFound):
|
2016-06-23 12:17:23 +03:00
|
|
|
project.get_drawing("test")
|
2016-06-20 19:45:31 +03:00
|
|
|
|
|
|
|
|
2017-06-27 11:09:21 +03:00
|
|
|
def test_delete_drawing(async_run, project, controller):
|
2016-06-23 12:17:23 +03:00
|
|
|
assert len(project._drawings) == 0
|
|
|
|
drawing = async_run(project.add_drawing())
|
|
|
|
assert len(project._drawings) == 1
|
2016-06-20 19:45:31 +03:00
|
|
|
controller._notification = MagicMock()
|
2016-06-23 12:17:23 +03:00
|
|
|
async_run(project.delete_drawing(drawing.id))
|
|
|
|
controller.notification.emit.assert_any_call("drawing.deleted", drawing.__json__())
|
|
|
|
assert len(project._drawings) == 0
|
2016-06-20 19:45:31 +03:00
|
|
|
|
|
|
|
|
2017-06-27 12:11:07 +03:00
|
|
|
def test_clean_pictures(async_run, project, controller):
|
2016-07-13 17:13:14 +03:00
|
|
|
"""
|
|
|
|
When a project is close old pictures should be removed
|
|
|
|
"""
|
|
|
|
|
|
|
|
drawing = async_run(project.add_drawing())
|
|
|
|
drawing._svg = "test.png"
|
|
|
|
open(os.path.join(project.pictures_directory, "test.png"), "w+").close()
|
|
|
|
open(os.path.join(project.pictures_directory, "test2.png"), "w+").close()
|
|
|
|
async_run(project.close())
|
|
|
|
assert os.path.exists(os.path.join(project.pictures_directory, "test.png"))
|
|
|
|
assert not os.path.exists(os.path.join(project.pictures_directory, "test2.png"))
|
|
|
|
|
|
|
|
|
2016-05-26 14:43:35 +03:00
|
|
|
def test_delete(async_run, project, controller):
|
|
|
|
assert os.path.exists(project.path)
|
|
|
|
async_run(project.delete())
|
|
|
|
assert not os.path.exists(project.path)
|
|
|
|
|
2016-06-14 13:04:23 +03:00
|
|
|
|
|
|
|
def test_dump():
|
|
|
|
directory = Config.instance().get_section_config("Server").get("projects_path")
|
|
|
|
|
|
|
|
with patch("gns3server.utils.path.get_default_project_directory", return_value=directory):
|
|
|
|
p = Project(project_id='00010203-0405-0607-0809-0a0b0c0d0e0f', name="Test")
|
|
|
|
p.dump()
|
|
|
|
with open(os.path.join(directory, p.id, "Test.gns3")) as f:
|
|
|
|
content = f.read()
|
|
|
|
assert "00010203-0405-0607-0809-0a0b0c0d0e0f" in content
|
2016-06-15 16:12:38 +03:00
|
|
|
|
|
|
|
|
|
|
|
def test_open_close(async_run, controller):
|
2016-07-11 16:36:52 +03:00
|
|
|
project = Project(controller=controller, status="closed", name="Test")
|
2016-06-15 16:12:38 +03:00
|
|
|
assert project.status == "closed"
|
2016-08-16 16:45:06 +03:00
|
|
|
project.start_all = AsyncioMagicMock()
|
2016-06-15 16:12:38 +03:00
|
|
|
async_run(project.open())
|
2016-08-16 16:45:06 +03:00
|
|
|
assert not project.start_all.called
|
2016-06-15 16:12:38 +03:00
|
|
|
assert project.status == "opened"
|
2016-08-15 13:06:26 +03:00
|
|
|
controller._notification = MagicMock()
|
2016-06-15 16:12:38 +03:00
|
|
|
async_run(project.close())
|
|
|
|
assert project.status == "closed"
|
2016-08-15 13:06:26 +03:00
|
|
|
controller.notification.emit.assert_any_call("project.closed", project.__json__())
|
2016-07-20 15:50:15 +03:00
|
|
|
|
|
|
|
|
2016-08-16 16:45:06 +03:00
|
|
|
def test_open_auto_start(async_run, controller):
|
2017-09-05 11:38:17 +03:00
|
|
|
project = Project(controller=controller, status="closed", name="Test", auto_start=True)
|
2016-08-16 16:45:06 +03:00
|
|
|
project.start_all = AsyncioMagicMock()
|
|
|
|
async_run(project.open())
|
|
|
|
assert project.start_all.called
|
|
|
|
|
|
|
|
|
2016-07-21 10:45:02 +03:00
|
|
|
def test_is_running(project, async_run, node):
|
|
|
|
"""
|
|
|
|
If a node is started or paused return True
|
|
|
|
"""
|
|
|
|
|
|
|
|
assert project.is_running() is False
|
|
|
|
node._status = "started"
|
|
|
|
assert project.is_running() is True
|
2016-07-25 15:47:37 +03:00
|
|
|
|
|
|
|
|
|
|
|
def test_duplicate(project, async_run, controller):
|
|
|
|
"""
|
|
|
|
Duplicate a project, the node should remain on the remote server
|
|
|
|
if they were on remote server
|
|
|
|
"""
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "remote"
|
|
|
|
compute.list_files = AsyncioMagicMock(return_value=[])
|
|
|
|
controller._computes["remote"] = compute
|
|
|
|
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
remote_vpcs = async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
|
|
|
|
# We allow node not allowed for standard import / export
|
2017-02-14 17:41:31 +02:00
|
|
|
remote_virtualbox = async_run(project.add_node(compute, "test", None, node_type="vmware", properties={"startup_config": "test.cfg"}))
|
2016-07-25 15:47:37 +03:00
|
|
|
|
|
|
|
new_project = async_run(project.duplicate(name="Hello"))
|
|
|
|
assert new_project.id != project.id
|
|
|
|
assert new_project.name == "Hello"
|
|
|
|
|
|
|
|
async_run(new_project.open())
|
|
|
|
|
2016-09-19 17:55:48 +03:00
|
|
|
assert list(new_project.nodes.values())[0].compute.id == "remote"
|
|
|
|
assert list(new_project.nodes.values())[1].compute.id == "remote"
|
2016-07-26 11:32:43 +03:00
|
|
|
|
|
|
|
|
2017-06-09 16:28:24 +03:00
|
|
|
def test_duplicate_with_zipfile_encoding_issues(project, async_run, controller):
|
|
|
|
zf = zipstream.ZipFile()
|
|
|
|
zf.writestr('test\udcc3', "data")
|
|
|
|
|
|
|
|
with asyncio_patch('gns3server.controller.project.export_project', return_value=zf):
|
|
|
|
with pytest.raises(aiohttp.web.HTTPConflict):
|
|
|
|
async_run(project.duplicate(name="Hello"))
|
|
|
|
|
|
|
|
|
2016-07-26 11:32:43 +03:00
|
|
|
def test_snapshots(project):
|
|
|
|
"""
|
|
|
|
List the snapshots
|
|
|
|
"""
|
|
|
|
os.makedirs(os.path.join(project.path, "snapshots"))
|
|
|
|
open(os.path.join(project.path, "snapshots", "test1_260716_103713.gns3project"), "w+").close()
|
|
|
|
project.reset()
|
|
|
|
|
|
|
|
assert len(project.snapshots) == 1
|
|
|
|
assert list(project.snapshots.values())[0].name == "test1"
|
|
|
|
|
|
|
|
|
|
|
|
def test_get_snapshot(project):
|
|
|
|
os.makedirs(os.path.join(project.path, "snapshots"))
|
|
|
|
open(os.path.join(project.path, "snapshots", "test1.gns3project"), "w+").close()
|
|
|
|
project.reset()
|
|
|
|
|
|
|
|
snapshot = list(project.snapshots.values())[0]
|
|
|
|
assert project.get_snapshot(snapshot.id) == snapshot
|
|
|
|
|
|
|
|
with pytest.raises(aiohttp.web_exceptions.HTTPNotFound):
|
|
|
|
project.get_snapshot("BLU")
|
|
|
|
|
|
|
|
|
|
|
|
def test_delete_snapshot(project, async_run):
|
|
|
|
os.makedirs(os.path.join(project.path, "snapshots"))
|
|
|
|
open(os.path.join(project.path, "snapshots", "test1_260716_103713.gns3project"), "w+").close()
|
|
|
|
project.reset()
|
|
|
|
|
|
|
|
snapshot = list(project.snapshots.values())[0]
|
|
|
|
assert project.get_snapshot(snapshot.id) == snapshot
|
|
|
|
|
|
|
|
async_run(project.delete_snapshot(snapshot.id))
|
|
|
|
|
|
|
|
with pytest.raises(aiohttp.web_exceptions.HTTPNotFound):
|
|
|
|
project.get_snapshot(snapshot.id)
|
|
|
|
|
|
|
|
assert not os.path.exists(os.path.join(project.path, "snapshots", "test1.gns3project"))
|
|
|
|
|
|
|
|
|
|
|
|
def test_snapshot(project, async_run):
|
|
|
|
"""
|
|
|
|
Create a snapshot
|
|
|
|
"""
|
|
|
|
assert len(project.snapshots) == 0
|
|
|
|
|
|
|
|
snapshot = async_run(project.snapshot("test1"))
|
|
|
|
assert snapshot.name == "test1"
|
|
|
|
|
|
|
|
assert len(project.snapshots) == 1
|
|
|
|
assert list(project.snapshots.values())[0].name == "test1"
|
|
|
|
|
|
|
|
# Raise a conflict if name is already use
|
|
|
|
with pytest.raises(aiohttp.web_exceptions.HTTPConflict):
|
|
|
|
snapshot = async_run(project.snapshot("test1"))
|
2016-08-16 16:45:06 +03:00
|
|
|
|
|
|
|
|
|
|
|
def test_start_all(project, async_run):
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
for node_i in range(0, 10):
|
|
|
|
async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
|
|
|
|
compute.post = AsyncioMagicMock()
|
|
|
|
async_run(project.start_all())
|
|
|
|
assert len(compute.post.call_args_list) == 10
|
|
|
|
|
|
|
|
|
|
|
|
def test_stop_all(project, async_run):
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
for node_i in range(0, 10):
|
|
|
|
async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
|
|
|
|
compute.post = AsyncioMagicMock()
|
|
|
|
async_run(project.stop_all())
|
|
|
|
assert len(compute.post.call_args_list) == 10
|
|
|
|
|
|
|
|
|
|
|
|
def test_suspend_all(project, async_run):
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
for node_i in range(0, 10):
|
|
|
|
async_run(project.add_node(compute, "test", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
|
|
|
|
compute.post = AsyncioMagicMock()
|
|
|
|
async_run(project.suspend_all())
|
|
|
|
assert len(compute.post.call_args_list) == 10
|
2016-10-18 12:11:45 +03:00
|
|
|
|
|
|
|
|
|
|
|
def test_node_name(project, async_run):
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
node = async_run(project.add_node(compute, "test-{0}", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert node.name == "test-1"
|
|
|
|
node = async_run(project.add_node(compute, "test-{0}", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert node.name == "test-2"
|
|
|
|
node = async_run(project.add_node(compute, "hello world-{0}", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert node.name == "helloworld-1"
|
|
|
|
node = async_run(project.add_node(compute, "hello world-{0}", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert node.name == "helloworld-2"
|
2017-07-20 18:29:42 +03:00
|
|
|
node = async_run(project.add_node(compute, "VPCS-1", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert node.name == "VPCS-1"
|
|
|
|
node = async_run(project.add_node(compute, "VPCS-1", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert node.name == "VPCS-2"
|
2017-06-27 11:09:21 +03:00
|
|
|
|
2017-07-27 17:06:52 +03:00
|
|
|
node = async_run(project.add_node(compute, "R3", None, node_type="vpcs", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert node.name == "R3"
|
|
|
|
|
2017-06-27 11:09:21 +03:00
|
|
|
|
|
|
|
def test_add_iou_node_and_check_if_gets_application_id(project, async_run):
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
# tests if get_next_application_id is called
|
|
|
|
with patch('gns3server.controller.project.get_next_application_id', return_value=222) as mocked_get_app_id:
|
2017-06-27 12:11:07 +03:00
|
|
|
node = async_run(project.add_node(
|
2017-06-27 11:09:21 +03:00
|
|
|
compute, "test", None, node_type="iou", properties={"startup_config": "test.cfg"}))
|
|
|
|
assert mocked_get_app_id.called
|
2017-06-27 12:11:07 +03:00
|
|
|
assert node.properties['application_id'] == 222
|
2017-06-27 11:09:21 +03:00
|
|
|
|
|
|
|
# tests if we can send property and it will be used
|
2017-06-27 12:11:07 +03:00
|
|
|
node = async_run(project.add_node(
|
2017-06-27 11:09:21 +03:00
|
|
|
compute, "test", None, node_type="iou", application_id=333, properties={"startup_config": "test.cfg"}))
|
|
|
|
assert mocked_get_app_id.called
|
2017-06-30 11:22:30 +03:00
|
|
|
assert node.properties['application_id'] == 333
|
2017-07-20 18:29:42 +03:00
|
|
|
|
|
|
|
|
|
|
|
def test_duplicate_node(project, async_run):
|
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "local"
|
|
|
|
response = MagicMock()
|
|
|
|
response.json = {"console": 2048}
|
|
|
|
compute.post = AsyncioMagicMock(return_value=response)
|
|
|
|
|
|
|
|
original = async_run(project.add_node(
|
|
|
|
compute,
|
|
|
|
"test",
|
|
|
|
None,
|
|
|
|
node_type="vpcs",
|
|
|
|
properties={
|
|
|
|
"startup_config": "test.cfg"
|
|
|
|
}))
|
|
|
|
new_node = async_run(project.duplicate_node(original, 42, 10, 11))
|
|
|
|
assert new_node.x == 42
|