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:
|
||||
"""The controller manage multiple gns3 servers"""
|
||||
|
||||
def __init__(self):
|
||||
self._servers = {}
|
||||
|
||||
def isEnabled(self):
|
||||
"""
|
||||
:returns: True if current instance is the controller
|
||||
@ -29,6 +32,19 @@ class 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
|
||||
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
|
||||
# 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 ...config import Config
|
||||
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
|
||||
|
||||
log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
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
|
||||
@Route.post(
|
||||
@ -60,7 +83,7 @@ class ServerHandler:
|
||||
continue
|
||||
|
||||
# then shutdown the server itself
|
||||
from gns3server.server import Server
|
||||
server = Server.instance()
|
||||
from gns3server.web.web_server import WebServer
|
||||
server = WebServer.instance()
|
||||
asyncio.async(server.shutdown_server())
|
||||
response.set_status(201)
|
||||
|
@ -28,7 +28,7 @@ import locale
|
||||
import argparse
|
||||
import asyncio
|
||||
|
||||
from gns3server.server import Server
|
||||
from gns3server.web.web_server import WebServer
|
||||
from gns3server.web.logger import init_logger
|
||||
from gns3server.version import __version__
|
||||
from gns3server.config import Config
|
||||
@ -233,7 +233,7 @@ def run():
|
||||
host = server_config["host"]
|
||||
port = int(server_config["port"])
|
||||
|
||||
server = Server.instance(host, port)
|
||||
server = WebServer.instance(host, port)
|
||||
try:
|
||||
server.run()
|
||||
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 atexit
|
||||
|
||||
from .web.route import Route
|
||||
from .web.request_handler import RequestHandler
|
||||
from .config import Config
|
||||
from .modules import MODULES
|
||||
from .modules.port_manager import PortManager
|
||||
from .route import Route
|
||||
from .request_handler import RequestHandler
|
||||
from ..config import Config
|
||||
from ..modules import MODULES
|
||||
from ..modules.port_manager import PortManager
|
||||
|
||||
# do not delete this import
|
||||
import gns3server.handlers
|
||||
@ -42,7 +42,7 @@ import logging
|
||||
log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class Server:
|
||||
class WebServer:
|
||||
|
||||
def __init__(self, host, port):
|
||||
|
||||
@ -61,11 +61,11 @@ class 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 port is not None
|
||||
Server._instance = Server(host, port)
|
||||
return Server._instance
|
||||
WebServer._instance = WebServer(host, port)
|
||||
return WebServer._instance
|
||||
|
||||
@asyncio.coroutine
|
||||
def _run_application(self, handler, ssl_context=None):
|
@ -40,6 +40,7 @@ from gns3server.handlers import *
|
||||
from gns3server.modules import MODULES
|
||||
from gns3server.modules.port_manager import PortManager
|
||||
from gns3server.modules.project_manager import ProjectManager
|
||||
from gns3server.controller import Controller
|
||||
from tests.handlers.api.base import Query
|
||||
|
||||
|
||||
@ -136,8 +137,14 @@ def ethernet_device():
|
||||
return sorted(psutil.net_if_addrs().keys())[0]
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def controller():
|
||||
Controller._instance = None
|
||||
return Controller.instance()
|
||||
|
||||
|
||||
@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
|
||||
"""
|
||||
@ -151,7 +158,7 @@ def run_around_tests(monkeypatch, port_manager):
|
||||
config.set("Server", "project_directory", os.path.join(tmppath, 'projects'))
|
||||
config.set("Server", "images_path", os.path.join(tmppath, 'images'))
|
||||
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
|
||||
config.set("VirtualBox", "vboxmanage_path", tmppath)
|
||||
|
@ -18,19 +18,23 @@
|
||||
import pytest
|
||||
|
||||
from gns3server.controller import Controller
|
||||
from gns3server.controller.server import Server
|
||||
from gns3server.config import Config
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def controller():
|
||||
Controller._instance = None
|
||||
return Controller.instance()
|
||||
|
||||
|
||||
|
||||
def test_isEnabled(controller):
|
||||
Config.instance().set("Server", "controller", False)
|
||||
assert not controller.isEnabled()
|
||||
Config.instance().set("Server", "controller", True)
|
||||
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