2015-05-22 15:21:46 +03:00
/*
* Copyright ( C ) 2015 Red Hat , Inc .
*
* 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/>.
*
* Author : Daniel P . Berrange < berrange @ redhat . com >
*/
# include <config.h>
# include "testutils.h"
# include "virerror.h"
# include "rpc/virnetserver.h"
# define VIR_FROM_THIS VIR_FROM_RPC
# ifdef HAVE_SOCKETPAIR
static virNetServerPtr
testCreateServer ( const char * host , int family )
{
virNetServerPtr srv = NULL ;
virNetServerServicePtr svc1 = NULL , svc2 = NULL ;
virNetServerClientPtr cln1 = NULL , cln2 = NULL ;
virNetSocketPtr sk1 = NULL , sk2 = NULL ;
int fdclient [ 2 ] ;
2015-06-15 12:28:11 +03:00
const char * mdns_entry = NULL ;
const char * mdns_group = NULL ;
# ifdef WITH_AVAHI
mdns_entry = " libvirt-ro " ;
mdns_group = " libvirtTest " ;
# endif
2015-05-22 15:21:46 +03:00
if ( socketpair ( PF_UNIX , SOCK_STREAM , 0 , fdclient ) < 0 ) {
virReportSystemError ( errno , " %s " ,
" Cannot create socket pair " ) ;
goto cleanup ;
}
if ( ! ( srv = virNetServerNew ( 10 , 50 , 5 , 100 , 10 ,
120 , 5 , true ,
2015-06-15 12:28:11 +03:00
mdns_group ,
2015-05-22 15:21:46 +03:00
NULL ,
NULL ,
NULL ,
NULL ) ) )
goto error ;
if ( ! ( svc1 = virNetServerServiceNewTCP ( host ,
NULL ,
family ,
VIR_NET_SERVER_SERVICE_AUTH_NONE ,
# ifdef WITH_GNUTLS
NULL ,
# endif
true ,
5 ,
2 ) ) )
goto error ;
if ( ! ( svc2 = virNetServerServiceNewTCP ( host ,
NULL ,
family ,
VIR_NET_SERVER_SERVICE_AUTH_POLKIT ,
# ifdef WITH_GNUTLS
NULL ,
# endif
false ,
25 ,
5 ) ) )
goto error ;
2015-06-15 12:28:11 +03:00
if ( virNetServerAddService ( srv , svc1 , mdns_entry ) < 0 )
2015-05-22 15:21:46 +03:00
goto error ;
2015-06-15 12:28:11 +03:00
if ( virNetServerAddService ( srv , svc2 , mdns_entry ) < 0 )
2015-05-22 15:21:46 +03:00
goto error ;
if ( virNetSocketNewConnectSockFD ( fdclient [ 0 ] , & sk1 ) < 0 )
goto error ;
if ( virNetSocketNewConnectSockFD ( fdclient [ 1 ] , & sk2 ) < 0 )
goto error ;
if ( ! ( cln1 = virNetServerClientNew ( sk1 ,
VIR_NET_SERVER_SERVICE_AUTH_SASL ,
true ,
15 ,
# ifdef WITH_GNUTLS
NULL ,
# endif
NULL , NULL , NULL , NULL ) ) )
goto error ;
if ( ! ( cln2 = virNetServerClientNew ( sk2 ,
VIR_NET_SERVER_SERVICE_AUTH_POLKIT ,
true ,
66 ,
# ifdef WITH_GNUTLS
NULL ,
# endif
NULL , NULL , NULL , NULL ) ) )
goto error ;
if ( virNetServerAddClient ( srv , cln1 ) < 0 )
goto error ;
if ( virNetServerAddClient ( srv , cln2 ) < 0 )
goto error ;
cleanup :
virObjectUnref ( cln1 ) ;
virObjectUnref ( cln2 ) ;
virObjectUnref ( svc1 ) ;
virObjectUnref ( svc2 ) ;
return srv ;
error :
virObjectUnref ( srv ) ;
srv = NULL ;
goto cleanup ;
}
static char * testGenerateJSON ( void )
{
virNetServerPtr srv = NULL ;
virJSONValuePtr json = NULL ;
char * jsonstr = NULL ;
bool has_ipv4 , has_ipv6 ;
/* Our pre-saved JSON file is created so that each service
* only has one socket . If we let libvirt bind to IPv4 and
* IPv6 we might end up with two sockets , so force one or
* the other based on what ' s available on thehost
*/
if ( virNetSocketCheckProtocols ( & has_ipv4 ,
& has_ipv6 ) < 0 )
return NULL ;
if ( ! has_ipv4 & & ! has_ipv6 )
return NULL ;
if ( ! ( srv = testCreateServer (
has_ipv4 ? " 127.0.0.1 " : " ::1 " ,
has_ipv4 ? AF_INET : AF_INET6 ) ) )
goto cleanup ;
if ( ! ( json = virNetServerPreExecRestart ( srv ) ) )
goto cleanup ;
if ( ! ( jsonstr = virJSONValueToString ( json , true ) ) )
goto cleanup ;
fprintf ( stderr , " %s \n " , jsonstr ) ;
cleanup :
virNetServerClose ( srv ) ;
virObjectUnref ( srv ) ;
virJSONValueFree ( json ) ;
return jsonstr ;
}
struct testExecRestartData {
const char * jsonfile ;
} ;
static int testExecRestart ( const void * opaque )
{
int ret = - 1 ;
virNetServerPtr srv = NULL ;
const struct testExecRestartData * data = opaque ;
char * infile = NULL , * outfile = NULL ;
char * injsonstr = NULL , * outjsonstr = NULL ;
virJSONValuePtr injson = NULL , outjson = NULL ;
int fdclient [ 2 ] = { - 1 , - 1 } , fdserver [ 2 ] = { - 1 , - 1 } ;
if ( socketpair ( PF_UNIX , SOCK_STREAM , 0 , fdclient ) < 0 ) {
virReportSystemError ( errno , " %s " ,
" Cannot create socket pair " ) ;
goto cleanup ;
}
if ( socketpair ( PF_UNIX , SOCK_STREAM , 0 , fdserver ) < 0 ) {
virReportSystemError ( errno , " %s " ,
" Cannot create socket pair " ) ;
goto cleanup ;
}
/* We're blindly assuming the test case isn't using
* fds 100 - > 103 for something else , which is probably
* fairly reasonable in general
*/
dup2 ( fdserver [ 0 ] , 100 ) ;
dup2 ( fdserver [ 1 ] , 101 ) ;
dup2 ( fdclient [ 0 ] , 102 ) ;
dup2 ( fdclient [ 1 ] , 103 ) ;
if ( virAsprintf ( & infile , " %s/virnetserverdata/input-data-%s.json " ,
abs_srcdir , data - > jsonfile ) < 0 )
goto cleanup ;
if ( virAsprintf ( & outfile , " %s/virnetserverdata/output-data-%s.json " ,
abs_srcdir , data - > jsonfile ) < 0 )
goto cleanup ;
if ( virFileReadAll ( infile , 8192 , & injsonstr ) < 0 )
goto cleanup ;
if ( ! ( injson = virJSONValueFromString ( injsonstr ) ) )
goto cleanup ;
if ( ! ( srv = virNetServerNewPostExecRestart ( injson ,
NULL , NULL , NULL ,
NULL , NULL ) ) )
goto cleanup ;
if ( ! ( outjson = virNetServerPreExecRestart ( srv ) ) )
goto cleanup ;
if ( ! ( outjsonstr = virJSONValueToString ( outjson , true ) ) )
goto cleanup ;
if ( virtTestCompareToFile ( outjsonstr , outfile ) < 0 )
goto fail ;
ret = 0 ;
cleanup :
if ( ret < 0 ) {
virErrorPtr err = virGetLastError ( ) ;
/* Rather be safe, we have lot of missing errors */
if ( err )
fprintf ( stderr , " %s \n " , err - > message ) ;
else
fprintf ( stderr , " %s \n " , " Unknown error " ) ;
}
fail :
VIR_FREE ( infile ) ;
VIR_FREE ( outfile ) ;
VIR_FREE ( injsonstr ) ;
VIR_FREE ( outjsonstr ) ;
virJSONValueFree ( injson ) ;
virJSONValueFree ( outjson ) ;
virObjectUnref ( srv ) ;
VIR_FORCE_CLOSE ( fdserver [ 0 ] ) ;
VIR_FORCE_CLOSE ( fdserver [ 1 ] ) ;
VIR_FORCE_CLOSE ( fdclient [ 0 ] ) ;
VIR_FORCE_CLOSE ( fdclient [ 1 ] ) ;
return ret ;
}
static int
mymain ( void )
{
int ret = 0 ;
/* Hack to make it easier to generate new JSON files when
* the RPC classes change . Just set this env var , save
* the generated JSON , and replace the file descriptor
* numbers with 100 , 101 , 102 , 103.
*/
if ( getenv ( " VIR_GENERATE_JSON " ) ) {
char * json = testGenerateJSON ( ) ;
fprintf ( stdout , " %s \n " , json ) ;
VIR_FREE ( json ) ;
return EXIT_SUCCESS ;
}
# define EXEC_RESTART_TEST(file) \
do { \
struct testExecRestartData data = { \
file \
} ; \
if ( virtTestRun ( " ExecRestart " file , \
testExecRestart , & data ) < 0 ) \
ret = - 1 ; \
} while ( 0 )
# ifdef WITH_AVAHI
EXEC_RESTART_TEST ( " initial " ) ;
# endif
EXEC_RESTART_TEST ( " initial-nomdns " ) ;
EXEC_RESTART_TEST ( " anon-clients " ) ;
return ret = = 0 ? EXIT_SUCCESS : EXIT_FAILURE ;
}
# else
static int
mymain ( void )
{
return EXIT_AM_SKIP ;
}
# endif
VIRT_TEST_MAIN ( mymain ) ;