2015-01-20 03:30:57 +02:00
# -*- 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/>.
2015-01-24 03:33:49 +02:00
import os
2015-07-22 07:58:28 +03:00
2015-04-27 23:19:17 +03:00
from aiohttp . web import HTTPConflict
2015-02-23 12:27:07 +02:00
from . . . web . route import Route
2015-04-27 23:19:17 +03:00
from . . . schemas . nio import NIO_SCHEMA
2015-02-23 12:27:07 +02:00
from . . . schemas . virtualbox import VBOX_CREATE_SCHEMA
from . . . schemas . virtualbox import VBOX_UPDATE_SCHEMA
from . . . schemas . virtualbox import VBOX_OBJECT_SCHEMA
2016-02-09 17:07:33 +02:00
from . . . schemas . vm import VM_CAPTURE_SCHEMA
2015-02-23 12:27:07 +02:00
from . . . modules . virtualbox import VirtualBox
from . . . modules . project_manager import ProjectManager
2015-01-20 03:30:57 +02:00
class VirtualBoxHandler :
2015-01-31 23:34:49 +02:00
2015-01-20 03:30:57 +02:00
"""
API entry points for VirtualBox .
"""
2015-01-24 02:57:54 +02:00
@classmethod
@Route.get (
2015-02-05 02:13:35 +02:00
r " /virtualbox/vms " ,
2015-01-24 02:57:54 +02:00
status_codes = {
200 : " Success " ,
} ,
description = " Get all VirtualBox VMs available " )
2015-11-09 13:14:25 +02:00
def index ( request , response ) :
2015-01-24 02:57:54 +02:00
vbox_manager = VirtualBox . instance ( )
2015-04-13 15:33:13 +03:00
vms = yield from vbox_manager . list_images ( )
2015-01-24 02:57:54 +02:00
response . json ( vms )
2015-01-20 03:30:57 +02:00
@classmethod
@Route.post (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms " ,
2015-02-05 02:13:35 +02:00
parameters = {
" project_id " : " UUID for the project "
} ,
2015-01-20 03:30:57 +02:00
status_codes = {
2015-01-24 01:38:46 +02:00
201 : " Instance created " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-20 03:30:57 +02:00
409 : " Conflict "
} ,
description = " Create a new VirtualBox VM instance " ,
input = VBOX_CREATE_SCHEMA ,
output = VBOX_OBJECT_SCHEMA )
def create ( request , response ) :
vbox_manager = VirtualBox . instance ( )
2015-02-03 03:56:13 +02:00
vm = yield from vbox_manager . create_vm ( request . json . pop ( " name " ) ,
2015-02-05 02:13:35 +02:00
request . match_info [ " project_id " ] ,
2015-02-04 22:48:29 +02:00
request . json . get ( " vm_id " ) ,
2015-02-03 03:56:13 +02:00
request . json . pop ( " vmname " ) ,
request . json . pop ( " linked_clone " ) ,
2015-03-03 02:28:28 +02:00
console = request . json . get ( " console " , None ) ,
2015-01-31 04:36:05 +02:00
adapters = request . json . get ( " adapters " , 0 ) )
2015-03-03 02:28:28 +02:00
if " enable_remote_console " in request . json :
yield from vm . set_enable_remote_console ( request . json . pop ( " enable_remote_console " ) )
2015-03-14 01:13:36 +02:00
if " ram " in request . json :
ram = request . json . pop ( " ram " )
if ram != vm . ram :
yield from vm . set_ram ( ram )
2015-01-31 04:36:05 +02:00
for name , value in request . json . items ( ) :
2015-04-14 15:24:13 +03:00
if name != " vm_id " :
if hasattr ( vm , name ) and getattr ( vm , name ) != value :
setattr ( vm , name , value )
2015-01-31 04:36:05 +02:00
2015-01-21 23:01:15 +02:00
response . set_status ( 201 )
2015-01-22 00:21:15 +02:00
response . json ( vm )
2015-01-20 03:30:57 +02:00
2015-01-24 01:38:46 +02:00
@classmethod
@Route.get (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} " ,
2015-01-24 01:38:46 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-24 01:38:46 +02:00
} ,
status_codes = {
200 : " Success " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
} ,
description = " Get a VirtualBox VM instance " ,
output = VBOX_OBJECT_SCHEMA )
def show ( request , response ) :
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-01-24 01:38:46 +02:00
response . json ( vm )
@classmethod
@Route.put (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} " ,
2015-01-24 01:38:46 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-24 01:38:46 +02:00
} ,
status_codes = {
200 : " Instance updated " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist " ,
409 : " Conflict "
} ,
description = " Update a VirtualBox VM instance " ,
input = VBOX_UPDATE_SCHEMA ,
output = VBOX_OBJECT_SCHEMA )
def update ( request , response ) :
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-01-24 01:38:46 +02:00
2015-03-08 20:32:36 +02:00
if " vmname " in request . json :
vmname = request . json . pop ( " vmname " )
if vmname != vm . vmname :
yield from vm . set_vmname ( vmname )
2015-03-03 02:28:28 +02:00
if " enable_remote_console " in request . json :
yield from vm . set_enable_remote_console ( request . json . pop ( " enable_remote_console " ) )
2015-03-05 03:24:15 +02:00
if " adapters " in request . json :
adapters = int ( request . json . pop ( " adapters " ) )
if adapters != vm . adapters :
yield from vm . set_adapters ( adapters )
2015-03-14 01:13:36 +02:00
if " ram " in request . json :
ram = request . json . pop ( " ram " )
if ram != vm . ram :
yield from vm . set_ram ( ram )
2015-01-24 01:38:46 +02:00
for name , value in request . json . items ( ) :
if hasattr ( vm , name ) and getattr ( vm , name ) != value :
setattr ( vm , name , value )
2015-02-07 02:31:13 +02:00
2015-01-24 01:38:46 +02:00
response . json ( vm )
@classmethod
@Route.delete (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} " ,
2015-01-24 01:38:46 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-24 01:38:46 +02:00
} ,
status_codes = {
204 : " Instance deleted " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
} ,
description = " Delete a VirtualBox VM instance " )
def delete ( request , response ) :
2015-02-05 02:13:35 +02:00
# check the project_id exists
ProjectManager . instance ( ) . get_project ( request . match_info [ " project_id " ] )
2015-02-04 22:48:29 +02:00
yield from VirtualBox . instance ( ) . delete_vm ( request . match_info [ " vm_id " ] )
2015-01-24 01:38:46 +02:00
response . set_status ( 204 )
2015-01-20 03:30:57 +02:00
@classmethod
@Route.post (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /start " ,
2015-01-20 03:30:57 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-20 03:30:57 +02:00
} ,
status_codes = {
2015-01-24 01:38:46 +02:00
204 : " Instance started " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
2015-01-20 03:30:57 +02:00
} ,
description = " Start a VirtualBox VM instance " )
2015-01-22 02:41:35 +02:00
def start ( request , response ) :
2015-01-20 03:30:57 +02:00
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-07-27 04:21:30 +03:00
if ( yield from vm . check_hw_virtualization ( ) ) :
2015-07-22 07:58:28 +03:00
pm = ProjectManager . instance ( )
if pm . check_hardware_virtualization ( vm ) is False :
2015-07-27 04:21:30 +03:00
raise HTTPConflict ( text = " Cannot start VM because hardware virtualization (VT-x/AMD-V) is already used by another software like VMware or KVM (on Linux) " )
2015-01-22 02:41:35 +02:00
yield from vm . start ( )
2015-01-21 23:01:15 +02:00
response . set_status ( 204 )
2015-01-20 03:30:57 +02:00
@classmethod
@Route.post (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /stop " ,
2015-01-20 03:30:57 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-20 03:30:57 +02:00
} ,
status_codes = {
2015-01-24 01:38:46 +02:00
204 : " Instance stopped " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
2015-01-20 03:30:57 +02:00
} ,
description = " Stop a VirtualBox VM instance " )
2015-01-22 02:41:35 +02:00
def stop ( request , response ) :
2015-01-20 03:30:57 +02:00
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-01-22 02:41:35 +02:00
yield from vm . stop ( )
2015-01-21 23:01:15 +02:00
response . set_status ( 204 )
2015-01-22 04:28:52 +02:00
@classmethod
@Route.post (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /suspend " ,
2015-01-22 04:28:52 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-22 04:28:52 +02:00
} ,
status_codes = {
2015-01-24 01:38:46 +02:00
204 : " Instance suspended " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
2015-01-22 04:28:52 +02:00
} ,
description = " Suspend a VirtualBox VM instance " )
def suspend ( request , response ) :
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-01-22 04:28:52 +02:00
yield from vm . suspend ( )
response . set_status ( 204 )
@classmethod
@Route.post (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /resume " ,
2015-01-22 04:28:52 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-22 04:28:52 +02:00
} ,
status_codes = {
2015-01-24 01:38:46 +02:00
204 : " Instance resumed " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
2015-01-22 04:28:52 +02:00
} ,
description = " Resume a suspended VirtualBox VM instance " )
2015-05-01 04:05:37 +03:00
def resume ( request , response ) :
2015-01-22 04:28:52 +02:00
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-01-22 04:28:52 +02:00
yield from vm . resume ( )
response . set_status ( 204 )
2015-01-23 04:06:17 +02:00
@classmethod
@Route.post (
2015-02-06 18:42:25 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /reload " ,
2015-01-23 04:06:17 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance "
2015-01-23 04:06:17 +02:00
} ,
status_codes = {
2015-01-24 01:38:46 +02:00
204 : " Instance reloaded " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
2015-01-23 04:06:17 +02:00
} ,
description = " Reload a VirtualBox VM instance " )
2015-02-05 02:13:35 +02:00
def reload ( request , response ) :
2015-01-23 04:06:17 +02:00
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-01-23 04:06:17 +02:00
yield from vm . reload ( )
response . set_status ( 204 )
2015-01-24 01:38:46 +02:00
@Route.post (
2015-02-14 00:11:14 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/nio " ,
2015-01-24 01:38:46 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance " ,
2015-02-14 00:11:14 +02:00
" adapter_number " : " Adapter where the nio should be added " ,
" port_number " : " Port on the adapter (always 0) "
2015-01-24 01:38:46 +02:00
} ,
status_codes = {
201 : " NIO created " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
} ,
description = " Add a NIO to a VirtualBox VM instance " ,
2015-04-27 23:19:17 +03:00
input = NIO_SCHEMA ,
output = NIO_SCHEMA )
2015-01-24 01:38:46 +02:00
def create_nio ( request , response ) :
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-04-27 23:19:17 +03:00
nio_type = request . json [ " type " ]
2015-05-07 00:21:39 +03:00
if nio_type not in ( " nio_udp " , " nio_nat " ) :
2015-04-27 23:19:17 +03:00
raise HTTPConflict ( text = " NIO of type {} is not supported " . format ( nio_type ) )
2015-01-24 02:57:54 +02:00
nio = vbox_manager . create_nio ( vbox_manager . vboxmanage_path , request . json )
2015-02-14 00:11:14 +02:00
yield from vm . adapter_add_nio_binding ( int ( request . match_info [ " adapter_number " ] ) , nio )
2015-01-24 01:38:46 +02:00
response . set_status ( 201 )
response . json ( nio )
@classmethod
@Route.delete (
2015-02-14 00:11:14 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/nio " ,
2015-01-24 01:38:46 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance " ,
2015-02-14 00:11:14 +02:00
" adapter_number " : " Adapter from where the nio should be removed " ,
2015-03-07 05:08:00 +02:00
" port_number " : " Port on the adapter (always 0) "
2015-01-24 01:38:46 +02:00
} ,
status_codes = {
204 : " NIO deleted " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 01:38:46 +02:00
404 : " Instance doesn ' t exist "
} ,
description = " Remove a NIO from a VirtualBox VM instance " )
def delete_nio ( request , response ) :
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-02-14 00:11:14 +02:00
yield from vm . adapter_remove_nio_binding ( int ( request . match_info [ " adapter_number " ] ) )
2015-01-24 01:38:46 +02:00
response . set_status ( 204 )
2015-01-24 03:33:49 +02:00
@Route.post (
2015-02-14 00:11:14 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/start_capture " ,
2015-01-24 03:33:49 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance " ,
2015-02-14 00:11:14 +02:00
" adapter_number " : " Adapter to start a packet capture " ,
" port_number " : " Port on the adapter (always 0) "
2015-01-24 03:33:49 +02:00
} ,
status_codes = {
200 : " Capture started " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 03:33:49 +02:00
404 : " Instance doesn ' t exist "
} ,
description = " Start a packet capture on a VirtualBox VM instance " ,
2016-02-09 17:07:33 +02:00
input = VM_CAPTURE_SCHEMA )
2015-01-24 03:33:49 +02:00
def start_capture ( request , response ) :
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-02-14 00:11:14 +02:00
adapter_number = int ( request . match_info [ " adapter_number " ] )
2015-02-08 23:44:56 +02:00
pcap_file_path = os . path . join ( vm . project . capture_working_directory ( ) , request . json [ " capture_file_name " ] )
2015-05-27 22:56:27 +03:00
yield from vm . start_capture ( adapter_number , pcap_file_path )
2015-01-31 21:01:23 +02:00
response . json ( { " pcap_file_path " : pcap_file_path } )
2015-01-24 03:33:49 +02:00
@Route.post (
2015-02-14 00:11:14 +02:00
r " /projects/ {project_id} /virtualbox/vms/ {vm_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/stop_capture " ,
2015-01-24 03:33:49 +02:00
parameters = {
2015-02-05 02:13:35 +02:00
" project_id " : " UUID for the project " ,
2015-02-04 22:48:29 +02:00
" vm_id " : " UUID for the instance " ,
2015-02-14 00:11:14 +02:00
" adapter_number " : " Adapter to stop a packet capture " ,
" port_number " : " Port on the adapter (always 0) "
2015-01-24 03:33:49 +02:00
} ,
status_codes = {
204 : " Capture stopped " ,
2015-02-05 02:13:35 +02:00
400 : " Invalid request " ,
2015-01-24 03:33:49 +02:00
404 : " Instance doesn ' t exist "
} ,
description = " Stop a packet capture on a VirtualBox VM instance " )
2015-02-17 01:53:50 +02:00
def stop_capture ( request , response ) :
2015-01-24 03:33:49 +02:00
vbox_manager = VirtualBox . instance ( )
2015-02-05 02:13:35 +02:00
vm = vbox_manager . get_vm ( request . match_info [ " vm_id " ] , project_id = request . match_info [ " project_id " ] )
2015-02-14 00:11:14 +02:00
vm . stop_capture ( int ( request . match_info [ " adapter_number " ] ) )
2015-01-24 03:33:49 +02:00
response . set_status ( 204 )