2015-01-19 15:23:41 +00:00
|
|
|
|
#!/usr/bin/env python
|
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
#
|
2020-06-16 04:29:03 +00:00
|
|
|
|
# Copyright (C) 2020 GNS3 Technologies Inc.
|
2015-01-19 15:23:41 +00:00
|
|
|
|
#
|
|
|
|
|
# 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 os
|
2020-06-29 09:13:35 +00:00
|
|
|
|
import sys
|
2016-04-05 16:32:48 +00:00
|
|
|
|
import uuid
|
2015-01-26 11:10:30 +00:00
|
|
|
|
import asyncio
|
2015-01-23 10:28:58 +00:00
|
|
|
|
import pytest
|
2015-01-23 15:57:41 +00:00
|
|
|
|
import aiohttp
|
2015-02-04 16:18:53 +00:00
|
|
|
|
from uuid import uuid4
|
2015-01-23 13:07:10 +00:00
|
|
|
|
from unittest.mock import patch
|
|
|
|
|
|
2015-02-06 10:31:54 +00:00
|
|
|
|
from tests.utils import asyncio_patch
|
2016-04-15 15:57:06 +00:00
|
|
|
|
from gns3server.compute.project import Project
|
|
|
|
|
from gns3server.compute.notification_manager import NotificationManager
|
|
|
|
|
from gns3server.compute.vpcs import VPCS, VPCSVM
|
2016-03-10 09:32:07 +00:00
|
|
|
|
from gns3server.config import Config
|
2015-01-23 10:28:58 +00:00
|
|
|
|
|
|
|
|
|
|
2016-10-24 19:39:35 +00:00
|
|
|
|
@pytest.fixture(scope="function")
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def manager(loop, port_manager):
|
|
|
|
|
|
2015-01-23 10:28:58 +00:00
|
|
|
|
m = VPCS.instance()
|
|
|
|
|
m.port_manager = port_manager
|
|
|
|
|
return m
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(scope="function")
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def node(compute_project, manager):
|
|
|
|
|
|
|
|
|
|
node = manager.create_node("test", compute_project.id, "00010203-0405-0607-0809-0a0b0c0d0e0f")
|
2020-06-19 09:35:23 +00:00
|
|
|
|
return await node
|
2020-06-16 04:29:03 +00:00
|
|
|
|
|
2015-01-20 12:04:20 +00:00
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_affect_uuid():
|
2015-01-19 15:23:41 +00:00
|
|
|
|
|
2015-02-04 20:48:29 +00:00
|
|
|
|
p = Project(project_id='00010203-0405-0607-0809-0a0b0c0d0e0f')
|
|
|
|
|
assert p.id == '00010203-0405-0607-0809-0a0b0c0d0e0f'
|
2015-01-19 15:23:41 +00:00
|
|
|
|
|
2015-01-20 12:04:20 +00:00
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_clean_tmp_directory():
|
2016-04-21 15:27:49 +00:00
|
|
|
|
"""
|
|
|
|
|
The tmp directory should be clean at project open and close
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
p = Project(project_id='00010203-0405-0607-0809-0a0b0c0d0e0f')
|
|
|
|
|
path = p.tmp_working_directory()
|
|
|
|
|
os.makedirs(path)
|
2020-06-16 04:29:03 +00:00
|
|
|
|
await p.close()
|
2016-04-21 15:27:49 +00:00
|
|
|
|
assert not os.path.exists(path)
|
|
|
|
|
|
|
|
|
|
os.makedirs(path)
|
|
|
|
|
p = Project(project_id='00010203-0405-0607-0809-0a0b0c0d0e0f')
|
|
|
|
|
assert not os.path.exists(path)
|
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_path(projects_dir):
|
2016-03-10 09:32:07 +00:00
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
directory = projects_dir
|
2016-04-15 15:57:06 +00:00
|
|
|
|
with patch("gns3server.compute.project.Project.is_local", return_value=True):
|
2016-05-11 16:42:55 +00:00
|
|
|
|
with patch("gns3server.utils.path.get_default_project_directory", return_value=directory):
|
2016-03-10 09:32:07 +00:00
|
|
|
|
p = Project(project_id=str(uuid4()))
|
|
|
|
|
assert p.path == os.path.join(directory, p.id)
|
|
|
|
|
assert os.path.exists(os.path.join(directory, p.id))
|
2015-02-05 10:39:32 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_init_path(tmpdir):
|
2015-02-05 16:52:37 +00:00
|
|
|
|
|
2016-04-15 15:57:06 +00:00
|
|
|
|
with patch("gns3server.compute.project.Project.is_local", return_value=True):
|
2016-03-10 09:32:07 +00:00
|
|
|
|
p = Project(path=str(tmpdir), project_id=str(uuid4()))
|
2015-02-05 16:52:37 +00:00
|
|
|
|
assert p.path == str(tmpdir)
|
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_changing_path_not_allowed(tmpdir):
|
|
|
|
|
|
2016-04-15 15:57:06 +00:00
|
|
|
|
with patch("gns3server.compute.project.Project.is_local", return_value=False):
|
2015-02-04 20:17:00 +00:00
|
|
|
|
with pytest.raises(aiohttp.web.HTTPForbidden):
|
2016-03-10 09:32:07 +00:00
|
|
|
|
p = Project(project_id=str(uuid4()))
|
2015-02-04 20:17:00 +00:00
|
|
|
|
p.path = str(tmpdir)
|
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_variables():
|
|
|
|
|
|
2018-05-04 12:34:44 +00:00
|
|
|
|
variables = [{"name": "VAR1", "value": "VAL1"}]
|
|
|
|
|
p = Project(project_id=str(uuid4()), variables=variables)
|
|
|
|
|
assert p.variables == variables
|
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_json():
|
|
|
|
|
|
2016-03-10 09:32:07 +00:00
|
|
|
|
p = Project(project_id=str(uuid4()))
|
2018-05-04 12:34:44 +00:00
|
|
|
|
assert p.__json__() == {
|
|
|
|
|
"name": p.name,
|
|
|
|
|
"project_id": p.id,
|
|
|
|
|
"variables": None
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_json_with_variables():
|
|
|
|
|
|
2018-05-04 12:34:44 +00:00
|
|
|
|
variables = [{"name": "VAR1", "value": "VAL1"}]
|
|
|
|
|
p = Project(project_id=str(uuid4()), variables=variables)
|
|
|
|
|
assert p.__json__() == {
|
|
|
|
|
"name": p.name,
|
|
|
|
|
"project_id": p.id,
|
|
|
|
|
"variables": variables
|
|
|
|
|
}
|
2015-01-20 13:31:47 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_node_working_directory(node, projects_dir):
|
2016-03-10 09:32:07 +00:00
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
directory = projects_dir
|
2016-04-15 15:57:06 +00:00
|
|
|
|
with patch("gns3server.compute.project.Project.is_local", return_value=True):
|
2016-03-10 09:32:07 +00:00
|
|
|
|
p = Project(project_id=str(uuid4()))
|
2016-05-11 17:35:36 +00:00
|
|
|
|
assert p.node_working_directory(node) == os.path.join(directory, p.id, 'project-files', node.module_name, node.id)
|
|
|
|
|
assert os.path.exists(p.node_working_directory(node))
|
2015-01-23 10:28:58 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_node_working_path(node, projects_dir):
|
2017-10-02 08:41:57 +00:00
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
directory = projects_dir
|
2017-10-02 08:41:57 +00:00
|
|
|
|
with patch("gns3server.compute.project.Project.is_local", return_value=True):
|
|
|
|
|
p = Project(project_id=str(uuid4()))
|
|
|
|
|
assert p.node_working_path(node) == os.path.join(directory, p.id, 'project-files', node.module_name, node.id)
|
|
|
|
|
# after this execution directory structure should not be created
|
|
|
|
|
assert not os.path.exists(p.node_working_path(node))
|
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_project_delete():
|
|
|
|
|
|
2016-03-10 09:32:07 +00:00
|
|
|
|
project = Project(project_id=str(uuid4()))
|
2015-01-23 10:48:20 +00:00
|
|
|
|
directory = project.path
|
|
|
|
|
assert os.path.exists(directory)
|
2020-06-16 04:29:03 +00:00
|
|
|
|
await project.delete()
|
2015-01-23 10:48:20 +00:00
|
|
|
|
assert os.path.exists(directory) is False
|
2015-01-23 13:07:10 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-29 09:13:35 +00:00
|
|
|
|
@pytest.mark.skipif(not sys.platform.startswith("win") and os.getuid() == 0, reason="Root can delete any project")
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_project_delete_permission_issue():
|
|
|
|
|
|
2016-03-10 09:32:07 +00:00
|
|
|
|
project = Project(project_id=str(uuid4()))
|
2015-01-26 12:54:44 +00:00
|
|
|
|
directory = project.path
|
|
|
|
|
assert os.path.exists(directory)
|
|
|
|
|
os.chmod(directory, 0)
|
|
|
|
|
with pytest.raises(aiohttp.web.HTTPInternalServerError):
|
2020-06-19 09:35:23 +00:00
|
|
|
|
await project.delete()
|
2015-01-26 12:54:44 +00:00
|
|
|
|
os.chmod(directory, 700)
|
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_project_add_node(manager):
|
|
|
|
|
|
2016-03-10 09:32:07 +00:00
|
|
|
|
project = Project(project_id=str(uuid4()))
|
2016-05-11 17:35:36 +00:00
|
|
|
|
node = VPCSVM("test", "00010203-0405-0607-0809-0a0b0c0d0e0f", project, manager)
|
|
|
|
|
project.add_node(node)
|
|
|
|
|
assert len(project.nodes) == 1
|
2015-01-23 13:07:10 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_project_close(node, compute_project):
|
2015-03-06 14:48:16 +00:00
|
|
|
|
|
2016-04-15 15:57:06 +00:00
|
|
|
|
with asyncio_patch("gns3server.compute.vpcs.vpcs_vm.VPCSVM.close") as mock:
|
2020-06-16 04:29:03 +00:00
|
|
|
|
await compute_project.close()
|
2015-01-23 13:07:10 +00:00
|
|
|
|
assert mock.called
|
2016-05-11 17:35:36 +00:00
|
|
|
|
assert node.id not in node.manager._nodes
|
2015-01-23 15:02:26 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_list_files(tmpdir):
|
2015-05-14 10:03:17 +00:00
|
|
|
|
|
2016-05-11 16:42:55 +00:00
|
|
|
|
with patch("gns3server.config.Config.get_section_config", return_value={"projects_path": str(tmpdir)}):
|
2016-03-10 09:32:07 +00:00
|
|
|
|
project = Project(project_id=str(uuid4()))
|
2015-05-14 10:03:17 +00:00
|
|
|
|
path = project.path
|
|
|
|
|
os.makedirs(os.path.join(path, "vm-1", "dynamips"))
|
|
|
|
|
with open(os.path.join(path, "vm-1", "dynamips", "test.bin"), "w+") as f:
|
|
|
|
|
f.write("test")
|
|
|
|
|
open(os.path.join(path, "vm-1", "dynamips", "test.ghost"), "w+").close()
|
|
|
|
|
with open(os.path.join(path, "test.txt"), "w+") as f:
|
|
|
|
|
f.write("test2")
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
files = await project.list_files()
|
2015-05-14 10:03:17 +00:00
|
|
|
|
|
|
|
|
|
assert files == [
|
|
|
|
|
{
|
|
|
|
|
"path": "test.txt",
|
|
|
|
|
"md5sum": "ad0234829205b9033196ba818f7a872b"
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
"path": os.path.join("vm-1", "dynamips", "test.bin"),
|
|
|
|
|
"md5sum": "098f6bcd4621d373cade4e832627b4f6"
|
|
|
|
|
}
|
|
|
|
|
]
|
2016-03-17 14:15:30 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_emit():
|
2016-03-17 14:15:30 +00:00
|
|
|
|
|
|
|
|
|
with NotificationManager.instance().queue() as queue:
|
2020-06-16 04:29:03 +00:00
|
|
|
|
await queue.get(0.5) # Ping
|
2016-03-17 14:15:30 +00:00
|
|
|
|
|
|
|
|
|
project = Project(project_id=str(uuid4()))
|
|
|
|
|
project.emit("test", {})
|
2020-06-16 04:29:03 +00:00
|
|
|
|
(action, event, context) = await queue.get(0.5)
|
2016-03-17 14:15:30 +00:00
|
|
|
|
assert action == "test"
|
|
|
|
|
assert context["project_id"] == project.id
|
2018-05-09 13:29:35 +00:00
|
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
async def test_update_project():
|
|
|
|
|
|
2018-05-09 13:29:35 +00:00
|
|
|
|
variables = [{"name": "TEST", "value": "VAL"}]
|
|
|
|
|
project = Project(project_id=str(uuid.uuid4()))
|
2020-06-16 04:29:03 +00:00
|
|
|
|
await project.update(variables=variables)
|
2018-05-09 13:29:35 +00:00
|
|
|
|
assert project.variables == variables
|