Support auth for network V2 hypervisors

This commit is contained in:
Julien Duponchelle 2016-03-16 15:55:07 +01:00
parent c0e452133d
commit 757ee34dac
No known key found for this signature in database
GPG Key ID: F1E2485547D4595D
3 changed files with 136 additions and 30 deletions

View File

@ -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

View File

@ -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)

View File

@ -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__
} }