mirror of
https://github.com/GNS3/gns3-server
synced 2025-01-27 08:21:24 +00:00
Create a /server API for register servers
This allow to push to the controller information about the connection to a server.
This commit is contained in:
parent
84eb8356e8
commit
aad69e9650
@ -22,6 +22,9 @@ from ..config import Config
|
|||||||
class Controller:
|
class Controller:
|
||||||
"""The controller manage multiple gns3 servers"""
|
"""The controller manage multiple gns3 servers"""
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self._servers = {}
|
||||||
|
|
||||||
def isEnabled(self):
|
def isEnabled(self):
|
||||||
"""
|
"""
|
||||||
:returns: True if current instance is the controller
|
:returns: True if current instance is the controller
|
||||||
@ -29,6 +32,19 @@ class Controller:
|
|||||||
"""
|
"""
|
||||||
return Config.instance().get_section_config("Server").getboolean("controller")
|
return Config.instance().get_section_config("Server").getboolean("controller")
|
||||||
|
|
||||||
|
def addServer(self, server):
|
||||||
|
"""
|
||||||
|
Add a server to the dictionnary of servers controlled by GNS3
|
||||||
|
"""
|
||||||
|
self._servers[server.id] = server
|
||||||
|
|
||||||
|
@property
|
||||||
|
def servers(self):
|
||||||
|
"""
|
||||||
|
:returns: The dictionnary of servers managed by GNS3
|
||||||
|
"""
|
||||||
|
return self._servers
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def instance():
|
def instance():
|
||||||
"""
|
"""
|
||||||
|
63
gns3server/controller/server.py
Normal file
63
gns3server/controller/server.py
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
#!/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 <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
|
class Server:
|
||||||
|
"""
|
||||||
|
A GNS3 server.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, server_id, protocol="http", host="localhost", port=8000, user=None, password=None):
|
||||||
|
self._id = server_id
|
||||||
|
self._protocol = protocol
|
||||||
|
self._host = host
|
||||||
|
self._port = port
|
||||||
|
self._user = user
|
||||||
|
self._password = password
|
||||||
|
self._connected = False
|
||||||
|
# The remote server version
|
||||||
|
self._version = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def id(self):
|
||||||
|
"""
|
||||||
|
:returns: Server identifier (string)
|
||||||
|
"""
|
||||||
|
return self._id
|
||||||
|
|
||||||
|
@property
|
||||||
|
def host(self):
|
||||||
|
"""
|
||||||
|
:returns: Server host (string)
|
||||||
|
"""
|
||||||
|
return self._host
|
||||||
|
|
||||||
|
def __json__(self):
|
||||||
|
return {
|
||||||
|
"server_id": self._id,
|
||||||
|
"protocol": self._protocol,
|
||||||
|
"host": self._host,
|
||||||
|
"port": self._port,
|
||||||
|
"user": self._user,
|
||||||
|
"connected": self._connected,
|
||||||
|
"version": self._version
|
||||||
|
}
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
if not isinstance(other, Server):
|
||||||
|
return False
|
||||||
|
return other._id == self._id
|
@ -15,18 +15,41 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
from aiohttp.web import HTTPForbidden
|
||||||
|
|
||||||
from ...web.route import Route
|
from ...web.route import Route
|
||||||
from ...config import Config
|
from ...config import Config
|
||||||
from ...modules.project_manager import ProjectManager
|
from ...modules.project_manager import ProjectManager
|
||||||
from aiohttp.web import HTTPForbidden
|
from ...schemas.server import SERVER_CREATE_SCHEMA, SERVER_OBJECT_SCHEMA
|
||||||
|
from ...controller import Controller
|
||||||
|
from ...controller.server import Server
|
||||||
|
|
||||||
|
|
||||||
import asyncio
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class ServerHandler:
|
class ServerHandler:
|
||||||
|
"""API entry points for server management."""
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
@Route.post(
|
||||||
|
r"/servers",
|
||||||
|
description="Register a server",
|
||||||
|
status_codes={
|
||||||
|
201: "Server added"
|
||||||
|
},
|
||||||
|
controller=True,
|
||||||
|
input=SERVER_CREATE_SCHEMA,
|
||||||
|
output=SERVER_OBJECT_SCHEMA)
|
||||||
|
def create(request, response):
|
||||||
|
|
||||||
|
server = Server(request.json.pop("server_id"), **request.json)
|
||||||
|
Controller.instance().addServer(server)
|
||||||
|
|
||||||
|
response.set_status(201)
|
||||||
|
response.json(server)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@Route.post(
|
@Route.post(
|
||||||
@ -60,7 +83,7 @@ class ServerHandler:
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
# then shutdown the server itself
|
# then shutdown the server itself
|
||||||
from gns3server.server import Server
|
from gns3server.web.web_server import WebServer
|
||||||
server = Server.instance()
|
server = WebServer.instance()
|
||||||
asyncio.async(server.shutdown_server())
|
asyncio.async(server.shutdown_server())
|
||||||
response.set_status(201)
|
response.set_status(201)
|
||||||
|
@ -28,7 +28,7 @@ import locale
|
|||||||
import argparse
|
import argparse
|
||||||
import asyncio
|
import asyncio
|
||||||
|
|
||||||
from gns3server.server import Server
|
from gns3server.web.web_server import WebServer
|
||||||
from gns3server.web.logger import init_logger
|
from gns3server.web.logger import init_logger
|
||||||
from gns3server.version import __version__
|
from gns3server.version import __version__
|
||||||
from gns3server.config import Config
|
from gns3server.config import Config
|
||||||
@ -233,7 +233,7 @@ def run():
|
|||||||
host = server_config["host"]
|
host = server_config["host"]
|
||||||
port = int(server_config["port"])
|
port = int(server_config["port"])
|
||||||
|
|
||||||
server = Server.instance(host, port)
|
server = WebServer.instance(host, port)
|
||||||
try:
|
try:
|
||||||
server.run()
|
server.run()
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
|
89
gns3server/schemas/server.py
Normal file
89
gns3server/schemas/server.py
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
# -*- 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/>.
|
||||||
|
|
||||||
|
|
||||||
|
SERVER_CREATE_SCHEMA = {
|
||||||
|
"$schema": "http://json-schema.org/draft-04/schema#",
|
||||||
|
"description": "Request validation to register a GNS3 server instance",
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"server_id": {
|
||||||
|
"description": "Server identifier",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"protocol": {
|
||||||
|
"description": "Server protocol",
|
||||||
|
"enum": ["http", "https"]
|
||||||
|
},
|
||||||
|
"host": {
|
||||||
|
"description": "Server host",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"port": {
|
||||||
|
"description": "Server port",
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"user": {
|
||||||
|
"description": "User for auth",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"password": {
|
||||||
|
"description": "Password for auth",
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"required": ["server_id", "protocol", "host", "port"]
|
||||||
|
}
|
||||||
|
|
||||||
|
SERVER_OBJECT_SCHEMA = {
|
||||||
|
"$schema": "http://json-schema.org/draft-04/schema#",
|
||||||
|
"description": "Request validation to a GNS3 server object instance",
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"server_id": {
|
||||||
|
"description": "Server identifier",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"protocol": {
|
||||||
|
"description": "Server protocol",
|
||||||
|
"enum": ["http", "https"]
|
||||||
|
},
|
||||||
|
"host": {
|
||||||
|
"description": "Server host",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"port": {
|
||||||
|
"description": "Server port",
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"user": {
|
||||||
|
"description": "User for auth",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"connected": {
|
||||||
|
"description": "True if controller is connected to the server",
|
||||||
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"version": {
|
||||||
|
"description": "Version of the GNS3 remote server",
|
||||||
|
"type": ["string", "null"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"required": ["server_id", "protocol", "host", "port"]
|
||||||
|
}
|
@ -29,11 +29,11 @@ import types
|
|||||||
import time
|
import time
|
||||||
import atexit
|
import atexit
|
||||||
|
|
||||||
from .web.route import Route
|
from .route import Route
|
||||||
from .web.request_handler import RequestHandler
|
from .request_handler import RequestHandler
|
||||||
from .config import Config
|
from ..config import Config
|
||||||
from .modules import MODULES
|
from ..modules import MODULES
|
||||||
from .modules.port_manager import PortManager
|
from ..modules.port_manager import PortManager
|
||||||
|
|
||||||
# do not delete this import
|
# do not delete this import
|
||||||
import gns3server.handlers
|
import gns3server.handlers
|
||||||
@ -42,7 +42,7 @@ import logging
|
|||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class Server:
|
class WebServer:
|
||||||
|
|
||||||
def __init__(self, host, port):
|
def __init__(self, host, port):
|
||||||
|
|
||||||
@ -61,11 +61,11 @@ class Server:
|
|||||||
:returns: instance of Server
|
:returns: instance of Server
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if not hasattr(Server, "_instance") or Server._instance is None:
|
if not hasattr(WebServer, "_instance") or WebServer._instance is None:
|
||||||
assert host is not None
|
assert host is not None
|
||||||
assert port is not None
|
assert port is not None
|
||||||
Server._instance = Server(host, port)
|
WebServer._instance = WebServer(host, port)
|
||||||
return Server._instance
|
return WebServer._instance
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def _run_application(self, handler, ssl_context=None):
|
def _run_application(self, handler, ssl_context=None):
|
@ -40,6 +40,7 @@ from gns3server.handlers import *
|
|||||||
from gns3server.modules import MODULES
|
from gns3server.modules import MODULES
|
||||||
from gns3server.modules.port_manager import PortManager
|
from gns3server.modules.port_manager import PortManager
|
||||||
from gns3server.modules.project_manager import ProjectManager
|
from gns3server.modules.project_manager import ProjectManager
|
||||||
|
from gns3server.controller import Controller
|
||||||
from tests.handlers.api.base import Query
|
from tests.handlers.api.base import Query
|
||||||
|
|
||||||
|
|
||||||
@ -136,8 +137,14 @@ def ethernet_device():
|
|||||||
return sorted(psutil.net_if_addrs().keys())[0]
|
return sorted(psutil.net_if_addrs().keys())[0]
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def controller():
|
||||||
|
Controller._instance = None
|
||||||
|
return Controller.instance()
|
||||||
|
|
||||||
|
|
||||||
@pytest.yield_fixture(autouse=True)
|
@pytest.yield_fixture(autouse=True)
|
||||||
def run_around_tests(monkeypatch, port_manager):
|
def run_around_tests(monkeypatch, port_manager, controller):
|
||||||
"""
|
"""
|
||||||
This setup a temporay project file environnement around tests
|
This setup a temporay project file environnement around tests
|
||||||
"""
|
"""
|
||||||
@ -151,7 +158,7 @@ def run_around_tests(monkeypatch, port_manager):
|
|||||||
config.set("Server", "project_directory", os.path.join(tmppath, 'projects'))
|
config.set("Server", "project_directory", os.path.join(tmppath, 'projects'))
|
||||||
config.set("Server", "images_path", os.path.join(tmppath, 'images'))
|
config.set("Server", "images_path", os.path.join(tmppath, 'images'))
|
||||||
config.set("Server", "auth", False)
|
config.set("Server", "auth", False)
|
||||||
config.set("Server", "controller", False)
|
config.set("Server", "controller", True)
|
||||||
|
|
||||||
# Prevent executions of the VM if we forgot to mock something
|
# Prevent executions of the VM if we forgot to mock something
|
||||||
config.set("VirtualBox", "vboxmanage_path", tmppath)
|
config.set("VirtualBox", "vboxmanage_path", tmppath)
|
||||||
|
@ -18,19 +18,23 @@
|
|||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from gns3server.controller import Controller
|
from gns3server.controller import Controller
|
||||||
|
from gns3server.controller.server import Server
|
||||||
from gns3server.config import Config
|
from gns3server.config import Config
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def controller():
|
|
||||||
Controller._instance = None
|
|
||||||
return Controller.instance()
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def test_isEnabled(controller):
|
def test_isEnabled(controller):
|
||||||
Config.instance().set("Server", "controller", False)
|
Config.instance().set("Server", "controller", False)
|
||||||
assert not controller.isEnabled()
|
assert not controller.isEnabled()
|
||||||
Config.instance().set("Server", "controller", True)
|
Config.instance().set("Server", "controller", True)
|
||||||
assert controller.isEnabled()
|
assert controller.isEnabled()
|
||||||
|
|
||||||
|
|
||||||
|
def test_addServer(controller):
|
||||||
|
server1 = Server("test1")
|
||||||
|
|
||||||
|
controller.addServer(server1)
|
||||||
|
assert len(controller.servers) == 1
|
||||||
|
controller.addServer(Server("test1"))
|
||||||
|
assert len(controller.servers) == 1
|
||||||
|
controller.addServer(Server("test2"))
|
||||||
|
assert len(controller.servers) == 2
|
||||||
|
48
tests/controller/test_server.py
Normal file
48
tests/controller/test_server.py
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
#!/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 <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from gns3server.controller.server import Server
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def server():
|
||||||
|
return Server("my_server_id", protocol="https", host="example.com", port=84, user="test", password="secure")
|
||||||
|
|
||||||
|
|
||||||
|
def test_init(server):
|
||||||
|
assert server.id == "my_server_id"
|
||||||
|
|
||||||
|
|
||||||
|
def test_json(server):
|
||||||
|
assert server.__json__() == {
|
||||||
|
"server_id": "my_server_id",
|
||||||
|
"protocol": "https",
|
||||||
|
"host": "example.com",
|
||||||
|
"port": 84,
|
||||||
|
"user": "test",
|
||||||
|
"connected": False,
|
||||||
|
"version": None
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def test__eq__(server):
|
||||||
|
assert server != 1
|
||||||
|
assert server == server
|
||||||
|
assert server == Server("my_server_id")
|
||||||
|
assert server != Server("test")
|
36
tests/handlers/api/test_server.py
Normal file
36
tests/handlers/api/test_server.py
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
#!/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 <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
|
def test_server_create(server, controller):
|
||||||
|
|
||||||
|
params = {
|
||||||
|
"server_id": "my_server_id",
|
||||||
|
"protocol": "http",
|
||||||
|
"host": "example.com",
|
||||||
|
"port": 84,
|
||||||
|
"user": "julien",
|
||||||
|
"password": "secure"
|
||||||
|
}
|
||||||
|
response = server.post("/servers", params)
|
||||||
|
assert response.status == 201
|
||||||
|
assert response.route == "/servers"
|
||||||
|
assert response.json["user"] == "julien"
|
||||||
|
assert "password" not in response.json
|
||||||
|
|
||||||
|
assert len(controller.servers) == 1
|
||||||
|
assert controller.servers["my_server_id"].host == "example.com"
|
Loading…
Reference in New Issue
Block a user