1
0
mirror of https://github.com/GNS3/gns3-server synced 2024-11-30 20:28:08 +00:00

Serialize NIO

This commit is contained in:
Julien Duponchelle 2015-01-16 17:09:45 +01:00
parent 0cdc1c3042
commit bf6f62e629
11 changed files with 65 additions and 26 deletions

View File

@ -1,22 +1,25 @@
curl -i -xPOST 'http://localhost:8000/vpcs/{vpcs_id}/ports/{port_id}/nio' -d '{"local_file": "/tmp/test", "remote_file": "/tmp/remote", "type": "nio_unix"}' curl -i -xPOST 'http://localhost:8000/vpcs/{vpcs_id}/ports/{port_id}/nio' -d '{"lport": 4242, "rhost": "127.0.0.1", "rport": 4343, "type": "nio_udp"}'
POST /vpcs/{vpcs_id}/ports/{port_id}/nio HTTP/1.1 POST /vpcs/{vpcs_id}/ports/{port_id}/nio HTTP/1.1
{ {
"local_file": "/tmp/test", "lport": 4242,
"remote_file": "/tmp/remote", "rhost": "127.0.0.1",
"type": "nio_unix" "rport": 4343,
"type": "nio_udp"
} }
HTTP/1.1 404 HTTP/1.1 200
CONNECTION: close CONNECTION: close
CONTENT-LENGTH: 59 CONTENT-LENGTH: 89
CONTENT-TYPE: application/json CONTENT-TYPE: application/json
DATE: Thu, 08 Jan 2015 16:09:15 GMT DATE: Thu, 08 Jan 2015 16:09:15 GMT
SERVER: Python/3.4 aiohttp/0.13.1 SERVER: Python/3.4 aiohttp/0.13.1
X-ROUTE: /vpcs/{vpcs_id}/ports/{port_id}/nio X-ROUTE: /vpcs/{vpcs_id}/ports/{port_id}/nio
{ {
"message": "ID 42 doesn't exist", "lport": 4242,
"status": 404 "rhost": "127.0.0.1",
"rport": 4343,
"type": "nio_udp"
} }

View File

@ -19,7 +19,7 @@
from ..web.route import Route from ..web.route import Route
from ..schemas.vpcs import VPCS_CREATE_SCHEMA from ..schemas.vpcs import VPCS_CREATE_SCHEMA
from ..schemas.vpcs import VPCS_OBJECT_SCHEMA from ..schemas.vpcs import VPCS_OBJECT_SCHEMA
from ..schemas.vpcs import VPCS_ADD_NIO_SCHEMA from ..schemas.vpcs import VPCS_NIO_SCHEMA
from ..modules.vpcs import VPCS from ..modules.vpcs import VPCS
@ -48,7 +48,8 @@ class VPCSHandler(object):
"vpcs_id": "Id of VPCS instance" "vpcs_id": "Id of VPCS instance"
}, },
status_codes={ status_codes={
201: "Success of creation of VPCS", 200: "Success of starting VPCS",
404: "If VPCS doesn't exist"
}, },
description="Start VPCS", description="Start VPCS",
) )
@ -64,7 +65,8 @@ class VPCSHandler(object):
"vpcs_id": "Id of VPCS instance" "vpcs_id": "Id of VPCS instance"
}, },
status_codes={ status_codes={
201: "Success of stopping VPCS", 200: "Success of stopping VPCS",
404: "If VPCS doesn't exist"
}, },
description="Stop VPCS", description="Stop VPCS",
) )
@ -81,17 +83,17 @@ class VPCSHandler(object):
}, },
status_codes={ status_codes={
201: "Success of creation of NIO", 201: "Success of creation of NIO",
409: "Conflict" 404: "If VPCS doesn't exist"
}, },
description="ADD NIO to a VPCS", description="ADD NIO to a VPCS",
input=VPCS_ADD_NIO_SCHEMA) input=VPCS_NIO_SCHEMA,
output=VPCS_NIO_SCHEMA)
def create_nio(request, response): def create_nio(request, response):
# TODO: raise 404 if VPCS not found GET VM can raise an exeption
# TODO: response with nio # TODO: response with nio
vpcs_manager = VPCS.instance() vpcs_manager = VPCS.instance()
vm = vpcs_manager.get_vm(int(request.match_info['vpcs_id'])) vm = vpcs_manager.get_vm(int(request.match_info['vpcs_id']))
vm.port_add_nio_binding(int(request.match_info['port_id']), request.json) nio = vm.port_add_nio_binding(int(request.match_info['port_id']), request.json)
response.json({'name': "PC 2", "vpcs_id": 42, "console": 4242}) response.json(nio)

View File

@ -39,7 +39,7 @@ class BaseManager:
:returns: instance of Manager :returns: instance of Manager
""" """
if not hasattr(cls, "_instance"): if not hasattr(cls, "_instance") or cls._instance is None:
cls._instance = cls() cls._instance = cls()
return cls._instance return cls._instance

View File

@ -44,3 +44,6 @@ class NIO_TAP(object):
def __str__(self): def __str__(self):
return "NIO TAP" return "NIO TAP"
def __json__(self):
return {"type": "nio_tap", "tap_device": self._tap_device}

View File

@ -70,3 +70,6 @@ class NIO_UDP(object):
def __str__(self): def __str__(self):
return "NIO UDP" return "NIO UDP"
def __json__(self):
return {"type": "nio_udp", "lport": self._lport, "rport": self._rport, "rhost": self._rhost}

View File

@ -265,7 +265,6 @@ class VPCSDevice(BaseVM):
nio = NIO_UDP(lport, rhost, rport) nio = NIO_UDP(lport, rhost, rport)
elif nio_settings["type"] == "nio_tap": elif nio_settings["type"] == "nio_tap":
tap_device = nio_settings["tap_device"] tap_device = nio_settings["tap_device"]
print(has_privileged_access)
if not has_privileged_access(self._path): if not has_privileged_access(self._path):
raise VPCSError("{} has no privileged access to {}.".format(self._path, tap_device)) raise VPCSError("{} has no privileged access to {}.".format(self._path, tap_device))
nio = NIO_TAP(tap_device) nio = NIO_TAP(tap_device)

View File

@ -42,7 +42,7 @@ VPCS_CREATE_SCHEMA = {
} }
VPCS_ADD_NIO_SCHEMA = { VPCS_NIO_SCHEMA = {
"$schema": "http://json-schema.org/draft-04/schema#", "$schema": "http://json-schema.org/draft-04/schema#",
"description": "Request validation to add a NIO for a VPCS instance", "description": "Request validation to add a NIO for a VPCS instance",
"type": "object", "type": "object",

View File

@ -18,7 +18,9 @@
import json import json
import jsonschema import jsonschema
import aiohttp.web import aiohttp.web
import logging
log = logging.getLogger(__name__)
class Response(aiohttp.web.Response): class Response(aiohttp.web.Response):
@ -29,13 +31,22 @@ class Response(aiohttp.web.Response):
headers['X-Route'] = self._route headers['X-Route'] = self._route
super().__init__(headers=headers, **kwargs) super().__init__(headers=headers, **kwargs)
"""
Set the response content type to application/json and serialize
the content.
:param anwser The response as a Python object
"""
def json(self, answer): def json(self, answer):
"""Pass a Python object and return a JSON as answer""" """Pass a Python object and return a JSON as answer"""
self.content_type = "application/json" self.content_type = "application/json"
if hasattr(answer, '__json__'):
answer = answer.__json__()
if self._output_schema is not None: if self._output_schema is not None:
try: try:
jsonschema.validate(answer, self._output_schema) jsonschema.validate(answer, self._output_schema)
except jsonschema.ValidationError as e: except jsonschema.ValidationError as e:
log.error("Invalid output schema")
raise aiohttp.web.HTTPBadRequest(text="{}".format(e)) raise aiohttp.web.HTTPBadRequest(text="{}".format(e))
self.body = json.dumps(answer, indent=4, sort_keys=True).encode('utf-8') self.body = json.dumps(answer, indent=4, sort_keys=True).encode('utf-8')

View File

@ -19,6 +19,9 @@ import json
import jsonschema import jsonschema
import asyncio import asyncio
import aiohttp import aiohttp
import logging
log = logging.getLogger(__name__)
from ..modules.vm_error import VMError from ..modules.vm_error import VMError
from .response import Response from .response import Response
@ -37,6 +40,7 @@ def parse_request(request, input_schema):
try: try:
jsonschema.validate(request.json, input_schema) jsonschema.validate(request.json, input_schema)
except jsonschema.ValidationError as e: except jsonschema.ValidationError as e:
log.error("Invalid input schema")
raise aiohttp.web.HTTPBadRequest(text="Request is not {} '{}' in schema: {}".format( raise aiohttp.web.HTTPBadRequest(text="Request is not {} '{}' in schema: {}".format(
e.validator, e.validator,
e.validator_value, e.validator_value,

View File

@ -21,7 +21,7 @@ It's also used for unittest the HTTP implementation.
""" """
from tests.utils import asyncio_patch from tests.utils import asyncio_patch
from tests.api.base import server, loop from tests.api.base import server, loop, port_manager
from gns3server.version import __version__ from gns3server.version import __version__

View File

@ -15,6 +15,7 @@
# 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/>.
from unittest.mock import patch
from tests.api.base import server, loop, port_manager from tests.api.base import server, loop, port_manager
from tests.utils import asyncio_patch from tests.utils import asyncio_patch
from gns3server import modules from gns3server import modules
@ -29,13 +30,26 @@ def test_vpcs_create(server):
assert response.json['vpcs_id'] == 84 assert response.json['vpcs_id'] == 84
def test_vpcs_nio_create(server): def test_vpcs_nio_create_udp(server):
response = server.post('/vpcs/42/ports/0/nio', { vm = server.post('/vpcs', {'name': 'PC TEST 1'})
'type': 'nio_unix', response = server.post('/vpcs/{}/ports/0/nio'.format(vm.json["vpcs_id"]), {
'local_file': '/tmp/test', 'type': 'nio_udp',
'remote_file': '/tmp/remote' 'lport': 4242,
'rport': 4343,
'rhost': '127.0.0.1'
}, },
example=True) example=True)
assert response.status == 200 assert response.status == 200
assert response.route == '/vpcs/{vpcs_id}/ports/{port_id}/nio' assert response.route == '/vpcs/{vpcs_id}/ports/{port_id}/nio'
assert response.json['name'] == 'PC 2' assert response.json['type'] == 'nio_udp'
@patch("gns3server.modules.vpcs.vpcs_device.has_privileged_access", return_value=True)
def test_vpcs_nio_create_tap(mock, server):
vm = server.post('/vpcs', {'name': 'PC TEST 1'})
response = server.post('/vpcs/{}/ports/0/nio'.format(vm.json["vpcs_id"]), {
'type': 'nio_tap',
'tap_device': 'test',
})
assert response.status == 200
assert response.route == '/vpcs/{vpcs_id}/ports/{port_id}/nio'
assert response.json['type'] == 'nio_tap'