2015-01-21 21:32:33 +00:00
|
|
|
import pytest
|
2020-06-16 08:57:16 +00:00
|
|
|
import asyncio
|
2015-01-23 16:39:17 +00:00
|
|
|
import tempfile
|
|
|
|
import shutil
|
2020-10-02 06:37:50 +00:00
|
|
|
import sys
|
|
|
|
import os
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2020-12-02 08:09:08 +00:00
|
|
|
from fastapi import FastAPI
|
|
|
|
from sqlalchemy.ext.asyncio import AsyncSession, create_async_engine
|
|
|
|
from httpx import AsyncClient
|
2020-06-16 04:29:03 +00:00
|
|
|
from unittest.mock import MagicMock, patch
|
2020-01-07 22:03:31 +00:00
|
|
|
from pathlib import Path
|
2015-05-14 10:03:17 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
from gns3server.controller import Controller
|
|
|
|
from gns3server.config import Config
|
2016-04-15 15:57:06 +00:00
|
|
|
from gns3server.compute import MODULES
|
|
|
|
from gns3server.compute.port_manager import PortManager
|
|
|
|
from gns3server.compute.project_manager import ProjectManager
|
2020-12-07 06:22:36 +00:00
|
|
|
from gns3server.db.models import Base, User
|
|
|
|
from gns3server.db.repositories.users import UsersRepository
|
|
|
|
from gns3server.api.routes.controller.dependencies.database import get_db_session
|
|
|
|
from gns3server.schemas.users import UserCreate
|
|
|
|
from gns3server.services import auth_service
|
2015-01-21 21:32:33 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
sys._called_from_test = True
|
|
|
|
sys.original_platform = sys.platform
|
2015-01-21 21:32:33 +00:00
|
|
|
|
2020-12-02 08:09:08 +00:00
|
|
|
|
2020-10-19 14:27:06 +00:00
|
|
|
if sys.platform.startswith("win") and sys.version_info < (3, 8):
|
|
|
|
@pytest.yield_fixture(scope="session")
|
|
|
|
def event_loop(request):
|
|
|
|
"""
|
|
|
|
Overwrite pytest_asyncio event loop on Windows for Python < 3.8
|
|
|
|
As of Python 3.8, the default event loop on Windows is Proactor
|
|
|
|
"""
|
|
|
|
|
|
|
|
loop = asyncio.ProactorEventLoop()
|
|
|
|
asyncio.set_event_loop(loop)
|
|
|
|
yield loop
|
2020-10-20 00:20:46 +00:00
|
|
|
asyncio.set_event_loop(None)
|
2020-06-16 04:38:17 +00:00
|
|
|
|
2020-12-18 06:21:54 +00:00
|
|
|
|
2020-12-07 06:22:36 +00:00
|
|
|
# https://github.com/pytest-dev/pytest-asyncio/issues/68
|
|
|
|
# this event_loop is used by pytest-asyncio, and redefining it
|
|
|
|
# is currently the only way of changing the scope of this fixture
|
2020-12-07 08:23:40 +00:00
|
|
|
@pytest.yield_fixture(scope="class")
|
2020-12-07 06:22:36 +00:00
|
|
|
def event_loop(request):
|
|
|
|
|
|
|
|
loop = asyncio.get_event_loop_policy().new_event_loop()
|
|
|
|
yield loop
|
|
|
|
loop.close()
|
|
|
|
|
|
|
|
|
2020-12-07 08:23:40 +00:00
|
|
|
@pytest.fixture(scope="class")
|
|
|
|
async def app() -> FastAPI:
|
2020-12-07 06:22:36 +00:00
|
|
|
|
2020-12-07 08:23:40 +00:00
|
|
|
# async with db_engine.begin() as conn:
|
|
|
|
# await conn.run_sync(Base.metadata.drop_all)
|
|
|
|
# await conn.run_sync(Base.metadata.create_all)
|
2020-12-07 06:22:36 +00:00
|
|
|
from gns3server.api.server import app as gns3app
|
|
|
|
yield gns3app
|
|
|
|
|
|
|
|
|
2020-12-07 08:23:40 +00:00
|
|
|
@pytest.fixture(scope="class")
|
2020-12-07 06:22:36 +00:00
|
|
|
def db_engine():
|
2015-01-21 21:32:33 +00:00
|
|
|
|
2021-03-28 10:47:29 +00:00
|
|
|
db_url = os.getenv("GNS3_TEST_DATABASE_URI", "sqlite+aiosqlite:///:memory:") # "sqlite:///./sql_test_app.db"
|
2020-12-07 06:22:36 +00:00
|
|
|
engine = create_async_engine(db_url, connect_args={"check_same_thread": False}, future=True)
|
|
|
|
yield engine
|
|
|
|
engine.sync_engine.dispose()
|
2020-12-02 08:09:08 +00:00
|
|
|
|
|
|
|
|
2020-12-07 06:22:36 +00:00
|
|
|
@pytest.fixture(scope="class")
|
2020-12-07 08:23:40 +00:00
|
|
|
async def db_session(db_engine):
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2020-12-07 06:22:36 +00:00
|
|
|
# recreate database tables for each class
|
|
|
|
# preferred and faster way would be to rollback the session/transaction
|
|
|
|
# but it doesn't work for some reason
|
|
|
|
async with db_engine.begin() as conn:
|
|
|
|
await conn.run_sync(Base.metadata.drop_all)
|
|
|
|
await conn.run_sync(Base.metadata.create_all)
|
2018-10-17 10:32:10 +00:00
|
|
|
|
2020-12-07 06:22:36 +00:00
|
|
|
session = AsyncSession(db_engine)
|
|
|
|
try:
|
|
|
|
yield session
|
|
|
|
finally:
|
|
|
|
await session.close()
|
2016-03-16 15:34:16 +00:00
|
|
|
|
2020-10-19 08:55:32 +00:00
|
|
|
|
2020-12-02 08:09:08 +00:00
|
|
|
@pytest.fixture
|
2020-12-07 06:22:36 +00:00
|
|
|
async def client(app: FastAPI, db_session: AsyncSession) -> AsyncClient:
|
|
|
|
|
|
|
|
async def _get_test_db():
|
|
|
|
try:
|
|
|
|
yield db_session
|
|
|
|
finally:
|
|
|
|
pass
|
|
|
|
|
|
|
|
app.dependency_overrides[get_db_session] = _get_test_db
|
2020-12-02 08:09:08 +00:00
|
|
|
|
|
|
|
async with AsyncClient(
|
|
|
|
app=app,
|
|
|
|
base_url="http://test-api",
|
|
|
|
headers={"Content-Type": "application/json"}
|
|
|
|
) as client:
|
|
|
|
yield client
|
2020-10-19 08:55:32 +00:00
|
|
|
|
|
|
|
|
2020-12-07 06:22:36 +00:00
|
|
|
@pytest.fixture
|
|
|
|
async def test_user(db_session: AsyncSession) -> User:
|
|
|
|
|
|
|
|
new_user = UserCreate(
|
|
|
|
username="user1",
|
|
|
|
email="user1@email.com",
|
|
|
|
password="user1_password",
|
|
|
|
)
|
|
|
|
user_repo = UsersRepository(db_session)
|
|
|
|
existing_user = await user_repo.get_user_by_username(new_user.username)
|
|
|
|
if existing_user:
|
|
|
|
return existing_user
|
|
|
|
return await user_repo.create_user(new_user)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def authorized_client(client: AsyncClient, test_user: User) -> AsyncClient:
|
|
|
|
|
|
|
|
access_token = auth_service.create_access_token(test_user.username)
|
|
|
|
client.headers = {
|
|
|
|
**client.headers,
|
|
|
|
"Authorization": f"Bearer {access_token}",
|
|
|
|
}
|
|
|
|
return client
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def controller_config_path(tmpdir):
|
2018-10-17 10:32:10 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
return str(tmpdir / "config" / "gns3_controller.conf")
|
2016-03-16 15:34:16 +00:00
|
|
|
|
2018-10-17 10:32:10 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def controller(tmpdir, controller_config_path):
|
2015-01-21 21:32:33 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
Controller._instance = None
|
|
|
|
controller = Controller.instance()
|
|
|
|
os.makedirs(os.path.dirname(controller_config_path), exist_ok=True)
|
|
|
|
Path(controller_config_path).touch()
|
|
|
|
controller._config_file = controller_config_path
|
|
|
|
controller._config_loaded = True
|
|
|
|
return controller
|
2016-09-02 12:09:01 +00:00
|
|
|
|
2018-10-17 10:32:10 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def compute(controller):
|
2018-10-17 10:32:10 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
compute = MagicMock()
|
|
|
|
compute.id = "example.com"
|
|
|
|
controller._computes = {"example.com": compute}
|
|
|
|
return compute
|
2016-03-07 14:01:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2020-10-02 06:37:50 +00:00
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def project(tmpdir, controller):
|
2020-06-16 04:29:03 +00:00
|
|
|
|
|
|
|
return await controller.add_project(name="Test")
|
2015-01-21 21:32:33 +00:00
|
|
|
|
|
|
|
|
2016-03-07 14:01:35 +00:00
|
|
|
@pytest.fixture
|
2020-06-16 04:29:03 +00:00
|
|
|
def compute_project(tmpdir):
|
|
|
|
|
|
|
|
return ProjectManager.instance().create_project(project_id="a1e920ca-338a-4e9f-b363-aa607b09dd80")
|
2016-03-07 14:01:35 +00:00
|
|
|
|
|
|
|
|
2016-04-19 08:47:53 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def config():
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2016-04-19 08:47:53 +00:00
|
|
|
config = Config.instance()
|
|
|
|
config.clear()
|
|
|
|
return config
|
|
|
|
|
|
|
|
|
2016-06-07 13:34:04 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def images_dir(config):
|
|
|
|
"""
|
|
|
|
Get the location of images
|
|
|
|
"""
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2016-06-07 13:34:04 +00:00
|
|
|
path = config.get_section_config("Server").get("images_path")
|
|
|
|
os.makedirs(path, exist_ok=True)
|
2020-08-13 07:48:45 +00:00
|
|
|
os.makedirs(os.path.join(path, "QEMU"), exist_ok=True)
|
|
|
|
os.makedirs(os.path.join(path, "IOU"), exist_ok=True)
|
2016-06-07 13:34:04 +00:00
|
|
|
return path
|
|
|
|
|
|
|
|
|
2016-06-28 17:58:57 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def symbols_dir(config):
|
|
|
|
"""
|
|
|
|
Get the location of symbols
|
|
|
|
"""
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2016-06-28 17:58:57 +00:00
|
|
|
path = config.get_section_config("Server").get("symbols_path")
|
|
|
|
os.makedirs(path, exist_ok=True)
|
|
|
|
print(path)
|
|
|
|
return path
|
|
|
|
|
|
|
|
|
2016-06-17 15:50:06 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def projects_dir(config):
|
|
|
|
"""
|
|
|
|
Get the location of images
|
|
|
|
"""
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2016-06-17 15:50:06 +00:00
|
|
|
path = config.get_section_config("Server").get("projects_path")
|
|
|
|
os.makedirs(path, exist_ok=True)
|
|
|
|
return path
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
@pytest.fixture(scope="function")
|
|
|
|
def port_manager():
|
|
|
|
"""An instance of port manager"""
|
2016-06-23 10:10:18 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
PortManager._instance = None
|
|
|
|
p = PortManager.instance()
|
|
|
|
p.console_host = "127.0.0.1"
|
|
|
|
return p
|
2016-06-23 10:10:18 +00:00
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
|
|
|
|
@pytest.fixture(scope="function")
|
|
|
|
def free_console_port(port_manager, compute_project):
|
|
|
|
"""Get a free TCP port"""
|
|
|
|
|
|
|
|
# In case of already use ports we will raise an exception
|
|
|
|
port = port_manager.get_free_tcp_port(compute_project)
|
|
|
|
# We release the port immediately in order to allow
|
|
|
|
# the test do whatever the test want
|
|
|
|
port_manager.release_tcp_port(port, compute_project)
|
|
|
|
return port
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2015-08-07 14:49:45 +00:00
|
|
|
def darwin_platform():
|
|
|
|
"""
|
|
|
|
Change sys.plaform to Darwin
|
|
|
|
"""
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2015-08-07 14:49:45 +00:00
|
|
|
old_platform = sys.platform
|
|
|
|
sys.platform = "darwin10.10"
|
|
|
|
yield
|
|
|
|
sys.plaform = old_platform
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
@pytest.fixture
|
2015-08-07 14:49:45 +00:00
|
|
|
def windows_platform():
|
|
|
|
"""
|
2017-09-11 08:09:32 +00:00
|
|
|
Change sys.platform to Windows
|
2015-08-07 14:49:45 +00:00
|
|
|
"""
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2015-08-07 14:49:45 +00:00
|
|
|
old_platform = sys.platform
|
|
|
|
sys.platform = "win10"
|
|
|
|
yield
|
|
|
|
sys.plaform = old_platform
|
|
|
|
|
|
|
|
|
2020-06-16 04:29:03 +00:00
|
|
|
@pytest.fixture
|
2015-08-07 14:49:45 +00:00
|
|
|
def linux_platform():
|
|
|
|
"""
|
2017-09-11 08:09:32 +00:00
|
|
|
Change sys.platform to Linux
|
2015-08-07 14:49:45 +00:00
|
|
|
"""
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2015-08-07 14:49:45 +00:00
|
|
|
old_platform = sys.platform
|
|
|
|
sys.platform = "linuxdebian"
|
|
|
|
yield
|
|
|
|
sys.plaform = old_platform
|
2016-03-10 09:32:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2016-09-05 16:40:49 +00:00
|
|
|
def on_gns3vm(linux_platform):
|
2016-08-19 17:02:39 +00:00
|
|
|
"""
|
|
|
|
Mock the hostname to emulate the GNS3 VM
|
|
|
|
"""
|
2020-06-16 04:29:03 +00:00
|
|
|
|
2016-09-22 15:46:32 +00:00
|
|
|
with patch("gns3server.utils.interfaces.interfaces", return_value=[
|
|
|
|
{"name": "eth0", "special": False, "type": "ethernet"},
|
|
|
|
{"name": "eth1", "special": False, "type": "ethernet"},
|
|
|
|
{"name": "virbr0", "special": True, "type": "ethernet"}]):
|
2016-08-30 16:27:04 +00:00
|
|
|
with patch("socket.gethostname", return_value="gns3vm"):
|
|
|
|
yield
|
2020-06-16 04:29:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def ethernet_device():
|
|
|
|
|
|
|
|
import psutil
|
|
|
|
return sorted(psutil.net_if_addrs().keys())[0]
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def ubridge_path(config):
|
|
|
|
"""
|
|
|
|
Get the location of a fake ubridge
|
|
|
|
"""
|
|
|
|
|
|
|
|
path = config.get_section_config("Server").get("ubridge_path")
|
|
|
|
os.makedirs(os.path.dirname(path), exist_ok=True)
|
|
|
|
open(path, 'w+').close()
|
|
|
|
return path
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def run_around_tests(monkeypatch, config, port_manager):#port_manager, controller, config):
|
|
|
|
"""
|
|
|
|
This setup a temporary project file environment around tests
|
|
|
|
"""
|
|
|
|
|
|
|
|
tmppath = tempfile.mkdtemp()
|
|
|
|
|
|
|
|
for module in MODULES:
|
|
|
|
module._instance = None
|
|
|
|
|
|
|
|
os.makedirs(os.path.join(tmppath, 'projects'))
|
|
|
|
config.set("Server", "projects_path", os.path.join(tmppath, 'projects'))
|
|
|
|
config.set("Server", "symbols_path", os.path.join(tmppath, 'symbols'))
|
|
|
|
config.set("Server", "images_path", os.path.join(tmppath, 'images'))
|
|
|
|
config.set("Server", "appliances_path", os.path.join(tmppath, 'appliances'))
|
|
|
|
config.set("Server", "ubridge_path", os.path.join(tmppath, 'bin', 'ubridge'))
|
|
|
|
config.set("Server", "auth", False)
|
2020-10-19 12:09:30 +00:00
|
|
|
config.set("Server", "local", True)
|
2020-06-16 04:29:03 +00:00
|
|
|
|
|
|
|
# Prevent executions of the VM if we forgot to mock something
|
|
|
|
config.set("VirtualBox", "vboxmanage_path", tmppath)
|
|
|
|
config.set("VPCS", "vpcs_path", tmppath)
|
|
|
|
config.set("VMware", "vmrun_path", tmppath)
|
|
|
|
config.set("Dynamips", "dynamips_path", tmppath)
|
|
|
|
|
|
|
|
# Force turn off KVM because it's not available on CI
|
|
|
|
config.set("Qemu", "enable_kvm", False)
|
|
|
|
|
|
|
|
monkeypatch.setattr("gns3server.utils.path.get_default_project_directory", lambda *args: os.path.join(tmppath, 'projects'))
|
|
|
|
|
|
|
|
# Force sys.platform to the original value. Because it seem not be restore correctly at each tests
|
|
|
|
sys.platform = sys.original_platform
|
|
|
|
|
|
|
|
yield
|
|
|
|
|
|
|
|
# An helper should not raise Exception
|
|
|
|
try:
|
|
|
|
shutil.rmtree(tmppath)
|
|
|
|
except BaseException:
|
|
|
|
pass
|
2020-12-02 08:09:08 +00:00
|
|
|
|