2014-07-12 13:18:25 -06:00
#
# Copyright (C) 2014 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/>.
"""
VirtualBox VM instance .
"""
2014-10-30 18:53:17 -06:00
import re
2014-07-12 13:18:25 -06:00
import os
2017-02-13 19:11:29 +01:00
import sys
2014-11-09 11:50:47 -07:00
import json
2017-02-13 19:11:29 +01:00
import uuid
import shlex
import shutil
2015-01-19 18:30:57 -07:00
import asyncio
2017-02-13 19:11:29 +01:00
import tempfile
2016-10-28 16:00:26 +02:00
import xml . etree . ElementTree as ET
2014-07-12 13:18:25 -06:00
2016-05-02 17:13:23 +02:00
from gns3server . utils import parse_version
2016-11-07 11:16:51 +01:00
from gns3server . utils . asyncio . telnet_server import AsyncioTelnetServer
from gns3server . utils . asyncio . serial import asyncio_open_serial
2018-08-25 14:10:47 +07:00
from gns3server . utils . asyncio import locking
2016-11-07 11:16:51 +01:00
from gns3server . compute . virtualbox . virtualbox_error import VirtualBoxError
from gns3server . compute . nios . nio_udp import NIOUDP
from gns3server . compute . adapters . ethernet_adapter import EthernetAdapter
from gns3server . compute . base_node import BaseNode
2014-10-30 18:53:17 -06:00
2014-07-12 13:18:25 -06:00
import logging
2021-04-13 18:46:50 +09:30
2014-07-12 13:18:25 -06:00
log = logging . getLogger ( __name__ )
2014-07-17 20:02:18 -06:00
2016-05-11 11:35:36 -06:00
class VirtualBoxVM ( BaseNode ) :
2015-01-31 14:34:49 -07:00
2014-07-12 13:18:25 -06:00
"""
VirtualBox VM implementation .
"""
2021-04-13 18:46:50 +09:30
def __init__ (
self ,
name ,
node_id ,
project ,
manager ,
vmname ,
linked_clone = False ,
console = None ,
console_type = " telnet " ,
adapters = 0 ,
) :
super ( ) . __init__ (
name , node_id , project , manager , console = console , linked_clone = linked_clone , console_type = console_type
)
2015-01-19 18:30:57 -07:00
2019-03-18 23:29:18 +07:00
self . _uuid = None # UUID in VirtualBox
2015-01-21 19:26:39 -07:00
self . _maximum_adapters = 8
2015-01-20 19:02:22 -07:00
self . _system_properties = { }
2016-11-07 11:16:51 +01:00
self . _telnet_server = None
2016-06-24 18:35:39 -06:00
self . _local_udp_tunnels = { }
2015-01-19 18:30:57 -07:00
2014-07-12 13:18:25 -06:00
# VirtualBox settings
2015-01-30 19:36:05 -07:00
self . _adapters = adapters
2015-04-03 12:08:18 +02:00
self . _ethernet_adapters = { }
2014-07-17 15:28:02 -06:00
self . _headless = False
2018-03-30 21:18:44 +07:00
self . _on_close = " power_off "
2014-07-17 15:28:02 -06:00
self . _vmname = vmname
2015-02-06 17:31:13 -07:00
self . _use_any_adapter = False
2015-03-13 17:13:36 -06:00
self . _ram = 0
2014-11-04 19:00:01 -07:00
self . _adapter_type = " Intel PRO/1000 MT Desktop (82540EM) "
2014-07-12 13:18:25 -06:00
2021-04-17 23:34:28 +09:30
def asdict ( self ) :
2014-11-09 11:50:47 -07:00
2021-04-13 18:46:50 +09:30
json = {
" name " : self . name ,
" usage " : self . usage ,
" node_id " : self . id ,
" console " : self . console ,
" console_type " : self . console_type ,
" project_id " : self . project . id ,
" vmname " : self . vmname ,
" headless " : self . headless ,
" on_close " : self . on_close ,
" adapters " : self . _adapters ,
" adapter_type " : self . adapter_type ,
" ram " : self . ram ,
" status " : self . status ,
" use_any_adapter " : self . use_any_adapter ,
" linked_clone " : self . linked_clone ,
}
2016-10-24 21:39:35 +02:00
if self . linked_clone :
2017-10-02 10:41:57 +02:00
json [ " node_directory " ] = self . working_path
2015-06-26 17:09:19 +02:00
else :
2016-05-12 10:39:50 +02:00
json [ " node_directory " ] = None
2015-06-26 17:09:19 +02:00
return json
2014-07-12 13:18:25 -06:00
2017-07-18 14:59:47 +02:00
@property
def ethernet_adapters ( self ) :
return self . _ethernet_adapters
2018-10-15 17:05:49 +07:00
async def _get_system_properties ( self ) :
2015-01-19 18:30:57 -07:00
2018-10-15 17:05:49 +07:00
properties = await self . manager . execute ( " list " , [ " systemproperties " ] )
2015-01-19 18:30:57 -07:00
for prop in properties :
try :
2021-04-13 18:46:50 +09:30
name , value = prop . split ( " : " , 1 )
2015-01-19 18:30:57 -07:00
except ValueError :
continue
self . _system_properties [ name . strip ( ) ] = value . strip ( )
2014-07-12 13:18:25 -06:00
2018-10-15 17:05:49 +07:00
async def _get_vm_state ( self ) :
2015-01-21 19:26:39 -07:00
"""
Returns the VM state ( e . g . running , paused etc . )
2014-07-12 13:18:25 -06:00
2015-01-21 19:26:39 -07:00
: returns : state ( string )
"""
2015-01-19 18:30:57 -07:00
2019-03-18 23:29:18 +07:00
results = await self . manager . execute ( " showvminfo " , [ self . _uuid , " --machinereadable " ] )
2015-01-21 19:26:39 -07:00
for info in results :
2021-04-13 18:46:50 +09:30
if " = " in info :
name , value = info . split ( " = " , 1 )
2015-04-14 14:35:48 +02:00
if name == " VMState " :
return value . strip ( ' " ' )
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Could not get VM state for { self . _vmname } " )
2015-01-21 19:26:39 -07:00
2018-10-15 17:05:49 +07:00
async def _control_vm ( self , params ) :
2015-01-21 19:26:39 -07:00
"""
Change setting in this VM when running .
: param params : params to use with sub - command controlvm
: returns : result of the command .
"""
args = shlex . split ( params )
2019-03-18 23:29:18 +07:00
result = await self . manager . execute ( " controlvm " , [ self . _uuid ] + args )
2015-01-21 19:26:39 -07:00
return result
2018-10-15 17:05:49 +07:00
async def _modify_vm ( self , params ) :
2015-01-21 19:26:39 -07:00
"""
Change setting in this VM when not running .
: param params : params to use with sub - command modifyvm
"""
2014-07-12 13:18:25 -06:00
2015-01-21 19:26:39 -07:00
args = shlex . split ( params )
2019-03-18 23:29:18 +07:00
await self . manager . execute ( " modifyvm " , [ self . _uuid ] + args )
2014-07-12 13:18:25 -06:00
2018-10-15 17:05:49 +07:00
async def _check_duplicate_linked_clone ( self ) :
2016-11-17 12:21:38 +01:00
"""
Without linked clone two VM using the same image can ' t run
at the same time .
To avoid issue like false detection when a project close
and another open we try multiple times .
"""
trial = 0
while True :
found = False
for node in self . manager . nodes :
if node != self and node . vmname == self . vmname :
found = True
if node . project != self . project :
if trial > = 30 :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
f " Sorry a node without the linked clone setting enabled can only be used once on your server. \n { self . vmname } is already used by { node . name } in project { self . project . name } "
)
2016-11-17 12:21:38 +01:00
else :
if trial > = 5 :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
f " Sorry a node without the linked clone setting enabled can only be used once on your server. \n { self . vmname } is already used by { node . name } in this project "
)
2016-11-17 12:21:38 +01:00
if not found :
return
trial + = 1
2018-10-15 17:05:49 +07:00
await asyncio . sleep ( 1 )
2016-11-17 12:21:38 +01:00
2019-11-03 16:34:24 +08:00
async def _refresh_vm_uuid ( self ) :
vm_info = await self . _get_vm_info ( )
2019-11-08 17:25:59 +08:00
self . _uuid = vm_info . get ( " UUID " , self . _uuid )
2019-11-03 16:34:24 +08:00
if not self . _uuid :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Could not find any UUID for VM ' { self . _vmname } ' " )
2019-11-03 16:34:24 +08:00
if " memory " in vm_info :
self . _ram = int ( vm_info [ " memory " ] )
2018-10-15 17:05:49 +07:00
async def create ( self ) :
2019-03-18 23:29:18 +07:00
2016-11-17 12:21:38 +01:00
if not self . linked_clone :
2018-10-15 17:05:49 +07:00
await self . _check_duplicate_linked_clone ( )
2015-01-22 19:07:09 -07:00
2018-10-15 17:05:49 +07:00
await self . _get_system_properties ( )
2015-04-14 14:32:44 +02:00
if " API version " not in self . _system_properties :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Can ' t access to VirtualBox API version: \n { self . _system_properties } " )
2015-01-21 19:26:39 -07:00
if parse_version ( self . _system_properties [ " API version " ] ) < parse_version ( " 4_3 " ) :
raise VirtualBoxError ( " The VirtualBox API version is lower than 4.3 " )
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] created " )
2015-01-21 19:26:39 -07:00
2016-10-24 21:39:35 +02:00
if self . linked_clone :
2015-02-04 13:48:29 -07:00
if self . id and os . path . isdir ( os . path . join ( self . working_dir , self . _vmname ) ) :
2016-10-28 16:00:26 +02:00
self . _patch_vm_uuid ( )
2018-10-15 17:05:49 +07:00
await self . manager . execute ( " registervm " , [ self . _linked_vbox_file ( ) ] )
2019-11-03 16:34:24 +08:00
await self . _refresh_vm_uuid ( )
2018-10-15 17:05:49 +07:00
await self . _reattach_linked_hdds ( )
2019-11-03 16:34:24 +08:00
2015-01-21 19:26:39 -07:00
else :
2019-11-03 16:34:24 +08:00
await self . _refresh_vm_uuid ( )
2018-10-15 17:05:49 +07:00
await self . _create_linked_clone ( )
2019-11-08 17:13:21 +08:00
else :
await self . _refresh_vm_uuid ( )
2014-07-12 13:18:25 -06:00
2015-01-31 12:01:23 -07:00
if self . _adapters :
2018-10-15 17:05:49 +07:00
await self . set_adapters ( self . _adapters )
2015-01-22 21:31:26 -07:00
2016-10-28 16:00:26 +02:00
def _linked_vbox_file ( self ) :
return os . path . join ( self . working_dir , self . _vmname , self . _vmname + " .vbox " )
def _patch_vm_uuid ( self ) :
"""
Fix the VM uuid in the case of linked clone
"""
2017-01-10 14:22:04 +01:00
if os . path . exists ( self . _linked_vbox_file ( ) ) :
2017-06-09 09:57:47 +02:00
try :
tree = ET . parse ( self . _linked_vbox_file ( ) )
except ET . ParseError :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
" Cannot modify VirtualBox linked nodes file. "
" File {} is corrupted. " . format ( self . _linked_vbox_file ( ) )
)
2018-09-11 15:06:01 +02:00
except OSError as e :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Cannot modify VirtualBox linked nodes file ' { self . _linked_vbox_file ( ) } ' : { e } " )
2017-06-09 09:57:47 +02:00
2017-01-10 14:22:04 +01:00
machine = tree . getroot ( ) . find ( " { http://www.virtualbox.org/}Machine " )
2017-02-13 19:11:29 +01:00
if machine is not None and machine . get ( " uuid " ) != " { " + self . id + " } " :
for image in tree . getroot ( ) . findall ( " { http://www.virtualbox.org/}Image " ) :
currentSnapshot = machine . get ( " currentSnapshot " )
if currentSnapshot :
2019-01-17 18:01:58 +07:00
newSnapshot = re . sub ( r " \ { .* \ } " , " { " + str ( uuid . uuid4 ( ) ) + " } " , currentSnapshot )
2021-04-13 18:46:50 +09:30
shutil . move (
os . path . join ( self . working_dir , self . _vmname , " Snapshots " , currentSnapshot ) + " .vdi " ,
os . path . join ( self . working_dir , self . _vmname , " Snapshots " , newSnapshot ) + " .vdi " ,
)
2017-02-13 19:11:29 +01:00
image . set ( " uuid " , newSnapshot )
2017-01-13 17:20:02 +01:00
machine . set ( " uuid " , " { " + self . id + " } " )
tree . write ( self . _linked_vbox_file ( ) )
2016-10-28 16:00:26 +02:00
2018-10-15 17:05:49 +07:00
async def check_hw_virtualization ( self ) :
2015-07-21 22:58:28 -06:00
"""
Returns either hardware virtualization is activated or not .
: returns : boolean
"""
2018-10-15 17:05:49 +07:00
vm_info = await self . _get_vm_info ( )
2015-07-21 22:58:28 -06:00
if " hwvirtex " in vm_info and vm_info [ " hwvirtex " ] == " on " :
return True
return False
2018-09-11 15:06:01 +02:00
@locking
2018-10-15 17:05:49 +07:00
async def start ( self ) :
2015-01-21 19:26:39 -07:00
"""
Starts this VirtualBox VM .
"""
2015-01-19 18:30:57 -07:00
2016-11-17 10:38:29 +01:00
if self . status == " started " :
return
2015-01-21 19:26:39 -07:00
# resume the VM if it is paused
2018-10-15 17:05:49 +07:00
vm_state = await self . _get_vm_state ( )
2015-01-21 19:26:39 -07:00
if vm_state == " paused " :
2018-10-15 17:05:49 +07:00
await self . resume ( )
2015-01-21 19:26:39 -07:00
return
2015-01-19 18:30:57 -07:00
2015-09-08 01:20:46 -06:00
# VM must be powered off to start it
2019-12-05 13:46:02 +08:00
if vm_state == " saved " :
result = await self . manager . execute ( " guestproperty " , [ " get " , self . _uuid , " SavedByGNS3 " ] )
2021-04-13 18:46:50 +09:30
if result == [ " No value set! " ] :
2019-12-05 13:46:02 +08:00
raise VirtualBoxError ( " VirtualBox VM was not saved from GNS3 " )
else :
await self . manager . execute ( " guestproperty " , [ " delete " , self . _uuid , " SavedByGNS3 " ] )
elif vm_state == " poweroff " :
await self . _set_network_options ( )
await self . _set_serial_console ( )
else :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " VirtualBox VM ' { self . name } ' is not powered off (current state is ' { vm_state } ' ) " )
2015-01-19 18:30:57 -07:00
2015-10-12 15:57:37 -06:00
# check if there is enough RAM to run
self . check_available_ram ( self . ram )
2019-03-18 23:29:18 +07:00
args = [ self . _uuid ]
2015-01-21 19:26:39 -07:00
if self . _headless :
args . extend ( [ " --type " , " headless " ] )
2018-10-15 17:05:49 +07:00
result = await self . manager . execute ( " startvm " , args )
2016-05-13 20:41:58 -06:00
self . status = " started "
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] started " )
log . debug ( f " Start result: { result } " )
2015-01-21 19:26:39 -07:00
# add a guest property to let the VM know about the GNS3 name
2019-03-18 23:29:18 +07:00
await self . manager . execute ( " guestproperty " , [ " set " , self . _uuid , " NameInGNS3 " , self . name ] )
2015-01-21 19:26:39 -07:00
# add a guest property to let the VM know about the GNS3 project directory
2019-03-18 23:29:18 +07:00
await self . manager . execute ( " guestproperty " , [ " set " , self . _uuid , " ProjectDirInGNS3 " , self . working_dir ] )
2015-01-21 19:26:39 -07:00
2018-10-15 17:05:49 +07:00
await self . _start_ubridge ( )
2017-07-11 13:42:47 +02:00
for adapter_number in range ( 0 , self . _adapters ) :
nio = self . _ethernet_adapters [ adapter_number ] . get_nio ( 0 )
if nio :
2021-04-13 18:46:50 +09:30
await self . add_ubridge_udp_connection (
f " VBOX- { self . _id } - { adapter_number } " , self . _local_udp_tunnels [ adapter_number ] [ 1 ] , nio
)
2016-06-24 18:35:39 -06:00
2018-10-15 17:05:49 +07:00
await self . _start_console ( )
2014-07-12 13:18:25 -06:00
2021-04-13 18:46:50 +09:30
if await self . check_hw_virtualization ( ) :
2015-07-21 22:58:28 -06:00
self . _hw_virtualization = True
2018-08-25 14:10:47 +07:00
@locking
2018-10-15 17:05:49 +07:00
async def stop ( self ) :
2015-01-21 19:26:39 -07:00
"""
Stops this VirtualBox VM .
2014-07-12 13:18:25 -06:00
"""
2015-07-21 22:58:28 -06:00
self . _hw_virtualization = False
2018-10-15 17:05:49 +07:00
await self . _stop_ubridge ( )
await self . _stop_remote_console ( )
vm_state = await self . _get_vm_state ( )
2021-04-13 18:37:58 +09:30
log . info ( f " Stopping VirtualBox VM ' { self . name } ' [ { self . id } ] (current state is { vm_state } ) " )
2020-01-23 18:00:33 +08:00
if vm_state in ( " running " , " paused " ) :
2018-03-30 21:18:44 +07:00
if self . on_close == " save_vm_state " :
2019-12-05 13:46:02 +08:00
# add a guest property to know the VM has been saved
await self . manager . execute ( " guestproperty " , [ " set " , self . _uuid , " SavedByGNS3 " , " yes " ] )
2018-10-15 17:05:49 +07:00
result = await self . _control_vm ( " savestate " )
2018-03-30 21:18:44 +07:00
self . status = " stopped "
2021-04-13 18:37:58 +09:30
log . debug ( f " Stop result: { result } " )
2018-03-30 21:18:44 +07:00
elif self . on_close == " shutdown_signal " :
2015-06-02 16:30:35 -06:00
# use ACPI to shutdown the VM
2018-10-15 17:05:49 +07:00
result = await self . _control_vm ( " acpipowerbutton " )
2017-02-24 13:58:03 +01:00
trial = 0
while True :
2018-10-15 17:05:49 +07:00
vm_state = await self . _get_vm_state ( )
2017-02-24 13:58:03 +01:00
if vm_state == " poweroff " :
break
2018-10-15 17:05:49 +07:00
await asyncio . sleep ( 1 )
2017-02-24 13:58:03 +01:00
trial + = 1
if trial > = 120 :
2018-10-15 17:05:49 +07:00
await self . _control_vm ( " poweroff " )
2017-02-24 13:58:03 +01:00
break
2016-05-13 20:41:58 -06:00
self . status = " stopped "
2021-04-13 18:37:58 +09:30
log . debug ( f " ACPI shutdown result: { result } " )
2015-06-02 16:30:35 -06:00
else :
# power off the VM
2018-10-15 17:05:49 +07:00
result = await self . _control_vm ( " poweroff " )
2016-05-13 20:41:58 -06:00
self . status = " stopped "
2021-04-13 18:37:58 +09:30
log . debug ( f " Stop result: { result } " )
2020-01-23 18:00:33 +08:00
elif vm_state == " aborted " :
self . status = " stopped "
2014-07-17 15:28:02 -06:00
2020-01-23 18:00:33 +08:00
if self . status == " stopped " :
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] stopped " )
2018-10-15 17:05:49 +07:00
await asyncio . sleep ( 0.5 ) # give some time for VirtualBox to unlock the VM
2019-12-05 13:46:02 +08:00
if self . on_close != " save_vm_state " :
# do some cleaning when the VM is powered off
try :
# deactivate the first serial port
await self . _modify_vm ( " --uart1 off " )
except VirtualBoxError as e :
2021-04-13 18:37:58 +09:30
log . warning ( f " Could not deactivate the first serial port: { e } " )
2019-12-05 13:46:02 +08:00
for adapter_number in range ( 0 , self . _adapters ) :
nio = self . _ethernet_adapters [ adapter_number ] . get_nio ( 0 )
if nio :
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --nictrace { adapter_number + 1 } off " )
await self . _modify_vm ( f " --cableconnected { adapter_number + 1 } off " )
await self . _modify_vm ( f " --nic { adapter_number + 1 } null " )
2018-10-15 17:05:49 +07:00
await super ( ) . stop ( )
2014-07-12 13:18:25 -06:00
2018-10-15 17:05:49 +07:00
async def suspend ( self ) :
2015-01-21 19:26:39 -07:00
"""
Suspends this VirtualBox VM .
2014-07-12 13:18:25 -06:00
"""
2018-10-15 17:05:49 +07:00
vm_state = await self . _get_vm_state ( )
2015-01-21 19:26:39 -07:00
if vm_state == " running " :
2018-10-15 17:05:49 +07:00
await self . _control_vm ( " pause " )
2016-05-13 20:41:58 -06:00
self . status = " suspended "
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] suspended " )
2015-01-21 19:26:39 -07:00
else :
2021-04-13 18:46:50 +09:30
log . warning (
" VirtualBox VM ' {name} ' [ {id} ] cannot be suspended, current state: {state} " . format (
name = self . name , id = self . id , state = vm_state
)
)
2014-07-12 13:18:25 -06:00
2018-10-15 17:05:49 +07:00
async def resume ( self ) :
2014-07-12 13:18:25 -06:00
"""
2015-01-21 19:26:39 -07:00
Resumes this VirtualBox VM .
2014-07-12 13:18:25 -06:00
"""
2018-10-15 17:05:49 +07:00
await self . _control_vm ( " resume " )
2016-05-13 20:41:58 -06:00
self . status = " started "
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] resumed " )
2015-01-21 19:26:39 -07:00
2018-10-15 17:05:49 +07:00
async def reload ( self ) :
2015-01-21 19:26:39 -07:00
"""
Reloads this VirtualBox VM .
"""
2014-07-12 13:18:25 -06:00
2018-10-15 17:05:49 +07:00
result = await self . _control_vm ( " reset " )
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] reloaded " )
log . debug ( f " Reload result: { result } " )
2014-07-12 13:18:25 -06:00
2018-10-15 17:05:49 +07:00
async def _get_all_hdd_files ( self ) :
2014-11-09 11:50:47 -07:00
hdds = [ ]
2018-10-15 17:05:49 +07:00
properties = await self . manager . execute ( " list " , [ " hdds " ] )
2014-11-09 11:50:47 -07:00
for prop in properties :
try :
2021-04-13 18:46:50 +09:30
name , value = prop . split ( " : " , 1 )
2014-11-09 11:50:47 -07:00
except ValueError :
continue
if name . strip ( ) == " Location " :
hdds . append ( value . strip ( ) )
return hdds
2018-10-15 17:05:49 +07:00
async def _reattach_linked_hdds ( self ) :
2016-04-26 15:44:11 -06:00
"""
Reattach linked cloned hard disks .
"""
2014-11-09 11:50:47 -07:00
2015-01-22 19:07:09 -07:00
hdd_info_file = os . path . join ( self . working_dir , self . _vmname , " hdd_info.json " )
2014-11-09 11:50:47 -07:00
try :
2021-04-13 18:37:58 +09:30
with open ( hdd_info_file , encoding = " utf-8 " ) as f :
2014-11-09 11:50:47 -07:00
hdd_table = json . load ( f )
2015-09-29 14:15:01 -06:00
except ( ValueError , OSError ) as e :
2016-11-02 12:50:10 +01:00
# The VM has never be started
return
2014-11-09 11:50:47 -07:00
for hdd_info in hdd_table :
2015-01-22 19:07:09 -07:00
hdd_file = os . path . join ( self . working_dir , self . _vmname , " Snapshots " , hdd_info [ " hdd " ] )
2014-11-09 11:50:47 -07:00
if os . path . exists ( hdd_file ) :
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ] attaching HDD {controller} {port} {device} {medium} " . format (
name = self . name ,
id = self . id ,
controller = hdd_info [ " controller " ] ,
port = hdd_info [ " port " ] ,
device = hdd_info [ " device " ] ,
medium = hdd_file ,
)
)
2015-02-06 17:31:13 -07:00
2016-04-26 15:44:11 -06:00
try :
2021-04-13 18:46:50 +09:30
await self . _storage_attach (
' --storagectl " {} " --port {} --device {} --type hdd --medium " {} " ' . format (
hdd_info [ " controller " ] , hdd_info [ " port " ] , hdd_info [ " device " ] , hdd_file
)
)
2016-04-26 15:44:11 -06:00
except VirtualBoxError as e :
2021-04-13 18:46:50 +09:30
log . warning (
" VirtualBox VM ' {name} ' [ {id} ] error reattaching HDD {controller} {port} {device} {medium} : {error} " . format (
name = self . name ,
id = self . id ,
controller = hdd_info [ " controller " ] ,
port = hdd_info [ " port " ] ,
device = hdd_info [ " device " ] ,
medium = hdd_file ,
error = e ,
)
)
2016-04-26 15:44:11 -06:00
continue
2014-11-09 11:50:47 -07:00
2018-10-15 17:05:49 +07:00
async def save_linked_hdds_info ( self ) :
2014-07-12 13:18:25 -06:00
"""
2016-04-26 15:44:11 -06:00
Save linked cloned hard disks information .
2014-07-12 13:18:25 -06:00
2016-04-26 15:44:11 -06:00
: returns : disk table information
"""
2015-02-23 19:00:34 -07:00
2016-04-26 15:44:11 -06:00
hdd_table = [ ]
2016-10-24 21:39:35 +02:00
if self . linked_clone :
2015-01-23 16:38:46 -07:00
if os . path . exists ( self . working_dir ) :
2018-10-15 17:05:49 +07:00
hdd_files = await self . _get_all_hdd_files ( )
vm_info = await self . _get_vm_info ( )
2014-11-15 16:05:55 -07:00
for entry , value in vm_info . items ( ) :
2021-04-13 18:46:50 +09:30
match = re . search (
r " ^([ \ s \ w]+) \ -( \ d) \ -( \ d)$ " , entry
) # match Controller-PortNumber-DeviceNumber entry
2014-11-15 16:05:55 -07:00
if match :
controller = match . group ( 1 )
port = match . group ( 2 )
device = match . group ( 3 )
2021-04-13 18:46:50 +09:30
if value in hdd_files and os . path . exists (
os . path . join ( self . working_dir , self . _vmname , " Snapshots " , os . path . basename ( value ) )
) :
log . info (
" VirtualBox VM ' {name} ' [ {id} ] detaching HDD {controller} {port} {device} " . format (
name = self . name , id = self . id , controller = controller , port = port , device = device
)
)
2014-11-15 16:05:55 -07:00
hdd_table . append (
{
" hdd " : os . path . basename ( value ) ,
" controller " : controller ,
" port " : port ,
" device " : device ,
}
)
2014-11-09 11:50:47 -07:00
2014-11-15 16:05:55 -07:00
if hdd_table :
try :
2015-01-23 16:38:46 -07:00
hdd_info_file = os . path . join ( self . working_dir , self . _vmname , " hdd_info.json " )
2015-04-25 11:58:34 -06:00
with open ( hdd_info_file , " w " , encoding = " utf-8 " ) as f :
2014-11-15 16:05:55 -07:00
json . dump ( hdd_table , f , indent = 4 )
except OSError as e :
2021-04-13 18:46:50 +09:30
log . warning (
" VirtualBox VM ' {name} ' [ {id} ] could not write HHD info file: {error} " . format (
name = self . name , id = self . id , error = e . strerror
)
)
2014-11-09 11:50:47 -07:00
2016-04-26 15:44:11 -06:00
return hdd_table
2018-10-15 17:05:49 +07:00
async def close ( self ) :
2016-04-26 15:44:11 -06:00
"""
Closes this VirtualBox VM .
"""
if self . _closed :
# VM is already closed
return
2018-10-15 17:05:49 +07:00
if not ( await super ( ) . close ( ) ) :
2016-04-28 16:52:29 +02:00
return False
2021-04-13 18:37:58 +09:30
log . debug ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] is closing " )
2016-04-26 15:44:11 -06:00
if self . _console :
self . _manager . port_manager . release_tcp_port ( self . _console , self . _project )
self . _console = None
for adapter in self . _ethernet_adapters . values ( ) :
if adapter is not None :
for nio in adapter . ports . values ( ) :
if nio and isinstance ( nio , NIOUDP ) :
self . manager . port_manager . release_udp_port ( nio . lport , self . _project )
2016-06-24 18:35:39 -06:00
for udp_tunnel in self . _local_udp_tunnels . values ( ) :
self . manager . port_manager . release_udp_port ( udp_tunnel [ 0 ] . lport , self . _project )
self . manager . port_manager . release_udp_port ( udp_tunnel [ 1 ] . lport , self . _project )
self . _local_udp_tunnels = { }
2018-03-30 21:18:44 +07:00
self . on_close = " power_off "
2018-10-15 17:05:49 +07:00
await self . stop ( )
2016-04-26 15:44:11 -06:00
2016-10-24 21:39:35 +02:00
if self . linked_clone :
2018-10-15 17:05:49 +07:00
hdd_table = await self . save_linked_hdds_info ( )
2016-04-26 15:44:11 -06:00
for hdd in hdd_table . copy ( ) :
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ] detaching HDD {controller} {port} {device} " . format (
name = self . name , id = self . id , controller = hdd [ " controller " ] , port = hdd [ " port " ] , device = hdd [ " device " ]
)
)
2016-04-26 15:44:11 -06:00
try :
2021-04-13 18:46:50 +09:30
await self . _storage_attach (
' --storagectl " {} " --port {} --device {} --type hdd --medium none ' . format (
hdd [ " controller " ] , hdd [ " port " ] , hdd [ " device " ]
)
)
2016-04-26 15:44:11 -06:00
except VirtualBoxError as e :
2021-04-13 18:46:50 +09:30
log . warning (
" VirtualBox VM ' {name} ' [ {id} ] error detaching HDD {controller} {port} {device} : {error} " . format (
name = self . name ,
id = self . id ,
controller = hdd [ " controller " ] ,
port = hdd [ " port " ] ,
device = hdd [ " device " ] ,
error = e ,
)
)
2016-04-26 15:44:11 -06:00
continue
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] unregistering " )
2018-10-15 17:05:49 +07:00
await self . manager . execute ( " unregistervm " , [ self . _name ] )
2016-04-26 15:44:11 -06:00
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] closed " )
2015-02-02 18:56:13 -07:00
self . _closed = True
2014-07-12 13:18:25 -06:00
2014-07-17 15:28:02 -06:00
@property
def headless ( self ) :
"""
Returns either the VM will start in headless mode
: returns : boolean
"""
return self . _headless
@headless.setter
def headless ( self , headless ) :
"""
Sets either the VM will start in headless mode
: param headless : boolean
"""
if headless :
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] has enabled the headless mode " )
2014-07-17 15:28:02 -06:00
else :
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] has disabled the headless mode " )
2014-07-17 15:28:02 -06:00
self . _headless = headless
2015-06-02 16:30:35 -06:00
@property
2018-03-30 21:18:44 +07:00
def on_close ( self ) :
2015-06-02 16:30:35 -06:00
"""
2018-03-30 21:18:44 +07:00
Returns the action to execute when the VM is stopped / closed
2015-06-02 16:30:35 -06:00
2018-03-30 21:18:44 +07:00
: returns : string
2015-06-02 16:30:35 -06:00
"""
2018-03-30 21:18:44 +07:00
return self . _on_close
2015-06-02 16:30:35 -06:00
2018-03-30 21:18:44 +07:00
@on_close.setter
def on_close ( self , on_close ) :
2015-06-02 16:30:35 -06:00
"""
2018-03-30 21:18:44 +07:00
Sets the action to execute when the VM is stopped / closed
2015-06-02 16:30:35 -06:00
2018-03-30 21:18:44 +07:00
: param on_close : string
2015-06-02 16:30:35 -06:00
"""
2021-04-13 18:37:58 +09:30
log . info ( f ' VirtualBox VM " { self . _name } " [ { self . _id } ] set the close action to " { on_close } " ' )
2018-03-30 21:18:44 +07:00
self . _on_close = on_close
2015-06-02 16:30:35 -06:00
2015-03-13 17:13:36 -06:00
@property
def ram ( self ) :
"""
Returns the amount of RAM allocated to this VirtualBox VM .
: returns : amount RAM in MB ( integer )
"""
return self . _ram
2018-10-15 17:05:49 +07:00
async def set_ram ( self , ram ) :
2015-03-13 17:13:36 -06:00
"""
Set the amount of RAM allocated to this VirtualBox VM .
: param ram : amount RAM in MB ( integer )
"""
if ram == 0 :
return
2021-04-13 18:46:50 +09:30
await self . _modify_vm ( f " --memory { ram } " )
2015-03-13 17:13:36 -06:00
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] has set amount of RAM to { ram } " )
2015-03-13 17:13:36 -06:00
self . _ram = ram
2014-07-17 15:28:02 -06:00
@property
def vmname ( self ) :
"""
2015-03-08 12:32:36 -06:00
Returns the VirtualBox VM name .
2014-07-17 15:28:02 -06:00
: returns : VirtualBox VM name
"""
return self . _vmname
2018-10-15 17:05:49 +07:00
async def set_vmname ( self , vmname ) :
2014-07-17 15:28:02 -06:00
"""
2015-03-08 12:32:36 -06:00
Renames the VirtualBox VM .
2014-07-17 15:28:02 -06:00
: param vmname : VirtualBox VM name
"""
2017-01-31 18:58:43 +01:00
if vmname == self . _vmname :
return
2016-10-24 21:39:35 +02:00
if self . linked_clone :
2017-01-30 15:19:46 +01:00
if self . status == " started " :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Cannot change the name of running VM { self . _name } " )
2017-01-31 18:58:43 +01:00
# We can't rename a VM to name that already exists
2018-10-15 17:05:49 +07:00
vms = await self . manager . list_vms ( allow_clone = True )
2017-01-31 18:58:43 +01:00
if vmname in [ vm [ " vmname " ] for vm in vms ] :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Cannot change the name to { vmname } , it is already used in VirtualBox " )
await self . _modify_vm ( f ' --name " { vmname } " ' )
2015-03-08 12:32:36 -06:00
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] has set the VM name to ' { vmname } ' " )
2014-07-17 15:28:02 -06:00
self . _vmname = vmname
2015-02-06 17:31:13 -07:00
@property
def adapters ( self ) :
"""
Returns the number of adapters configured for this VirtualBox VM .
: returns : number of adapters
"""
2015-02-05 11:58:10 -07:00
2015-02-06 17:31:13 -07:00
return self . _adapters
2015-02-05 11:58:10 -07:00
2018-10-15 17:05:49 +07:00
async def set_adapters ( self , adapters ) :
2014-07-17 15:28:02 -06:00
"""
Sets the number of Ethernet adapters for this VirtualBox VM instance .
: param adapters : number of adapters
"""
2014-10-31 17:41:12 -06:00
# check for the maximum adapters supported by the VM
2018-10-15 17:05:49 +07:00
vm_info = await self . _get_vm_info ( )
2017-10-22 15:23:43 +07:00
chipset = " piix3 " # default chipset for VirtualBox VMs
self . _maximum_adapters = 8 # default maximum network adapter count for PIIX3 chipset
if " chipset " in vm_info :
chipset = vm_info [ " chipset " ]
2021-04-13 18:37:58 +09:30
max_adapter_string = f " Maximum { chipset . upper ( ) } Network Adapter count "
2017-10-22 15:23:43 +07:00
if max_adapter_string in self . _system_properties :
try :
self . _maximum_adapters = int ( self . _system_properties [ max_adapter_string ] )
except ValueError :
2021-04-13 18:46:50 +09:30
log . error (
f " Could not convert system property to integer: { max_adapter_string } = { self . _system_properties [ max_adapter_string ] } "
)
2017-10-22 15:23:43 +07:00
else :
2021-04-13 18:37:58 +09:30
log . warning ( f " Could not find system property ' { max_adapter_string } ' for chipset { chipset } " )
2017-10-22 15:23:43 +07:00
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ] can have a maximum of {max} network adapters for chipset {chipset} " . format (
name = self . name , id = self . id , max = self . _maximum_adapters , chipset = chipset . upper ( )
)
)
2015-04-03 12:08:18 +02:00
if adapters > self . _maximum_adapters :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
" The configured {} chipset limits the VM to {} network adapters. The chipset can be changed outside GNS3 in the VirtualBox VM settings. " . format (
chipset . upper ( ) , self . _maximum_adapters
)
)
2014-10-31 17:41:12 -06:00
2014-07-17 15:28:02 -06:00
self . _ethernet_adapters . clear ( )
2015-02-13 15:41:56 -07:00
for adapter_number in range ( 0 , adapters ) :
2015-04-03 12:08:18 +02:00
self . _ethernet_adapters [ adapter_number ] = EthernetAdapter ( )
2014-07-17 15:28:02 -06:00
2015-01-30 19:36:05 -07:00
self . _adapters = len ( self . _ethernet_adapters )
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ] has changed the number of Ethernet adapters to {adapters} " . format (
name = self . name , id = self . id , adapters = adapters
)
)
2014-08-26 15:27:43 -06:00
@property
2015-02-06 17:31:13 -07:00
def use_any_adapter ( self ) :
2014-08-26 15:27:43 -06:00
"""
2015-02-06 17:31:13 -07:00
Returns either GNS3 can use any VirtualBox adapter on this instance .
2014-08-26 15:27:43 -06:00
2015-05-21 21:48:59 -06:00
: returns : boolean
2014-08-26 15:27:43 -06:00
"""
2015-02-06 17:31:13 -07:00
return self . _use_any_adapter
2014-08-26 15:27:43 -06:00
2015-02-06 17:31:13 -07:00
@use_any_adapter.setter
def use_any_adapter ( self , use_any_adapter ) :
2014-08-26 15:27:43 -06:00
"""
2015-02-06 17:31:13 -07:00
Allows GNS3 to use any VirtualBox adapter on this instance .
2014-08-26 15:27:43 -06:00
2015-02-06 17:31:13 -07:00
: param use_any_adapter : boolean
2014-08-26 15:27:43 -06:00
"""
2015-02-06 17:31:13 -07:00
if use_any_adapter :
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] is allowed to use any adapter " )
2015-02-06 17:31:13 -07:00
else :
2021-04-13 18:37:58 +09:30
log . info ( f " VirtualBox VM ' { self . name } ' [ { self . id } ] is not allowed to use any adapter " )
2015-02-06 17:31:13 -07:00
self . _use_any_adapter = use_any_adapter
2014-07-17 15:28:02 -06:00
@property
def adapter_type ( self ) :
"""
Returns the adapter type for this VirtualBox VM instance .
: returns : adapter type ( string )
"""
return self . _adapter_type
@adapter_type.setter
def adapter_type ( self , adapter_type ) :
"""
Sets the adapter type for this VirtualBox VM instance .
: param adapter_type : adapter type ( string )
"""
self . _adapter_type = adapter_type
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ]: adapter type changed to {adapter_type} " . format (
name = self . name , id = self . id , adapter_type = adapter_type
)
)
2014-10-30 18:53:17 -06:00
2018-10-15 17:05:49 +07:00
async def _get_vm_info ( self ) :
2014-10-30 18:53:17 -06:00
"""
Returns this VM info .
: returns : dict of info
"""
vm_info = { }
2021-04-13 18:46:50 +09:30
results = await self . manager . execute (
" showvminfo " , [ " --machinereadable " , " -- " , self . _vmname ]
) # "--" is to protect against vm names containing the "-" character
2014-10-30 18:53:17 -06:00
for info in results :
try :
2021-04-13 18:46:50 +09:30
name , value = info . split ( " = " , 1 )
2014-10-30 18:53:17 -06:00
except ValueError :
continue
2014-11-09 11:50:47 -07:00
vm_info [ name . strip ( ' " ' ) ] = value . strip ( ' " ' )
2014-10-30 18:53:17 -06:00
return vm_info
def _get_pipe_name ( self ) :
"""
Returns the pipe name to create a serial connection .
: returns : pipe path ( string )
"""
2022-01-19 22:28:36 +10:30
pipe_name = os . path . join ( tempfile . gettempdir ( ) , " gns3_vbox " , f " { self . id } " )
try :
os . makedirs ( os . path . dirname ( pipe_name ) , exist_ok = True )
except OSError as e :
raise VirtualBoxError ( f " Could not create the VirtualBox pipe directory: { e } " )
2014-10-30 18:53:17 -06:00
return pipe_name
2018-10-15 17:05:49 +07:00
async def _set_serial_console ( self ) :
2014-10-30 18:53:17 -06:00
"""
Configures the first serial port to allow a serial console connection .
"""
# activate the first serial port
2018-10-15 17:05:49 +07:00
await self . _modify_vm ( " --uart1 0x3F8 4 " )
2014-10-30 18:53:17 -06:00
# set server mode with a pipe on the first serial port
pipe_name = self . _get_pipe_name ( )
2019-03-18 23:29:18 +07:00
args = [ self . _uuid , " --uartmode1 " , " server " , pipe_name ]
2018-10-15 17:05:49 +07:00
await self . manager . execute ( " modifyvm " , args )
2014-10-30 18:53:17 -06:00
2018-10-15 17:05:49 +07:00
async def _storage_attach ( self , params ) :
2014-11-09 11:50:47 -07:00
"""
Change storage medium in this VM .
: param params : params to use with sub - command storageattach
"""
args = shlex . split ( params )
2019-03-18 23:29:18 +07:00
await self . manager . execute ( " storageattach " , [ self . _uuid ] + args )
2014-11-09 11:50:47 -07:00
2018-10-15 17:05:49 +07:00
async def _get_nic_attachements ( self , maximum_adapters ) :
2014-10-30 18:53:17 -06:00
"""
Returns NIC attachements .
: param maximum_adapters : maximum number of supported adapters
: returns : list of adapters with their Attachment setting ( NAT , bridged etc . )
"""
nics = [ ]
2018-10-15 17:05:49 +07:00
vm_info = await self . _get_vm_info ( )
2015-02-13 15:41:56 -07:00
for adapter_number in range ( 0 , maximum_adapters ) :
2021-04-13 18:37:58 +09:30
entry = f " nic { adapter_number + 1 } "
2014-10-30 18:53:17 -06:00
if entry in vm_info :
value = vm_info [ entry ]
2015-03-11 12:05:22 -06:00
nics . append ( value . lower ( ) )
2014-10-30 18:53:17 -06:00
else :
nics . append ( None )
return nics
2018-10-15 17:05:49 +07:00
async def _set_network_options ( self ) :
2014-10-30 18:53:17 -06:00
"""
Configures network options .
"""
2018-10-15 17:05:49 +07:00
nic_attachments = await self . _get_nic_attachements ( self . _maximum_adapters )
2015-04-03 12:08:18 +02:00
for adapter_number in range ( 0 , self . _adapters ) :
2015-03-05 19:11:33 -07:00
attachment = nic_attachments [ adapter_number ]
if attachment == " null " :
# disconnect the cable if no backend is attached.
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --cableconnected { adapter_number + 1 } off " )
2015-04-03 19:08:29 -06:00
if attachment == " none " :
# set the backend to null to avoid a difference in the number of interfaces in the Guest.
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --nic { adapter_number + 1 } null " )
await self . _modify_vm ( f " --cableconnected { adapter_number + 1 } off " )
2016-06-24 18:35:39 -06:00
2017-07-11 13:42:47 +02:00
# use a local UDP tunnel to connect to uBridge instead
if adapter_number not in self . _local_udp_tunnels :
self . _local_udp_tunnels [ adapter_number ] = self . _create_local_udp_tunnel ( )
nio = self . _local_udp_tunnels [ adapter_number ] [ 0 ]
2016-06-24 18:35:39 -06:00
2015-02-06 17:31:13 -07:00
if nio :
2017-11-15 16:41:33 +07:00
if not self . _use_any_adapter and attachment in ( " nat " , " bridged " , " intnet " , " hostonly " , " natnetwork " ) :
continue
2014-10-30 18:53:17 -06:00
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --nictrace { adapter_number + 1 } off " )
2018-04-02 22:27:12 +07:00
custom_adapter = self . _get_custom_adapter_settings ( adapter_number )
adapter_type = custom_adapter . get ( " adapter_type " , self . _adapter_type )
2014-10-30 18:53:17 -06:00
vbox_adapter_type = " 82540EM "
2018-04-02 22:27:12 +07:00
if adapter_type == " PCnet-PCI II (Am79C970A) " :
2015-02-06 17:31:13 -07:00
vbox_adapter_type = " Am79C970A "
2018-04-02 22:27:12 +07:00
if adapter_type == " PCNet-FAST III (Am79C973) " :
2015-02-06 17:31:13 -07:00
vbox_adapter_type = " Am79C973 "
2018-04-02 22:27:12 +07:00
if adapter_type == " Intel PRO/1000 MT Desktop (82540EM) " :
2015-02-06 17:31:13 -07:00
vbox_adapter_type = " 82540EM "
2018-04-02 22:27:12 +07:00
if adapter_type == " Intel PRO/1000 T Server (82543GC) " :
2015-02-06 17:31:13 -07:00
vbox_adapter_type = " 82543GC "
2018-04-02 22:27:12 +07:00
if adapter_type == " Intel PRO/1000 MT Server (82545EM) " :
2015-02-06 17:31:13 -07:00
vbox_adapter_type = " 82545EM "
2018-04-02 22:27:12 +07:00
if adapter_type == " Paravirtualized Network (virtio-net) " :
2015-02-06 17:31:13 -07:00
vbox_adapter_type = " virtio "
2021-04-13 18:37:58 +09:30
args = [ self . _uuid , f " --nictype { adapter_number + 1 } " , vbox_adapter_type ]
2018-10-15 17:05:49 +07:00
await self . manager . execute ( " modifyvm " , args )
2014-10-30 18:53:17 -06:00
2015-05-06 15:21:39 -06:00
if isinstance ( nio , NIOUDP ) :
2021-04-13 18:37:58 +09:30
log . debug ( f " setting UDP params on adapter { adapter_number } " )
await self . _modify_vm ( f " --nic { adapter_number + 1 } generic " )
await self . _modify_vm ( f " --nicgenericdrv { adapter_number + 1 } UDPTunnel " )
await self . _modify_vm ( f " --nicproperty { adapter_number + 1 } sport= { nio . lport } " )
await self . _modify_vm ( f " --nicproperty { adapter_number + 1 } dest= { nio . rhost } " )
await self . _modify_vm ( f " --nicproperty { adapter_number + 1 } dport= { nio . rport } " )
2018-03-19 16:26:12 +07:00
if nio . suspend :
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --cableconnected { adapter_number + 1 } off " )
2018-03-19 16:26:12 +07:00
else :
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --cableconnected { adapter_number + 1 } on " )
2014-10-30 18:53:17 -06:00
if nio . capturing :
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --nictrace { adapter_number + 1 } on " )
await self . _modify_vm ( f ' --nictracefile { adapter_number + 1 } " { nio . pcap_output_file } " ' )
2014-10-30 18:53:17 -06:00
2017-07-11 13:42:47 +02:00
if not self . _ethernet_adapters [ adapter_number ] . get_nio ( 0 ) :
2021-04-13 18:37:58 +09:30
await self . _modify_vm ( f " --cableconnected { adapter_number + 1 } off " )
2017-01-05 14:30:23 +08:00
2015-04-03 12:08:18 +02:00
for adapter_number in range ( self . _adapters , self . _maximum_adapters ) :
2021-04-13 18:37:58 +09:30
log . debug ( f " disabling remaining adapter { adapter_number } " )
await self . _modify_vm ( f " --nic { adapter_number + 1 } none " )
2014-10-30 18:53:17 -06:00
2018-10-15 17:05:49 +07:00
async def _create_linked_clone ( self ) :
2014-11-15 16:05:55 -07:00
"""
Creates a new linked clone .
"""
2014-11-09 11:50:47 -07:00
gns3_snapshot_exists = False
2018-10-15 17:05:49 +07:00
vm_info = await self . _get_vm_info ( )
2014-11-09 11:50:47 -07:00
for entry , value in vm_info . items ( ) :
if entry . startswith ( " SnapshotName " ) and value == " GNS3 Linked Base for clones " :
gns3_snapshot_exists = True
if not gns3_snapshot_exists :
2019-03-18 23:29:18 +07:00
result = await self . manager . execute ( " snapshot " , [ self . _uuid , " take " , " GNS3 Linked Base for clones " ] )
2021-04-13 18:37:58 +09:30
log . debug ( f " GNS3 snapshot created: { result } " )
2014-11-09 11:50:47 -07:00
2021-04-13 18:46:50 +09:30
args = [
self . _uuid ,
" --snapshot " ,
" GNS3 Linked Base for clones " ,
" --options " ,
" link " ,
" --name " ,
self . name ,
" --basefolder " ,
self . working_dir ,
" --register " ,
]
2014-11-09 11:50:47 -07:00
2018-10-15 17:05:49 +07:00
result = await self . manager . execute ( " clonevm " , args )
2021-04-13 18:37:58 +09:30
log . debug ( f " VirtualBox VM: { result } cloned " )
2015-01-11 12:22:59 +00:00
2019-11-03 16:34:24 +08:00
# refresh the UUID and vmname to match with the clone
2014-11-09 11:50:47 -07:00
self . _vmname = self . _name
2019-11-03 16:34:24 +08:00
await self . _refresh_vm_uuid ( )
2019-03-18 23:29:18 +07:00
await self . manager . execute ( " setextradata " , [ self . _uuid , " GNS3/Clone " , " yes " ] )
2015-01-11 18:24:13 -07:00
2016-11-04 11:07:43 +01:00
# We create a reset snapshot in order to simplify life of user who want to rollback their VM
# Warning: Do not document this it's seem buggy we keep it because Raizo students use it.
try :
2019-03-18 23:29:18 +07:00
args = [ self . _uuid , " take " , " reset " ]
2018-10-15 17:05:49 +07:00
result = await self . manager . execute ( " snapshot " , args )
2021-04-13 18:37:58 +09:30
log . debug ( f " Snapshot ' reset ' created: { result } " )
2016-11-04 11:07:43 +01:00
# It seem sometimes this failed due to internal race condition of Vbox
# we have no real explanation of this.
except VirtualBoxError :
2018-03-15 14:17:39 +07:00
log . warning ( " Snapshot ' reset ' not created " )
2014-11-09 11:50:47 -07:00
2016-11-07 14:10:44 +01:00
os . makedirs ( os . path . join ( self . working_dir , self . _vmname ) , exist_ok = True )
2018-10-15 17:05:49 +07:00
async def _start_console ( self ) :
2015-01-04 14:56:17 -07:00
"""
Starts remote console support for this VM .
"""
2018-03-24 18:11:21 +07:00
if self . console and self . console_type == " telnet " :
2018-10-04 15:44:13 +02:00
pipe_name = self . _get_pipe_name ( )
try :
2018-10-15 17:05:49 +07:00
self . _remote_pipe = await asyncio_open_serial ( pipe_name )
2018-10-04 15:44:13 +02:00
except OSError as e :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Could not open serial pipe ' { pipe_name } ' : { e } " )
2021-04-13 18:46:50 +09:30
server = AsyncioTelnetServer ( reader = self . _remote_pipe , writer = self . _remote_pipe , binary = True , echo = True )
2018-04-18 17:08:42 +08:00
try :
2021-04-13 18:46:50 +09:30
self . _telnet_server = await asyncio . start_server (
server . run , self . _manager . port_manager . console_host , self . console
)
2018-04-18 17:08:42 +08:00
except OSError as e :
2021-04-13 18:46:50 +09:30
self . project . emit (
" log.warning " ,
{
" message " : f " Could not start Telnet server on socket { self . _manager . port_manager . console_host } : { self . console } : { e } "
} ,
)
2015-01-04 14:56:17 -07:00
2018-10-15 17:05:49 +07:00
async def _stop_remote_console ( self ) :
2015-01-04 14:56:17 -07:00
"""
Stops remote console support for this VM .
"""
2016-11-07 11:16:51 +01:00
if self . _telnet_server :
self . _telnet_server . close ( )
2018-10-15 17:05:49 +07:00
await self . _telnet_server . wait_closed ( )
2016-11-18 11:27:50 +01:00
self . _remote_pipe . close ( )
self . _telnet_server = None
2015-01-04 14:56:17 -07:00
2020-07-26 18:27:18 +09:30
async def reset_console ( self ) :
"""
Reset the console .
"""
await self . _stop_remote_console ( )
await self . _start_console ( )
2018-03-24 18:11:21 +07:00
@BaseNode.console_type.setter
def console_type ( self , new_console_type ) :
"""
Sets the console type for this VirtualBox VM .
: param new_console_type : console type ( string )
"""
if self . is_running ( ) and self . console_type != new_console_type :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f ' " { self . _name } " must be stopped to change the console type to { new_console_type } ' )
2018-03-24 18:11:21 +07:00
super ( VirtualBoxVM , VirtualBoxVM ) . console_type . __set__ ( self , new_console_type )
2018-10-15 17:05:49 +07:00
async def adapter_add_nio_binding ( self , adapter_number , nio ) :
2014-07-17 15:28:02 -06:00
"""
2015-02-06 17:31:13 -07:00
Adds an adapter NIO binding .
2014-07-17 15:28:02 -06:00
2015-02-13 15:41:56 -07:00
: param adapter_number : adapter number
2014-07-17 15:28:02 -06:00
: param nio : NIO instance to add to the slot / port
"""
try :
2015-02-13 15:41:56 -07:00
adapter = self . _ethernet_adapters [ adapter_number ]
2015-05-27 16:38:57 +02:00
except KeyError :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
" Adapter {adapter_number} doesn ' t exist on VirtualBox VM ' {name} ' " . format (
name = self . name , adapter_number = adapter_number
)
)
2014-07-17 15:28:02 -06:00
2017-11-15 16:41:33 +07:00
# check if trying to connect to a nat, bridged, host-only or any other special adapter
2018-10-15 17:05:49 +07:00
nic_attachments = await self . _get_nic_attachements ( self . _maximum_adapters )
2017-11-15 16:41:33 +07:00
attachment = nic_attachments [ adapter_number ]
if attachment in ( " nat " , " bridged " , " intnet " , " hostonly " , " natnetwork " ) :
if not self . _use_any_adapter :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
" Attachment ' {attachment} ' is already configured on adapter {adapter_number} . "
" Please remove it or allow VirtualBox VM ' {name} ' to use any adapter. " . format (
attachment = attachment , adapter_number = adapter_number , name = self . name
)
)
2017-11-15 16:41:33 +07:00
elif self . is_running ( ) :
# dynamically configure an UDP tunnel attachment if the VM is already running
local_nio = self . _local_udp_tunnels [ adapter_number ] [ 0 ]
if local_nio and isinstance ( local_nio , NIOUDP ) :
2021-04-13 18:37:58 +09:30
await self . _control_vm ( f " nic { adapter_number + 1 } generic UDPTunnel " )
await self . _control_vm ( f " nicproperty { adapter_number + 1 } sport= { local_nio . lport } " )
await self . _control_vm ( f " nicproperty { adapter_number + 1 } dest= { local_nio . rhost } " )
await self . _control_vm ( f " nicproperty { adapter_number + 1 } dport= { local_nio . rport } " )
await self . _control_vm ( f " setlinkstate { adapter_number + 1 } on " )
2017-11-15 16:41:33 +07:00
2017-07-12 10:32:02 +02:00
if self . is_running ( ) :
try :
2021-04-13 18:46:50 +09:30
await self . add_ubridge_udp_connection (
f " VBOX- { self . _id } - { adapter_number } " , self . _local_udp_tunnels [ adapter_number ] [ 1 ] , nio
)
2017-07-12 10:32:02 +02:00
except KeyError :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
" Adapter {adapter_number} doesn ' t exist on VirtualBox VM ' {name} ' " . format (
name = self . name , adapter_number = adapter_number
)
)
2021-04-13 18:37:58 +09:30
await self . _control_vm ( f " setlinkstate { adapter_number + 1 } on " )
2016-02-10 19:08:34 -07:00
2014-07-17 15:28:02 -06:00
adapter . add_nio ( 0 , nio )
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ]: {nio} added to adapter {adapter_number} " . format (
name = self . name , id = self . id , nio = nio , adapter_number = adapter_number
)
)
2017-07-18 14:59:47 +02:00
2018-10-15 17:05:49 +07:00
async def adapter_update_nio_binding ( self , adapter_number , nio ) :
2017-07-18 14:59:47 +02:00
"""
2018-10-27 14:47:17 +07:00
Update an adapter NIO binding .
2017-07-18 14:59:47 +02:00
: param adapter_number : adapter number
2018-10-27 14:47:17 +07:00
: param nio : NIO instance to update on the adapter
2017-07-18 14:59:47 +02:00
"""
if self . is_running ( ) :
try :
2021-04-13 18:46:50 +09:30
await self . update_ubridge_udp_connection (
f " VBOX- { self . _id } - { adapter_number } " , self . _local_udp_tunnels [ adapter_number ] [ 1 ] , nio
)
2018-03-19 16:26:12 +07:00
if nio . suspend :
2021-04-13 18:37:58 +09:30
await self . _control_vm ( f " setlinkstate { adapter_number + 1 } off " )
2018-03-19 16:26:12 +07:00
else :
2021-04-13 18:37:58 +09:30
await self . _control_vm ( f " setlinkstate { adapter_number + 1 } on " )
2017-07-18 14:59:47 +02:00
except IndexError :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
' Adapter {adapter_number} does not exist on VirtualBox VM " {name} " ' . format (
name = self . _name , adapter_number = adapter_number
)
)
2014-07-17 15:28:02 -06:00
2018-10-15 17:05:49 +07:00
async def adapter_remove_nio_binding ( self , adapter_number ) :
2014-07-17 15:28:02 -06:00
"""
2015-02-06 17:31:13 -07:00
Removes an adapter NIO binding .
2014-07-17 15:28:02 -06:00
2015-02-13 15:41:56 -07:00
: param adapter_number : adapter number
2014-07-17 15:28:02 -06:00
: returns : NIO instance
"""
try :
2015-02-13 15:41:56 -07:00
adapter = self . _ethernet_adapters [ adapter_number ]
2015-05-27 16:38:57 +02:00
except KeyError :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
" Adapter {adapter_number} doesn ' t exist on VirtualBox VM ' {name} ' " . format (
name = self . name , adapter_number = adapter_number
)
)
2014-07-17 15:28:02 -06:00
2019-04-01 19:47:31 +07:00
await self . stop_capture ( adapter_number )
2017-07-12 10:32:02 +02:00
if self . is_running ( ) :
2021-04-13 18:37:58 +09:30
await self . _ubridge_send ( " bridge delete {name} " . format ( name = f " VBOX- { self . _id } - { adapter_number } " ) )
2018-10-15 17:05:49 +07:00
vm_state = await self . _get_vm_state ( )
2017-07-11 13:42:47 +02:00
if vm_state == " running " :
2021-04-13 18:37:58 +09:30
await self . _control_vm ( f " setlinkstate { adapter_number + 1 } off " )
2014-07-17 15:28:02 -06:00
nio = adapter . get_nio ( 0 )
2015-02-23 19:00:34 -07:00
if isinstance ( nio , NIOUDP ) :
2015-03-21 17:19:12 -06:00
self . manager . port_manager . release_udp_port ( nio . lport , self . _project )
2014-07-17 15:28:02 -06:00
adapter . remove_nio ( 0 )
2015-01-24 15:32:58 -07:00
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ]: {nio} removed from adapter {adapter_number} " . format (
name = self . name , id = self . id , nio = nio , adapter_number = adapter_number
)
)
2014-07-17 15:28:02 -06:00
return nio
2018-10-27 14:47:17 +07:00
def get_nio ( self , adapter_number ) :
2014-07-17 15:28:02 -06:00
"""
2018-10-27 14:47:17 +07:00
Gets an adapter NIO binding .
2014-07-17 15:28:02 -06:00
2015-02-13 15:41:56 -07:00
: param adapter_number : adapter number
2018-10-27 14:47:17 +07:00
: returns : NIO instance
2014-07-17 15:28:02 -06:00
"""
try :
2018-10-27 14:47:17 +07:00
adapter = self . ethernet_adapters [ adapter_number ]
2015-05-27 16:38:57 +02:00
except KeyError :
2021-04-13 18:46:50 +09:30
raise VirtualBoxError (
" Adapter {adapter_number} doesn ' t exist on VirtualBox VM ' {name} ' " . format (
name = self . name , adapter_number = adapter_number
)
)
2014-07-17 15:28:02 -06:00
nio = adapter . get_nio ( 0 )
2015-06-01 15:42:17 -06:00
if not nio :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Adapter { adapter_number } is not connected " )
2015-06-01 15:42:17 -06:00
2018-10-27 14:47:17 +07:00
return nio
def is_running ( self ) :
"""
: returns : True if the vm is not stopped
"""
return self . ubridge is not None
async def start_capture ( self , adapter_number , output_file ) :
"""
Starts a packet capture .
: param adapter_number : adapter number
: param output_file : PCAP destination file for the capture
"""
nio = self . get_nio ( adapter_number )
2014-07-17 15:28:02 -06:00
if nio . capturing :
2021-04-13 18:37:58 +09:30
raise VirtualBoxError ( f " Packet capture is already activated on adapter { adapter_number } " )
2014-07-17 15:28:02 -06:00
2019-04-01 20:58:18 +07:00
nio . start_packet_capture ( output_file )
2016-12-14 12:01:34 +01:00
if self . ubridge :
2021-04-13 18:46:50 +09:30
await self . _ubridge_send (
' bridge start_capture {name} " {output_file} " ' . format (
name = f " VBOX- { self . _id } - { adapter_number } " , output_file = output_file
)
)
log . info (
" VirtualBox VM ' {name} ' [ {id} ]: starting packet capture on adapter {adapter_number} " . format (
name = self . name , id = self . id , adapter_number = adapter_number
)
)
2014-07-17 15:28:02 -06:00
2018-10-15 17:05:49 +07:00
async def stop_capture ( self , adapter_number ) :
2014-07-17 15:28:02 -06:00
"""
Stops a packet capture .
2015-02-13 15:41:56 -07:00
: param adapter_number : adapter number
2014-07-17 15:28:02 -06:00
"""
2018-10-27 14:47:17 +07:00
nio = self . get_nio ( adapter_number )
2019-04-01 19:47:31 +07:00
if not nio . capturing :
return
2014-07-17 15:28:02 -06:00
2019-04-01 20:58:18 +07:00
nio . stop_packet_capture ( )
2016-12-14 12:01:34 +01:00
if self . ubridge :
2021-04-13 18:46:50 +09:30
await self . _ubridge_send ( " bridge stop_capture {name} " . format ( name = f " VBOX- { self . _id } - { adapter_number } " ) )
2016-06-24 18:35:39 -06:00
2021-04-13 18:46:50 +09:30
log . info (
" VirtualBox VM ' {name} ' [ {id} ]: stopping packet capture on adapter {adapter_number} " . format (
name = self . name , id = self . id , adapter_number = adapter_number
)
)