2015-02-11 14:31:21 +01: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/>.
import pytest
import aiohttp
import asyncio
import os
import stat
2015-03-16 11:52:22 +01:00
import socket
2015-04-27 15:09:42 +02:00
import sys
2015-10-21 14:28:39 +02:00
import uuid
import shutil
2016-11-06 21:22:48 +01:00
from tests . utils import asyncio_patch , AsyncioMagicMock
2015-02-11 14:31:21 +01:00
2016-11-06 21:22:48 +01:00
from unittest . mock import patch , MagicMock , PropertyMock , call
2015-04-27 15:09:42 +02:00
pytestmark = pytest . mark . skipif ( sys . platform . startswith ( " win " ) , reason = " Not supported on Windows " )
if not sys . platform . startswith ( " win " ) :
2016-04-15 17:57:06 +02:00
from gns3server . compute . iou . iou_vm import IOUVM
from gns3server . compute . iou . iou_error import IOUError
from gns3server . compute . iou import IOU
2015-04-27 15:09:42 +02:00
2015-03-19 15:36:06 +01:00
from gns3server . config import Config
2015-02-11 14:31:21 +01:00
2015-03-19 17:42:43 +01:00
2016-10-24 21:39:35 +02:00
@pytest.fixture
2015-02-11 14:31:21 +01:00
def manager ( port_manager ) :
m = IOU . instance ( )
m . port_manager = port_manager
return m
@pytest.fixture ( scope = " function " )
2015-03-16 11:52:22 +01:00
def vm ( project , manager , tmpdir , fake_iou_bin , iourc_file ) :
2015-10-21 14:28:39 +02:00
vm = IOUVM ( " test " , str ( uuid . uuid4 ( ) ) , project , manager )
2015-02-11 14:31:21 +01:00
config = manager . config . get_section_config ( " IOU " )
2015-03-16 11:52:22 +01:00
config [ " iourc_path " ] = iourc_file
2015-02-11 14:31:21 +01:00
manager . config . set_section_config ( " IOU " , config )
2015-11-10 15:21:10 +01:00
vm . path = " iou.bin "
2015-02-11 14:31:21 +01:00
return vm
2015-03-16 11:52:22 +01:00
@pytest.fixture
def iourc_file ( tmpdir ) :
path = str ( tmpdir / " iourc " )
with open ( path , " w+ " ) as f :
hostname = socket . gethostname ( )
f . write ( " [license] \n {} = aaaaaaaaaaaaaaaa; " . format ( hostname ) )
return path
2015-02-11 14:31:21 +01:00
@pytest.fixture
2016-06-07 15:34:04 +02:00
def fake_iou_bin ( images_dir ) :
2015-02-11 14:31:21 +01:00
""" Create a fake IOU image on disk """
2016-06-07 15:34:04 +02:00
path = os . path . join ( images_dir , " iou.bin " )
2015-02-11 14:31:21 +01:00
with open ( path , " w+ " ) as f :
f . write ( ' \x7f ELF \x01 \x01 \x01 ' )
os . chmod ( path , stat . S_IREAD | stat . S_IEXEC )
return path
def test_vm ( project , manager ) :
vm = IOUVM ( " test " , " 00010203-0405-0607-0809-0a0b0c0d0e0f " , project , manager )
assert vm . name == " test "
assert vm . id == " 00010203-0405-0607-0809-0a0b0c0d0e0f "
2015-06-06 15:15:03 -06:00
def test_vm_startup_config_content ( project , manager ) :
2015-04-12 15:09:37 -06:00
vm = IOUVM ( " test " , " 00010203-0405-0607-0808-0a0b0c0d0e0f " , project , manager )
2015-06-06 15:15:03 -06:00
vm . startup_config_content = " hostname % h "
2015-02-16 10:05:17 +01:00
assert vm . name == " test "
2015-06-06 15:15:03 -06:00
assert vm . startup_config_content == " hostname test "
2015-02-16 10:05:17 +01:00
assert vm . id == " 00010203-0405-0607-0808-0a0b0c0d0e0f "
2015-02-13 22:16:43 +01:00
2016-11-06 21:22:48 +01:00
def test_start ( loop , vm ) :
2015-02-11 14:31:21 +01:00
2016-11-06 21:22:48 +01:00
mock_process = MagicMock ( )
2015-02-11 14:31:21 +01:00
2016-11-06 21:22:48 +01:00
vm . _check_requirements = AsyncioMagicMock ( return_value = True )
vm . _check_iou_licence = AsyncioMagicMock ( return_value = True )
vm . _start_ubridge = AsyncioMagicMock ( return_value = True )
vm . _ubridge_send = AsyncioMagicMock ( )
2015-03-11 23:09:01 -06:00
2016-11-06 21:22:48 +01:00
with asyncio_patch ( " asyncio.create_subprocess_exec " , return_value = mock_process ) as mock_exec :
mock_process . returncode = None
loop . run_until_complete ( asyncio . async ( vm . start ( ) ) )
assert vm . is_running ( )
assert vm . command_line == ' ' . join ( mock_exec . call_args [ 0 ] )
2015-02-11 14:31:21 +01:00
2016-11-06 21:22:48 +01:00
assert vm . _check_requirements . called
assert vm . _check_iou_licence . called
assert vm . _start_ubridge . called
vm . _ubridge_send . assert_any_call ( " iol_bridge delete IOL-BRIDGE-513 " )
vm . _ubridge_send . assert_any_call ( " iol_bridge create IOL-BRIDGE-513 513 " )
vm . _ubridge_send . assert_any_call ( " iol_bridge start IOL-BRIDGE-513 " )
2016-03-08 16:12:46 +01:00
2016-11-06 21:22:48 +01:00
def test_start_with_iourc ( loop , vm , tmpdir ) :
2015-02-17 16:40:45 +01:00
fake_file = str ( tmpdir / " iourc " )
with open ( fake_file , " w+ " ) as f :
f . write ( " 1 " )
2015-03-26 17:43:00 +01:00
mock_process = MagicMock ( )
2016-11-06 21:22:48 +01:00
vm . _check_requirements = AsyncioMagicMock ( return_value = True )
vm . _check_iou_licence = AsyncioMagicMock ( return_value = True )
vm . _start_ioucon = AsyncioMagicMock ( return_value = True )
vm . _start_ubridge = AsyncioMagicMock ( return_value = True )
vm . _ubridge_send = AsyncioMagicMock ( )
with patch ( " gns3server.config.Config.get_section_config " , return_value = { " iourc_path " : fake_file } ) :
with asyncio_patch ( " asyncio.create_subprocess_exec " , return_value = mock_process ) as exec_mock :
mock_process . returncode = None
loop . run_until_complete ( asyncio . async ( vm . start ( ) ) )
assert vm . is_running ( )
arsgs , kwargs = exec_mock . call_args
assert kwargs [ " env " ] [ " IOURC " ] == fake_file
def test_rename_nvram_file ( loop , vm ) :
2015-02-13 20:57:20 +01:00
"""
It should rename the nvram file to the correct name before launching the VM
"""
with open ( os . path . join ( vm . working_dir , " nvram_0000 {} " . format ( vm . application_id + 1 ) ) , ' w+ ' ) as f :
f . write ( " 1 " )
2015-03-05 17:00:25 +01:00
with open ( os . path . join ( vm . working_dir , " vlan.dat-0000 {} " . format ( vm . application_id + 1 ) ) , ' w+ ' ) as f :
f . write ( " 1 " )
2015-02-13 20:57:20 +01:00
vm . _rename_nvram_file ( )
assert os . path . exists ( os . path . join ( vm . working_dir , " nvram_0000 {} " . format ( vm . application_id ) ) )
2015-03-05 17:00:25 +01:00
assert os . path . exists ( os . path . join ( vm . working_dir , " vlan.dat-0000 {} " . format ( vm . application_id ) ) )
2015-02-13 20:57:20 +01:00
2015-02-11 14:31:21 +01:00
def test_stop ( loop , vm ) :
process = MagicMock ( )
2016-11-06 21:22:48 +01:00
vm . _check_requirements = AsyncioMagicMock ( return_value = True )
vm . _check_iou_licence = AsyncioMagicMock ( return_value = True )
vm . _start_ioucon = AsyncioMagicMock ( return_value = True )
vm . _start_ubridge = AsyncioMagicMock ( return_value = True )
vm . _ubridge_send = AsyncioMagicMock ( )
2015-02-11 14:31:21 +01:00
# Wait process kill success
future = asyncio . Future ( )
future . set_result ( True )
process . wait . return_value = future
2016-11-06 21:22:48 +01:00
with asyncio_patch ( " asyncio.create_subprocess_exec " , return_value = process ) :
with asyncio_patch ( " gns3server.utils.asyncio.wait_for_process_termination " ) :
loop . run_until_complete ( asyncio . async ( vm . start ( ) ) )
process . returncode = None
assert vm . is_running ( )
loop . run_until_complete ( asyncio . async ( vm . stop ( ) ) )
assert vm . is_running ( ) is False
process . terminate . assert_called_with ( )
2015-02-11 14:31:21 +01:00
def test_reload ( loop , vm , fake_iou_bin ) :
process = MagicMock ( )
2016-11-06 21:22:48 +01:00
vm . _check_requirements = AsyncioMagicMock ( return_value = True )
vm . _check_iou_licence = AsyncioMagicMock ( return_value = True )
vm . _start_ioucon = AsyncioMagicMock ( return_value = True )
vm . _start_ubridge = AsyncioMagicMock ( return_value = True )
vm . _ubridge_send = AsyncioMagicMock ( )
2015-02-11 14:31:21 +01:00
# Wait process kill success
future = asyncio . Future ( )
future . set_result ( True )
process . wait . return_value = future
2015-03-26 17:43:00 +01:00
process . returncode = None
2015-02-11 14:31:21 +01:00
2016-11-06 21:22:48 +01:00
with asyncio_patch ( " asyncio.create_subprocess_exec " , return_value = process ) :
with asyncio_patch ( " gns3server.utils.asyncio.wait_for_process_termination " ) :
loop . run_until_complete ( asyncio . async ( vm . start ( ) ) )
assert vm . is_running ( )
loop . run_until_complete ( asyncio . async ( vm . reload ( ) ) )
assert vm . is_running ( ) is True
process . terminate . assert_called_with ( )
2015-02-11 14:31:21 +01:00
2015-02-16 19:14:45 +01:00
def test_close ( vm , port_manager , loop ) :
2016-04-15 17:57:06 +02:00
with asyncio_patch ( " gns3server.compute.iou.iou_vm.IOUVM._check_requirements " , return_value = True ) :
2015-02-11 14:31:21 +01:00
with asyncio_patch ( " asyncio.create_subprocess_exec " , return_value = MagicMock ( ) ) :
vm . start ( )
port = vm . console
2015-02-16 19:14:45 +01:00
loop . run_until_complete ( asyncio . async ( vm . close ( ) ) )
2015-02-11 14:31:21 +01:00
# Raise an exception if the port is not free
2015-03-21 17:19:12 -06:00
port_manager . reserve_tcp_port ( port , vm . project )
2015-02-11 14:31:21 +01:00
assert vm . is_running ( ) is False
2016-11-04 15:42:29 +01:00
def test_path ( vm , fake_iou_bin , config ) :
config . set_section_config ( " Server " , { " local " : True } )
vm . path = fake_iou_bin
assert vm . path == fake_iou_bin
2015-02-11 14:31:21 +01:00
2015-02-25 16:26:17 +01:00
def test_path_relative ( vm , fake_iou_bin , tmpdir ) :
2015-03-19 15:36:06 +01:00
vm . path = " iou.bin "
2015-02-25 16:26:17 +01:00
assert vm . path == fake_iou_bin
2016-11-04 15:42:29 +01:00
def test_path_invalid_bin ( vm , tmpdir , config ) :
2015-02-11 14:31:21 +01:00
2016-11-04 15:42:29 +01:00
config . set_section_config ( " Server " , { " local " : True } )
path = str ( tmpdir / " test.bin " )
2015-02-11 14:31:21 +01:00
2016-11-04 15:42:29 +01:00
with open ( path , " w+ " ) as f :
f . write ( " BUG " )
2015-02-11 14:31:21 +01:00
2016-11-04 15:42:29 +01:00
with pytest . raises ( IOUError ) :
vm . path = path
vm . _check_requirements ( )
2015-02-11 14:31:21 +01:00
def test_create_netmap_config ( vm ) :
vm . _create_netmap_config ( )
netmap_path = os . path . join ( vm . working_dir , " NETMAP " )
with open ( netmap_path ) as f :
content = f . read ( )
assert " 513:0/0 1:0/0 " in content
assert " 513:15/3 1:15/3 " in content
2015-02-16 17:20:07 +01:00
def test_build_command ( vm , loop ) :
2015-02-11 14:31:21 +01:00
2016-11-08 10:21:20 +01:00
assert loop . run_until_complete ( asyncio . async ( vm . _build_command ( ) ) ) == [ vm . path , str ( vm . application_id ) ]
2015-02-13 22:16:43 +01:00
2015-06-19 16:35:19 +02:00
2015-06-06 15:15:03 -06:00
def test_get_startup_config ( vm ) :
2015-02-13 22:16:43 +01:00
content = " service timestamps debug datetime msec \n service timestamps log datetime msec \n no service password-encryption "
2015-06-06 15:15:03 -06:00
vm . startup_config = content
assert vm . startup_config == content
2015-02-13 22:16:43 +01:00
2015-06-06 15:15:03 -06:00
def test_update_startup_config ( vm ) :
2015-02-13 22:16:43 +01:00
content = " service timestamps debug datetime msec \n service timestamps log datetime msec \n no service password-encryption "
2015-06-06 15:15:03 -06:00
vm . startup_config_content = content
filepath = os . path . join ( vm . working_dir , " startup-config.cfg " )
2015-02-13 22:16:43 +01:00
assert os . path . exists ( filepath )
with open ( filepath ) as f :
assert f . read ( ) == content
2015-06-06 15:15:03 -06:00
def test_update_startup_config_empty ( vm ) :
2015-04-06 21:30:57 +02:00
content = " service timestamps debug datetime msec \n service timestamps log datetime msec \n no service password-encryption "
2015-06-06 15:15:03 -06:00
vm . startup_config_content = content
filepath = os . path . join ( vm . working_dir , " startup-config.cfg " )
2015-04-06 21:30:57 +02:00
assert os . path . exists ( filepath )
with open ( filepath ) as f :
assert f . read ( ) == content
2015-06-06 15:15:03 -06:00
vm . startup_config_content = " "
2015-04-06 21:30:57 +02:00
with open ( filepath ) as f :
assert f . read ( ) == content
2015-06-06 15:15:03 -06:00
def test_update_startup_config_content_hostname ( vm ) :
2015-02-13 22:16:43 +01:00
content = " hostname % h \n "
vm . name = " pc1 "
2015-06-06 15:15:03 -06:00
vm . startup_config_content = content
with open ( vm . startup_config_file ) as f :
2015-02-13 22:16:43 +01:00
assert f . read ( ) == " hostname pc1 \n "
def test_change_name ( vm , tmpdir ) :
2015-06-06 15:15:03 -06:00
path = os . path . join ( vm . working_dir , " startup-config.cfg " )
2015-02-13 22:16:43 +01:00
vm . name = " world "
with open ( path , ' w+ ' ) as f :
f . write ( " hostname world " )
vm . name = " hello "
assert vm . name == " hello "
with open ( path ) as f :
assert f . read ( ) == " hostname hello "
2017-02-27 12:48:05 +01:00
# support hostname not sync
vm . name = " alpha "
with open ( path , ' w+ ' ) as f :
f . write ( " no service password-encryption \n hostname beta \n no ip icmp rate-limit unreachable " )
vm . name = " charlie "
assert vm . name == " charlie "
with open ( path ) as f :
assert f . read ( ) == " no service password-encryption \n hostname charlie \n no ip icmp rate-limit unreachable "
2015-02-16 17:20:07 +01:00
def test_library_check ( loop , vm ) :
with asyncio_patch ( " gns3server.utils.asyncio.subprocess_check_output " , return_value = " " ) as mock :
loop . run_until_complete ( asyncio . async ( vm . _library_check ( ) ) )
with asyncio_patch ( " gns3server.utils.asyncio.subprocess_check_output " , return_value = " libssl => not found " ) as mock :
with pytest . raises ( IOUError ) :
loop . run_until_complete ( asyncio . async ( vm . _library_check ( ) ) )
def test_enable_l1_keepalives ( loop , vm ) :
with asyncio_patch ( " gns3server.utils.asyncio.subprocess_check_output " , return_value = " *************************************************************** \n \n -l Enable Layer 1 keepalive messages \n -u <n> UDP port base for distributed networks \n " ) as mock :
command = [ " test " ]
loop . run_until_complete ( asyncio . async ( vm . _enable_l1_keepalives ( command ) ) )
assert command == [ " test " , " -l " ]
with asyncio_patch ( " gns3server.utils.asyncio.subprocess_check_output " , return_value = " *************************************************************** \n \n -u <n> UDP port base for distributed networks \n " ) as mock :
command = [ " test " ]
with pytest . raises ( IOUError ) :
loop . run_until_complete ( asyncio . async ( vm . _enable_l1_keepalives ( command ) ) )
assert command == [ " test " ]
2015-02-16 20:08:04 +01:00
2015-02-17 10:37:09 +01:00
def test_start_capture ( vm , tmpdir , manager , free_console_port , loop ) :
2015-02-16 20:08:04 +01:00
output_file = str ( tmpdir / " test.pcap " )
2016-06-24 18:35:39 -06:00
nio = manager . create_nio ( { " type " : " nio_udp " , " lport " : free_console_port , " rport " : free_console_port , " rhost " : " 127.0.0.1 " } )
2016-01-11 15:19:15 +01:00
loop . run_until_complete ( asyncio . async ( vm . adapter_add_nio_binding ( 0 , 0 , nio ) ) )
2015-02-17 10:37:09 +01:00
loop . run_until_complete ( asyncio . async ( vm . start_capture ( 0 , 0 , output_file ) ) )
2015-02-16 20:08:04 +01:00
assert vm . _adapters [ 0 ] . get_nio ( 0 ) . capturing
2015-02-17 10:37:09 +01:00
def test_stop_capture ( vm , tmpdir , manager , free_console_port , loop ) :
2015-02-16 20:08:04 +01:00
output_file = str ( tmpdir / " test.pcap " )
2016-06-24 18:35:39 -06:00
nio = manager . create_nio ( { " type " : " nio_udp " , " lport " : free_console_port , " rport " : free_console_port , " rhost " : " 127.0.0.1 " } )
2016-01-11 15:19:15 +01:00
loop . run_until_complete ( asyncio . async ( vm . adapter_add_nio_binding ( 0 , 0 , nio ) ) )
2015-02-17 10:37:09 +01:00
loop . run_until_complete ( vm . start_capture ( 0 , 0 , output_file ) )
2015-02-16 20:08:04 +01:00
assert vm . _adapters [ 0 ] . get_nio ( 0 ) . capturing
2015-02-17 10:37:09 +01:00
loop . run_until_complete ( asyncio . async ( vm . stop_capture ( 0 , 0 ) ) )
2015-02-17 16:40:45 +01:00
assert vm . _adapters [ 0 ] . get_nio ( 0 ) . capturing is False
2015-02-25 16:35:13 +01:00
2015-02-25 18:55:35 -07:00
def test_get_legacy_vm_workdir ( ) :
2015-02-25 16:35:13 +01:00
2015-02-26 10:45:37 +01:00
assert IOU . get_legacy_vm_workdir ( 42 , " bla " ) == " iou/device-42 "
2015-03-16 11:52:22 +01:00
def test_invalid_iou_file ( loop , vm , iourc_file ) :
hostname = socket . gethostname ( )
loop . run_until_complete ( asyncio . async ( vm . _check_iou_licence ( ) ) )
# Missing ;
with pytest . raises ( IOUError ) :
with open ( iourc_file , " w+ " ) as f :
f . write ( " [license] \n {} = aaaaaaaaaaaaaaaa " . format ( hostname ) )
loop . run_until_complete ( asyncio . async ( vm . _check_iou_licence ( ) ) )
# Key too short
with pytest . raises ( IOUError ) :
with open ( iourc_file , " w+ " ) as f :
f . write ( " [license] \n {} = aaaaaaaaaaaaaa; " . format ( hostname ) )
loop . run_until_complete ( asyncio . async ( vm . _check_iou_licence ( ) ) )
# Invalid hostname
with pytest . raises ( IOUError ) :
with open ( iourc_file , " w+ " ) as f :
f . write ( " [license] \n bla = aaaaaaaaaaaaaa; " )
loop . run_until_complete ( asyncio . async ( vm . _check_iou_licence ( ) ) )
# Missing licence section
with pytest . raises ( IOUError ) :
with open ( iourc_file , " w+ " ) as f :
f . write ( " [licensetest] \n {} = aaaaaaaaaaaaaaaa; " )
loop . run_until_complete ( asyncio . async ( vm . _check_iou_licence ( ) ) )
# Broken config file
with pytest . raises ( IOUError ) :
with open ( iourc_file , " w+ " ) as f :
f . write ( " [ " )
loop . run_until_complete ( asyncio . async ( vm . _check_iou_licence ( ) ) )
# Missing file
with pytest . raises ( IOUError ) :
os . remove ( iourc_file )
loop . run_until_complete ( asyncio . async ( vm . _check_iou_licence ( ) ) )
2015-03-17 16:31:45 +01:00
def test_iourc_content ( vm ) :
vm . iourc_content = " test "
with open ( os . path . join ( vm . temporary_directory , " iourc " ) ) as f :
assert f . read ( ) == " test "
2015-10-21 14:28:39 +02:00
2016-03-21 16:53:11 +01:00
def test_iourc_content_fix_carriage_return ( vm ) :
vm . iourc_content = " test \r \n 12 "
with open ( os . path . join ( vm . temporary_directory , " iourc " ) ) as f :
assert f . read ( ) == " test \n 12 "
2015-10-21 14:28:39 +02:00
def test_extract_configs ( vm ) :
assert vm . extract_configs ( ) == ( None , None )
with open ( os . path . join ( vm . working_dir , " nvram_00001 " ) , " w+ " ) as f :
f . write ( " CORRUPTED " )
assert vm . extract_configs ( ) == ( None , None )
with open ( os . path . join ( vm . working_dir , " nvram_00001 " ) , " w+ " ) as f :
f . write ( " CORRUPTED " )
assert vm . extract_configs ( ) == ( None , None )
shutil . copy ( " tests/resources/nvram_iou " , os . path . join ( vm . working_dir , " nvram_00001 " ) )
startup_config , private_config = vm . extract_configs ( )
assert len ( startup_config ) == 1392
assert len ( private_config ) == 0