2015-05-01 04:05:37 +03: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-09-13 18:40:09 +03:00
import os
2015-05-16 04:09:48 +03:00
from aiohttp . web import HTTPConflict
2016-05-14 03:00:07 +03:00
from gns3server . web . route import Route
from gns3server . schemas . node import NODE_CAPTURE_SCHEMA
from gns3server . schemas . nio import NIO_SCHEMA
from gns3server . compute . vmware import VMware
from gns3server . compute . project_manager import ProjectManager
from gns3server . schemas . vmware import (
VMWARE_CREATE_SCHEMA ,
VMWARE_OBJECT_SCHEMA
)
2015-05-01 04:05:37 +03:00
class VMwareHandler :
"""
API entry points for VMware .
"""
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
201 : " Instance created " ,
400 : " Invalid request " ,
409 : " Conflict "
} ,
description = " Create a new VMware VM instance " ,
input = VMWARE_CREATE_SCHEMA ,
output = VMWARE_OBJECT_SCHEMA )
2018-10-15 13:05:49 +03:00
async def create ( request , response ) :
2015-05-01 04:05:37 +03:00
vmware_manager = VMware . instance ( )
2018-10-15 13:05:49 +03:00
vm = await vmware_manager . create_node ( request . json . pop ( " name " ) ,
2016-05-11 20:35:36 +03:00
request . match_info [ " project_id " ] ,
request . json . get ( " node_id " ) ,
request . json . pop ( " vmx_path " ) ,
2016-10-24 22:39:35 +03:00
linked_clone = request . json . pop ( " linked_clone " ) ,
2018-03-24 14:11:21 +03:00
console = request . json . get ( " console " , None ) ,
console_type = request . json . get ( " console_type " , " telnet " ) )
2015-05-01 04:05:37 +03:00
2015-05-02 03:47:46 +03:00
for name , value in request . json . items ( ) :
2016-05-11 20:35:36 +03:00
if name != " node_id " :
2015-05-02 03:47:46 +03:00
if hasattr ( vm , name ) and getattr ( vm , name ) != value :
setattr ( vm , name , value )
2015-05-01 04:05:37 +03:00
response . set_status ( 201 )
response . json ( vm )
@Route.get (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
200 : " Success " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Get a VMware VM instance " ,
output = VMWARE_OBJECT_SCHEMA )
def show ( request , response ) :
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2015-05-01 04:05:37 +03:00
response . json ( vm )
@Route.put (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
200 : " Instance updated " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist " ,
409 : " Conflict "
} ,
description = " Update a VMware VM instance " ,
2016-07-12 17:29:58 +03:00
input = VMWARE_OBJECT_SCHEMA ,
2015-05-01 04:05:37 +03:00
output = VMWARE_OBJECT_SCHEMA )
def update ( request , response ) :
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2018-06-09 17:47:36 +03:00
# update the console first to avoid issue if updating console type
vm . console = request . json . pop ( " console " , vm . console )
2015-05-02 03:47:46 +03:00
for name , value in request . json . items ( ) :
if hasattr ( vm , name ) and getattr ( vm , name ) != value :
setattr ( vm , name , value )
2015-05-01 04:05:37 +03:00
2016-05-18 12:00:35 +03:00
vm . updated ( )
2015-05-01 04:05:37 +03:00
response . json ( vm )
@Route.delete (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
204 : " Instance deleted " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Delete a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def delete ( request , response ) :
2015-05-01 04:05:37 +03:00
# check the project_id exists
ProjectManager . instance ( ) . get_project ( request . match_info [ " project_id " ] )
2018-10-15 13:05:49 +03:00
await VMware . instance ( ) . delete_node ( request . match_info [ " node_id " ] )
2015-05-01 04:05:37 +03:00
response . set_status ( 204 )
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /start " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
204 : " Instance started " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Start a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def start ( request , response ) :
2015-05-01 04:05:37 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2015-07-27 04:21:30 +03:00
if vm . check_hw_virtualization ( ) :
pm = ProjectManager . instance ( )
if pm . check_hardware_virtualization ( vm ) is False :
raise HTTPConflict ( text = " Cannot start VM because hardware virtualization (VT-x/AMD-V) is already used by another software like VirtualBox or KVM (on Linux) " )
2018-10-15 13:05:49 +03:00
await vm . start ( )
2015-05-01 04:05:37 +03:00
response . set_status ( 204 )
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /stop " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
204 : " Instance stopped " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Stop a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def stop ( request , response ) :
2015-05-01 04:05:37 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2018-10-15 13:05:49 +03:00
await vm . stop ( )
2015-05-01 04:05:37 +03:00
response . set_status ( 204 )
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /suspend " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
204 : " Instance suspended " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Suspend a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def suspend ( request , response ) :
2015-05-01 04:05:37 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2018-10-15 13:05:49 +03:00
await vm . suspend ( )
2015-05-01 04:05:37 +03:00
response . set_status ( 204 )
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /resume " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
204 : " Instance resumed " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Resume a suspended VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def resume ( request , response ) :
2015-05-01 04:05:37 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2018-10-15 13:05:49 +03:00
await vm . resume ( )
2015-05-01 04:05:37 +03:00
response . set_status ( 204 )
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /reload " ,
2015-05-01 04:05:37 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID "
2015-05-01 04:05:37 +03:00
} ,
status_codes = {
204 : " Instance reloaded " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Reload a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def reload ( request , response ) :
2015-05-01 04:05:37 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2018-10-15 13:05:49 +03:00
await vm . reload ( )
2015-05-01 04:05:37 +03:00
response . set_status ( 204 )
2015-05-16 04:09:48 +03:00
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/nio " ,
2015-05-16 04:09:48 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
2015-05-16 04:09:48 +03:00
" adapter_number " : " Adapter where the nio should be added " ,
" port_number " : " Port on the adapter (always 0) "
} ,
status_codes = {
201 : " NIO created " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Add a NIO to a VMware VM instance " ,
input = NIO_SCHEMA ,
output = NIO_SCHEMA )
2018-10-15 13:05:49 +03:00
async def create_nio ( request , response ) :
2015-05-16 04:09:48 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2015-05-16 04:09:48 +03:00
nio_type = request . json [ " type " ]
2016-07-07 07:00:39 +03:00
if nio_type not in ( " nio_udp " , " nio_vmnet " , " nio_nat " , " nio_tap " ) :
2015-05-16 04:09:48 +03:00
raise HTTPConflict ( text = " NIO of type {} is not supported " . format ( nio_type ) )
2016-06-25 03:35:39 +03:00
nio = vmware_manager . create_nio ( request . json )
2018-10-15 13:05:49 +03:00
await vm . adapter_add_nio_binding ( int ( request . match_info [ " adapter_number " ] ) , nio )
2015-05-16 04:09:48 +03:00
response . set_status ( 201 )
response . json ( nio )
2017-07-18 19:04:03 +03:00
@Route.put (
r " /projects/ {project_id} /vmware/nodes/ {node_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/nio " ,
parameters = {
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
" adapter_number " : " Network adapter where the nio is located " ,
" port_number " : " Port from where the nio should be updated "
} ,
status_codes = {
201 : " NIO updated " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
input = NIO_SCHEMA ,
output = NIO_SCHEMA ,
2018-10-27 10:47:17 +03:00
description = " Update a NIO on a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def update_nio ( request , response ) :
2017-07-18 19:04:03 +03:00
vmware_manager = VMware . instance ( )
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2018-10-27 10:47:17 +03:00
adapter_number = int ( request . match_info [ " adapter_number " ] )
nio = vm . get_nio ( adapter_number )
if " filters " in request . json :
2017-07-18 19:04:03 +03:00
nio . filters = request . json [ " filters " ]
2018-10-27 10:47:17 +03:00
await vm . adapter_update_nio_binding ( adapter_number , nio )
response . set_status ( 201 )
response . json ( request . json )
2017-07-18 19:04:03 +03:00
2015-05-16 04:09:48 +03:00
@Route.delete (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/nio " ,
2015-05-16 04:09:48 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
2015-05-16 04:09:48 +03:00
" adapter_number " : " Adapter from where the nio should be removed " ,
" port_number " : " Port on the adapter (always 0) "
} ,
status_codes = {
204 : " NIO deleted " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist "
} ,
description = " Remove a NIO from a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def delete_nio ( request , response ) :
2015-05-16 04:09:48 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2018-10-27 10:47:17 +03:00
adapter_number = int ( request . match_info [ " adapter_number " ] )
await vm . adapter_remove_nio_binding ( adapter_number )
2015-05-16 04:09:48 +03:00
response . set_status ( 204 )
2015-09-05 23:38:11 +03:00
2015-09-13 18:40:09 +03:00
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/start_capture " ,
2015-09-13 18:40:09 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
2015-09-13 18:40:09 +03:00
" adapter_number " : " Adapter to start a packet capture " ,
" port_number " : " Port on the adapter (always 0) "
} ,
status_codes = {
200 : " Capture started " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist " ,
} ,
description = " Start a packet capture on a VMware VM instance " ,
2016-05-11 20:35:36 +03:00
input = NODE_CAPTURE_SCHEMA )
2018-10-15 13:05:49 +03:00
async def start_capture ( request , response ) :
2015-09-13 18:40:09 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2015-09-13 18:40:09 +03:00
adapter_number = int ( request . match_info [ " adapter_number " ] )
pcap_file_path = os . path . join ( vm . project . capture_working_directory ( ) , request . json [ " capture_file_name " ] )
2018-10-15 13:05:49 +03:00
await vm . start_capture ( adapter_number , pcap_file_path )
2015-09-13 18:40:09 +03:00
response . json ( { " pcap_file_path " : pcap_file_path } )
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/stop_capture " ,
2015-09-13 18:40:09 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
2015-09-13 18:40:09 +03:00
" adapter_number " : " Adapter to stop a packet capture " ,
" port_number " : " Port on the adapter (always 0) "
} ,
status_codes = {
204 : " Capture stopped " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist " ,
} ,
description = " Stop a packet capture on a VMware VM instance " )
2018-10-15 13:05:49 +03:00
async def stop_capture ( request , response ) :
2015-09-13 18:40:09 +03:00
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2015-09-13 18:40:09 +03:00
adapter_number = int ( request . match_info [ " adapter_number " ] )
2018-10-15 13:05:49 +03:00
await vm . stop_capture ( adapter_number )
2015-09-13 18:40:09 +03:00
response . set_status ( 204 )
2018-10-27 10:47:17 +03:00
@Route.get (
r " /projects/ {project_id} /vmware/nodes/ {node_id} /adapters/ { adapter_number: \ d+}/ports/ { port_number: \ d+}/pcap " ,
description = " Stream the pcap capture file " ,
parameters = {
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
" adapter_number " : " Adapter to steam a packet capture " ,
" port_number " : " Port on the adapter (always 0) "
} ,
status_codes = {
200 : " File returned " ,
403 : " Permission denied " ,
404 : " The file doesn ' t exist "
} )
async def stream_pcap_file ( request , response ) :
vmware_manager = VMware . instance ( )
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
adapter_number = int ( request . match_info [ " adapter_number " ] )
nio = vm . get_nio ( adapter_number )
await vmware_manager . stream_pcap_file ( nio , vm . project . id , request , response )
2015-09-05 23:38:11 +03:00
@Route.post (
2016-05-11 20:35:36 +03:00
r " /projects/ {project_id} /vmware/nodes/ {node_id} /interfaces/vmnet " ,
2015-09-05 23:38:11 +03:00
parameters = {
2016-05-14 03:00:07 +03:00
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
2015-09-05 23:38:11 +03:00
} ,
status_codes = {
201 : " VMnet interface allocated " ,
} ,
description = " Allocate a VMware VMnet interface on the server " )
def allocate_vmnet ( request , response ) :
vmware_manager = VMware . instance ( )
2016-05-11 20:35:36 +03:00
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
2015-09-20 22:19:57 +03:00
vmware_manager . refresh_vmnet_list ( ubridge = False )
2015-09-05 23:38:11 +03:00
vmnet = vmware_manager . allocate_vmnet ( )
2015-10-15 06:39:17 +03:00
vm . vmnets . append ( vmnet )
2015-09-05 23:38:11 +03:00
response . set_status ( 201 )
response . json ( { " vmnet " : vmnet } )
2016-05-14 03:00:07 +03:00
@Route.get (
r " /vmware/vms " ,
status_codes = {
200 : " Success " ,
} ,
description = " Get all VMware VMs available " )
2018-10-15 13:05:49 +03:00
async def get_vms ( request , response ) :
2016-05-14 03:00:07 +03:00
vmware_manager = VMware . instance ( )
2018-10-15 13:05:49 +03:00
vms = await vmware_manager . list_vms ( )
2016-05-14 03:00:07 +03:00
response . json ( vms )
2020-01-31 11:31:27 +02:00
@Route.get (
r " /projects/ {project_id} /vmware/nodes/ {node_id} /console/ws " ,
description = " WebSocket for console " ,
parameters = {
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
} )
async def console_ws ( request , response ) :
vmware_manager = VMware . instance ( )
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
return await vm . start_websocket_console ( request )
2020-07-26 11:57:18 +03:00
@Route.post (
r " /projects/ {project_id} /vmware/nodes/ {node_id} /console/reset " ,
description = " Reset console " ,
parameters = {
" project_id " : " Project UUID " ,
" node_id " : " Node UUID " ,
} ,
status_codes = {
204 : " Console has been reset " ,
400 : " Invalid request " ,
404 : " Instance doesn ' t exist " ,
409 : " Container not started "
} )
async def reset_console ( request , response ) :
vmware_manager = VMware . instance ( )
vm = vmware_manager . get_node ( request . match_info [ " node_id " ] , project_id = request . match_info [ " project_id " ] )
await vm . reset_console ( )
response . set_status ( 204 )