mirror of
https://github.com/GNS3/gns3-server.git
synced 2024-12-19 04:47:54 +00:00
Support auth for network V2 hypervisors
This commit is contained in:
parent
c0e452133d
commit
757ee34dac
@ -18,6 +18,7 @@
|
|||||||
import aiohttp
|
import aiohttp
|
||||||
import asyncio
|
import asyncio
|
||||||
import json
|
import json
|
||||||
|
from pkg_resources import parse_version
|
||||||
|
|
||||||
from ..controller.controller_error import ControllerError
|
from ..controller.controller_error import ControllerError
|
||||||
from ..config import Config
|
from ..config import Config
|
||||||
@ -43,8 +44,9 @@ class Hypervisor:
|
|||||||
self._protocol = protocol
|
self._protocol = protocol
|
||||||
self._host = host
|
self._host = host
|
||||||
self._port = port
|
self._port = port
|
||||||
self._user = user
|
self._user = None
|
||||||
self._password = password
|
self._password = None
|
||||||
|
self._setAuth(user, password)
|
||||||
self._connected = False
|
self._connected = False
|
||||||
# The remote hypervisor version
|
# The remote hypervisor version
|
||||||
# TODO: For the moment it's fake we return the controller version
|
# TODO: For the moment it's fake we return the controller version
|
||||||
@ -55,6 +57,17 @@ class Hypervisor:
|
|||||||
if hypervisor_id == "local" and Config.instance().get_section_config("Server")["local"] is False:
|
if hypervisor_id == "local" and Config.instance().get_section_config("Server")["local"] is False:
|
||||||
raise HypervisorError("The local hypervisor is started without --local")
|
raise HypervisorError("The local hypervisor is started without --local")
|
||||||
|
|
||||||
|
def _setAuth(self, user, password):
|
||||||
|
"""
|
||||||
|
Set authentication parameters
|
||||||
|
"""
|
||||||
|
self._user = user
|
||||||
|
self._password = password
|
||||||
|
if self._user and self._password:
|
||||||
|
self._auth = aiohttp.BasicAuth(self._user, self._password)
|
||||||
|
else:
|
||||||
|
self._auth = None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def id(self):
|
def id(self):
|
||||||
"""
|
"""
|
||||||
@ -69,6 +82,22 @@ class Hypervisor:
|
|||||||
"""
|
"""
|
||||||
return self._host
|
return self._host
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user(self):
|
||||||
|
return self._user
|
||||||
|
|
||||||
|
@user.setter
|
||||||
|
def user(self, value):
|
||||||
|
self._setAuth(value, self._password)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def password(self):
|
||||||
|
return self._password
|
||||||
|
|
||||||
|
@user.setter
|
||||||
|
def password(self, value):
|
||||||
|
self._setAuth(self._user, value)
|
||||||
|
|
||||||
def __json__(self):
|
def __json__(self):
|
||||||
return {
|
return {
|
||||||
"hypervisor_id": self._id,
|
"hypervisor_id": self._id,
|
||||||
@ -76,38 +105,55 @@ class Hypervisor:
|
|||||||
"host": self._host,
|
"host": self._host,
|
||||||
"port": self._port,
|
"port": self._port,
|
||||||
"user": self._user,
|
"user": self._user,
|
||||||
"connected": self._connected,
|
"connected": self._connected
|
||||||
"version": self._version
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def httpQuery(self, method, path, data=None):
|
def httpQuery(self, method, path, data=None):
|
||||||
|
if not self._connected:
|
||||||
|
response = yield from self._runHttpQuery("GET", "/version")
|
||||||
|
if "version" not in response.json:
|
||||||
|
raise aiohttp.web.HTTPConflict(text="The server {} is not a GNS3 server".format(self._id))
|
||||||
|
if parse_version(__version__)[:2] != parse_version(response.json["version"])[:2]:
|
||||||
|
raise aiohttp.web.HTTPConflict(text="The server {} versions are not compatible {} != {}".format(self._id, __version__, response.json["version"]))
|
||||||
|
|
||||||
|
self._connected = True
|
||||||
|
return (yield from self._runHttpQuery(method, path, data=data))
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def _runHttpQuery(self, method, path, data=None):
|
||||||
with aiohttp.Timeout(10):
|
with aiohttp.Timeout(10):
|
||||||
with aiohttp.ClientSession() as session:
|
with aiohttp.ClientSession() as session:
|
||||||
url = "{}://{}:{}/v2/hypervisor{}".format(self._protocol, self._host, self._port, path)
|
url = "{}://{}:{}/v2/hypervisor{}".format(self._protocol, self._host, self._port, path)
|
||||||
headers = {'content-type': 'application/json'}
|
headers = {'content-type': 'application/json'}
|
||||||
if hasattr(data, '__json__'):
|
if data:
|
||||||
data = data.__json__()
|
if hasattr(data, '__json__'):
|
||||||
data = json.dumps(data)
|
data = data.__json__()
|
||||||
response = yield from session.request(method, url, headers=headers, data=data)
|
data = json.dumps(data)
|
||||||
|
response = yield from session.request(method, url, headers=headers, data=data, auth=self._auth)
|
||||||
body = yield from response.read()
|
body = yield from response.read()
|
||||||
if body:
|
if body:
|
||||||
body = body.decode()
|
body = body.decode()
|
||||||
if response.status == 400:
|
|
||||||
raise aiohttp.web.HTTPBadRequest(text=body)
|
if response.status >= 300:
|
||||||
elif response.status == 401:
|
if response.status == 400:
|
||||||
raise aiohttp.web.HTTPUnauthorized(text=body)
|
raise aiohttp.web.HTTPBadRequest(text="Bad request {} {}".format(url, body))
|
||||||
elif response.status == 403:
|
elif response.status == 401:
|
||||||
raise aiohttp.web.HTTPForbidden(text=body)
|
raise aiohttp.web.HTTPUnauthorized(text="Invalid authentication for hypervisor {}".format(self.id))
|
||||||
elif response.status == 404:
|
elif response.status == 403:
|
||||||
raise aiohttp.web.HTTPNotFound(text="{} not found on hypervisor".format(url))
|
raise aiohttp.web.HTTPForbidden(text="Forbidden {} {}".format(url, body))
|
||||||
elif response.status == 409:
|
elif response.status == 404:
|
||||||
raise aiohttp.web.HTTPConflict(text=body)
|
raise aiohttp.web.HTTPNotFound(text="{} not found on hypervisor".format(url))
|
||||||
elif response.status >= 300:
|
elif response.status == 409:
|
||||||
raise NotImplemented("{} status code is not supported".format(e.status))
|
raise aiohttp.web.HTTPConflict(text="Conflict {} {}".format(url, body))
|
||||||
if body and len(body):
|
else:
|
||||||
response.json = json.loads(body)
|
raise NotImplemented("{} status code is not supported".format(e.status))
|
||||||
else:
|
if len(body):
|
||||||
|
try:
|
||||||
|
response.json = json.loads(body)
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
raise aiohttp.web.HTTPConflict(text="The server {} is not a GNS3 server".format(self._id))
|
||||||
|
if response.json is None:
|
||||||
response.json = {}
|
response.json = {}
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
@ -87,7 +87,6 @@ class VM:
|
|||||||
"""
|
"""
|
||||||
return self._hypervisor.host
|
return self._hypervisor.host
|
||||||
|
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def create(self):
|
def create(self):
|
||||||
data = copy.copy(self._properties)
|
data = copy.copy(self._properties)
|
||||||
|
@ -24,12 +24,14 @@ from unittest.mock import patch, MagicMock
|
|||||||
from gns3server.controller.project import Project
|
from gns3server.controller.project import Project
|
||||||
from gns3server.controller.hypervisor import Hypervisor, HypervisorError
|
from gns3server.controller.hypervisor import Hypervisor, HypervisorError
|
||||||
from gns3server.version import __version__
|
from gns3server.version import __version__
|
||||||
from tests.utils import asyncio_patch
|
from tests.utils import asyncio_patch, AsyncioMagicMock
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def hypervisor():
|
def hypervisor():
|
||||||
return Hypervisor("my_hypervisor_id", protocol="https", host="example.com", port=84, user="test", password="secure")
|
hypervisor = Hypervisor("my_hypervisor_id", protocol="https", host="example.com", port=84)
|
||||||
|
hypervisor._connected = True
|
||||||
|
return hypervisor
|
||||||
|
|
||||||
|
|
||||||
def test_init(hypervisor):
|
def test_init(hypervisor):
|
||||||
@ -56,7 +58,66 @@ def test_hypervisor_httpQuery(hypervisor, async_run):
|
|||||||
response.status = 200
|
response.status = 200
|
||||||
|
|
||||||
async_run(hypervisor.post("/projects", {"a": "b"}))
|
async_run(hypervisor.post("/projects", {"a": "b"}))
|
||||||
mock.assert_called_with("POST", "https://example.com:84/v2/hypervisor/projects", data='{"a": "b"}', headers={'content-type': 'application/json'})
|
mock.assert_called_with("POST", "https://example.com:84/v2/hypervisor/projects", data='{"a": "b"}', headers={'content-type': 'application/json'}, auth=None)
|
||||||
|
assert hypervisor._auth is None
|
||||||
|
|
||||||
|
|
||||||
|
def test_hypervisor_httpQueryAuth(hypervisor, async_run):
|
||||||
|
response = MagicMock()
|
||||||
|
with asyncio_patch("aiohttp.ClientSession.request", return_value=response) as mock:
|
||||||
|
response.status = 200
|
||||||
|
|
||||||
|
hypervisor.user = "root"
|
||||||
|
hypervisor.password = "toor"
|
||||||
|
async_run(hypervisor.post("/projects", {"a": "b"}))
|
||||||
|
mock.assert_called_with("POST", "https://example.com:84/v2/hypervisor/projects", data='{"a": "b"}', headers={'content-type': 'application/json'}, auth=hypervisor._auth)
|
||||||
|
assert hypervisor._auth.login == "root"
|
||||||
|
assert hypervisor._auth.password == "toor"
|
||||||
|
|
||||||
|
|
||||||
|
def test_hypervisor_httpQueryNotConnected(hypervisor, async_run):
|
||||||
|
hypervisor._connected = False
|
||||||
|
response = AsyncioMagicMock()
|
||||||
|
response.read = AsyncioMagicMock(return_value = json.dumps({"version": __version__}).encode())
|
||||||
|
response.status = 200
|
||||||
|
with asyncio_patch("aiohttp.ClientSession.request", return_value=response) as mock:
|
||||||
|
async_run(hypervisor.post("/projects", {"a": "b"}))
|
||||||
|
mock.assert_any_call("GET", "https://example.com:84/v2/hypervisor/version", headers={'content-type': 'application/json'}, data=None, auth=None)
|
||||||
|
mock.assert_any_call("POST", "https://example.com:84/v2/hypervisor/projects", data='{"a": "b"}', headers={'content-type': 'application/json'}, auth=None)
|
||||||
|
assert hypervisor._connected
|
||||||
|
|
||||||
|
|
||||||
|
def test_hypervisor_httpQueryNotConnectedInvalidVersion(hypervisor, async_run):
|
||||||
|
hypervisor._connected = False
|
||||||
|
response = AsyncioMagicMock()
|
||||||
|
response.read = AsyncioMagicMock(return_value = json.dumps({"version": "1.42.4"}).encode())
|
||||||
|
response.status = 200
|
||||||
|
with asyncio_patch("aiohttp.ClientSession.request", return_value=response) as mock:
|
||||||
|
with pytest.raises(aiohttp.web.HTTPConflict):
|
||||||
|
async_run(hypervisor.post("/projects", {"a": "b"}))
|
||||||
|
mock.assert_any_call("GET", "https://example.com:84/v2/hypervisor/version", headers={'content-type': 'application/json'}, data=None, auth=None)
|
||||||
|
|
||||||
|
|
||||||
|
def test_hypervisor_httpQueryNotConnectedNonGNS3Server(hypervisor, async_run):
|
||||||
|
hypervisor._connected = False
|
||||||
|
response = AsyncioMagicMock()
|
||||||
|
response.read = AsyncioMagicMock(return_value = b'Blocked by super antivirus')
|
||||||
|
response.status = 200
|
||||||
|
with asyncio_patch("aiohttp.ClientSession.request", return_value=response) as mock:
|
||||||
|
with pytest.raises(aiohttp.web.HTTPConflict):
|
||||||
|
async_run(hypervisor.post("/projects", {"a": "b"}))
|
||||||
|
mock.assert_any_call("GET", "https://example.com:84/v2/hypervisor/version", headers={'content-type': 'application/json'}, data=None, auth=None)
|
||||||
|
|
||||||
|
|
||||||
|
def test_hypervisor_httpQueryNotConnectedNonGNS3Server2(hypervisor, async_run):
|
||||||
|
hypervisor._connected = False
|
||||||
|
response = AsyncioMagicMock()
|
||||||
|
response.read = AsyncioMagicMock(return_value = b'{}')
|
||||||
|
response.status = 200
|
||||||
|
with asyncio_patch("aiohttp.ClientSession.request", return_value=response) as mock:
|
||||||
|
with pytest.raises(aiohttp.web.HTTPConflict):
|
||||||
|
async_run(hypervisor.post("/projects", {"a": "b"}))
|
||||||
|
mock.assert_any_call("GET", "https://example.com:84/v2/hypervisor/version", headers={'content-type': 'application/json'}, data=None, auth=None)
|
||||||
|
|
||||||
|
|
||||||
def test_hypervisor_httpQueryError(hypervisor, async_run):
|
def test_hypervisor_httpQueryError(hypervisor, async_run):
|
||||||
@ -75,16 +136,16 @@ def test_hypervisor_httpQuery_project(hypervisor, async_run):
|
|||||||
|
|
||||||
project = Project()
|
project = Project()
|
||||||
async_run(hypervisor.post("/projects", project))
|
async_run(hypervisor.post("/projects", project))
|
||||||
mock.assert_called_with("POST", "https://example.com:84/v2/hypervisor/projects", data=json.dumps(project.__json__()), headers={'content-type': 'application/json'})
|
mock.assert_called_with("POST", "https://example.com:84/v2/hypervisor/projects", data=json.dumps(project.__json__()), headers={'content-type': 'application/json'}, auth=None)
|
||||||
|
|
||||||
|
|
||||||
def test_json(hypervisor):
|
def test_json(hypervisor):
|
||||||
|
hypervisor.user = "test"
|
||||||
assert hypervisor.__json__() == {
|
assert hypervisor.__json__() == {
|
||||||
"hypervisor_id": "my_hypervisor_id",
|
"hypervisor_id": "my_hypervisor_id",
|
||||||
"protocol": "https",
|
"protocol": "https",
|
||||||
"host": "example.com",
|
"host": "example.com",
|
||||||
"port": 84,
|
"port": 84,
|
||||||
"user": "test",
|
"user": "test",
|
||||||
"connected": False,
|
"connected": True
|
||||||
"version": __version__
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user