2005-11-10 19:12:31 +03:00
/*
2005-12-05 14:16:07 +03:00
* virsh . c : a Xen shell used to exercise the libvir API
2005-11-10 19:12:31 +03:00
*
* Copyright ( C ) 2005 Red Hat , Inc .
*
* See COPYING . LIB for the License of this software
*
* Daniel Veillard < veillard @ redhat . com >
2005-12-08 17:22:52 +03:00
* Karel Zak < kzak @ redhat . com >
*
* $ Id $
2005-11-10 19:12:31 +03:00
*/
2005-12-08 13:23:34 +03:00
# define _GNU_SOURCE /* isblank() */
2006-02-09 20:45:11 +03:00
# include "libvirt.h"
2006-02-28 00:34:28 +03:00
# include "virterror.h"
2005-11-10 19:12:31 +03:00
# include <stdio.h>
2005-12-08 13:23:34 +03:00
# include <stdlib.h>
# include <string.h>
# include <stdarg.h>
2005-12-01 19:35:42 +03:00
# include <unistd.h>
2005-12-08 13:23:34 +03:00
# include <getopt.h>
2005-12-01 19:35:42 +03:00
# include <sys/types.h>
2005-12-08 17:22:52 +03:00
# include <sys/time.h>
2005-12-08 13:23:34 +03:00
# include <ctype.h>
# include <readline/readline.h>
# include <readline/history.h>
# include "config.h"
2005-12-08 18:08:46 +03:00
# include "internal.h"
2005-12-08 13:23:34 +03:00
static char * progname ;
# ifndef TRUE
# define TRUE 1
# define FALSE 0
# endif
# define VSH_PROMPT_RW "virsh # "
# define VSH_PROMPT_RO "virsh > "
# define GETTIMEOFDAY(T) gettimeofday(T, NULL)
# define DIFF_MSEC(T, U) \
( ( ( ( int ) ( ( T ) - > tv_sec - ( U ) - > tv_sec ) ) * 1000000.0 + \
( ( int ) ( ( T ) - > tv_usec - ( U ) - > tv_usec ) ) ) / 1000.0 )
typedef enum {
VSH_MESG , /* standard output */
VSH_HEADER , /* header for standard output */
VSH_FOOTER , /* timing, last command state, or whatever */
VSH_DEBUG1 , /* debugN where 'N' = level */
VSH_DEBUG2 ,
VSH_DEBUG3 ,
VSH_DEBUG4 ,
VSH_DEBUG5
} vshOutType ;
2006-02-28 00:34:28 +03:00
/*
* The error handler for virtsh
*/
static void
virshErrorHandler ( void * unused , virErrorPtr error ) {
if ( ( unused ! = NULL ) | | ( error = = NULL ) )
return ;
/* Suppress the VIR_ERR_NO_XEN error which fails as non-root */
if ( ( error - > code = = VIR_ERR_NO_XEN ) | | ( error - > code = = VIR_ERR_OK ) )
return ;
virDefaultErrorFunc ( error ) ;
}
2005-12-08 13:23:34 +03:00
/*
* virsh command line grammar :
*
* command_line = < command > \ n | < command > ; < command > ; . . .
*
* command = < keyword > < option > < data >
*
* option = < bool_option > | < int_option > | < string_option >
* data = < string >
*
* bool_option = - - optionname
* int_option = - - optionname < number >
* string_option = - - optionname < string >
*
2006-01-25 12:46:22 +03:00
* keyword = [ a - zA - Z ]
* number = [ 0 - 9 ] +
* string = [ ^ [ : blank : ] ] | " [[:alnum:]] " $
2005-12-08 13:23:34 +03:00
*
*/
/*
* vshCmdOptType - command option type
*/
typedef enum {
VSH_OT_NONE = 0 , /* none */
VSH_OT_BOOL , /* boolean option */
VSH_OT_STRING , /* string option */
VSH_OT_INT , /* int option */
VSH_OT_DATA /* string data (as non-option) */
} vshCmdOptType ;
/*
* Command Option Flags
*/
# define VSH_OFLAG_NONE 0 /* without flags */
# define VSH_OFLAG_REQ (1 << 1) /* option required */
/* dummy */
typedef struct __vshControl vshControl ;
typedef struct __vshCmd vshCmd ;
/*
* vshCmdInfo - - information about command
*/
typedef struct {
2005-12-08 17:22:52 +03:00
const char * name ; /* name of information */
const char * data ; /* information */
2005-12-08 13:23:34 +03:00
} vshCmdInfo ;
/*
* vshCmdOptDef - command option definition
*/
typedef struct {
2005-12-08 17:22:52 +03:00
const char * name ; /* the name of option */
2005-12-08 13:23:34 +03:00
vshCmdOptType type ; /* option type */
int flag ; /* flags */
2005-12-08 17:22:52 +03:00
const char * help ; /* help string */
2005-12-08 13:23:34 +03:00
} vshCmdOptDef ;
/*
* vshCmdOpt - command options
*/
typedef struct vshCmdOpt {
vshCmdOptDef * def ; /* pointer to relevant option */
char * data ; /* allocated data */
struct vshCmdOpt * next ;
} vshCmdOpt ;
/*
* vshCmdDef - command definition
*/
typedef struct {
2005-12-08 17:22:52 +03:00
const char * name ;
2005-12-08 13:23:34 +03:00
int ( * handler ) ( vshControl * , vshCmd * ) ; /* command handler */
vshCmdOptDef * opts ; /* definition of command options */
vshCmdInfo * info ; /* details about command */
} vshCmdDef ;
/*
* vshCmd - parsed command
*/
typedef struct __vshCmd {
vshCmdDef * def ; /* command definition */
vshCmdOpt * opts ; /* list of command arguments */
struct __vshCmd * next ; /* next command */
} __vshCmd ;
/*
* vshControl
*/
typedef struct __vshControl {
virConnectPtr conn ; /* connection to hypervisor */
vshCmd * cmd ; /* the current command */
char * cmdstr ; /* string with command */
uid_t uid ; /* process owner */
int imode ; /* interactive mode? */
int quiet ; /* quiet mode */
int debug ; /* print debug messages? */
int timing ; /* print timing info? */
} __vshControl ;
2005-11-10 19:12:31 +03:00
2005-12-02 17:16:21 +03:00
2005-12-08 13:23:34 +03:00
static vshCmdDef commands [ ] ;
2005-12-08 17:22:52 +03:00
static void vshError ( vshControl * ctl , int doexit , const char * format , . . . ) ;
2005-12-08 13:23:34 +03:00
static int vshInit ( vshControl * ctl ) ;
static int vshDeinit ( vshControl * ctl ) ;
2005-12-08 17:22:52 +03:00
static void vshUsage ( vshControl * ctl , const char * cmdname ) ;
2005-12-08 13:23:34 +03:00
static int vshParseArgv ( vshControl * ctl , int argc , char * * argv ) ;
2005-12-08 17:22:52 +03:00
static const char * vshCmddefGetInfo ( vshCmdDef * cmd , const char * info ) ;
static vshCmdDef * vshCmddefSearch ( const char * cmdname ) ;
static int vshCmddefHelp ( vshControl * ctl , const char * name , int withprog ) ;
2005-12-08 13:23:34 +03:00
2005-12-08 17:22:52 +03:00
static vshCmdOpt * vshCommandOpt ( vshCmd * cmd , const char * name ) ;
static int vshCommandOptInt ( vshCmd * cmd , const char * name , int * found ) ;
static char * vshCommandOptString ( vshCmd * cmd , const char * name , int * found ) ;
static int vshCommandOptBool ( vshCmd * cmd , const char * name ) ;
2005-12-15 20:00:43 +03:00
static virDomainPtr vshCommandOptDomain ( vshControl * ctl , vshCmd * cmd , const char * optname , char * * name ) ;
2005-12-08 13:23:34 +03:00
2005-12-08 17:22:52 +03:00
static void vshPrint ( vshControl * ctl , vshOutType out , const char * format , . . . ) ;
2005-12-08 13:23:34 +03:00
2005-12-15 20:00:43 +03:00
2005-12-08 17:22:52 +03:00
static const char * vshDomainStateToString ( int state ) ;
2005-12-08 13:23:34 +03:00
static int vshConnectionUsability ( vshControl * ctl , virConnectPtr conn , int showerror ) ;
/* ---------------
* Commands
* - - - - - - - - - - - - - - -
*/
/*
* " help " command
*/
static vshCmdInfo info_help [ ] = {
{ " syntax " , " help [<command>] " } ,
{ " help " , " print help " } ,
{ " desc " , " Prints global help or command specific help. " } ,
2005-12-08 16:26:52 +03:00
{ " version " , " Prints versionning informations. " } ,
2005-12-08 13:23:34 +03:00
{ NULL , NULL }
} ;
static vshCmdOptDef opts_help [ ] = {
{ " command " , VSH_OT_DATA , 0 , " name of command " } ,
{ NULL , 0 , 0 , NULL }
} ;
static int
cmdHelp ( vshControl * ctl , vshCmd * cmd ) {
2005-12-08 17:22:52 +03:00
const char * cmdname = vshCommandOptString ( cmd , " command " , NULL ) ;
2005-12-08 13:23:34 +03:00
if ( ! cmdname ) {
vshCmdDef * def ;
vshPrint ( ctl , VSH_HEADER , " Commands: \n \n " ) ;
for ( def = commands ; def - > name ; def + + )
vshPrint ( ctl , VSH_MESG , " %-15s %s \n " , def - > name ,
vshCmddefGetInfo ( def , " help " ) ) ;
return TRUE ;
}
return vshCmddefHelp ( ctl , cmdname , FALSE ) ;
}
/*
* " connect " command
*/
static vshCmdInfo info_connect [ ] = {
{ " syntax " , " connect [--readonly] " } ,
{ " help " , " (re)connect to hypervisor " } ,
{ " desc " , " Connect to local hypervisor. This is build-in command after shell start up. " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_connect [ ] = {
{ " readonly " , VSH_OT_BOOL , 0 , " read-only connection " } ,
{ NULL , 0 , 0 , NULL }
} ;
static int
cmdConnect ( vshControl * ctl , vshCmd * cmd ) {
int ro = vshCommandOptBool ( cmd , " readonly " ) ;
if ( ctl - > conn ) {
if ( virConnectClose ( ctl - > conn ) ! = 0 ) {
vshError ( ctl , FALSE , " failed to disconnect from the hypervisor " ) ;
return FALSE ;
}
ctl - > conn = NULL ;
}
if ( ! ro )
ctl - > conn = virConnectOpen ( NULL ) ;
else
ctl - > conn = virConnectOpenReadOnly ( NULL ) ;
if ( ! ctl - > conn )
vshError ( ctl , FALSE , " failed to connect to the hypervisor " ) ;
return ctl - > conn ? TRUE : FALSE ;
}
/*
* " list " command
*/
static vshCmdInfo info_list [ ] = {
{ " syntax " , " list " } ,
{ " help " , " list domains " } ,
{ " desc " , " Returns list of domains. " } ,
{ NULL , NULL }
} ;
static int
2005-12-08 18:08:46 +03:00
cmdList ( vshControl * ctl , vshCmd * cmd ATTRIBUTE_UNUSED ) {
2005-12-08 13:23:34 +03:00
int * ids , maxid , i ;
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
maxid = virConnectNumOfDomains ( ctl - > conn ) ;
if ( maxid < = 0 ) {
/* strange, there should be at least dom0... */
vshError ( ctl , FALSE , " failed to list active domains. " ) ;
return FALSE ;
}
ids = malloc ( sizeof ( int ) * maxid ) ;
virConnectListDomains ( ctl - > conn , & ids [ 0 ] , maxid ) ;
vshPrint ( ctl , VSH_HEADER , " %3s %-20s %s \n " , " Id " , " Name " , " State " ) ;
vshPrint ( ctl , VSH_HEADER , " ---------------------------------- \n " ) ;
for ( i = 0 ; i < maxid ; i + + ) {
int ret ;
virDomainInfo info ;
virDomainPtr dom = virDomainLookupByID ( ctl - > conn , ids [ i ] ) ;
/* this kind of work with domains is not atomic operation */
if ( ! dom )
continue ;
ret = virDomainGetInfo ( dom , & info ) ;
vshPrint ( ctl , VSH_MESG , " %3d %-20s %s \n " ,
virDomainGetID ( dom ) ,
virDomainGetName ( dom ) ,
ret < 0 ? " no state " : vshDomainStateToString ( info . state ) ) ;
2005-12-09 02:01:48 +03:00
virDomainFree ( dom ) ;
2005-12-08 13:23:34 +03:00
}
free ( ids ) ;
return TRUE ;
}
/*
* " dstate " command
*/
static vshCmdInfo info_dstate [ ] = {
2005-12-15 20:00:43 +03:00
{ " syntax " , " dstate <domain> " } ,
2005-12-08 13:23:34 +03:00
{ " help " , " domain state " } ,
2005-12-15 20:00:43 +03:00
{ " desc " , " Returns state about a running domain. " } ,
2005-12-08 13:23:34 +03:00
{ NULL , NULL }
} ;
static vshCmdOptDef opts_dstate [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2005-12-15 20:00:43 +03:00
{ NULL , 0 , 0 , NULL }
2005-12-08 13:23:34 +03:00
} ;
static int
cmdDstate ( vshControl * ctl , vshCmd * cmd ) {
2005-12-05 21:14:37 +03:00
virDomainInfo info ;
2005-12-08 13:23:34 +03:00
virDomainPtr dom ;
2005-12-15 20:00:43 +03:00
int ret = TRUE ;
2005-12-08 13:23:34 +03:00
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
2005-12-05 21:14:37 +03:00
2005-12-15 20:00:43 +03:00
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , NULL ) ) )
2005-12-08 13:23:34 +03:00
return FALSE ;
if ( virDomainGetInfo ( dom , & info ) = = 0 )
vshPrint ( ctl , VSH_MESG , " %s \n " , vshDomainStateToString ( info . state ) ) ;
else
ret = FALSE ;
2005-12-09 02:01:48 +03:00
virDomainFree ( dom ) ;
return ret ;
}
/*
* " suspend " command
*/
static vshCmdInfo info_suspend [ ] = {
2005-12-15 20:00:43 +03:00
{ " syntax " , " suspend <domain> " } ,
{ " help " , " suspend a domain " } ,
2005-12-09 02:01:48 +03:00
{ " desc " , " Suspend a running domain. " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_suspend [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2005-12-15 20:00:43 +03:00
{ NULL , 0 , 0 , NULL }
2005-12-09 02:01:48 +03:00
} ;
static int
cmdSuspend ( vshControl * ctl , vshCmd * cmd ) {
virDomainPtr dom ;
2005-12-15 20:00:43 +03:00
char * name ;
int ret = TRUE ;
2005-12-09 02:01:48 +03:00
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
2005-12-15 20:00:43 +03:00
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , & name ) ) )
2005-12-09 02:01:48 +03:00
return FALSE ;
if ( virDomainSuspend ( dom ) = = 0 ) {
2005-12-15 20:00:43 +03:00
vshPrint ( ctl , VSH_MESG , " Domain %s suspended \n " , name ) ;
2005-12-09 02:01:48 +03:00
} else {
vshError ( ctl , FALSE , " Failed to suspend domain \n " ) ;
ret = FALSE ;
}
virDomainFree ( dom ) ;
return ret ;
}
2006-01-20 13:00:08 +03:00
/*
* " save " command
*/
static vshCmdInfo info_save [ ] = {
2006-01-25 12:46:22 +03:00
{ " syntax " , " save <domain> <file> " } ,
2006-01-20 13:00:08 +03:00
{ " help " , " save a domain state to a file " } ,
{ " desc " , " Save a running domain. " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_save [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2006-02-03 17:49:41 +03:00
{ " file " , VSH_OT_DATA , VSH_OFLAG_REQ , " where to save the data " } ,
2006-01-20 13:00:08 +03:00
{ NULL , 0 , 0 , NULL }
} ;
static int
cmdSave ( vshControl * ctl , vshCmd * cmd ) {
virDomainPtr dom ;
char * name ;
char * to ;
int ret = TRUE ;
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
2006-01-25 12:46:22 +03:00
if ( ! ( to = vshCommandOptString ( cmd , " file " , NULL ) ) )
2006-01-20 13:00:08 +03:00
return FALSE ;
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , & name ) ) )
return FALSE ;
if ( virDomainSave ( dom , to ) = = 0 ) {
vshPrint ( ctl , VSH_MESG , " Domain %s saved \n " , name ) ;
} else {
vshError ( ctl , FALSE , " Failed to save domain \n " ) ;
ret = FALSE ;
}
virDomainFree ( dom ) ;
return ret ;
}
/*
* " restore " command
*/
static vshCmdInfo info_restore [ ] = {
{ " syntax " , " restore a domain from <file> " } ,
{ " help " , " restore a domain from a saved state in a file " } ,
{ " desc " , " Restore a domain. " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_restore [ ] = {
2006-01-25 12:46:22 +03:00
{ " file " , VSH_OT_DATA , VSH_OFLAG_REQ , " the state to restore " } ,
2006-01-20 13:00:08 +03:00
{ NULL , 0 , 0 , NULL }
} ;
static int
cmdRestore ( vshControl * ctl , vshCmd * cmd ) {
char * from ;
int found ;
int ret = TRUE ;
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
from = vshCommandOptString ( cmd , " file " , & found ) ;
if ( ! found )
return FALSE ;
if ( virDomainRestore ( ctl - > conn , from ) = = 0 ) {
vshPrint ( ctl , VSH_MESG , " Domain restored from %s \n " , from ) ;
} else {
vshError ( ctl , FALSE , " Failed to restore domain \n " ) ;
ret = FALSE ;
}
return ret ;
}
2005-12-09 02:01:48 +03:00
/*
* " resume " command
*/
static vshCmdInfo info_resume [ ] = {
2005-12-15 20:00:43 +03:00
{ " syntax " , " resume <domain> " } ,
{ " help " , " resume a domain " } ,
2005-12-09 02:01:48 +03:00
{ " desc " , " Resume a previously suspended domain. " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_resume [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2005-12-15 20:00:43 +03:00
{ NULL , 0 , 0 , NULL }
2005-12-09 02:01:48 +03:00
} ;
static int
cmdResume ( vshControl * ctl , vshCmd * cmd ) {
virDomainPtr dom ;
2005-12-15 20:00:43 +03:00
int ret = TRUE ;
char * name ;
2005-12-09 02:01:48 +03:00
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
2005-12-15 20:00:43 +03:00
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , & name ) ) )
2005-12-09 02:01:48 +03:00
return FALSE ;
if ( virDomainResume ( dom ) = = 0 ) {
2005-12-15 20:00:43 +03:00
vshPrint ( ctl , VSH_MESG , " Domain %s resumed \n " , name ) ;
2005-12-09 02:01:48 +03:00
} else {
vshError ( ctl , FALSE , " Failed to resume domain \n " ) ;
ret = FALSE ;
}
virDomainFree ( dom ) ;
return ret ;
}
2005-12-16 15:16:41 +03:00
/*
* " shutdown " command
*/
static vshCmdInfo info_shutdown [ ] = {
{ " syntax " , " shutdown <domain> " } ,
{ " help " , " gracefully shutdown a domain " } ,
{ " desc " , " Run shutdown in the targetted domain " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_shutdown [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2005-12-16 15:16:41 +03:00
{ NULL , 0 , 0 , NULL }
} ;
static int
cmdShutdown ( vshControl * ctl , vshCmd * cmd ) {
virDomainPtr dom ;
int ret = TRUE ;
char * name ;
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , & name ) ) )
return FALSE ;
if ( virDomainShutdown ( dom ) = = 0 ) {
vshPrint ( ctl , VSH_MESG , " Domain %s is being shutdown \n " , name ) ;
} else {
vshError ( ctl , FALSE , " Failed to shutdown domain \n " ) ;
ret = FALSE ;
}
virDomainFree ( dom ) ;
return ret ;
}
2005-12-09 02:01:48 +03:00
/*
* " destroy " command
*/
static vshCmdInfo info_destroy [ ] = {
2005-12-15 20:00:43 +03:00
{ " syntax " , " destroy <domain> " } ,
{ " help " , " destroy a domain " } ,
2005-12-09 02:01:48 +03:00
{ " desc " , " Destroy a given domain. " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_destroy [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2005-12-15 20:00:43 +03:00
{ NULL , 0 , 0 , NULL }
2005-12-09 02:01:48 +03:00
} ;
static int
cmdDestroy ( vshControl * ctl , vshCmd * cmd ) {
virDomainPtr dom ;
2005-12-15 20:00:43 +03:00
int ret = TRUE ;
char * name ;
2005-12-09 02:01:48 +03:00
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
2005-12-15 20:00:43 +03:00
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , & name ) ) )
2005-12-09 02:01:48 +03:00
return FALSE ;
if ( virDomainDestroy ( dom ) = = 0 ) {
2005-12-15 20:00:43 +03:00
vshPrint ( ctl , VSH_MESG , " Domain %s destroyed \n " , name ) ;
2005-12-09 02:01:48 +03:00
} else {
vshError ( ctl , FALSE , " Failed to destroy domain \n " ) ;
ret = FALSE ;
virDomainFree ( dom ) ;
}
2005-12-08 13:23:34 +03:00
return ret ;
}
/*
* " dinfo " command
*/
static vshCmdInfo info_dinfo [ ] = {
2005-12-15 20:00:43 +03:00
{ " syntax " , " dinfo <domain> " } ,
2005-12-08 13:23:34 +03:00
{ " help " , " domain information " } ,
{ " desc " , " Returns basic information about the domain. " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_dinfo [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2005-12-09 02:01:48 +03:00
{ NULL , 0 , 0 , NULL }
2005-12-08 13:23:34 +03:00
} ;
static int
cmdDinfo ( vshControl * ctl , vshCmd * cmd ) {
virDomainInfo info ;
virDomainPtr dom ;
2005-12-15 20:00:43 +03:00
int ret = TRUE ;
2005-12-16 03:51:27 +03:00
char * str ;
2005-12-15 20:00:43 +03:00
2005-12-08 13:23:34 +03:00
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
2005-12-15 20:00:43 +03:00
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , NULL ) ) )
2005-12-08 13:23:34 +03:00
return FALSE ;
2005-12-16 03:51:27 +03:00
vshPrint ( ctl , VSH_MESG , " %-15s %d \n " , " Id: " ,
virDomainGetID ( dom ) ) ;
vshPrint ( ctl , VSH_MESG , " %-15s %s \n " , " Name: " ,
virDomainGetName ( dom ) ) ;
if ( ( str = virDomainGetOSType ( dom ) ) ) {
vshPrint ( ctl , VSH_MESG , " %-15s %s \n " , " OS Type: " , str ) ;
free ( str ) ;
}
2005-12-08 13:23:34 +03:00
if ( virDomainGetInfo ( dom , & info ) = = 0 ) {
vshPrint ( ctl , VSH_MESG , " %-15s %s \n " , " State: " ,
vshDomainStateToString ( info . state ) ) ;
2005-12-16 03:51:27 +03:00
2005-12-08 13:23:34 +03:00
vshPrint ( ctl , VSH_MESG , " %-15s %d \n " , " CPU(s): " ,
info . nrVirtCpu ) ;
2005-12-15 20:00:43 +03:00
if ( info . cpuTime ! = 0 )
{
float cpuUsed = info . cpuTime ;
cpuUsed / = 1000000000 ;
2005-12-08 13:23:34 +03:00
2005-12-15 20:00:43 +03:00
vshPrint ( ctl , VSH_MESG , " %-15s %.1fs \n " , " CPU time: " , cpuUsed ) ;
2005-12-08 13:23:34 +03:00
}
vshPrint ( ctl , VSH_MESG , " %-15s %lu kB \n " , " Max memory: " ,
info . maxMem ) ;
vshPrint ( ctl , VSH_MESG , " %-15s %lu kB \n " , " Used memory: " ,
info . memory ) ;
} else {
ret = FALSE ;
}
2005-12-09 02:01:48 +03:00
virDomainFree ( dom ) ;
2005-12-08 13:23:34 +03:00
return ret ;
}
2005-12-13 19:22:05 +03:00
/*
* " dumpxml " command
*/
static vshCmdInfo info_dumpxml [ ] = {
2005-12-15 20:00:43 +03:00
{ " syntax " , " dumpxml <name> " } ,
2005-12-13 19:22:05 +03:00
{ " help " , " domain information in XML " } ,
{ " desc " , " Ouput the domain informations as an XML dump to stdout " } ,
{ NULL , NULL }
} ;
static vshCmdOptDef opts_dumpxml [ ] = {
2006-01-25 12:46:22 +03:00
{ " domain " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name or id " } ,
2005-12-13 19:22:05 +03:00
{ NULL , 0 , 0 , NULL }
} ;
static int
cmdDumpXML ( vshControl * ctl , vshCmd * cmd ) {
virDomainPtr dom ;
2005-12-15 20:00:43 +03:00
int ret = TRUE ;
2005-12-13 19:22:05 +03:00
char * dump ;
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
2005-12-15 20:00:43 +03:00
if ( ! ( dom = vshCommandOptDomain ( ctl , cmd , " domain " , NULL ) ) )
2005-12-13 19:22:05 +03:00
return FALSE ;
dump = virDomainGetXMLDesc ( dom , 0 ) ;
if ( dump ! = NULL ) {
printf ( " %s " , dump ) ;
free ( dump ) ;
} else {
ret = FALSE ;
}
virDomainFree ( dom ) ;
return ret ;
}
2005-12-08 13:23:34 +03:00
/*
* " nameof " command
*/
static vshCmdInfo info_nameof [ ] = {
{ " syntax " , " nameof <id> " } ,
2005-12-15 20:00:43 +03:00
{ " help " , " convert a domain Id to domain name " } ,
2005-12-08 13:23:34 +03:00
{ NULL , NULL }
} ;
static vshCmdOptDef opts_nameof [ ] = {
2006-01-25 12:46:22 +03:00
{ " id " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain Id " } ,
2006-01-20 13:00:08 +03:00
{ NULL , 0 , 0 , NULL }
2005-12-08 13:23:34 +03:00
} ;
static int
cmdNameof ( vshControl * ctl , vshCmd * cmd ) {
int found ;
int id = vshCommandOptInt ( cmd , " id " , & found ) ;
virDomainPtr dom ;
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
if ( ! found )
return FALSE ;
dom = virDomainLookupByID ( ctl - > conn , id ) ;
if ( dom ) {
vshPrint ( ctl , VSH_MESG , " %s \n " , virDomainGetName ( dom ) ) ;
2005-12-09 02:01:48 +03:00
virDomainFree ( dom ) ;
2005-12-08 13:23:34 +03:00
} else {
vshError ( ctl , FALSE , " failed to get domain '%d' " , id ) ;
return FALSE ;
}
return TRUE ;
}
/*
* " idof " command
*/
static vshCmdInfo info_idof [ ] = {
{ " syntax " , " idof <name> " } ,
2005-12-15 20:00:43 +03:00
{ " help " , " convert a domain name to domain Id " } ,
2005-12-08 13:23:34 +03:00
{ NULL , NULL }
} ;
static vshCmdOptDef opts_idof [ ] = {
2006-01-25 12:46:22 +03:00
{ " name " , VSH_OT_DATA , VSH_OFLAG_REQ , " domain name " } ,
2005-12-08 13:23:34 +03:00
{ NULL , 0 , 0 , NULL }
} ;
static int
cmdIdof ( vshControl * ctl , vshCmd * cmd ) {
char * name = vshCommandOptString ( cmd , " name " , NULL ) ;
2005-12-05 14:16:07 +03:00
virDomainPtr dom ;
2005-12-08 13:23:34 +03:00
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
if ( ! name )
return FALSE ;
dom = virDomainLookupByName ( ctl - > conn , name ) ;
if ( dom ) {
2006-01-17 19:56:17 +03:00
vshPrint ( ctl , VSH_MESG , " %d \n " , virDomainGetID ( dom ) ) ;
2005-12-09 02:01:48 +03:00
virDomainFree ( dom ) ;
2005-12-08 13:23:34 +03:00
} else {
vshError ( ctl , FALSE , " failed to get domain '%s' " , name ) ;
return FALSE ;
}
return TRUE ;
}
2005-12-08 16:26:52 +03:00
/*
* " version " command
*/
static vshCmdInfo info_version [ ] = {
{ " syntax " , " version " } ,
{ " help " , " show versions " } ,
{ " desc " , " Display the version informations available " } ,
{ NULL , NULL }
} ;
static int
2005-12-08 18:08:46 +03:00
cmdVersion ( vshControl * ctl , vshCmd * cmd ATTRIBUTE_UNUSED ) {
2005-12-08 16:26:52 +03:00
unsigned long hvVersion ;
const char * hvType ;
2005-12-08 18:08:46 +03:00
unsigned long libVersion ;
unsigned long includeVersion ;
unsigned long apiVersion ;
int ret ;
unsigned int major ;
unsigned int minor ;
unsigned int rel ;
2005-12-08 16:26:52 +03:00
if ( ! vshConnectionUsability ( ctl , ctl - > conn , TRUE ) )
return FALSE ;
hvType = virConnectGetType ( ctl - > conn ) ;
if ( hvType = = NULL ) {
2005-12-08 17:22:52 +03:00
vshError ( ctl , FALSE , " failed to get hypervisor type \n " ) ;
2005-12-08 16:26:52 +03:00
return FALSE ;
}
2005-12-08 18:08:46 +03:00
includeVersion = LIBVIR_VERSION_NUMBER ;
major = includeVersion / 1000000 ;
includeVersion % = 1000000 ;
minor = includeVersion / 1000 ;
rel = includeVersion % 1000 ;
vshPrint ( ctl , VSH_MESG , " Compiled against library: libvir %d.%d.%d \n " ,
major , minor , rel ) ;
ret = virGetVersion ( & libVersion , hvType , & apiVersion ) ;
if ( ret < 0 ) {
vshError ( ctl , FALSE , " failed to get the library version " ) ;
return FALSE ;
}
major = libVersion / 1000000 ;
libVersion % = 1000000 ;
minor = libVersion / 1000 ;
rel = libVersion % 1000 ;
vshPrint ( ctl , VSH_MESG , " Using library: libvir %d.%d.%d \n " ,
major , minor , rel ) ;
major = apiVersion / 1000000 ;
apiVersion % = 1000000 ;
minor = apiVersion / 1000 ;
rel = apiVersion % 1000 ;
vshPrint ( ctl , VSH_MESG , " Using API: %s %d.%d.%d \n " , hvType ,
major , minor , rel ) ;
ret = virConnectGetVersion ( ctl - > conn , & hvVersion ) ;
if ( ret < 0 ) {
vshError ( ctl , FALSE , " failed to get the hypervisor version " ) ;
2005-12-08 16:26:52 +03:00
return FALSE ;
}
if ( hvVersion = = 0 ) {
vshPrint ( ctl , VSH_MESG ,
2005-12-08 17:22:52 +03:00
" cannot extract running %s hypervisor version \n " ,
2005-12-08 16:26:52 +03:00
hvType ) ;
} else {
2005-12-08 18:08:46 +03:00
major = hvVersion / 1000000 ;
2005-12-08 16:26:52 +03:00
hvVersion % = 1000000 ;
2005-12-08 18:08:46 +03:00
minor = hvVersion / 1000 ;
rel = hvVersion % 1000 ;
2005-12-08 16:26:52 +03:00
vshPrint ( ctl , VSH_MESG , " Running hypervisor: %s %d.%d.%d \n " , hvType ,
major , minor , rel ) ;
}
return TRUE ;
}
2005-12-08 13:23:34 +03:00
/*
* " quit " command
*/
static vshCmdInfo info_quit [ ] = {
{ " syntax " , " quit " } ,
{ " help " , " quit this interactive terminal " } ,
{ NULL , NULL }
} ;
static int
2005-12-08 18:08:46 +03:00
cmdQuit ( vshControl * ctl , vshCmd * cmd ATTRIBUTE_UNUSED ) {
2005-12-08 13:23:34 +03:00
ctl - > imode = FALSE ;
return TRUE ;
}
/*
* Commands
*/
static vshCmdDef commands [ ] = {
{ " connect " , cmdConnect , opts_connect , info_connect } ,
{ " dinfo " , cmdDinfo , opts_dinfo , info_dinfo } ,
2005-12-13 19:22:05 +03:00
{ " dumpxml " , cmdDumpXML , opts_dumpxml , info_dumpxml } ,
2005-12-08 13:23:34 +03:00
{ " dstate " , cmdDstate , opts_dstate , info_dstate } ,
2005-12-09 02:01:48 +03:00
{ " suspend " , cmdSuspend , opts_suspend , info_suspend } ,
{ " resume " , cmdResume , opts_resume , info_resume } ,
2006-01-20 13:00:08 +03:00
{ " save " , cmdSave , opts_save , info_save } ,
{ " restore " , cmdRestore , opts_restore , info_restore } ,
2005-12-16 15:16:41 +03:00
{ " shutdown " , cmdShutdown , opts_shutdown , info_shutdown } ,
2005-12-09 02:01:48 +03:00
{ " destroy " , cmdDestroy , opts_destroy , info_destroy } ,
2005-12-08 13:23:34 +03:00
{ " help " , cmdHelp , opts_help , info_help } ,
{ " idof " , cmdIdof , opts_idof , info_idof } ,
{ " list " , cmdList , NULL , info_list } ,
{ " nameof " , cmdNameof , opts_nameof , info_nameof } ,
2005-12-08 16:26:52 +03:00
{ " version " , cmdVersion , NULL , info_version } ,
2005-12-08 13:23:34 +03:00
{ " quit " , cmdQuit , NULL , info_quit } ,
{ NULL , NULL , NULL , NULL }
} ;
/* ---------------
* Utils for work with command definition
* - - - - - - - - - - - - - - -
*/
2005-12-08 17:22:52 +03:00
static const char *
vshCmddefGetInfo ( vshCmdDef * cmd , const char * name ) {
2005-12-08 13:23:34 +03:00
vshCmdInfo * info ;
for ( info = cmd - > info ; info & & info - > name ; info + + ) {
if ( strcmp ( info - > name , name ) = = 0 )
return info - > data ;
}
return NULL ;
}
static vshCmdOptDef *
2005-12-08 17:22:52 +03:00
vshCmddefGetOption ( vshCmdDef * cmd , const char * name ) {
2005-12-08 13:23:34 +03:00
vshCmdOptDef * opt ;
for ( opt = cmd - > opts ; opt & & opt - > name ; opt + + )
if ( strcmp ( opt - > name , name ) = = 0 )
return opt ;
return NULL ;
}
static vshCmdOptDef *
2006-01-25 12:46:22 +03:00
vshCmddefGetData ( vshCmdDef * cmd , int data_ct ) {
2005-12-08 13:23:34 +03:00
vshCmdOptDef * opt ;
2006-01-25 12:46:22 +03:00
for ( opt = cmd - > opts ; opt & & opt - > name ; opt + + ) {
if ( opt - > type = = VSH_OT_DATA ) {
if ( data_ct = = 0 )
return opt ;
else
data_ct - - ;
}
}
2005-12-08 13:23:34 +03:00
return NULL ;
}
2006-01-25 12:46:22 +03:00
/*
* Checks for required options
*/
static int
vshCommandCheckOpts ( vshControl * ctl , vshCmd * cmd )
{
vshCmdDef * def = cmd - > def ;
vshCmdOptDef * d ;
int err = 0 ;
for ( d = def - > opts ; d & & d - > name ; d + + ) {
if ( d - > flag & VSH_OFLAG_REQ ) {
vshCmdOpt * o = cmd - > opts ;
int ok = 0 ;
while ( o & & ok = = 0 ) {
if ( o - > def = = d )
ok = 1 ;
o = o - > next ;
}
if ( ! ok ) {
vshError ( ctl , FALSE ,
d - > type = = VSH_OT_DATA ?
" command '%s' requires <%s> option " :
" command '%s' requires --%s option " ,
def - > name , d - > name ) ;
err = 1 ;
}
}
}
return ! err ;
}
2005-12-08 13:23:34 +03:00
static vshCmdDef *
2005-12-08 17:22:52 +03:00
vshCmddefSearch ( const char * cmdname ) {
2005-12-08 13:23:34 +03:00
vshCmdDef * c ;
for ( c = commands ; c - > name ; c + + )
if ( strcmp ( c - > name , cmdname ) = = 0 )
return c ;
return NULL ;
}
static int
2005-12-08 17:22:52 +03:00
vshCmddefHelp ( vshControl * ctl , const char * cmdname , int withprog ) {
2005-12-08 13:23:34 +03:00
vshCmdDef * def = vshCmddefSearch ( cmdname ) ;
if ( ! def ) {
vshError ( ctl , FALSE , " command '%s' doesn't exist " , cmdname ) ;
return FALSE ;
} else {
vshCmdOptDef * opt ;
2005-12-08 17:22:52 +03:00
const char * desc = vshCmddefGetInfo ( def , " desc " ) ;
const char * help = vshCmddefGetInfo ( def , " help " ) ;
const char * syntax = vshCmddefGetInfo ( def , " syntax " ) ;
2005-12-08 13:23:34 +03:00
fputs ( " NAME \n " , stdout ) ;
fprintf ( stdout , " %s - %s \n " , def - > name , help ) ;
if ( syntax ) {
fputs ( " \n SYNOPSIS \n " , stdout ) ;
if ( ! withprog )
fprintf ( stdout , " %s \n " , syntax ) ;
else
fprintf ( stdout , " %s %s \n " , progname , syntax ) ;
}
if ( desc ) {
fputs ( " \n DESCRIPTION \n " , stdout ) ;
fprintf ( stdout , " %s \n " , desc ) ;
}
if ( def - > opts ) {
fputs ( " \n OPTIONS \n " , stdout ) ;
for ( opt = def - > opts ; opt - > name ; opt + + ) {
char buf [ 256 ] ;
if ( opt - > type = = VSH_OT_BOOL )
snprintf ( buf , sizeof ( buf ) , " --%s " , opt - > name ) ;
else if ( opt - > type = = VSH_OT_INT )
snprintf ( buf , sizeof ( buf ) , " --%s <number> " , opt - > name ) ;
else if ( opt - > type = = VSH_OT_STRING )
snprintf ( buf , sizeof ( buf ) , " --%s <string> " , opt - > name ) ;
else if ( opt - > type = = VSH_OT_DATA )
snprintf ( buf , sizeof ( buf ) , " <%s> " , opt - > name ) ;
fprintf ( stdout , " %-15s %s \n " , buf , opt - > help ) ;
}
}
fputc ( ' \n ' , stdout ) ;
}
return TRUE ;
}
/* ---------------
* Utils for work with runtime commands data
* - - - - - - - - - - - - - - -
*/
static void
vshCommandOptFree ( vshCmdOpt * arg ) {
vshCmdOpt * a = arg ;
while ( a ) {
vshCmdOpt * tmp = a ;
a = a - > next ;
if ( tmp - > data )
free ( tmp - > data ) ;
free ( tmp ) ;
}
}
static void
vshCommandFree ( vshCmd * cmd ) {
vshCmd * c = cmd ;
while ( c ) {
vshCmd * tmp = c ;
c = c - > next ;
if ( tmp - > opts )
vshCommandOptFree ( tmp - > opts ) ;
free ( tmp ) ;
}
}
/*
* Returns option by name
*/
static vshCmdOpt *
2005-12-08 17:22:52 +03:00
vshCommandOpt ( vshCmd * cmd , const char * name ) {
2005-12-08 13:23:34 +03:00
vshCmdOpt * opt = cmd - > opts ;
while ( opt ) {
if ( opt - > def & & strcmp ( opt - > def - > name , name ) = = 0 )
return opt ;
opt = opt - > next ;
}
return NULL ;
}
/*
* Returns option as INT
*/
static int
2005-12-08 17:22:52 +03:00
vshCommandOptInt ( vshCmd * cmd , const char * name , int * found ) {
2005-12-08 13:23:34 +03:00
vshCmdOpt * arg = vshCommandOpt ( cmd , name ) ;
int res = 0 ;
if ( arg )
res = atoi ( arg - > data ) ;
if ( found )
* found = arg ? TRUE : FALSE ;
return res ;
}
/*
* Returns option as STRING
*/
static char *
2005-12-08 17:22:52 +03:00
vshCommandOptString ( vshCmd * cmd , const char * name , int * found ) {
2005-12-08 13:23:34 +03:00
vshCmdOpt * arg = vshCommandOpt ( cmd , name ) ;
if ( found )
* found = arg ? TRUE : FALSE ;
2006-01-25 12:46:22 +03:00
return arg & & arg - > data & & * arg - > data ? arg - > data : NULL ;
2005-12-08 13:23:34 +03:00
}
/*
* Returns TRUE / FALSE if the option exists
*/
static int
2005-12-08 17:22:52 +03:00
vshCommandOptBool ( vshCmd * cmd , const char * name ) {
2005-12-08 13:23:34 +03:00
return vshCommandOpt ( cmd , name ) ? TRUE : FALSE ;
}
2006-01-25 12:46:22 +03:00
2005-12-15 20:00:43 +03:00
static virDomainPtr
vshCommandOptDomain ( vshControl * ctl , vshCmd * cmd , const char * optname , char * * name ) {
virDomainPtr dom = NULL ;
char * n , * end = NULL ;
int id ;
if ( ! ( n = vshCommandOptString ( cmd , optname , NULL ) ) ) {
vshError ( ctl , FALSE , " undefined domain name or id " ) ;
return NULL ;
}
2006-01-25 12:46:22 +03:00
vshPrint ( ctl , VSH_DEBUG5 , " %s: found option <%s>: %s \n " , cmd - > def - > name , optname , n ) ;
2005-12-15 20:00:43 +03:00
if ( name )
* name = n ;
/* try it by ID */
id = ( int ) strtol ( n , & end , 10 ) ;
2006-01-25 12:46:22 +03:00
if ( id > = 0 & & end & & * end = = ' \0 ' ) {
vshPrint ( ctl , VSH_DEBUG5 , " %s: <%s> seems like domain ID \n " , cmd - > def - > name , optname ) ;
2005-12-15 20:00:43 +03:00
dom = virDomainLookupByID ( ctl - > conn , id ) ;
2006-01-25 12:46:22 +03:00
}
2005-12-15 20:00:43 +03:00
/* try it by NAME */
2006-01-25 12:46:22 +03:00
if ( ! dom ) {
vshPrint ( ctl , VSH_DEBUG5 , " %s: <%s> tring as domain NAME \n " , cmd - > def - > name , optname ) ;
2005-12-15 20:00:43 +03:00
dom = virDomainLookupByName ( ctl - > conn , n ) ;
2006-01-25 12:46:22 +03:00
}
2005-12-15 20:00:43 +03:00
if ( ! dom )
vshError ( ctl , FALSE , " failed to get domain '%s' " , n ) ;
return dom ;
}
2005-12-08 13:23:34 +03:00
/*
* Executes command ( s ) and returns return code from last command
*/
static int
vshCommandRun ( vshControl * ctl , vshCmd * cmd ) {
int ret = TRUE ;
while ( cmd ) {
struct timeval before , after ;
if ( ctl - > timing )
GETTIMEOFDAY ( & before ) ;
ret = cmd - > def - > handler ( ctl , cmd ) ;
if ( ctl - > timing )
GETTIMEOFDAY ( & after ) ;
if ( strcmp ( cmd - > def - > name , " quit " ) = = 0 ) /* hack ... */
return ret ;
if ( ctl - > timing )
vshPrint ( ctl , VSH_MESG , " \n (Time: %.3f ms) \n \n " , DIFF_MSEC ( & after , & before ) ) ;
else
vshPrint ( ctl , VSH_FOOTER , " \n " ) ;
cmd = cmd - > next ;
}
return ret ;
}
/* ---------------
* Command string parsing
* - - - - - - - - - - - - - - -
*/
# define VSH_TK_ERROR -1
# define VSH_TK_NONE 0
# define VSH_TK_OPTION 1
# define VSH_TK_DATA 2
# define VSH_TK_END 3
static int
vshCommandGetToken ( vshControl * ctl , char * str , char * * end , char * * res ) {
int tk = VSH_TK_NONE ;
int quote = FALSE ;
int sz = 0 ;
char * p = str ;
char * tkstr = NULL ;
* end = NULL ;
while ( p & & * p & & isblank ( ( unsigned char ) * p ) )
p + + ;
2005-11-10 19:12:31 +03:00
2005-12-08 13:23:34 +03:00
if ( p = = NULL | | * p = = ' \0 ' )
return VSH_TK_END ;
if ( * p = = ' ; ' ) {
* end = + + p ; /* = \0 or begi of next command */
return VSH_TK_END ;
}
while ( * p ) {
/* end of token is blank space or ';' */
if ( ( quote = = FALSE & & isblank ( ( unsigned char ) * p ) ) | | * p = = ' ; ' )
break ;
2006-01-25 12:46:22 +03:00
/* end of option name could be '=' */
if ( tk = = VSH_TK_OPTION & & * p = = ' = ' ) {
p + + ; /* skip '=' */
break ;
}
2005-12-08 13:23:34 +03:00
if ( tk = = VSH_TK_NONE ) {
if ( * p = = ' - ' & & * ( p + 1 ) = = ' - ' & & * ( p + 2 ) & & isalnum ( ( unsigned char ) * ( p + 2 ) ) ) {
tk = VSH_TK_OPTION ;
p + = 2 ;
} else {
tk = VSH_TK_DATA ;
if ( * p = = ' " ' ) {
quote = TRUE ;
p + + ;
} else {
quote = FALSE ;
}
}
tkstr = p ; /* begin of token */
} else if ( quote & & * p = = ' " ' ) {
quote = FALSE ;
p + + ;
break ; /* end of "..." token */
}
p + + ;
sz + + ;
}
if ( quote ) {
vshError ( ctl , FALSE , " missing \" " ) ;
return VSH_TK_ERROR ;
}
if ( tkstr = = NULL | | * tkstr = = ' \0 ' | | p = = NULL )
return VSH_TK_END ;
if ( sz = = 0 )
return VSH_TK_END ;
* res = malloc ( sz + 1 ) ;
memcpy ( * res , tkstr , sz ) ;
* ( * res + sz ) = ' \0 ' ;
* end = p ;
return tk ;
}
static int
vshCommandParse ( vshControl * ctl , char * cmdstr ) {
char * str ;
char * tkdata = NULL ;
vshCmd * clast = NULL ;
vshCmdOpt * first = NULL ;
if ( ctl - > cmd ) {
vshCommandFree ( ctl - > cmd ) ;
ctl - > cmd = NULL ;
}
if ( cmdstr = = NULL | | * cmdstr = = ' \0 ' )
return FALSE ;
str = cmdstr ;
while ( str & & * str )
{
vshCmdOpt * last = NULL ;
vshCmdDef * cmd = NULL ;
int tk = VSH_TK_NONE ;
2006-01-25 12:46:22 +03:00
int data_ct = 0 ;
2005-12-08 13:23:34 +03:00
first = NULL ;
while ( tk ! = VSH_TK_END ) {
char * end = NULL ;
vshCmdOptDef * opt = NULL ;
tkdata = NULL ;
/* get token */
tk = vshCommandGetToken ( ctl , str , & end , & tkdata ) ;
str = end ;
if ( tk = = VSH_TK_END )
break ;
if ( tk = = VSH_TK_ERROR )
goto syntaxError ;
if ( cmd = = NULL ) {
/* first token must be command name */
if ( tk ! = VSH_TK_DATA ) {
vshError ( ctl , FALSE ,
" unexpected token (command name): '%s' " ,
tkdata ) ;
goto syntaxError ;
}
if ( ! ( cmd = vshCmddefSearch ( tkdata ) ) ) {
vshError ( ctl , FALSE ,
" unknown command: '%s' " , tkdata ) ;
goto syntaxError ; /* ... or ignore this command only? */
}
free ( tkdata ) ;
} else if ( tk = = VSH_TK_OPTION ) {
if ( ! ( opt = vshCmddefGetOption ( cmd , tkdata ) ) ) {
vshError ( ctl , FALSE ,
" command '%s' doesn't support option --%s " ,
cmd - > name , tkdata ) ;
goto syntaxError ;
}
free ( tkdata ) ; /* option name */
tkdata = NULL ;
if ( opt - > type ! = VSH_OT_BOOL ) {
/* option data */
tk = vshCommandGetToken ( ctl , str , & end , & tkdata ) ;
str = end ;
if ( tk = = VSH_TK_ERROR )
goto syntaxError ;
if ( tk ! = VSH_TK_DATA ) {
vshError ( ctl , FALSE ,
" expected syntax: --%s <%s> " ,
opt - > name ,
opt - > type = = VSH_OT_INT ? " number " : " string " ) ;
goto syntaxError ;
}
}
} else if ( tk = = VSH_TK_DATA ) {
2006-01-25 12:46:22 +03:00
if ( ! ( opt = vshCmddefGetData ( cmd , data_ct + + ) ) ) {
2005-12-08 13:23:34 +03:00
vshError ( ctl , FALSE ,
" unexpected data '%s' " ,
tkdata ) ;
goto syntaxError ;
}
}
if ( opt ) {
/* save option */
vshCmdOpt * arg = malloc ( sizeof ( vshCmdOpt ) ) ;
arg - > def = opt ;
arg - > data = tkdata ;
arg - > next = NULL ;
tkdata = NULL ;
if ( ! first )
first = arg ;
if ( last )
last - > next = arg ;
last = arg ;
vshPrint ( ctl , VSH_DEBUG4 , " %s: %s(%s): %s \n " ,
cmd - > name ,
opt - > name ,
tk = = VSH_TK_OPTION ? " OPTION " : " DATA " ,
arg - > data ) ;
}
if ( ! str )
break ;
}
/* commad parsed -- allocate new struct for the command */
if ( cmd ) {
vshCmd * c = malloc ( sizeof ( vshCmd ) ) ;
c - > opts = first ;
c - > def = cmd ;
c - > next = NULL ;
2006-01-25 12:46:22 +03:00
if ( ! vshCommandCheckOpts ( ctl , c ) )
goto syntaxError ;
2005-12-08 13:23:34 +03:00
if ( ! ctl - > cmd )
ctl - > cmd = c ;
if ( clast )
clast - > next = c ;
clast = c ;
}
}
return TRUE ;
syntaxError :
if ( ctl - > cmd )
vshCommandFree ( ctl - > cmd ) ;
if ( first )
vshCommandOptFree ( first ) ;
if ( tkdata )
free ( tkdata ) ;
return FALSE ;
}
/* ---------------
* Misc utils
* - - - - - - - - - - - - - - -
*/
2005-12-08 17:22:52 +03:00
static const char *
2005-12-08 13:23:34 +03:00
vshDomainStateToString ( int state ) {
switch ( state ) {
case VIR_DOMAIN_RUNNING :
return " running " ;
case VIR_DOMAIN_BLOCKED :
return " blocked " ;
case VIR_DOMAIN_PAUSED :
return " paused " ;
case VIR_DOMAIN_SHUTDOWN :
return " in shutdown " ;
case VIR_DOMAIN_SHUTOFF :
return " shut off " ;
default :
return " no state " ; /* = dom0 state */
}
return NULL ;
}
static int
vshConnectionUsability ( vshControl * ctl , virConnectPtr conn , int showerror ) {
/* TODO: use something like virConnectionState() to
* check usability of the connection
*/
if ( ! conn ) {
if ( showerror )
vshError ( ctl , FALSE , " no valid connection. " ) ;
return FALSE ;
}
return TRUE ;
}
static int
vshWantedDebug ( vshOutType type , int mode ) {
switch ( type ) {
case VSH_DEBUG5 :
if ( mode < 5 )
return FALSE ;
return TRUE ;
case VSH_DEBUG4 :
if ( mode < 4 )
return FALSE ;
return TRUE ;
case VSH_DEBUG3 :
if ( mode < 3 )
return FALSE ;
return TRUE ;
case VSH_DEBUG2 :
if ( mode < 2 )
return FALSE ;
return TRUE ;
case VSH_DEBUG1 :
if ( mode < 1 )
return FALSE ;
return TRUE ;
default :
/* it's right, all others types have to pass */
return TRUE ;
}
return FALSE ;
}
static void
2005-12-08 17:22:52 +03:00
vshPrint ( vshControl * ctl , vshOutType type , const char * format , . . . ) {
2005-12-08 13:23:34 +03:00
va_list ap ;
if ( ctl - > quiet = = TRUE & & ( type = = VSH_HEADER | | type = = VSH_FOOTER ) )
return ;
if ( ! vshWantedDebug ( type , ctl - > debug ) )
return ;
va_start ( ap , format ) ;
vfprintf ( stderr , format , ap ) ;
va_end ( ap ) ;
}
static void
2005-12-08 17:22:52 +03:00
vshError ( vshControl * ctl , int doexit , const char * format , . . . ) {
2005-12-08 13:23:34 +03:00
va_list ap ;
if ( doexit )
fprintf ( stderr , " %s: error: " , progname ) ;
else
fputs ( " error: " , stderr ) ;
va_start ( ap , format ) ;
vfprintf ( stderr , format , ap ) ;
va_end ( ap ) ;
fputc ( ' \n ' , stderr ) ;
if ( doexit ) {
vshDeinit ( ctl ) ;
exit ( EXIT_FAILURE ) ;
}
}
/*
* Initialize vistsh
*/
static int
vshInit ( vshControl * ctl ) {
if ( ctl - > conn )
return FALSE ;
ctl - > uid = getuid ( ) ;
2006-02-28 00:34:28 +03:00
/* set up the library error handler */
virSetErrorFunc ( NULL , virshErrorHandler ) ;
2005-12-08 13:23:34 +03:00
/* basic connection to hypervisor */
if ( ctl - > uid = = 0 )
ctl - > conn = virConnectOpen ( NULL ) ;
else
ctl - > conn = virConnectOpenReadOnly ( NULL ) ;
if ( ! ctl - > conn )
vshError ( ctl , TRUE , " failed to connect to the hypervisor " ) ;
return TRUE ;
}
/* -----------------
* Readline stuff
* - - - - - - - - - - - - - - - - -
*/
/*
* Generator function for command completion . STATE lets us
* know whether to start from scratch ; without any state
* ( i . e . STATE = = 0 ) , then we start at the top of the list .
*/
static char *
vshReadlineCommandGenerator ( const char * text , int state ) {
static int list_index , len ;
2005-12-08 17:22:52 +03:00
const char * name ;
2005-12-08 13:23:34 +03:00
/* If this is a new word to complete, initialize now. This
* includes saving the length of TEXT for efficiency , and
* initializing the index variable to 0.
*/
if ( ! state ) {
list_index = 0 ;
len = strlen ( text ) ;
}
/* Return the next name which partially matches from the
* command list .
*/
2005-12-08 17:22:52 +03:00
while ( ( name = commands [ list_index ] . name ) ) {
2005-12-08 13:23:34 +03:00
list_index + + ;
if ( strncmp ( name , text , len ) = = 0 )
return strdup ( name ) ;
}
/* If no names matched, then return NULL. */
return NULL ;
}
static char *
vshReadlineOptionsGenerator ( const char * text , int state ) {
static int list_index , len ;
static vshCmdDef * cmd = NULL ;
2005-12-08 17:22:52 +03:00
const char * name ;
2005-12-08 13:23:34 +03:00
if ( ! state ) {
/* determine command name */
char * p ;
char * cmdname ;
if ( ! ( p = strchr ( rl_line_buffer , ' ' ) ) )
return NULL ;
cmdname = calloc ( ( p - rl_line_buffer ) + 1 , 1 ) ;
memcpy ( cmdname , rl_line_buffer , p - rl_line_buffer ) ;
cmd = vshCmddefSearch ( cmdname ) ;
list_index = 0 ;
len = strlen ( text ) ;
free ( cmdname ) ;
}
if ( ! cmd )
return NULL ;
2005-12-08 17:22:52 +03:00
while ( ( name = cmd - > opts [ list_index ] . name ) ) {
2005-12-08 13:23:34 +03:00
vshCmdOptDef * opt = & cmd - > opts [ list_index ] ;
char * res ;
list_index + + ;
2005-12-08 17:22:52 +03:00
if ( opt - > type = = VSH_OT_DATA )
2005-12-08 13:23:34 +03:00
/* ignore non --option */
continue ;
if ( len > 2 ) {
if ( strncmp ( name , text + 2 , len - 2 ) )
continue ;
}
res = malloc ( strlen ( name ) + 3 ) ;
sprintf ( res , " --%s " , name ) ;
return res ;
}
/* If no names matched, then return NULL. */
return NULL ;
}
static char * *
2005-12-08 18:08:46 +03:00
vshReadlineCompletion ( const char * text , int start , int end ATTRIBUTE_UNUSED ) {
2005-12-08 13:23:34 +03:00
char * * matches = ( char * * ) NULL ;
if ( start = = 0 )
/* command name generator */
matches = rl_completion_matches ( text , vshReadlineCommandGenerator ) ;
else
/* commands options */
matches = rl_completion_matches ( text , vshReadlineOptionsGenerator ) ;
return matches ;
}
static void
2005-12-08 17:22:52 +03:00
vshReadlineInit ( void ) {
2005-12-08 13:23:34 +03:00
/* Allow conditional parsing of the ~/.inputrc file. */
rl_readline_name = " virsh " ;
/* Tell the completer that we want a crack first. */
rl_attempted_completion_function = vshReadlineCompletion ;
}
/*
* Deinitliaze virsh
*/
static int
vshDeinit ( vshControl * ctl ) {
if ( ctl - > conn ) {
if ( virConnectClose ( ctl - > conn ) ! = 0 ) {
ctl - > conn = NULL ; /* prevent recursive call from vshError() */
vshError ( ctl , TRUE , " failed to disconnect from the hypervisor " ) ;
}
}
return TRUE ;
}
/*
* Print usage
*/
static void
2005-12-08 17:22:52 +03:00
vshUsage ( vshControl * ctl , const char * cmdname ) {
2005-12-08 13:23:34 +03:00
vshCmdDef * cmd ;
/* global help */
if ( ! cmdname ) {
fprintf ( stdout , " \n %s [options] [commands] \n \n "
" options: \n "
" -d | --debug <num> debug level [0-5] \n "
" -h | --help this help \n "
" -q | --quiet quiet mode \n "
" -t | --timing print timing information \n "
" -v | --version program version \n \n "
" commands (non interactive mode): \n " , progname ) ;
for ( cmd = commands ; cmd - > name ; cmd + + )
fprintf ( stdout ,
" %-15s %s \n " , cmd - > name , vshCmddefGetInfo ( cmd , " help " ) ) ;
fprintf ( stdout , " \n (specify --help <command> for details about the command) \n \n " ) ;
return ;
}
if ( ! vshCmddefHelp ( ctl , cmdname , TRUE ) )
exit ( EXIT_FAILURE ) ;
}
/*
* argv [ ] : virsh [ options ] [ command ]
*
*/
static int
vshParseArgv ( vshControl * ctl , int argc , char * * argv ) {
char * last = NULL ;
int i , end = 0 , help = 0 ;
int arg , idx = 0 ;
struct option opt [ ] = {
{ " debug " , 1 , 0 , ' d ' } ,
{ " help " , 0 , 0 , ' h ' } ,
{ " quiet " , 0 , 0 , ' q ' } ,
{ " timing " , 0 , 0 , ' t ' } ,
{ " version " , 0 , 0 , ' v ' } ,
{ 0 , 0 , 0 , 0 }
} ;
if ( argc < 2 )
2005-12-08 17:22:52 +03:00
return TRUE ;
2005-12-08 13:23:34 +03:00
2006-01-25 12:46:22 +03:00
/* look for begin of the command, for example:
2005-12-08 13:23:34 +03:00
* . / virsh - - debug 5 - q command - - cmdoption
* < - - - ^ - - - >
* getopt ( ) stuff | command suff
*/
for ( i = 1 ; i < argc ; i + + ) {
if ( * argv [ i ] ! = ' - ' ) {
int valid = FALSE ;
/* non "--option" argv, is it command? */
if ( last ) {
struct option * o ;
int sz = strlen ( last ) ;
for ( o = opt ; o - > name ; o + + ) {
if ( sz = = 2 & & * ( last + 1 ) = = o - > val )
/* valid virsh short option */
valid = TRUE ;
else if ( sz > 2 & & strcmp ( o - > name , last + 2 ) = = 0 )
/* valid virsh long option */
valid = TRUE ;
}
}
if ( ! valid ) {
end = i ;
break ;
}
}
last = argv [ i ] ;
}
end = end ? : argc ;
/* standard (non-command) options */
while ( ( arg = getopt_long ( end , argv , " d:hqtv " , opt , & idx ) ) ! = - 1 ) {
switch ( arg ) {
case ' d ' :
ctl - > debug = atoi ( optarg ) ;
break ;
case ' h ' :
help = 1 ;
break ;
case ' q ' :
ctl - > quiet = TRUE ;
break ;
case ' t ' :
ctl - > timing = TRUE ;
break ;
case ' v ' :
fprintf ( stdout , " %s \n " , VERSION ) ;
exit ( EXIT_SUCCESS ) ;
default :
vshError ( ctl , TRUE , " unsupported option '-%c'. See --help. " , arg ) ;
break ;
}
}
if ( help ) {
/* global or command specific help */
vshUsage ( ctl , argc > end ? argv [ end ] : NULL ) ;
exit ( EXIT_SUCCESS ) ;
}
if ( argc > end ) {
/* parse command */
char * cmdstr ;
2005-12-08 17:22:52 +03:00
int sz = 0 , ret ;
2005-12-08 13:23:34 +03:00
ctl - > imode = FALSE ;
for ( i = end ; i < argc ; i + + )
sz + = strlen ( argv [ i ] ) + 1 ; /* +1 is for blank space between items */
cmdstr = calloc ( sz + 1 , 1 ) ;
for ( i = end ; i < argc ; i + + ) {
strncat ( cmdstr , argv [ i ] , sz ) ;
sz - = strlen ( argv [ i ] ) ;
strncat ( cmdstr , " " , sz - - ) ;
}
vshPrint ( ctl , VSH_DEBUG2 , " command: \" %s \" \n " , cmdstr ) ;
ret = vshCommandParse ( ctl , cmdstr ) ;
free ( cmdstr ) ;
return ret ;
}
return TRUE ;
}
int
main ( int argc , char * * argv ) {
vshControl _ctl , * ctl = & _ctl ;
int ret = TRUE ;
if ( ! ( progname = strrchr ( argv [ 0 ] , ' / ' ) ) )
progname = argv [ 0 ] ;
else
progname + + ;
memset ( ctl , 0 , sizeof ( vshControl ) ) ;
ctl - > imode = TRUE ; /* default is interactive mode */
if ( ! vshParseArgv ( ctl , argc , argv ) )
exit ( EXIT_FAILURE ) ;
if ( ! vshInit ( ctl ) )
exit ( EXIT_FAILURE ) ;
if ( ! ctl - > imode ) {
ret = vshCommandRun ( ctl , ctl - > cmd ) ;
2005-12-01 19:35:42 +03:00
} else {
2005-12-08 13:23:34 +03:00
/* interactive mode */
if ( ! ctl - > quiet ) {
vshPrint ( ctl , VSH_MESG , " Welcome to %s, the virtualization interactive terminal. \n \n " ,
progname ) ;
vshPrint ( ctl , VSH_MESG , " Type: 'help' for help with commands \n "
" 'quit' to quit \n \n " ) ;
}
2005-12-08 17:22:52 +03:00
vshReadlineInit ( ) ;
2005-12-08 13:23:34 +03:00
do {
ctl - > cmdstr = readline ( ctl - > uid = = 0 ? VSH_PROMPT_RW : VSH_PROMPT_RO ) ;
if ( ctl - > cmdstr = = NULL )
break ; /* EOF */
if ( * ctl - > cmdstr ) {
add_history ( ctl - > cmdstr ) ;
if ( vshCommandParse ( ctl , ctl - > cmdstr ) )
vshCommandRun ( ctl , ctl - > cmd ) ;
}
free ( ctl - > cmdstr ) ;
ctl - > cmdstr = NULL ;
} while ( ctl - > imode ) ;
if ( ctl - > cmdstr = = NULL )
fputc ( ' \n ' , stdout ) ; /* line break after alone prompt */
}
vshDeinit ( ctl ) ;
exit ( ret ? EXIT_SUCCESS : EXIT_FAILURE ) ;
2005-11-10 19:12:31 +03:00
}
2005-12-08 13:23:34 +03:00
/*
* vim : set tabstop = 4 :
* vim : set shiftwidth = 4 :
* vim : set expandtab :
*/