2013-07-24 10:15:37 +02:00
/*
2014-02-27 17:48:23 -07:00
* Copyright ( C ) 2013 , 2014 Red Hat , Inc .
2013-07-24 10:15:37 +02:00
*
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
* version 2.1 of the License , or ( at your option ) any later version .
*
* This library 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
* Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library . If not , see
* < http : //www.gnu.org/licenses/>.
*
*/
# include <config.h>
# include "testutils.h"
# include "testutilsqemu.h"
# include "qemumonitortestutils.h"
# include "qemu/qemu_conf.h"
# include "qemu/qemu_agent.h"
# include "virerror.h"
# define VIR_FROM_THIS VIR_FROM_NONE
2016-12-19 23:35:02 +01:00
virQEMUDriver driver ;
2020-11-07 13:12:53 +04:00
static int
testQemuAgentSSHKeys ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-06-14 13:51:17 +02:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
g_auto ( GStrv ) keys = NULL ;
2020-11-07 13:12:53 +04:00
int nkeys = 0 ;
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-06-14 13:51:17 +02:00
return - 1 ;
2020-11-07 13:12:53 +04:00
if ( qemuMonitorTestAddItem ( test , " guest-ssh-get-authorized-keys " ,
" { \" return \" : { "
" \" keys \" : [ "
" \" algo1 key1 comments1 \" , "
" \" algo2 key2 comments2 \" "
" ] "
" }} " ) < 0 )
2021-06-14 13:51:17 +02:00
return - 1 ;
2020-11-07 13:12:53 +04:00
if ( qemuMonitorTestAddItem ( test , " guest-ssh-add-authorized-keys " ,
" { \" return \" : {} } " ) < 0 )
2021-06-14 13:51:17 +02:00
return - 1 ;
2020-11-07 13:12:53 +04:00
if ( qemuMonitorTestAddItem ( test , " guest-ssh-remove-authorized-keys " ,
" { \" return \" : {} } " ) < 0 )
2021-06-14 13:51:17 +02:00
return - 1 ;
2020-11-07 13:12:53 +04:00
if ( ( nkeys = qemuAgentSSHGetAuthorizedKeys ( qemuMonitorTestGetAgent ( test ) ,
" user " ,
& keys ) ) < 0 )
2021-06-14 13:51:17 +02:00
return - 1 ;
2020-11-07 13:12:53 +04:00
if ( nkeys ! = 2 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" expected 2 keys, got %d " , nkeys ) ;
2021-06-14 13:51:17 +02:00
return - 1 ;
2020-11-07 13:12:53 +04:00
}
if ( STRNEQ ( keys [ 1 ] , " algo2 key2 comments2 " ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " Unexpected key returned: %s " , keys [ 1 ] ) ;
2021-06-14 13:51:17 +02:00
return - 1 ;
2020-11-07 13:12:53 +04:00
}
2021-06-14 13:51:17 +02:00
if ( qemuAgentSSHAddAuthorizedKeys ( qemuMonitorTestGetAgent ( test ) ,
" user " ,
( const char * * ) keys ,
nkeys ,
true ) < 0 )
return - 1 ;
2020-11-07 13:12:53 +04:00
2021-06-14 13:51:17 +02:00
if ( qemuAgentSSHRemoveAuthorizedKeys ( qemuMonitorTestGetAgent ( test ) ,
" user " ,
( const char * * ) keys ,
nkeys ) < 0 )
return - 1 ;
2020-11-07 13:12:53 +04:00
2021-06-14 13:51:17 +02:00
return 0 ;
2020-11-07 13:12:53 +04:00
}
2013-07-24 10:15:37 +02:00
static int
testQemuAgentFSFreeze ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2014-05-01 20:06:19 -04:00
const char * mountpoints [ ] = { " /fs1 " , " /fs2 " , " /fs3 " , " /fs4 " , " /fs5 " } ;
2021-11-01 10:20:57 +01:00
int rc ;
2013-07-24 10:15:37 +02:00
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-24 10:15:37 +02:00
2014-08-08 16:03:41 -04:00
if ( qemuMonitorTestAddItem ( test , " guest-fsfreeze-freeze-list " ,
2013-07-24 10:15:37 +02:00
" { \" return \" : 5 } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-24 10:15:37 +02:00
if ( qemuMonitorTestAddItem ( test , " guest-fsfreeze-freeze " ,
" { \" return \" : 7 } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-24 10:15:37 +02:00
2021-11-01 10:20:57 +01:00
if ( ( rc = qemuAgentFSFreeze ( qemuMonitorTestGetAgent ( test ) ,
mountpoints , 5 ) ) < 0 )
return - 1 ;
2013-07-24 10:15:37 +02:00
2021-11-01 10:20:57 +01:00
if ( rc ! = 5 ) {
2013-07-24 10:15:37 +02:00
virReportError ( VIR_ERR_INTERNAL_ERROR ,
2021-11-01 10:20:57 +01:00
" expected 5 frozen filesystems, got %d " , rc ) ;
return - 1 ;
2013-07-24 10:15:37 +02:00
}
2021-11-01 10:20:57 +01:00
if ( ( rc = qemuAgentFSFreeze ( qemuMonitorTestGetAgent ( test ) , NULL , 0 ) ) < 0 )
return - 1 ;
2013-07-24 10:15:37 +02:00
2021-11-01 10:20:57 +01:00
if ( rc ! = 7 ) {
2013-07-24 10:15:37 +02:00
virReportError ( VIR_ERR_INTERNAL_ERROR ,
2021-11-01 10:20:57 +01:00
" expected 7 frozen filesystems, got %d " , rc ) ;
return - 1 ;
2013-07-24 10:15:37 +02:00
}
2021-11-01 10:20:57 +01:00
return 0 ;
2013-07-24 10:15:37 +02:00
}
static int
testQemuAgentFSThaw ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2021-11-01 10:20:57 +01:00
int rc ;
2013-07-24 10:15:37 +02:00
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-24 10:15:37 +02:00
if ( qemuMonitorTestAddItem ( test , " guest-fsfreeze-thaw " ,
" { \" return \" : 5 } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-24 10:15:37 +02:00
if ( qemuMonitorTestAddItem ( test , " guest-fsfreeze-thaw " ,
" { \" return \" : 7 } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-24 10:15:37 +02:00
2021-11-01 10:20:57 +01:00
if ( ( rc = qemuAgentFSThaw ( qemuMonitorTestGetAgent ( test ) ) ) < 0 )
return - 1 ;
2013-07-24 10:15:37 +02:00
2021-11-01 10:20:57 +01:00
if ( rc ! = 5 ) {
2013-07-24 10:15:37 +02:00
virReportError ( VIR_ERR_INTERNAL_ERROR ,
2021-11-01 10:20:57 +01:00
" expected 5 thawed filesystems, got %d " , rc ) ;
return - 1 ;
2013-07-24 10:15:37 +02:00
}
2021-11-01 10:20:57 +01:00
if ( ( rc = qemuAgentFSThaw ( qemuMonitorTestGetAgent ( test ) ) ) < 0 )
return - 1 ;
2013-07-24 10:15:37 +02:00
2021-11-01 10:20:57 +01:00
if ( rc ! = 7 ) {
2013-07-24 10:15:37 +02:00
virReportError ( VIR_ERR_INTERNAL_ERROR ,
2021-11-01 10:20:57 +01:00
" expected 7 thawed filesystems, got %d " , rc ) ;
return - 1 ;
2013-07-24 10:15:37 +02:00
}
2021-11-01 10:20:57 +01:00
return 0 ;
2013-07-24 10:15:37 +02:00
}
2013-07-29 11:02:19 +02:00
static int
testQemuAgentFSTrim ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2013-07-29 11:02:19 +02:00
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 11:02:19 +02:00
2023-05-22 17:08:27 +02:00
if ( qemuMonitorTestAddItemVerbatim ( test ,
" { \" execute \" : \" guest-fstrim \" , "
" \" arguments \" : { \" minimum \" :1337}} " ,
NULL ,
" { \" return \" : {}} " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 11:02:19 +02:00
if ( qemuAgentFSTrim ( qemuMonitorTestGetAgent ( test ) , 1337 ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 11:02:19 +02:00
2021-11-01 10:20:57 +01:00
return 0 ;
2013-07-29 11:02:19 +02:00
}
2014-11-21 20:27:45 -05:00
static int
2021-03-11 08:16:13 +01:00
testQemuAgentGetFSInfoCommon ( virDomainXMLOption * xmlopt ,
qemuMonitorTest * * test ,
virDomainDef * * def )
2014-11-21 20:27:45 -05:00
{
2020-07-28 21:57:28 +02:00
g_autofree char * domain_filename = NULL ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) ret_test = NULL ;
2020-07-28 21:58:18 +02:00
g_autoptr ( virDomainDef ) ret_def = NULL ;
2014-11-21 20:27:45 -05:00
2019-08-23 11:31:21 -05:00
if ( ! test | | ! def )
return - 1 ;
if ( ! ( ret_test = qemuMonitorTestNewAgent ( xmlopt ) ) )
2014-11-21 20:27:45 -05:00
return - 1 ;
2019-10-22 15:26:14 +02:00
domain_filename = g_strdup_printf ( " %s/qemuagentdata/fsinfo.xml " , abs_srcdir ) ;
2014-11-21 20:27:45 -05:00
2019-11-27 12:29:21 +00:00
if ( ! ( ret_def = virDomainDefParseFile ( domain_filename , xmlopt ,
2019-08-23 11:31:21 -05:00
NULL , VIR_DOMAIN_DEF_PARSE_INACTIVE ) ) )
2021-11-01 10:20:57 +01:00
return - 1 ;
2014-11-21 20:27:45 -05:00
2019-08-23 11:31:21 -05:00
if ( qemuMonitorTestAddAgentSyncResponse ( ret_test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2014-11-21 20:27:45 -05:00
2019-08-23 11:31:21 -05:00
if ( qemuMonitorTestAddItem ( ret_test , " guest-get-fsinfo " ,
2014-11-21 20:27:45 -05:00
" { \" return \" : [ "
" { \" name \" : \" sda1 \" , \" mountpoint \" : \" / \" , "
2019-08-23 11:31:21 -05:00
" \" total-bytes \" :952840192, "
" \" used-bytes \" :229019648, "
2014-11-21 20:27:45 -05:00
" \" disk \" : [ "
2019-08-23 11:31:21 -05:00
" { \" serial \" : \" ARBITRARYSTRING \" , "
" \" bus-type \" : \" ide \" , "
2014-11-21 20:27:45 -05:00
" \" bus \" : 1, \" unit \" : 0, "
" \" pci-controller \" : { "
" \" bus \" : 0, \" slot \" : 1, "
" \" domain \" : 0, \" function \" : 1}, "
2019-08-23 11:31:21 -05:00
" \" dev \" : \" /dev/sda1 \" , "
2014-11-21 20:27:45 -05:00
" \" target \" : 0}], "
" \" type \" : \" ext4 \" }, "
" { \" name \" : \" dm-1 \" , "
" \" mountpoint \" : \" /opt \" , "
" \" disk \" : [ "
" { \" bus-type \" : \" virtio \" , "
" \" bus \" : 0, \" unit \" : 0, "
" \" pci-controller \" : { "
" \" bus \" : 0, \" slot \" : 6, "
" \" domain \" : 0, \" function \" : 0}, "
" \" target \" : 0}, "
" { \" bus-type \" : \" virtio \" , "
" \" bus \" : 0, \" unit \" : 0, "
" \" pci-controller \" : { "
" \" bus \" : 0, \" slot \" : 7, "
" \" domain \" : 0, \" function \" : 0}, "
" \" target \" : 0}], "
" \" type \" : \" vfat \" }, "
" { \" name \" : \" sdb1 \" , "
" \" mountpoint \" : \" /mnt/disk \" , "
" \" disk \" : [], \" type \" : \" xfs \" }]} " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2019-08-23 11:31:21 -05:00
2019-10-16 13:45:15 +02:00
* test = g_steal_pointer ( & ret_test ) ;
* def = g_steal_pointer ( & ret_def ) ;
2021-11-01 10:20:57 +01:00
return 0 ;
2019-08-23 11:31:21 -05:00
}
static int
testQemuAgentGetFSInfo ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = NULL ;
2021-09-04 21:50:02 +02:00
g_autoptr ( virDomainDef ) def = NULL ;
2021-03-11 08:16:13 +01:00
qemuAgentFSInfo * * info = NULL ;
2019-08-23 11:31:21 -05:00
int ret = - 1 , ninfo = 0 , i ;
if ( testQemuAgentGetFSInfoCommon ( xmlopt , & test , & def ) < 0 )
2014-11-21 20:27:45 -05:00
goto cleanup ;
if ( ( ninfo = qemuAgentGetFSInfo ( qemuMonitorTestGetAgent ( test ) ,
2020-03-16 08:37:13 +01:00
& info , true ) ) < 0 )
2014-11-21 20:27:45 -05:00
goto cleanup ;
if ( ninfo ! = 3 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" expected 3 filesystems information, got %d " , ninfo ) ;
ret = - 1 ;
goto cleanup ;
}
if ( STRNEQ ( info [ 2 ] - > name , " sda1 " ) | |
STRNEQ ( info [ 2 ] - > mountpoint , " / " ) | |
STRNEQ ( info [ 2 ] - > fstype , " ext4 " ) | |
2020-01-10 17:32:13 -06:00
info [ 2 ] - > ndisks ! = 1 | |
! info [ 2 ] - > disks | | ! info [ 2 ] - > disks [ 0 ] ) {
2014-11-21 20:27:45 -05:00
virReportError ( VIR_ERR_INTERNAL_ERROR ,
2020-01-10 17:32:13 -06:00
" unexpected filesystems information returned for sda1 (%s) " ,
info [ 2 ] - > name ) ;
2014-11-21 20:27:45 -05:00
ret = - 1 ;
goto cleanup ;
}
if ( STRNEQ ( info [ 1 ] - > name , " dm-1 " ) | |
STRNEQ ( info [ 1 ] - > mountpoint , " /opt " ) | |
STRNEQ ( info [ 1 ] - > fstype , " vfat " ) | |
2020-01-10 17:32:13 -06:00
info [ 1 ] - > ndisks ! = 2 | |
! info [ 1 ] - > disks | | ! info [ 1 ] - > disks [ 0 ] | | ! info [ 1 ] - > disks [ 1 ] | |
STRNEQ ( info [ 1 ] - > disks [ 0 ] - > bus_type , " virtio " ) | |
info [ 1 ] - > disks [ 0 ] - > bus ! = 0 | |
info [ 1 ] - > disks [ 0 ] - > target ! = 0 | |
info [ 1 ] - > disks [ 0 ] - > unit ! = 0 | |
info [ 1 ] - > disks [ 0 ] - > pci_controller . domain ! = 0 | |
info [ 1 ] - > disks [ 0 ] - > pci_controller . bus ! = 0 | |
info [ 1 ] - > disks [ 0 ] - > pci_controller . slot ! = 6 | |
info [ 1 ] - > disks [ 0 ] - > pci_controller . function ! = 0 | |
STRNEQ ( info [ 1 ] - > disks [ 1 ] - > bus_type , " virtio " ) | |
info [ 1 ] - > disks [ 1 ] - > bus ! = 0 | |
info [ 1 ] - > disks [ 1 ] - > target ! = 0 | |
info [ 1 ] - > disks [ 1 ] - > unit ! = 0 | |
info [ 1 ] - > disks [ 1 ] - > pci_controller . domain ! = 0 | |
info [ 1 ] - > disks [ 1 ] - > pci_controller . bus ! = 0 | |
info [ 1 ] - > disks [ 1 ] - > pci_controller . slot ! = 7 | |
info [ 1 ] - > disks [ 1 ] - > pci_controller . function ! = 0 ) {
2014-11-21 20:27:45 -05:00
virReportError ( VIR_ERR_INTERNAL_ERROR ,
2020-01-10 17:32:13 -06:00
" unexpected filesystems information returned for dm-1 (%s) " ,
info [ 0 ] - > name ) ;
2014-11-21 20:27:45 -05:00
ret = - 1 ;
goto cleanup ;
}
if ( STRNEQ ( info [ 0 ] - > name , " sdb1 " ) | |
STRNEQ ( info [ 0 ] - > mountpoint , " /mnt/disk " ) | |
STRNEQ ( info [ 0 ] - > fstype , " xfs " ) | |
2020-01-10 17:32:13 -06:00
info [ 0 ] - > ndisks ! = 0 | | info [ 0 ] - > disks ) {
2014-11-21 20:27:45 -05:00
virReportError ( VIR_ERR_INTERNAL_ERROR ,
2020-01-10 17:32:13 -06:00
" unexpected filesystems information returned for sdb1 (%s) " ,
info [ 0 ] - > name ) ;
2014-11-21 20:27:45 -05:00
ret = - 1 ;
goto cleanup ;
}
if ( qemuMonitorTestAddItem ( test , " guest-get-fsinfo " ,
" { \" error \" : "
" { \" class \" : \" CommandDisabled \" , "
" \" desc \" : \" The command guest-get-fsinfo "
" has been disabled for "
" this instance \" , "
" \" data \" :{ \" name \" : \" guest-get-fsinfo \" } "
" } "
" } " ) < 0 )
goto cleanup ;
2020-03-16 08:37:13 +01:00
if ( qemuAgentGetFSInfo ( qemuMonitorTestGetAgent ( test ) , & info , true ) > = 0 ) {
2014-11-21 20:27:45 -05:00
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" agent get-fsinfo command should have failed " ) ;
goto cleanup ;
}
ret = 0 ;
cleanup :
for ( i = 0 ; i < ninfo ; i + + )
2020-01-10 17:32:13 -06:00
qemuAgentFSInfoFree ( info [ i ] ) ;
2014-11-21 20:27:45 -05:00
VIR_FREE ( info ) ;
2019-08-23 11:31:21 -05:00
return ret ;
}
2013-07-29 15:24:32 +02:00
static int
testQemuAgentSuspend ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2013-07-29 15:24:32 +02:00
size_t i ;
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 15:24:32 +02:00
if ( qemuMonitorTestAddItem ( test , " guest-suspend-ram " ,
" { \" return \" : {} } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 15:24:32 +02:00
if ( qemuMonitorTestAddItem ( test , " guest-suspend-disk " ,
" { \" return \" : {} } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 15:24:32 +02:00
if ( qemuMonitorTestAddItem ( test , " guest-suspend-hybrid " ,
" { \" return \" : {} } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 15:24:32 +02:00
/* try the commands - fail if ordering changes */
for ( i = 0 ; i < VIR_NODE_SUSPEND_TARGET_LAST ; i + + ) {
if ( qemuAgentSuspend ( qemuMonitorTestGetAgent ( test ) , i ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 15:24:32 +02:00
}
2021-11-01 10:20:57 +01:00
return 0 ;
2013-07-29 15:24:32 +02:00
}
2013-07-29 17:22:32 +02:00
struct qemuAgentShutdownTestData {
const char * mode ;
qemuAgentEvent event ;
} ;
static int
2021-03-11 08:16:13 +01:00
qemuAgentShutdownTestMonitorHandler ( qemuMonitorTest * test ,
qemuMonitorTestItem * item ,
2013-07-29 17:22:32 +02:00
const char * cmdstr )
{
struct qemuAgentShutdownTestData * data ;
2020-07-28 21:58:18 +02:00
g_autoptr ( virJSONValue ) val = NULL ;
2021-03-11 08:16:13 +01:00
virJSONValue * args ;
2013-07-29 17:22:32 +02:00
const char * cmdname ;
const char * mode ;
data = qemuMonitorTestItemGetPrivateData ( item ) ;
if ( ! ( val = virJSONValueFromString ( cmdstr ) ) )
return - 1 ;
2021-09-04 22:38:39 +02:00
if ( ! ( cmdname = virJSONValueObjectGetString ( val , " execute " ) ) )
return qemuMonitorTestAddErrorResponse ( test , " Missing command name in %s " , cmdstr ) ;
2013-07-29 17:22:32 +02:00
if ( STRNEQ ( cmdname , " guest-shutdown " ) ) {
2021-09-04 22:38:39 +02:00
return qemuMonitorTestAddInvalidCommandResponse ( test , " guest-shutdown " ,
2016-12-01 09:22:44 +01:00
cmdname ) ;
2013-07-29 17:22:32 +02:00
}
if ( ! ( args = virJSONValueObjectGet ( val , " arguments " ) ) ) {
2021-09-04 22:38:39 +02:00
return qemuMonitorTestAddErrorResponse ( test ,
2020-04-23 11:50:59 +02:00
" Missing arguments section " ) ;
2013-07-29 17:22:32 +02:00
}
2021-09-04 22:38:39 +02:00
if ( ! ( mode = virJSONValueObjectGetString ( args , " mode " ) ) )
return qemuMonitorTestAddErrorResponse ( test , " Missing shutdown mode " ) ;
2013-07-29 17:22:32 +02:00
2013-08-01 16:19:09 +02:00
if ( STRNEQ ( mode , data - > mode ) ) {
2021-09-04 22:38:39 +02:00
return qemuMonitorTestAddErrorResponse ( test ,
2020-04-23 11:50:59 +02:00
" expected shutdown mode '%s' got '%s' " ,
data - > mode , mode ) ;
2013-08-01 16:19:09 +02:00
}
2013-07-29 17:22:32 +02:00
/* now don't reply but return a qemu agent event */
qemuAgentNotifyEvent ( qemuMonitorTestGetAgent ( test ) ,
data - > event ) ;
2021-09-04 22:38:39 +02:00
return 0 ;
2013-07-29 17:22:32 +02:00
}
static int
testQemuAgentShutdown ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2013-07-29 17:22:32 +02:00
struct qemuAgentShutdownTestData priv ;
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 17:22:32 +02:00
priv . event = QEMU_AGENT_EVENT_SHUTDOWN ;
2013-08-01 16:19:09 +02:00
priv . mode = " halt " ;
2013-07-29 17:22:32 +02:00
2023-05-22 14:07:50 +02:00
qemuMonitorTestAddHandler ( test , " guest-shutdown " ,
qemuAgentShutdownTestMonitorHandler ,
& priv , NULL ) ;
2013-07-29 17:22:32 +02:00
if ( qemuAgentShutdown ( qemuMonitorTestGetAgent ( test ) ,
QEMU_AGENT_SHUTDOWN_HALT ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 17:22:32 +02:00
priv . event = QEMU_AGENT_EVENT_SHUTDOWN ;
priv . mode = " powerdown " ;
2023-05-22 14:07:50 +02:00
qemuMonitorTestAddHandler ( test , " guest-shutdown " ,
qemuAgentShutdownTestMonitorHandler ,
& priv , NULL ) ;
2013-07-29 17:22:32 +02:00
if ( qemuAgentShutdown ( qemuMonitorTestGetAgent ( test ) ,
QEMU_AGENT_SHUTDOWN_POWERDOWN ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 17:22:32 +02:00
priv . event = QEMU_AGENT_EVENT_RESET ;
priv . mode = " reboot " ;
2023-05-22 14:07:50 +02:00
qemuMonitorTestAddHandler ( test ,
" guest-shutdown " ,
qemuAgentShutdownTestMonitorHandler ,
& priv , NULL ) ;
2013-07-29 17:22:32 +02:00
if ( qemuAgentShutdown ( qemuMonitorTestGetAgent ( test ) ,
QEMU_AGENT_SHUTDOWN_REBOOT ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-29 17:22:32 +02:00
2013-08-01 16:28:29 +02:00
/* check negative response, so that we can verify that the agent breaks
* out from sleep */
if ( qemuMonitorTestAddItem ( test , " guest-shutdown " ,
" { \" error \" : "
" { \" class \" : \" CommandDisabled \" , "
" \" desc \" : \" The command guest-shutdown has "
" been disabled for this instance \" , "
" \" data \" :{ \" name \" : \" guest-shutdown \" } "
" } "
" } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 16:28:29 +02:00
if ( qemuAgentShutdown ( qemuMonitorTestGetAgent ( test ) ,
QEMU_AGENT_SHUTDOWN_REBOOT ) ! = - 1 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" agent shutdown command should have failed " ) ;
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 16:28:29 +02:00
}
2021-11-01 10:20:57 +01:00
return 0 ;
2013-07-29 17:22:32 +02:00
}
2013-07-30 14:09:16 +02:00
static const char testQemuAgentCPUResponse [ ] =
" { \" return \" : "
" [ "
" { \" online \" : true, "
" \" can-offline \" : false, "
" \" logical-id \" : 0 "
" }, "
" { \" online \" : true, "
" \" can-offline \" : true, "
" \" logical-id \" : 1 "
" }, "
" { \" online \" : true, "
" \" can-offline \" : true, "
" \" logical-id \" : 2 "
" }, "
" { \" online \" : false, "
" \" can-offline \" : true, "
" \" logical-id \" : 3 "
" } "
" ] "
" } " ;
static int
testQemuAgentCPU ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2021-11-01 10:33:49 +01:00
g_autofree qemuAgentCPUInfo * cpuinfo = NULL ;
2013-07-30 14:09:16 +02:00
int nvcpus ;
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
if ( qemuMonitorTestAddItem ( test , " guest-get-vcpus " ,
testQemuAgentCPUResponse ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
/* get cpus */
if ( ( nvcpus = qemuAgentGetVCPUs ( qemuMonitorTestGetAgent ( test ) ,
& cpuinfo ) ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
if ( nvcpus ! = 4 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected '4' cpus, got '%d' " , nvcpus ) ;
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
}
/* try to unplug one */
if ( qemuAgentUpdateCPUInfo ( 2 , cpuinfo , nvcpus ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
2023-05-22 17:08:27 +02:00
if ( qemuMonitorTestAddItemVerbatim ( test ,
" { \" execute \" : \" guest-set-vcpus \" , "
" \" arguments \" : { "
" \" vcpus \" :[{ \" logical-id \" :1, \" online \" :false}] "
" }} " ,
NULL ,
" { \" return \" : 1 } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
2016-06-20 14:15:50 +02:00
if ( qemuAgentSetVCPUs ( qemuMonitorTestGetAgent ( test ) , cpuinfo , nvcpus ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
2020-01-24 21:30:04 +01:00
/* try to hotplug two, second one will fail */
2023-05-22 17:08:27 +02:00
if ( qemuMonitorTestAddItemVerbatim ( test ,
" { \" execute \" : \" guest-set-vcpus \" , "
" \" arguments \" : { "
" \" vcpus \" :[{ \" logical-id \" :1, \" online \" :true}, "
" { \" logical-id \" :3, \" online \" :true}] "
" }} " ,
NULL ,
" { \" return \" : 1 } " ) < 0 )
return - 1 ;
if ( qemuMonitorTestAddItemVerbatim ( test ,
" { \" execute \" : \" guest-set-vcpus \" , "
" \" arguments \" : { "
" \" vcpus \" :[{ \" logical-id \" :3, \" online \" :true}] "
" }} " ,
NULL ,
" { \" error \" : \" random error \" } " ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
2016-06-20 14:15:50 +02:00
if ( qemuAgentUpdateCPUInfo ( 4 , cpuinfo , nvcpus ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2016-06-20 14:15:50 +02:00
/* this should fail */
if ( qemuAgentSetVCPUs ( qemuMonitorTestGetAgent ( test ) , cpuinfo , nvcpus ) ! = - 1 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-07-30 14:09:16 +02:00
2021-11-01 10:20:57 +01:00
return 0 ;
2013-07-30 14:09:16 +02:00
}
2013-08-01 11:38:21 +02:00
static const char testQemuAgentArbitraryCommandResponse [ ] =
" { \" return \" : \" bla \" } " ;
static int
testQemuAgentArbitraryCommand ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2020-07-28 21:57:28 +02:00
g_autofree char * reply = NULL ;
2013-08-01 11:38:21 +02:00
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 11:38:21 +02:00
if ( qemuMonitorTestAddItem ( test , " ble " ,
testQemuAgentArbitraryCommandResponse ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 11:38:21 +02:00
if ( qemuAgentArbitraryCommand ( qemuMonitorTestGetAgent ( test ) ,
" { \" execute \" : \" ble \" } " ,
& reply ,
VIR_DOMAIN_QEMU_AGENT_COMMAND_BLOCK ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 11:38:21 +02:00
if ( STRNEQ ( reply , testQemuAgentArbitraryCommandResponse ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" invalid processing of guest agent reply: "
" got '%s' expected '%s' " ,
reply , testQemuAgentArbitraryCommandResponse ) ;
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 11:38:21 +02:00
}
2021-11-01 10:20:57 +01:00
return 0 ;
2013-08-01 11:38:21 +02:00
}
2013-08-01 11:43:41 +02:00
static int
2021-03-11 08:16:13 +01:00
qemuAgentTimeoutTestMonitorHandler ( qemuMonitorTest * test G_GNUC_UNUSED ,
qemuMonitorTestItem * item G_GNUC_UNUSED ,
2019-10-14 14:45:03 +02:00
const char * cmdstr G_GNUC_UNUSED )
2013-08-01 11:43:41 +02:00
{
return 0 ;
}
static int
testQemuAgentTimeout ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2020-07-28 21:57:28 +02:00
g_autofree char * reply = NULL ;
2013-08-01 11:43:41 +02:00
if ( ! test )
return - 1 ;
2021-11-01 10:20:57 +01:00
if ( virTestGetExpensive ( ) = = 0 )
return EXIT_AM_SKIP ;
2013-08-01 11:43:41 +02:00
2023-05-22 14:07:50 +02:00
qemuMonitorTestAddHandler ( test , NULL ,
qemuAgentTimeoutTestMonitorHandler ,
NULL , NULL ) ;
2013-08-01 11:43:41 +02:00
2014-05-01 20:06:19 -04:00
if ( qemuAgentFSFreeze ( qemuMonitorTestGetAgent ( test ) , NULL , 0 ) ! = - 1 ) {
2013-08-01 11:43:41 +02:00
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" agent command should have failed " ) ;
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 11:43:41 +02:00
}
/* test timeout */
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 11:43:41 +02:00
2023-05-22 14:07:50 +02:00
qemuMonitorTestAddHandler ( test ,
NULL ,
qemuAgentTimeoutTestMonitorHandler ,
NULL , NULL ) ;
2013-08-01 11:43:41 +02:00
if ( qemuAgentArbitraryCommand ( qemuMonitorTestGetAgent ( test ) ,
" { \" execute \" : \" ble \" } " ,
& reply ,
1 ) ! = - 2 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" agent command didn't time out " ) ;
2021-11-01 10:20:57 +01:00
return - 1 ;
2013-08-01 11:43:41 +02:00
}
2021-11-01 10:20:57 +01:00
return 0 ;
2013-08-01 11:43:41 +02:00
}
2015-01-26 00:08:48 +05:30
static const char testQemuAgentGetInterfacesResponse [ ] =
" { \" return \" : "
" [ "
" { \" name \" : \" eth2 \" , "
" \" hardware-address \" : \" 52:54:00:36:2a:e5 \" "
" }, "
" { \" name \" : \" eth1:0 \" , "
" \" ip-addresses \" : "
" [ "
" { \" ip-address-type \" : \" ipv4 \" , "
" \" ip-address \" : \" 192.168.10.91 \" , "
" \" prefix \" :24 "
" }, "
" { \" ip-address-type \" : \" ipv6 \" , "
" \" ip-address \" : \" fe80::fc54:ff:fefe:4c4f \" , "
" \" prefix \" :64 "
" } "
" ], "
" \" hardware-address \" : \" 52:54:00:d3:39:ee \" "
" }, "
" { \" name \" : \" eth0 \" , "
" \" ip-addresses \" : "
" [ "
" { \" ip-address-type \" : \" ipv6 \" , "
" \" ip-address \" : \" fe80::5054:ff:fe89:ad35 \" , "
" \" prefix \" :64 "
" }, "
" { \" ip-address-type \" : \" ipv4 \" , "
" \" ip-address \" : \" 192.168.102.142 \" , "
" \" prefix \" :24 "
" }, "
" { \" ip-address-type \" : \" ipv4 \" , "
" \" ip-address \" : \" 192.168.234.152 \" , "
" \" prefix \" :16 "
" }, "
" { \" ip-address-type \" : \" ipv6 \" , "
" \" ip-address \" : \" fe80::5054:ff:fec3:68bb \" , "
" \" prefix \" :64 "
" } "
" ], "
" \" hardware-address \" : \" 52:54:00:89:ad:35 \" "
" }, "
" { \" name \" : \" eth1 \" , "
" \" ip-addresses \" : "
" [ "
" { \" ip-address-type \" : \" ipv4 \" , "
" \" ip-address \" : \" 192.168.103.83 \" , "
" \" prefix \" :32 "
" }, "
" { \" ip-address-type \" : \" ipv6 \" , "
" \" ip-address \" : \" fe80::5054:ff:fed3:39ee \" , "
" \" prefix \" :64 "
" } "
" ], "
" \" hardware-address \" : \" 52:54:00:d3:39:ee \" "
" }, "
" { \" name \" : \" lo \" , "
" \" ip-addresses \" : "
" [ "
" { \" ip-address-type \" : \" ipv4 \" , "
" \" ip-address \" : \" 127.0.0.1 \" , "
" \" prefix \" :8 "
" }, "
" { \" ip-address-type \" : \" ipv6 \" , "
" \" ip-address \" : \" ::1 \" , "
" \" prefix \" :128 "
" } "
" ], "
" \" hardware-address \" : \" 00:00:00:00:00:00 \" "
" } "
" ] "
" } " ;
static int
testQemuAgentGetInterfaces ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2015-01-26 00:08:48 +05:30
size_t i ;
int ret = - 1 ;
int ifaces_count = 0 ;
virDomainInterfacePtr * ifaces = NULL ;
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
goto cleanup ;
if ( qemuMonitorTestAddItem ( test , " guest-network-get-interfaces " ,
testQemuAgentGetInterfacesResponse ) < 0 )
goto cleanup ;
if ( ( ifaces_count = qemuAgentGetInterfaces ( qemuMonitorTestGetAgent ( test ) ,
2021-10-15 18:07:49 +08:00
& ifaces , true ) ) < 0 )
2015-01-26 00:08:48 +05:30
goto cleanup ;
if ( ifaces_count ! = 4 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" expected 4 interfaces, got %d " , ret ) ;
goto cleanup ;
}
if ( STRNEQ ( ifaces [ 0 ] - > name , " eth2 " ) | |
STRNEQ ( ifaces [ 1 ] - > name , " eth1 " ) | |
STRNEQ ( ifaces [ 2 ] - > name , " eth0 " ) | |
STRNEQ ( ifaces [ 3 ] - > name , " lo " ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for interface names " ) ;
goto cleanup ;
}
if ( STRNEQ ( ifaces [ 0 ] - > hwaddr , " 52:54:00:36:2a:e5 " ) | |
STRNEQ ( ifaces [ 1 ] - > hwaddr , " 52:54:00:d3:39:ee " ) | |
STRNEQ ( ifaces [ 2 ] - > hwaddr , " 52:54:00:89:ad:35 " ) | |
STRNEQ ( ifaces [ 3 ] - > hwaddr , " 00:00:00:00:00:00 " ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for MAC addresses " ) ;
goto cleanup ;
}
if ( ifaces [ 0 ] - > naddrs ! = 0 | |
ifaces [ 1 ] - > naddrs ! = 4 | |
ifaces [ 2 ] - > naddrs ! = 4 | |
ifaces [ 3 ] - > naddrs ! = 2 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for number of IP addresses " ) ;
goto cleanup ;
}
if ( ifaces [ 1 ] - > addrs [ 0 ] . type ! = VIR_IP_ADDR_TYPE_IPV4 | |
ifaces [ 1 ] - > addrs [ 1 ] . type ! = VIR_IP_ADDR_TYPE_IPV6 | |
ifaces [ 1 ] - > addrs [ 2 ] . type ! = VIR_IP_ADDR_TYPE_IPV4 | |
ifaces [ 1 ] - > addrs [ 3 ] . type ! = VIR_IP_ADDR_TYPE_IPV6 | |
ifaces [ 2 ] - > addrs [ 0 ] . type ! = VIR_IP_ADDR_TYPE_IPV6 | |
ifaces [ 2 ] - > addrs [ 1 ] . type ! = VIR_IP_ADDR_TYPE_IPV4 | |
ifaces [ 2 ] - > addrs [ 2 ] . type ! = VIR_IP_ADDR_TYPE_IPV4 | |
ifaces [ 2 ] - > addrs [ 3 ] . type ! = VIR_IP_ADDR_TYPE_IPV6 | |
ifaces [ 3 ] - > addrs [ 0 ] . type ! = VIR_IP_ADDR_TYPE_IPV4 | |
ifaces [ 3 ] - > addrs [ 1 ] . type ! = VIR_IP_ADDR_TYPE_IPV6 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for IP address types " ) ;
goto cleanup ;
}
if ( ifaces [ 1 ] - > addrs [ 0 ] . prefix ! = 24 | |
ifaces [ 1 ] - > addrs [ 1 ] . prefix ! = 64 | |
ifaces [ 1 ] - > addrs [ 2 ] . prefix ! = 32 | |
ifaces [ 1 ] - > addrs [ 3 ] . prefix ! = 64 | |
ifaces [ 2 ] - > addrs [ 0 ] . prefix ! = 64 | |
ifaces [ 2 ] - > addrs [ 1 ] . prefix ! = 24 | |
ifaces [ 2 ] - > addrs [ 2 ] . prefix ! = 16 | |
ifaces [ 2 ] - > addrs [ 3 ] . prefix ! = 64 | |
ifaces [ 3 ] - > addrs [ 0 ] . prefix ! = 8 | |
ifaces [ 3 ] - > addrs [ 1 ] . prefix ! = 128 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for IP address prefix " ) ;
goto cleanup ;
}
if ( STRNEQ ( ifaces [ 1 ] - > addrs [ 0 ] . addr , " 192.168.10.91 " ) | |
STRNEQ ( ifaces [ 1 ] - > addrs [ 1 ] . addr , " fe80::fc54:ff:fefe:4c4f " ) | |
STRNEQ ( ifaces [ 1 ] - > addrs [ 2 ] . addr , " 192.168.103.83 " ) | |
STRNEQ ( ifaces [ 1 ] - > addrs [ 3 ] . addr , " fe80::5054:ff:fed3:39ee " ) | |
STRNEQ ( ifaces [ 2 ] - > addrs [ 0 ] . addr , " fe80::5054:ff:fe89:ad35 " ) | |
STRNEQ ( ifaces [ 2 ] - > addrs [ 1 ] . addr , " 192.168.102.142 " ) | |
STRNEQ ( ifaces [ 2 ] - > addrs [ 2 ] . addr , " 192.168.234.152 " ) | |
STRNEQ ( ifaces [ 2 ] - > addrs [ 3 ] . addr , " fe80::5054:ff:fec3:68bb " ) | |
STRNEQ ( ifaces [ 3 ] - > addrs [ 0 ] . addr , " 127.0.0.1 " ) | |
STRNEQ ( ifaces [ 3 ] - > addrs [ 1 ] . addr , " ::1 " ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for IP address values " ) ;
goto cleanup ;
}
ret = 0 ;
cleanup :
if ( ifaces ) {
for ( i = 0 ; i < ifaces_count ; i + + )
virDomainInterfaceFree ( ifaces [ i ] ) ;
}
VIR_FREE ( ifaces ) ;
return ret ;
}
2013-08-01 11:43:41 +02:00
2020-11-20 22:09:45 +04:00
/* this is a bit of a pathological response on a real hw */
static const char testQemuAgentGetDisksResponse [ ] =
" { \" return \" : "
" [ "
" { \" alias \" : \" fedora_localhost--live-home \" , "
" \" dependencies \" : "
" [ "
" \" /dev/dm-0 \" "
" ], "
" \" name \" : \" /dev/dm-3 \" , "
" \" partition \" : false "
" }, "
" { \" address \" : "
" { \" bus \" : 0, "
" \" bus-type \" : \" unknown \" , "
" \" dev \" : \" /dev/nvme0n1 \" , "
" \" pci-controller \" : "
" { \" bus \" : -1, "
" \" domain \" : -1, "
" \" function \" : -1, "
" \" slot \" : -1 "
" }, "
" \" serial \" : \" GIGABYTE GP-ASM2NE6100TTTD_SN202208900567 \" , "
" \" target \" : 0, "
" \" unit \" : 0 "
" }, "
" \" dependencies \" : [], "
" \" name \" : \" /dev/nvme0n1 \" , "
" \" partition \" : false "
" } "
" ] "
" } " ;
static int
testQemuAgentGetDisks ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2020-11-20 22:09:45 +04:00
size_t i ;
int ret = - 1 ;
int disks_count = 0 ;
2021-03-11 08:16:13 +01:00
qemuAgentDiskInfo * * disks = NULL ;
2020-11-20 22:09:45 +04:00
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
goto cleanup ;
if ( qemuMonitorTestAddItem ( test , " guest-get-disks " ,
testQemuAgentGetDisksResponse ) < 0 )
goto cleanup ;
if ( ( disks_count = qemuAgentGetDisks ( qemuMonitorTestGetAgent ( test ) ,
& disks , true ) ) < 0 )
goto cleanup ;
if ( disks_count ! = 2 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" expected 2 disks, got %d " , ret ) ;
goto cleanup ;
}
if ( STRNEQ ( disks [ 0 ] - > name , " /dev/dm-3 " ) | |
STRNEQ ( disks [ 1 ] - > name , " /dev/nvme0n1 " ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for disks names " ) ;
goto cleanup ;
}
if ( STRNEQ ( disks [ 0 ] - > alias , " fedora_localhost--live-home " ) | |
disks [ 1 ] - > alias ! = NULL ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for disks aliases " ) ;
goto cleanup ;
}
if ( STRNEQ ( disks [ 0 ] - > dependencies [ 0 ] , " /dev/dm-0 " ) | |
disks [ 0 ] - > dependencies [ 1 ] ! = NULL | |
disks [ 1 ] - > dependencies [ 0 ] ! = NULL ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for disks dependencies " ) ;
goto cleanup ;
}
if ( disks [ 0 ] - > address ! = NULL | |
disks [ 1 ] - > address - > bus ! = 0 | |
disks [ 1 ] - > address - > target ! = 0 | |
disks [ 1 ] - > address - > unit ! = 0 | |
STRNEQ ( disks [ 1 ] - > address - > serial , " GIGABYTE GP-ASM2NE6100TTTD_SN202208900567 " ) | |
STRNEQ ( disks [ 1 ] - > address - > bus_type , " unknown " ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR , " %s " ,
" unexpected return values for disks addresses " ) ;
goto cleanup ;
}
ret = 0 ;
cleanup :
if ( disks ) {
for ( i = 0 ; i < disks_count ; i + + )
qemuAgentDiskInfoFree ( disks [ i ] ) ;
}
VIR_FREE ( disks ) ;
return ret ;
}
2019-08-23 11:31:17 -05:00
static const char testQemuAgentUsersResponse [ ] =
" { \" return \" : "
" [ "
" { \" user \" : \" test \" , "
" \" login-time \" : 1561739203.584038 "
" }, "
" { \" user \" : \" test2 \" , "
" \" login-time \" : 1561739229.190697 "
" } "
" ] "
" } " ;
static const char testQemuAgentUsersResponse2 [ ] =
" { \" return \" : "
" [ "
" { \" user \" : \" test \" , "
" \" domain \" : \" DOMAIN \" , "
" \" login-time \" : 1561739203.584038 "
" } "
" ] "
" } " ;
static int
checkUserInfo ( virTypedParameterPtr params ,
int nparams ,
size_t nth ,
const char * expUsername ,
const char * expDomain ,
unsigned long long expLogintime )
{
char param_name [ VIR_TYPED_PARAM_FIELD_LENGTH ] ;
const char * username = NULL ;
const char * domain = NULL ;
unsigned long long logintime = 0 ;
2019-11-13 14:53:42 +01:00
g_snprintf ( param_name , VIR_TYPED_PARAM_FIELD_LENGTH ,
" user.%zu.name " , nth ) ;
2019-08-23 11:31:17 -05:00
if ( virTypedParamsGetString ( params , nparams , param_name , & username ) < 0 )
return - 1 ;
if ( STRNEQ_NULLABLE ( expUsername , username ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected user name '%s', got '%s' " ,
expUsername , username ) ;
return - 1 ;
}
2019-11-13 14:53:42 +01:00
g_snprintf ( param_name , VIR_TYPED_PARAM_FIELD_LENGTH ,
" user.%zu.domain " , nth ) ;
2019-11-03 07:39:22 -05:00
if ( virTypedParamsGetString ( params , nparams , param_name , & domain ) < 0 )
return - 1 ;
2019-08-23 11:31:17 -05:00
if ( STRNEQ_NULLABLE ( expDomain , domain ) ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected domain '%s', got '%s' " ,
NULLSTR ( expDomain ) , NULLSTR ( domain ) ) ;
return - 1 ;
}
2019-11-13 14:53:42 +01:00
g_snprintf ( param_name , VIR_TYPED_PARAM_FIELD_LENGTH ,
" user.%zu.login-time " , nth ) ;
2019-08-23 11:31:17 -05:00
if ( virTypedParamsGetULLong ( params , nparams , param_name , & logintime ) < 0 )
return - 1 ;
if ( expLogintime ! = logintime ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected login time of '%llu', got '%llu' " ,
expLogintime , logintime ) ;
return - 1 ;
}
return 0 ;
}
static int
testQemuAgentUsers ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2019-08-23 11:31:17 -05:00
virTypedParameterPtr params = NULL ;
int nparams = 0 ;
int maxparams = 0 ;
int ret = - 1 ;
unsigned int count ;
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
goto cleanup ;
if ( qemuMonitorTestAddItem ( test , " guest-get-users " ,
testQemuAgentUsersResponse ) < 0 )
goto cleanup ;
/* get users */
if ( qemuAgentGetUsers ( qemuMonitorTestGetAgent ( test ) ,
2020-03-16 08:37:13 +01:00
& params , & nparams , & maxparams , true ) < 0 )
2019-08-23 11:31:17 -05:00
goto cleanup ;
if ( virTypedParamsGetUInt ( params , nparams , " user.count " , & count ) < 0 )
goto cleanup ;
if ( count ! = 2 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected '2' users, got '%u' " , count ) ;
goto cleanup ;
}
if ( checkUserInfo ( params , nparams , 0 , " test " , NULL , 1561739203584 ) < 0 | |
checkUserInfo ( params , nparams , 1 , " test2 " , NULL , 1561739229190 ) < 0 )
goto cleanup ;
if ( qemuMonitorTestAddItem ( test , " guest-get-users " ,
testQemuAgentUsersResponse2 ) < 0 )
goto cleanup ;
virTypedParamsFree ( params , nparams ) ;
params = NULL ;
nparams = 0 ;
maxparams = 0 ;
/* get users with domain */
if ( qemuAgentGetUsers ( qemuMonitorTestGetAgent ( test ) ,
2020-03-16 08:37:13 +01:00
& params , & nparams , & maxparams , true ) < 0 )
2019-08-23 11:31:17 -05:00
goto cleanup ;
if ( virTypedParamsGetUInt ( params , nparams , " user.count " , & count ) < 0 )
goto cleanup ;
if ( count ! = 1 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected '1' user, got '%u' " , count ) ;
goto cleanup ;
}
if ( checkUserInfo ( params , nparams , 0 , " test " , " DOMAIN " , 1561739203584 ) < 0 )
goto cleanup ;
ret = 0 ;
cleanup :
virTypedParamsFree ( params , nparams ) ;
return ret ;
}
2019-08-23 11:31:18 -05:00
static const char testQemuAgentOSInfoResponse [ ] =
" { \" return \" : "
" { \" name \" : \" CentOS Linux \" , "
" \" kernel-release \" : \" 3.10.0-862.14.4.el7.x86_64 \" , "
" \" version \" : \" 7 (Core) \" , "
" \" pretty-name \" : \" CentOS Linux 7 (Core) \" , "
" \" version-id \" : \" 7 \" , "
" \" kernel-version \" : \" #1 SMP Wed Sep 26 15:12:11 UTC 2018 \" , "
" \" machine \" : \" x86_64 \" , "
" \" id \" : \" centos \" } "
" } " ;
static const char testQemuAgentOSInfoResponse2 [ ] =
" { \" return \" : "
" { \" name \" : \" Microsoft Windows \" , "
" \" kernel-release \" : \" 7601 \" , "
" \" version \" : \" Microsoft Windows 77 \" , "
" \" variant \" : \" client \" , "
" \" pretty-name \" : \" Windows 7 Professional \" , "
" \" version-id \" : \" \" , "
" \" variant-id \" : \" client \" , "
" \" kernel-version \" : \" 6.1 \" , "
" \" machine \" : \" x86_64 \" , "
" \" id \" : \" mswindows \" } "
" } " ;
static int
testQemuAgentOSInfo ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2019-08-23 11:31:18 -05:00
virTypedParameterPtr params = NULL ;
int nparams = 0 ;
int maxparams = 0 ;
int ret = - 1 ;
if ( ! test )
return - 1 ;
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
goto cleanup ;
if ( qemuMonitorTestAddItem ( test , " guest-get-osinfo " ,
testQemuAgentOSInfoResponse ) < 0 )
goto cleanup ;
/* get osinfo */
if ( qemuAgentGetOSInfo ( qemuMonitorTestGetAgent ( test ) ,
2020-03-16 08:37:13 +01:00
& params , & nparams , & maxparams , true ) < 0 )
2019-08-23 11:31:18 -05:00
goto cleanup ;
if ( nparams ! = 8 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected 8 params, got %d " , nparams ) ;
goto cleanup ;
}
# define VALIDATE_PARAM(param_name_, expected_) \
do { \
const char * value_ = NULL ; \
if ( virTypedParamsGetString ( params , nparams , param_name_ , & value_ ) < 0 | | \
value_ = = NULL ) { \
virReportError ( VIR_ERR_INTERNAL_ERROR , " missing param '%s' " , param_name_ ) ; \
goto cleanup ; \
} \
if ( STRNEQ ( value_ , expected_ ) ) { \
virReportError ( VIR_ERR_INTERNAL_ERROR , \
" Expected name '%s', got '%s' " , expected_ , value_ ) ; \
goto cleanup ; \
} \
} while ( 0 )
VALIDATE_PARAM ( " os.id " , " centos " ) ;
VALIDATE_PARAM ( " os.name " , " CentOS Linux " ) ;
VALIDATE_PARAM ( " os.version " , " 7 (Core) " ) ;
VALIDATE_PARAM ( " os.version-id " , " 7 " ) ;
VALIDATE_PARAM ( " os.pretty-name " , " CentOS Linux 7 (Core) " ) ;
VALIDATE_PARAM ( " os.kernel-release " , " 3.10.0-862.14.4.el7.x86_64 " ) ;
VALIDATE_PARAM ( " os.kernel-version " , " #1 SMP Wed Sep 26 15:12:11 UTC 2018 " ) ;
VALIDATE_PARAM ( " os.machine " , " x86_64 " ) ;
virTypedParamsFree ( params , nparams ) ;
params = NULL ;
nparams = 0 ;
maxparams = 0 ;
if ( qemuMonitorTestAddItem ( test , " guest-get-osinfo " ,
testQemuAgentOSInfoResponse2 ) < 0 )
goto cleanup ;
/* get users with domain */
if ( qemuAgentGetOSInfo ( qemuMonitorTestGetAgent ( test ) ,
2020-03-16 08:37:13 +01:00
& params , & nparams , & maxparams , true ) < 0 )
2019-08-23 11:31:18 -05:00
goto cleanup ;
if ( nparams ! = 10 ) {
virReportError ( VIR_ERR_INTERNAL_ERROR ,
" Expected 10 params, got %d " , nparams ) ;
goto cleanup ;
}
VALIDATE_PARAM ( " os.id " , " mswindows " ) ;
VALIDATE_PARAM ( " os.name " , " Microsoft Windows " ) ;
VALIDATE_PARAM ( " os.pretty-name " , " Windows 7 Professional " ) ;
VALIDATE_PARAM ( " os.version " , " Microsoft Windows 77 " ) ;
VALIDATE_PARAM ( " os.version-id " , " " ) ;
VALIDATE_PARAM ( " os.variant " , " client " ) ;
VALIDATE_PARAM ( " os.variant-id " , " client " ) ;
VALIDATE_PARAM ( " os.kernel-release " , " 7601 " ) ;
VALIDATE_PARAM ( " os.kernel-version " , " 6.1 " ) ;
VALIDATE_PARAM ( " os.machine " , " x86_64 " ) ;
ret = 0 ;
cleanup :
2021-11-01 10:34:33 +01:00
virTypedParamsFree ( params , nparams ) ;
2019-08-23 11:31:18 -05:00
return ret ;
}
2019-08-23 11:31:19 -05:00
static const char testQemuAgentTimezoneResponse1 [ ] =
" { \" return \" :{ \" zone \" : \" IST \" , \" offset \" :19800}} " ;
static const char testQemuAgentTimezoneResponse2 [ ] =
" { \" return \" :{ \" zone \" : \" CEST \" , \" offset \" :7200}} " ;
static const char testQemuAgentTimezoneResponse3 [ ] =
" { \" return \" :{ \" zone \" : \" NDT \" , \" offset \" :-9000}} " ;
static const char testQemuAgentTimezoneResponse4 [ ] =
" { \" return \" :{ \" zone \" : \" PDT \" , \" offset \" :-25200}} " ;
2019-08-23 11:31:17 -05:00
2019-08-23 11:31:19 -05:00
static int
testQemuAgentTimezone ( const void * data )
{
2021-03-11 08:16:13 +01:00
virDomainXMLOption * xmlopt = ( virDomainXMLOption * ) data ;
2021-11-01 09:51:01 +01:00
g_autoptr ( qemuMonitorTest ) test = qemuMonitorTestNewAgent ( xmlopt ) ;
2021-11-01 10:34:33 +01:00
virTypedParameterPtr params = NULL ;
int nparams = 0 ;
2019-08-23 11:31:19 -05:00
int ret = - 1 ;
if ( ! test )
return - 1 ;
2022-11-07 18:42:32 +01:00
if ( qemuMonitorTestAddAgentSyncResponse ( test ) < 0 )
goto cleanup ;
2019-08-23 11:31:19 -05:00
# define VALIDATE_TIMEZONE(response_, expected_name_, expected_offset_) \
do { \
int maxparams_ = 0 ; \
const char * name_ = NULL ; \
int offset_ ; \
if ( qemuMonitorTestAddItem ( test , " guest-get-timezone " , \
response_ ) < 0 ) \
goto cleanup ; \
2021-11-01 10:34:33 +01:00
virTypedParamsFree ( params , nparams ) ; \
params = NULL ; \
nparams = 0 ; \
2019-08-23 11:31:19 -05:00
if ( qemuAgentGetTimezone ( qemuMonitorTestGetAgent ( test ) , \
2021-11-01 10:34:33 +01:00
& params , & nparams , & maxparams_ , true ) < 0 ) \
2019-08-23 11:31:19 -05:00
goto cleanup ; \
2021-11-01 10:34:33 +01:00
if ( nparams ! = 2 ) { \
2019-08-23 11:31:19 -05:00
virReportError ( VIR_ERR_INTERNAL_ERROR , \
2021-11-01 10:34:33 +01:00
" Expected 2 params, got %d " , nparams ) ; \
2019-08-23 11:31:19 -05:00
goto cleanup ; \
} \
2021-11-01 10:34:33 +01:00
if ( virTypedParamsGetString ( params , nparams , \
2019-08-23 11:31:19 -05:00
" timezone.name " , & name_ ) < 0 ) { \
virReportError ( VIR_ERR_INTERNAL_ERROR , " missing param '%s' " , \
" tiemzone.name " ) ; \
goto cleanup ; \
} \
if ( STRNEQ ( name_ , expected_name_ ) ) { \
virReportError ( VIR_ERR_INTERNAL_ERROR , \
" Expected name '%s', got '%s' " , expected_name_ , name_ ) ; \
goto cleanup ; \
} \
2021-11-01 10:34:33 +01:00
if ( virTypedParamsGetInt ( params , nparams , \
2019-08-23 11:31:19 -05:00
" timezone.offset " , & offset_ ) < 0 ) { \
virReportError ( VIR_ERR_INTERNAL_ERROR , " missing param '%s' " , \
" tiemzone.offset " ) ; \
goto cleanup ; \
} \
if ( offset_ ! = expected_offset_ ) { \
virReportError ( VIR_ERR_INTERNAL_ERROR , \
" Expected offset '%i', got '%i' " , offset_ , \
expected_offset_ ) ; \
goto cleanup ; \
} \
} while ( 0 )
VALIDATE_TIMEZONE ( testQemuAgentTimezoneResponse1 , " IST " , 19800 ) ;
VALIDATE_TIMEZONE ( testQemuAgentTimezoneResponse2 , " CEST " , 7200 ) ;
VALIDATE_TIMEZONE ( testQemuAgentTimezoneResponse3 , " NDT " , - 9000 ) ;
VALIDATE_TIMEZONE ( testQemuAgentTimezoneResponse4 , " PDT " , - 25200 ) ;
ret = 0 ;
cleanup :
2021-11-01 10:34:33 +01:00
virTypedParamsFree ( params , nparams ) ;
2019-08-23 11:31:19 -05:00
return ret ;
}
2013-07-24 10:15:37 +02:00
static int
mymain ( void )
{
2023-07-04 13:39:24 +02:00
g_autoptr ( GHashTable ) capslatest = testQemuGetLatestCaps ( ) ;
g_autoptr ( GHashTable ) capscache = virHashNew ( virObjectUnref ) ;
2013-07-24 10:15:37 +02:00
int ret = 0 ;
2019-09-16 17:44:23 +01:00
if ( qemuTestDriverInit ( & driver ) < 0 )
2013-07-24 10:15:37 +02:00
return EXIT_FAILURE ;
2023-07-04 13:39:24 +02:00
/* Some test cases need a real definition thus parse a XML. We need
* qemu capabilities for that . */
if ( testQemuInsertRealCaps ( driver . qemuCapsCache , " x86_64 " , " latest " , " " ,
capslatest , capscache , NULL , NULL ) < 0 )
return EXIT_FAILURE ;
2013-07-24 10:15:37 +02:00
virEventRegisterDefaultImpl ( ) ;
2017-11-03 13:09:47 +01:00
# define DO_TEST(name) \
if ( virTestRun ( # name , testQemuAgent # # name , driver . xmlopt ) < 0 ) \
2013-07-24 10:15:37 +02:00
ret = - 1
DO_TEST ( FSFreeze ) ;
DO_TEST ( FSThaw ) ;
2013-07-29 11:02:19 +02:00
DO_TEST ( FSTrim ) ;
2014-11-21 20:27:45 -05:00
DO_TEST ( GetFSInfo ) ;
2013-07-29 15:24:32 +02:00
DO_TEST ( Suspend ) ;
2013-07-29 17:22:32 +02:00
DO_TEST ( Shutdown ) ;
2013-07-30 14:09:16 +02:00
DO_TEST ( CPU ) ;
2013-08-01 11:38:21 +02:00
DO_TEST ( ArbitraryCommand ) ;
2015-01-26 00:08:48 +05:30
DO_TEST ( GetInterfaces ) ;
2019-08-23 11:31:17 -05:00
DO_TEST ( Users ) ;
2019-08-23 11:31:18 -05:00
DO_TEST ( OSInfo ) ;
2019-08-23 11:31:19 -05:00
DO_TEST ( Timezone ) ;
2020-11-07 13:12:53 +04:00
DO_TEST ( SSHKeys ) ;
2020-11-20 22:09:45 +04:00
DO_TEST ( GetDisks ) ;
2013-08-23 01:13:57 +05:30
DO_TEST ( Timeout ) ; /* Timeout should always be called last */
2013-07-24 10:15:37 +02:00
2015-09-15 08:16:02 +02:00
qemuTestDriverFree ( & driver ) ;
2013-07-24 10:15:37 +02:00
return ( ret = = 0 ) ? EXIT_SUCCESS : EXIT_FAILURE ;
}
2017-03-29 16:45:42 +02:00
VIR_TEST_MAIN ( mymain )