2007-04-26 16:27:49 +04:00
/*
ctdb control tool
Copyright ( C ) Andrew Tridgell 2007
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 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 , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include "includes.h"
# include "lib/events/events.h"
# include "system/filesys.h"
# include "popt.h"
# include "cmdline.h"
# include "../include/ctdb_private.h"
/*
show usage message
*/
static void usage ( void )
{
printf ( " Usage: ctdb_control [options] <control> \n " ) ;
2007-04-26 16:51:41 +04:00
printf ( " \n Controls: \n " ) ;
2007-04-26 21:27:07 +04:00
printf ( " ping \n " ) ;
2007-04-27 17:14:36 +04:00
printf ( " process-exists <vnn:pid> see if a process exists \n " ) ;
printf ( " status <vnn> show ctdb status on a node \n " ) ;
printf ( " debug <vnn> <level> set ctdb debug level on a node \n " ) ;
printf ( " debuglevel display ctdb debug levels \n " ) ;
printf ( " getvnnmap <vnn> display ctdb vnnmap \n " ) ;
2007-04-27 16:08:12 +04:00
printf ( " setvnnmap <vnn> <generation> <numslots> <lmaster>* \n " ) ;
2007-04-28 14:00:50 +04:00
printf ( " getdbmap <vnn> lists databases on a node \n " ) ;
2007-04-28 14:40:26 +04:00
printf ( " getnodemap <vnn> lists nodes known to a ctdb daemon \n " ) ;
2007-04-28 23:47:13 +04:00
printf ( " getkeys <vnn> <dbid> lists all keys in a remote tdb \n " ) ;
2007-04-26 16:27:49 +04:00
exit ( 1 ) ;
}
2007-04-27 17:14:36 +04:00
/*
see if a process exists
*/
2007-04-26 16:27:49 +04:00
static int control_process_exists ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
2007-04-26 16:51:41 +04:00
uint32_t vnn , pid ;
2007-04-26 16:27:49 +04:00
int ret ;
2007-04-26 16:51:41 +04:00
if ( argc < 1 ) {
2007-04-26 16:27:49 +04:00
usage ( ) ;
}
2007-04-26 16:51:41 +04:00
if ( sscanf ( argv [ 0 ] , " %u:%u " , & vnn , & pid ) ! = 2 ) {
printf ( " Badly formed vnn:pid \n " ) ;
return - 1 ;
}
2007-04-26 16:27:49 +04:00
2007-04-26 16:51:41 +04:00
ret = ctdb_process_exists ( ctdb , vnn , pid ) ;
2007-04-26 16:27:49 +04:00
if ( ret = = 0 ) {
2007-04-26 16:51:41 +04:00
printf ( " %u:%u exists \n " , vnn , pid ) ;
2007-04-26 16:27:49 +04:00
} else {
2007-04-26 16:51:41 +04:00
printf ( " %u:%u does not exist \n " , vnn , pid ) ;
2007-04-26 16:27:49 +04:00
}
return ret ;
}
2007-04-26 16:51:41 +04:00
/*
display status structure
*/
static void show_status ( struct ctdb_status * s )
{
printf ( " CTDB version %u \n " , CTDB_VERSION ) ;
printf ( " client_packets_sent %u \n " , s - > client_packets_sent ) ;
printf ( " client_packets_recv %u \n " , s - > client_packets_recv ) ;
printf ( " req_call %u \n " , s - > client . req_call ) ;
printf ( " req_message %u \n " , s - > client . req_message ) ;
printf ( " req_finished %u \n " , s - > client . req_finished ) ;
printf ( " req_register %u \n " , s - > client . req_register ) ;
printf ( " req_connect_wait %u \n " , s - > client . req_connect_wait ) ;
printf ( " req_shutdown %u \n " , s - > client . req_shutdown ) ;
printf ( " req_control %u \n " , s - > client . req_control ) ;
printf ( " node_packets_sent %u \n " , s - > node_packets_sent ) ;
printf ( " node_packets_recv %u \n " , s - > node_packets_recv ) ;
printf ( " req_call %u \n " , s - > count . req_call ) ;
printf ( " reply_call %u \n " , s - > count . reply_call ) ;
printf ( " reply_redirect %u \n " , s - > count . reply_redirect ) ;
printf ( " req_dmaster %u \n " , s - > count . req_dmaster ) ;
printf ( " reply_dmaster %u \n " , s - > count . reply_dmaster ) ;
printf ( " reply_error %u \n " , s - > count . reply_error ) ;
printf ( " reply_redirect %u \n " , s - > count . reply_redirect ) ;
printf ( " req_message %u \n " , s - > count . req_message ) ;
printf ( " req_finished %u \n " , s - > count . req_finished ) ;
printf ( " total_calls %u \n " , s - > total_calls ) ;
printf ( " pending_calls %u \n " , s - > pending_calls ) ;
printf ( " lockwait_calls %u \n " , s - > lockwait_calls ) ;
printf ( " pending_lockwait_calls %u \n " , s - > pending_lockwait_calls ) ;
printf ( " max_redirect_count %u \n " , s - > max_redirect_count ) ;
printf ( " max_call_latency %.6f sec \n " , s - > max_call_latency ) ;
printf ( " max_lockwait_latency %.6f sec \n " , s - > max_lockwait_latency ) ;
}
2007-04-27 17:14:36 +04:00
/*
display remote ctdb status
*/
2007-04-26 16:51:41 +04:00
static int control_status ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
uint32_t vnn ;
int ret ;
struct ctdb_status status ;
if ( argc < 1 ) {
usage ( ) ;
}
vnn = strtoul ( argv [ 0 ] , NULL , 0 ) ;
ret = ctdb_status ( ctdb , vnn , & status ) ;
if ( ret ! = 0 ) {
printf ( " Unable to get status from node %u \n " , vnn ) ;
return ret ;
}
show_status ( & status ) ;
return 0 ;
}
2007-04-27 17:14:36 +04:00
/*
display remote ctdb vnn map
*/
2007-04-27 14:56:10 +04:00
static int control_getvnnmap ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
uint32_t vnn ;
int i , ret ;
struct ctdb_vnn_map * vnnmap ;
if ( argc < 1 ) {
usage ( ) ;
}
vnn = strtoul ( argv [ 0 ] , NULL , 0 ) ;
vnnmap = talloc_zero ( ctdb , struct ctdb_vnn_map ) ;
ret = ctdb_getvnnmap ( ctdb , vnn , vnnmap ) ;
if ( ret ! = 0 ) {
printf ( " Unable to get vnnmap from node %u \n " , vnn ) ;
return ret ;
}
printf ( " Generation:%d \n " , vnnmap - > generation ) ;
printf ( " Size:%d \n " , vnnmap - > size ) ;
for ( i = 0 ; i < vnnmap - > size ; i + + ) {
printf ( " hash:%d lmaster:%d \n " , i , vnnmap - > map [ i ] ) ;
}
return 0 ;
}
2007-04-26 21:27:07 +04:00
2007-04-28 23:47:13 +04:00
/*
display remote list of keys for a tdb
*/
static int control_getkeys ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
uint32_t vnn , dbid ;
int i , j , ret ;
struct ctdb_key_list keys ;
TALLOC_CTX * mem_ctx ;
if ( argc < 2 ) {
usage ( ) ;
}
vnn = strtoul ( argv [ 0 ] , NULL , 0 ) ;
dbid = strtoul ( argv [ 1 ] , NULL , 0 ) ;
mem_ctx = talloc_new ( ctdb ) ;
ret = ctdb_getkeys ( ctdb , vnn , dbid , mem_ctx , & keys ) ;
if ( ret ! = 0 ) {
printf ( " Unable to get keys from node %u \n " , vnn ) ;
return ret ;
}
printf ( " Number of keys:%d \n " , keys . num ) ;
for ( i = 0 ; i < keys . num ; i + + ) {
printf ( " key: " ) ;
for ( j = 0 ; j < keys . keys [ i ] . dsize ; j + + ) {
printf ( " %02x " , keys . keys [ i ] . dptr [ j ] ) ;
}
printf ( " lmaster:%d rsn:%llu dmaster:%d laccessor:%d lacount:%d " , keys . lmasters [ i ] , keys . headers [ i ] . rsn , keys . headers [ i ] . dmaster , keys . headers [ i ] . laccessor , keys . headers [ i ] . lacount ) ;
printf ( " data: " ) ;
for ( j = 0 ; j < keys . data [ i ] . dsize ; j + + ) {
printf ( " %02x " , keys . data [ i ] . dptr [ j ] ) ;
}
printf ( " \n " ) ;
}
talloc_free ( mem_ctx ) ;
return 0 ;
}
2007-04-28 14:00:50 +04:00
/*
display a list of the databases on a remote ctdb
*/
static int control_getdbmap ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
uint32_t vnn ;
int i , ret ;
struct ctdb_dbid_map * dbmap ;
if ( argc < 1 ) {
usage ( ) ;
}
vnn = strtoul ( argv [ 0 ] , NULL , 0 ) ;
dbmap = talloc_zero ( ctdb , struct ctdb_dbid_map ) ;
ret = ctdb_getdbmap ( ctdb , vnn , dbmap ) ;
if ( ret ! = 0 ) {
printf ( " Unable to get dbids from node %u \n " , vnn ) ;
talloc_free ( dbmap ) ;
return ret ;
}
printf ( " Number of databases:%d \n " , dbmap - > num ) ;
for ( i = 0 ; i < dbmap - > num ; i + + ) {
const char * path ;
ctdb_getdbpath ( ctdb , dbmap - > dbids [ i ] , dbmap , & path ) ;
printf ( " dbid:0x%08x path:%s \n " , dbmap - > dbids [ i ] , path ) ;
}
talloc_free ( dbmap ) ;
return 0 ;
}
2007-04-28 14:40:26 +04:00
/*
display a list nodes known to a remote ctdb
*/
static int control_getnodemap ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
uint32_t vnn ;
int i , ret ;
struct ctdb_node_map * nodemap ;
if ( argc < 1 ) {
usage ( ) ;
}
vnn = strtoul ( argv [ 0 ] , NULL , 0 ) ;
nodemap = talloc_zero ( ctdb , struct ctdb_node_map ) ;
ret = ctdb_getnodemap ( ctdb , vnn , nodemap ) ;
if ( ret ! = 0 ) {
printf ( " Unable to get nodemap from node %u \n " , vnn ) ;
talloc_free ( nodemap ) ;
return ret ;
}
printf ( " Number of nodes:%d \n " , nodemap - > num ) ;
for ( i = 0 ; i < nodemap - > num ; i + + ) {
2007-04-28 17:11:23 +04:00
printf ( " vnn:%d %s \n " , nodemap - > nodes [ i ] . vnn ,
nodemap - > nodes [ i ] . vnn = = vnn ? " THIS NODE " :
nodemap - > nodes [ i ] . flags & NODE_FLAGS_CONNECTED ?
" CONNECTED " : " UNAVAILABLE " ) ;
2007-04-28 14:40:26 +04:00
}
talloc_free ( nodemap ) ;
return 0 ;
}
2007-04-27 17:14:36 +04:00
/*
set remote ctdb vnn map
*/
2007-04-27 16:08:12 +04:00
static int control_setvnnmap ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
uint32_t vnn ;
struct ctdb_vnn_map * vnnmap ;
int i , ret ;
if ( argc < 3 ) {
usage ( ) ;
}
vnn = strtoul ( argv [ 0 ] , NULL , 0 ) ;
vnnmap = talloc_zero ( ctdb , struct ctdb_vnn_map ) ;
vnnmap - > generation = strtoul ( argv [ 1 ] , NULL , 0 ) ;
vnnmap - > size = strtoul ( argv [ 2 ] , NULL , 0 ) ;
vnnmap - > map = talloc_array ( vnnmap , uint32_t , vnnmap - > size ) ;
for ( i = 0 ; i < vnnmap - > size ; i + + ) {
vnnmap - > map [ i ] = strtoul ( argv [ 3 + i ] , NULL , 0 ) ;
}
ret = ctdb_setvnnmap ( ctdb , vnn , vnnmap ) ;
if ( ret ! = 0 ) {
printf ( " Unable to set vnnmap for node %u \n " , vnn ) ;
return ret ;
}
return 0 ;
}
2007-04-27 17:14:36 +04:00
/*
ping all node
*/
2007-04-26 21:27:07 +04:00
static int control_ping ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
int ret , i ;
for ( i = 0 ; i < ctdb - > num_nodes ; i + + ) {
struct timeval tv = timeval_current ( ) ;
ret = ctdb_ping ( ctdb , i ) ;
if ( ret ! = 0 ) {
printf ( " Unable to get ping response from node %u \n " , i ) ;
} else {
printf ( " response from %u time=%.6f sec \n " ,
i , timeval_elapsed ( & tv ) ) ;
}
}
return 0 ;
}
2007-04-27 17:14:36 +04:00
/*
display debug level on all node
*/
static int control_debuglevel ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
int ret , i ;
for ( i = 0 ; i < ctdb - > num_nodes ; i + + ) {
uint32_t level ;
ret = ctdb_get_debuglevel ( ctdb , i , & level ) ;
if ( ret ! = 0 ) {
printf ( " Unable to get debuglevel response from node %u \n " , i ) ;
} else {
printf ( " Node %u is at debug level %u \n " , i , level ) ;
}
}
return 0 ;
}
/*
set debug level on a node
*/
static int control_debug ( struct ctdb_context * ctdb , int argc , const char * * argv )
{
int ret ;
uint32_t vnn , level ;
if ( argc < 2 ) {
usage ( ) ;
}
vnn = strtoul ( argv [ 0 ] , NULL , 0 ) ;
level = strtoul ( argv [ 1 ] , NULL , 0 ) ;
ret = ctdb_set_debuglevel ( ctdb , vnn , level ) ;
if ( ret ! = 0 ) {
printf ( " Unable to set debug level on node %u \n " , vnn ) ;
}
return 0 ;
}
2007-04-26 16:27:49 +04:00
/*
main program
*/
int main ( int argc , const char * argv [ ] )
{
struct ctdb_context * ctdb ;
struct poptOption popt_options [ ] = {
POPT_AUTOHELP
POPT_CTDB_CMDLINE
POPT_TABLEEND
} ;
int opt ;
const char * * extra_argv ;
int extra_argc = 0 ;
int ret ;
poptContext pc ;
struct event_context * ev ;
const char * control ;
pc = poptGetContext ( argv [ 0 ] , argc , argv , popt_options , POPT_CONTEXT_KEEP_FIRST ) ;
while ( ( opt = poptGetNextOpt ( pc ) ) ! = - 1 ) {
switch ( opt ) {
default :
fprintf ( stderr , " Invalid option %s: %s \n " ,
poptBadOption ( pc , 0 ) , poptStrerror ( opt ) ) ;
exit ( 1 ) ;
}
}
/* setup the remaining options for the main program to use */
extra_argv = poptGetArgs ( pc ) ;
if ( extra_argv ) {
extra_argv + + ;
while ( extra_argv [ extra_argc ] ) extra_argc + + ;
}
if ( extra_argc < 1 ) {
usage ( ) ;
}
control = extra_argv [ 0 ] ;
ev = event_context_init ( NULL ) ;
/* initialise ctdb */
ctdb = ctdb_cmdline_client ( ev ) ;
if ( ctdb = = NULL ) {
printf ( " Failed to init ctdb \n " ) ;
exit ( 1 ) ;
}
if ( strcmp ( control , " process-exists " ) = = 0 ) {
ret = control_process_exists ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-26 16:51:41 +04:00
} else if ( strcmp ( control , " status " ) = = 0 ) {
ret = control_status ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-27 14:56:10 +04:00
} else if ( strcmp ( control , " getvnnmap " ) = = 0 ) {
ret = control_getvnnmap ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-28 14:00:50 +04:00
} else if ( strcmp ( control , " getdbmap " ) = = 0 ) {
ret = control_getdbmap ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-28 14:40:26 +04:00
} else if ( strcmp ( control , " getnodemap " ) = = 0 ) {
ret = control_getnodemap ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-28 23:47:13 +04:00
} else if ( strcmp ( control , " getkeys " ) = = 0 ) {
ret = control_getkeys ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-27 16:08:12 +04:00
} else if ( strcmp ( control , " setvnnmap " ) = = 0 ) {
ret = control_setvnnmap ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-26 21:27:07 +04:00
} else if ( strcmp ( control , " ping " ) = = 0 ) {
ret = control_ping ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-27 17:14:36 +04:00
} else if ( strcmp ( control , " debug " ) = = 0 ) {
ret = control_debug ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
} else if ( strcmp ( control , " debuglevel " ) = = 0 ) {
ret = control_debuglevel ( ctdb , extra_argc - 1 , extra_argv + 1 ) ;
2007-04-26 16:27:49 +04:00
} else {
printf ( " Unknown control '%s' \n " , control ) ;
exit ( 1 ) ;
}
return ret ;
}