API for deleting a list

This commit is contained in:
Julien Duponchelle 2016-03-14 17:40:27 +01:00
parent 9771b33b68
commit 9062490be3
No known key found for this signature in database
GPG Key ID: F1E2485547D4595D
7 changed files with 105 additions and 8 deletions

View File

@ -37,6 +37,20 @@ class Link:
"port_number": port_number "port_number": port_number
}) })
@asyncio.coroutine
def create(self):
"""
Create the link
"""
raise NotImplemented
@asyncio.coroutine
def delete(self):
"""
Delete the link
"""
raise NotImplemented
@property @property
def id(self): def id(self):
return self._id return self._id

View File

@ -22,11 +22,16 @@ from .link import Link
class UDPLink(Link): class UDPLink(Link):
def __init__(self, project):
super().__init__(project)
@asyncio.coroutine @asyncio.coroutine
def create(self): def create(self):
""" """
Create the link on the VMs Create the link on the VMs
""" """
vm1 = self._vms[0]["vm"] vm1 = self._vms[0]["vm"]
adapter_number1 = self._vms[0]["adapter_number"] adapter_number1 = self._vms[0]["adapter_number"]
port_number1 = self._vms[0]["port_number"] port_number1 = self._vms[0]["port_number"]
@ -36,23 +41,39 @@ class UDPLink(Link):
# Reserve a UDP port on both side # Reserve a UDP port on both side
response = yield from vm1.hypervisor.post("/projects/{}/ports/udp".format(self._project.id)) response = yield from vm1.hypervisor.post("/projects/{}/ports/udp".format(self._project.id))
vm1_port = response.json["udp_port"] self._vm1_port = response.json["udp_port"]
response = yield from vm2.hypervisor.post("/projects/{}/ports/udp".format(self._project.id)) response = yield from vm2.hypervisor.post("/projects/{}/ports/udp".format(self._project.id))
vm2_port = response.json["udp_port"] self._vm2_port = response.json["udp_port"]
# Create the tunnel on both side # Create the tunnel on both side
data = { data = {
"lport": vm1_port, "lport": self._vm1_port,
"rhost": vm2.hypervisor.host, "rhost": vm2.hypervisor.host,
"rport": vm2_port, "rport": self._vm2_port,
"type": "nio_udp" "type": "nio_udp"
} }
yield from vm1.post("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number1, port_number=port_number1), data=data) yield from vm1.post("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number1, port_number=port_number1), data=data)
data = { data = {
"lport": vm2_port, "lport": self._vm2_port,
"rhost": vm1.hypervisor.host, "rhost": vm1.hypervisor.host,
"rport": vm1_port, "rport": self._vm1_port,
"type": "nio_udp" "type": "nio_udp"
} }
yield from vm2.post("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number2, port_number=port_number2), data=data) yield from vm2.post("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number2, port_number=port_number2), data=data)
@asyncio.coroutine
def delete(self):
"""
Delete the link and free the ressources
"""
vm1 = self._vms[0]["vm"]
adapter_number1 = self._vms[0]["adapter_number"]
port_number1 = self._vms[0]["port_number"]
vm2 = self._vms[1]["vm"]
adapter_number2 = self._vms[1]["adapter_number"]
port_number2 = self._vms[1]["port_number"]
yield from vm1.delete("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number1, port_number=port_number1))
yield from vm2.delete("/adapters/{adapter_number}/ports/{port_number}/nio".format(adapter_number=adapter_number2, port_number=port_number2))

View File

@ -92,6 +92,13 @@ class VM:
""" """
return (yield from self._hypervisor.post("/projects/{}/{}/vms/{}{}".format(self._project.id, self._vm_type, self._id, path), data=data)) return (yield from self._hypervisor.post("/projects/{}/{}/vms/{}{}".format(self._project.id, self._vm_type, self._id, path), data=data))
@asyncio.coroutine
def delete(self, path):
"""
HTTP post on the VM
"""
return (yield from self._hypervisor.delete("/projects/{}/{}/vms/{}{}".format(self._project.id, self._vm_type, self._id, path)))
def __json__(self): def __json__(self):
return { return {
"hypervisor_id": self._hypervisor.id, "hypervisor_id": self._hypervisor.id,

View File

@ -51,3 +51,24 @@ class LinkHandler:
yield from link.create() yield from link.create()
response.set_status(201) response.set_status(201)
response.json(link) response.json(link)
@classmethod
@Route.delete(
r"/projects/{project_id}/links/{link_id}",
parameters={
"project_id": "UUID for the project",
"link_id": "UUID of the link"
},
status_codes={
201: "Link deleted",
400: "Invalid request"
},
description="Delete a link instance")
def delete(request, response):
controller = Controller.instance()
project = controller.getProject(request.match_info["project_id"])
link = project.getLink(request.match_info["link_id"])
yield from link.delete()
response.set_status(201)
response.json(link)

View File

@ -17,6 +17,7 @@
import pytest import pytest
import asyncio import asyncio
import aiohttp
from unittest.mock import MagicMock from unittest.mock import MagicMock
from gns3server.controller.project import Project from gns3server.controller.project import Project
@ -79,3 +80,21 @@ def test_create(async_run, project):
"rport": 1024, "rport": 1024,
"type": "nio_udp" "type": "nio_udp"
}) })
def test_delete(async_run, project):
hypervisor1 = MagicMock()
hypervisor2 = MagicMock()
vm1 = VM(project, hypervisor1, vm_type="vpcs")
vm2 = VM(project, hypervisor2, vm_type="vpcs")
link = UDPLink(project)
async_run(link.addVM(vm1, 0, 4))
async_run(link.addVM(vm2, 3, 1))
async_run(link.delete())
hypervisor1.delete.assert_any_call("/projects/{}/vpcs/vms/{}/adapters/0/ports/4/nio".format(project.id, vm1.id))
hypervisor2.delete.assert_any_call("/projects/{}/vpcs/vms/{}/adapters/3/ports/1/nio".format(project.id, vm2.id))

View File

@ -78,3 +78,8 @@ def test_create(vm, hypervisor, project, async_run):
def test_post(vm, hypervisor, async_run): def test_post(vm, hypervisor, async_run):
async_run(vm.post("/test", {"a": "b"})) async_run(vm.post("/test", {"a": "b"}))
hypervisor.post.assert_called_with("/projects/{}/vpcs/vms/{}/test".format(vm.project.id, vm.id), data={"a": "b"}) hypervisor.post.assert_called_with("/projects/{}/vpcs/vms/{}/test".format(vm.project.id, vm.id), data={"a": "b"})
def test_delete(vm, hypervisor, async_run):
async_run(vm.delete("/test"))
hypervisor.delete.assert_called_with("/projects/{}/vpcs/vms/{}/test".format(vm.project.id, vm.id))

View File

@ -32,6 +32,7 @@ from tests.utils import asyncio_patch
from gns3server.handlers.api.controller.project_handler import ProjectHandler from gns3server.handlers.api.controller.project_handler import ProjectHandler
from gns3server.controller import Controller from gns3server.controller import Controller
from gns3server.controller.vm import VM from gns3server.controller.vm import VM
from gns3server.controller.link import Link
@pytest.fixture @pytest.fixture
@ -51,8 +52,7 @@ def test_create_link(http_controller, tmpdir, project, hypervisor, async_run):
vm1 = async_run(project.addVM(hypervisor, None)) vm1 = async_run(project.addVM(hypervisor, None))
vm2 = async_run(project.addVM(hypervisor, None)) vm2 = async_run(project.addVM(hypervisor, None))
with asyncio_patch("gns3server.controller.udp_link.UDPLink.create") as mock:
with asyncio_patch("gns3server.controller.udp_link.UDPLink.create"):
response = http_controller.post("/projects/{}/links".format(project.id), { response = http_controller.post("/projects/{}/links".format(project.id), {
"vms": [ "vms": [
{ {
@ -67,6 +67,16 @@ def test_create_link(http_controller, tmpdir, project, hypervisor, async_run):
} }
] ]
}, example=True) }, example=True)
assert mock.called
assert response.status == 201 assert response.status == 201
assert response.json["link_id"] is not None assert response.json["link_id"] is not None
assert len(response.json["vms"]) == 2 assert len(response.json["vms"]) == 2
def test_delete_link(http_controller, tmpdir, project, hypervisor, async_run):
link = Link(project)
project._links = {link.id: link}
with asyncio_patch("gns3server.controller.udp_link.Link.delete"):
response = http_controller.delete("/projects/{}/links/{}".format(project.id, link.id), example=True)
assert response.status == 201