2004-03-04 10:57:30 +02:00
/* Copyright (C) 2000-2004 MySQL AB
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA */
/* maintaince of mysql databases */
2001-09-12 23:53:31 +03:00
# include "client_priv.h"
2000-07-31 21:29:14 +02:00
# include <signal.h>
# ifdef THREAD
# include <my_pthread.h> /* because of signal() */
# endif
2002-08-23 02:48:52 +03:00
# include <sys/stat.h>
2002-11-15 00:16:30 +05:00
# include <mysql.h>
2000-07-31 21:29:14 +02:00
2004-11-19 17:21:27 +00:00
# ifdef LATER_HAVE_NDBCLUSTER_DB
2004-11-15 12:40:32 +00:00
# include "../ndb/src/mgmclient/ndb_mgmclient.h"
# endif
2004-10-07 22:08:17 +03:00
# define ADMIN_VERSION "8.41"
2003-01-24 14:59:36 +02:00
# define MAX_MYSQL_VAR 256
2001-03-11 21:20:15 +02:00
# define SHUTDOWN_DEF_TIMEOUT 3600 /* Wait for shutdown */
2000-07-31 21:29:14 +02:00
# define MAX_TRUNC_LENGTH 3
2005-01-06 01:08:03 +01:00
char * host = NULL , * user = 0 , * opt_password = 0 ,
* default_charset = NULL ;
2000-07-31 21:29:14 +02:00
char truncated_var_names [ MAX_MYSQL_VAR ] [ MAX_TRUNC_LENGTH ] ;
char ex_var_names [ MAX_MYSQL_VAR ] [ FN_REFLEN ] ;
ulonglong last_values [ MAX_MYSQL_VAR ] ;
static int interval = 0 ;
2001-01-22 15:33:34 +02:00
static my_bool option_force = 0 , interrupted = 0 , new_line = 0 ,
2002-04-02 20:29:53 +03:00
opt_compress = 0 , opt_relative = 0 , opt_verbose = 0 , opt_vertical = 0 ,
tty_password = 0 ;
2002-08-05 18:37:43 +03:00
static uint tcp_port = 0 , option_wait = 0 , option_silent = 0 , nr_iterations ,
opt_count_iterations = 0 ;
2001-03-11 21:20:15 +02:00
static ulong opt_connect_timeout , opt_shutdown_timeout ;
2000-07-31 21:29:14 +02:00
static my_string unix_port = 0 ;
2004-11-19 17:21:27 +00:00
# ifdef LATER_HAVE_NDBCLUSTER_DB
2004-11-15 12:40:32 +00:00
static my_bool opt_ndbcluster = 0 ;
static char * opt_ndb_connectstring = 0 ;
# endif
2000-07-31 21:29:14 +02:00
2002-11-15 00:16:30 +05:00
# ifdef HAVE_SMEM
static char * shared_memory_base_name = 0 ;
# endif
static uint opt_protocol = 0 ;
2002-06-11 11:20:31 +03:00
/*
When using extended - status relatively , ex_val_max_len is the estimated
maximum length for any relative value printed by extended - status . The
idea is to try to keep the length of output as short as possible .
*/
2000-07-31 21:29:14 +02:00
static uint ex_val_max_len [ MAX_MYSQL_VAR ] ;
static my_bool ex_status_printed = 0 ; /* First output is not relative. */
static uint ex_var_count , max_var_length , max_val_length ;
2002-09-17 23:46:53 +03:00
# include <sslopt-vars.h>
2000-07-31 21:29:14 +02:00
static void print_version ( void ) ;
static void usage ( void ) ;
2002-04-02 20:29:53 +03:00
static my_bool sql_connect ( MYSQL * mysql , uint wait ) ;
2001-10-07 14:18:09 +03:00
static int execute_commands ( MYSQL * mysql , int argc , char * * argv ) ;
2000-07-31 21:29:14 +02:00
static int drop_db ( MYSQL * mysql , const char * db ) ;
static sig_handler endprog ( int signal_number ) ;
static void nice_time ( ulong sec , char * buff ) ;
static void print_header ( MYSQL_RES * result ) ;
static void print_top ( MYSQL_RES * result ) ;
static void print_row ( MYSQL_RES * result , MYSQL_ROW cur , uint row ) ;
static void print_relative_row ( MYSQL_RES * result , MYSQL_ROW cur , uint row ) ;
static void print_relative_row_vert ( MYSQL_RES * result , MYSQL_ROW cur , uint row ) ;
static void print_relative_header ( ) ;
static void print_relative_line ( ) ;
static void truncate_names ( ) ;
static my_bool get_pidfile ( MYSQL * mysql , char * pidfile ) ;
2003-01-07 16:53:10 +02:00
static my_bool wait_pidfile ( char * pidfile , time_t last_modified ,
struct stat * pidfile_status ) ;
2000-07-31 21:29:14 +02:00
static void store_values ( MYSQL_RES * result ) ;
/*
The order of commands must be the same as command_names ,
except ADMIN_ERROR
*/
2002-12-05 01:14:51 +03:00
enum commands {
ADMIN_ERROR ,
2000-08-11 18:41:53 -05:00
ADMIN_CREATE , ADMIN_DROP , ADMIN_SHUTDOWN ,
2002-12-05 01:14:51 +03:00
ADMIN_RELOAD , ADMIN_REFRESH , ADMIN_VER ,
ADMIN_PROCESSLIST , ADMIN_STATUS , ADMIN_KILL ,
ADMIN_DEBUG , ADMIN_VARIABLES , ADMIN_FLUSH_LOGS ,
ADMIN_FLUSH_HOSTS , ADMIN_FLUSH_TABLES , ADMIN_PASSWORD ,
ADMIN_PING , ADMIN_EXTENDED_STATUS , ADMIN_FLUSH_STATUS ,
ADMIN_FLUSH_PRIVILEGES , ADMIN_START_SLAVE , ADMIN_STOP_SLAVE ,
2002-11-24 17:07:53 +03:00
ADMIN_FLUSH_THREADS , ADMIN_OLD_PASSWORD
2004-11-19 17:21:27 +00:00
# ifdef LATER_HAVE_NDBCLUSTER_DB
2004-11-15 12:40:32 +00:00
, ADMIN_NDB_MGM
# endif
2000-07-31 21:29:14 +02:00
} ;
2000-08-11 18:41:53 -05:00
static const char * command_names [ ] = {
" create " , " drop " , " shutdown " ,
" reload " , " refresh " , " version " ,
" processlist " , " status " , " kill " ,
" debug " , " variables " , " flush-logs " ,
" flush-hosts " , " flush-tables " , " password " ,
" ping " , " extended-status " , " flush-status " ,
2002-12-05 01:14:51 +03:00
" flush-privileges " , " start-slave " , " stop-slave " ,
" flush-threads " , " old-password " ,
2004-11-19 17:21:27 +00:00
# ifdef LATER_HAVE_NDBCLUSTER_DB
2004-11-15 12:40:32 +00:00
" ndb-mgm " ,
# endif
2000-08-11 18:41:53 -05:00
NullS
} ;
2000-07-31 21:29:14 +02:00
static TYPELIB command_typelib =
2004-11-05 08:08:38 +04:00
{ array_elements ( command_names ) - 1 , " commands " , command_names , NULL } ;
2000-07-31 21:29:14 +02:00
2002-04-02 20:29:53 +03:00
static struct my_option my_long_options [ ] =
{
2004-11-29 18:32:52 +02:00
# ifdef __NETWARE__
2004-11-29 17:41:47 +02:00
{ " autoclose " , ' a ' , " Auto close the screen on exit for NetWare " ,
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2004-11-29 18:32:52 +02:00
# endif
2002-08-05 18:37:43 +03:00
{ " count " , ' c ' ,
2003-06-13 10:59:02 +02:00
" Number of iterations to make. This works with -i (--sleep) only. " ,
2002-08-05 18:37:43 +03:00
( gptr * ) & nr_iterations , ( gptr * ) & nr_iterations , 0 , GET_UINT ,
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " debug " , ' # ' , " Output debug log. Often this is 'd:t:o,filename'. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_STR , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " force " , ' f ' ,
" Don't ask for confirmation on drop database; with multiple commands, continue even if an error occurs. " ,
( gptr * ) & option_force , ( gptr * ) & option_force , 0 , GET_BOOL , NO_ARG , 0 , 0 ,
0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " compress " , ' C ' , " Use compression in server/client protocol. " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & opt_compress , ( gptr * ) & opt_compress , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 ,
0 , 0 , 0 } ,
{ " character-sets-dir " , OPT_CHARSETS_DIR ,
" Directory where character sets are. " , ( gptr * ) & charsets_dir ,
( gptr * ) & charsets_dir , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2005-01-06 01:08:03 +01:00
{ " default-character-set " , OPT_DEFAULT_CHARSET ,
" Set the default character set. " , ( gptr * ) & default_charset ,
( gptr * ) & default_charset , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " help " , ' ? ' , " Display this help and exit. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " host " , ' h ' , " Connect to host. " , ( gptr * ) & host , ( gptr * ) & host , 0 , GET_STR ,
2002-04-02 20:29:53 +03:00
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " password " , ' p ' ,
" Password to use when connecting to server. If password is not given it's asked from the tty. " ,
0 , 0 , 0 , GET_STR , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# ifdef __WIN__
2002-04-02 20:29:53 +03:00
{ " pipe " , ' W ' , " Use named pipes to connect to server. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# endif
2002-05-11 14:36:34 +03:00
{ " port " , ' P ' , " Port number to use for connection. " , ( gptr * ) & tcp_port ,
2004-02-26 19:09:49 +01:00
( gptr * ) & tcp_port , 0 , GET_UINT , REQUIRED_ARG , MYSQL_PORT , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " protocol " , OPT_MYSQL_PROTOCOL , " The protocol of connection (tcp,socket,pipe,memory). " ,
2002-11-15 00:16:30 +05:00
0 , 0 , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " relative " , ' r ' ,
" Show difference between current and previous values when used with -i. Currently works only with extended-status. " ,
( gptr * ) & opt_relative , ( gptr * ) & opt_relative , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 ,
0 , 0 , 0 } ,
{ " set-variable " , ' O ' ,
2002-06-01 11:46:06 +03:00
" Change the value of a variable. Please note that this option is deprecated; you can set variables directly with --variable-name=value. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-11-15 00:16:30 +05:00
# ifdef HAVE_SMEM
2004-08-30 11:11:10 -05:00
{ " shared-memory-base-name " , OPT_SHARED_MEMORY_BASE_NAME ,
2003-06-13 10:59:02 +02:00
" Base name of shared memory. " , ( gptr * ) & shared_memory_base_name , ( gptr * ) & shared_memory_base_name ,
2002-11-15 00:16:30 +05:00
0 , GET_STR_ALLOC , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2003-06-13 10:59:02 +02:00
{ " silent " , ' s ' , " Silently exit if one can't connect to server. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " socket " , ' S ' , " Socket file to use for connection. " ,
2002-05-14 21:41:55 +03:00
( gptr * ) & unix_port , ( gptr * ) & unix_port , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 ,
0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " sleep " , ' i ' , " Execute commands again and again with a sleep between. " ,
2002-05-11 14:36:34 +03:00
( gptr * ) & interval , ( gptr * ) & interval , 0 , GET_INT , REQUIRED_ARG , 0 , 0 , 0 , 0 ,
0 , 0 } ,
2002-09-17 23:46:53 +03:00
# include <sslopt-longopts.h>
2000-07-31 21:29:14 +02:00
# ifndef DONT_ALLOW_USER_CHANGE
2002-04-02 20:29:53 +03:00
{ " user " , ' u ' , " User for login if not current user. " , ( gptr * ) & user ,
2002-05-15 20:24:00 +03:00
( gptr * ) & user , 0 , GET_STR_ALLOC , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# endif
2002-04-02 20:29:53 +03:00
{ " verbose " , ' v ' , " Write more information. " , ( gptr * ) & opt_verbose ,
( gptr * ) & opt_verbose , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " version " , ' V ' , " Output version information and exit. " , 0 , 0 , 0 , GET_NO_ARG ,
2002-04-02 20:29:53 +03:00
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-12-05 01:14:51 +03:00
{ " vertical " , ' E ' ,
2002-04-02 20:29:53 +03:00
" Print output vertically. Is similar to --relative, but prints output vertically. " ,
( gptr * ) & opt_vertical , ( gptr * ) & opt_vertical , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 ,
0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " wait " , ' w ' , " Wait and retry if connection is down. " , 0 , 0 , 0 , GET_UINT ,
2002-10-15 23:34:55 +03:00
OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " connect_timeout " , OPT_CONNECT_TIMEOUT , " " , ( gptr * ) & opt_connect_timeout ,
2002-05-11 14:36:34 +03:00
( gptr * ) & opt_connect_timeout , 0 , GET_ULONG , REQUIRED_ARG , 3600 * 12 , 0 ,
2002-04-02 20:29:53 +03:00
3600 * 12 , 0 , 1 , 0 } ,
{ " shutdown_timeout " , OPT_SHUTDOWN_TIMEOUT , " " , ( gptr * ) & opt_shutdown_timeout ,
2002-05-11 14:36:34 +03:00
( gptr * ) & opt_shutdown_timeout , 0 , GET_ULONG , REQUIRED_ARG ,
2002-04-02 20:29:53 +03:00
SHUTDOWN_DEF_TIMEOUT , 0 , 3600 * 12 , 0 , 1 , 0 } ,
2004-11-19 17:21:27 +00:00
# ifdef LATER_HAVE_NDBCLUSTER_DB
2004-11-15 12:40:32 +00:00
{ " ndbcluster " , OPT_NDBCLUSTER , " "
" " , ( gptr * ) & opt_ndbcluster ,
( gptr * ) & opt_ndbcluster , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " ndb-connectstring " , OPT_NDB_CONNECTSTRING , " "
" " , ( gptr * ) & opt_ndb_connectstring ,
( gptr * ) & opt_ndb_connectstring , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2002-04-02 20:29:53 +03:00
{ 0 , 0 , 0 , 0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 }
2000-07-31 21:29:14 +02:00
} ;
2000-11-27 02:28:41 +02:00
2000-07-31 21:29:14 +02:00
static const char * load_default_groups [ ] = { " mysqladmin " , " client " , 0 } ;
2002-04-02 20:29:53 +03:00
static my_bool
get_one_option ( int optid , const struct my_option * opt __attribute__ ( ( unused ) ) ,
char * argument )
2000-07-31 21:29:14 +02:00
{
2002-04-02 20:29:53 +03:00
int error = 0 ;
switch ( optid ) {
2004-11-29 18:32:52 +02:00
# ifdef __NETWARE__
case ' a ' :
setscreenmode ( SCR_AUTOCLOSE_ON_EXIT ) ; // auto close the screen /
break ;
# endif
2002-08-05 18:37:43 +03:00
case ' c ' :
opt_count_iterations = 1 ;
break ;
2002-04-02 20:29:53 +03:00
case ' p ' :
if ( argument )
{
char * start = argument ;
my_free ( opt_password , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
opt_password = my_strdup ( argument , MYF ( MY_FAE ) ) ;
while ( * argument ) * argument + + = ' x ' ; /* Destroy argument */
if ( * start )
start [ 1 ] = 0 ; /* Cut length of argument */
2004-10-07 22:08:17 +03:00
tty_password = 0 ;
2002-04-02 20:29:53 +03:00
}
else
tty_password = 1 ;
break ;
case ' s ' :
option_silent + + ;
break ;
case ' W ' :
2000-07-31 21:29:14 +02:00
# ifdef __WIN__
2002-11-15 00:16:30 +05:00
opt_protocol = MYSQL_PROTOCOL_PIPE ;
2000-07-31 21:29:14 +02:00
# endif
2002-04-02 20:29:53 +03:00
break ;
case ' # ' :
DBUG_PUSH ( argument ? argument : " d:t:o,/tmp/mysqladmin.trace " ) ;
break ;
2002-09-17 23:46:53 +03:00
# include <sslopt-case.h>
2002-04-02 20:29:53 +03:00
case ' V ' :
print_version ( ) ;
exit ( 0 ) ;
break ;
case ' w ' :
if ( argument )
{
if ( ( option_wait = atoi ( argument ) ) < = 0 )
option_wait = 1 ;
}
else
2005-06-01 17:34:10 -07:00
option_wait = ~ ( uint ) 0 ;
2002-04-02 20:29:53 +03:00
break ;
case ' ? ' :
case ' I ' : /* Info */
error + + ;
break ;
case OPT_CHARSETS_DIR :
2000-07-31 21:29:14 +02:00
# if MYSQL_VERSION_ID > 32300
2002-04-02 20:29:53 +03:00
charsets_dir = argument ;
2000-07-31 21:29:14 +02:00
# endif
2002-04-02 20:29:53 +03:00
break ;
2002-11-15 00:16:30 +05:00
case OPT_MYSQL_PROTOCOL :
{
2004-08-24 20:13:31 +05:00
if ( ( opt_protocol = find_type ( argument , & sql_protocol_typelib , 0 ) ) < = 0 )
2002-11-15 00:16:30 +05:00
{
fprintf ( stderr , " Unknown option to protocol: %s \n " , argument ) ;
exit ( 1 ) ;
}
break ;
}
2000-07-31 21:29:14 +02:00
}
2002-04-02 20:29:53 +03:00
if ( error )
2000-07-31 21:29:14 +02:00
{
usage ( ) ;
exit ( 1 ) ;
}
2002-04-02 20:29:53 +03:00
return 0 ;
}
int main ( int argc , char * argv [ ] )
{
2004-09-08 16:45:03 +05:00
int error = 0 , ho_error ;
2002-04-02 20:29:53 +03:00
MYSQL mysql ;
2002-06-11 11:20:31 +03:00
char * * commands , * * save_argv ;
2002-04-02 20:29:53 +03:00
MY_INIT ( argv [ 0 ] ) ;
mysql_init ( & mysql ) ;
load_defaults ( " my " , load_default_groups , & argc , & argv ) ;
2002-06-11 11:20:31 +03:00
save_argv = argv ; /* Save for free_defaults */
2004-08-31 21:27:58 +05:00
if ( ( ho_error = handle_options ( & argc , & argv , my_long_options , get_one_option ) ) )
2002-08-24 02:08:10 +03:00
{
free_defaults ( save_argv ) ;
2002-05-29 15:07:30 +03:00
exit ( ho_error ) ;
2002-08-24 02:08:10 +03:00
}
2002-05-29 15:07:30 +03:00
2002-04-02 20:29:53 +03:00
if ( argc = = 0 )
{
usage ( ) ;
exit ( 1 ) ;
}
commands = argv ;
2000-07-31 21:29:14 +02:00
if ( tty_password )
2000-11-13 23:55:10 +02:00
opt_password = get_tty_password ( NullS ) ;
2000-07-31 21:29:14 +02:00
VOID ( signal ( SIGINT , endprog ) ) ; /* Here if abort */
VOID ( signal ( SIGTERM , endprog ) ) ; /* Here if abort */
if ( opt_compress )
mysql_options ( & mysql , MYSQL_OPT_COMPRESS , NullS ) ;
2000-11-27 02:28:41 +02:00
if ( opt_connect_timeout )
{
uint tmp = opt_connect_timeout ;
mysql_options ( & mysql , MYSQL_OPT_CONNECT_TIMEOUT , ( char * ) & tmp ) ;
}
2000-07-31 21:29:14 +02:00
# ifdef HAVE_OPENSSL
if ( opt_use_ssl )
mysql_ssl_set ( & mysql , opt_ssl_key , opt_ssl_cert , opt_ssl_ca ,
2001-09-30 10:46:20 +08:00
opt_ssl_capath , opt_ssl_cipher ) ;
2002-08-09 16:47:16 +03:00
# endif
2002-11-15 00:16:30 +05:00
if ( opt_protocol )
mysql_options ( & mysql , MYSQL_OPT_PROTOCOL , ( char * ) & opt_protocol ) ;
# ifdef HAVE_SMEM
if ( shared_memory_base_name )
mysql_options ( & mysql , MYSQL_SHARED_MEMORY_BASE_NAME , shared_memory_base_name ) ;
2002-12-05 01:14:51 +03:00
# endif
2005-01-06 01:08:03 +01:00
if ( default_charset )
mysql_options ( & mysql , MYSQL_SET_CHARSET_NAME , default_charset ) ;
2002-04-02 20:29:53 +03:00
if ( sql_connect ( & mysql , option_wait ) )
2004-09-08 16:45:03 +05:00
{
unsigned int err = mysql_errno ( & mysql ) ;
if ( err > = CR_MIN_ERROR & & err < = CR_MAX_ERROR )
error = 1 ;
else
{
/* Return 0 if all commands are PING */
for ( ; argc > 0 ; argv + + , argc - - )
{
if ( find_type ( argv [ 0 ] , & command_typelib , 2 ) ! = ADMIN_PING )
{
error = 1 ;
break ;
}
}
}
}
2000-07-31 21:29:14 +02:00
else
{
2002-08-05 18:37:43 +03:00
while ( ! interrupted & & ( ! opt_count_iterations | | nr_iterations ) )
2000-07-31 21:29:14 +02:00
{
new_line = 0 ;
2001-10-07 14:18:09 +03:00
if ( ( error = execute_commands ( & mysql , argc , commands ) ) )
2000-07-31 21:29:14 +02:00
{
2001-10-07 14:18:09 +03:00
if ( error > 0 )
break ; /* Wrong command error */
if ( ! option_force )
2000-07-31 21:29:14 +02:00
{
2001-10-07 14:18:09 +03:00
if ( option_wait & & ! interrupted )
{
mysql_close ( & mysql ) ;
2002-04-02 20:29:53 +03:00
if ( ! sql_connect ( & mysql , option_wait ) )
2001-10-07 14:18:09 +03:00
{
sleep ( 1 ) ; /* Don't retry too rapidly */
continue ; /* Retry */
}
}
error = 1 ;
break ;
2000-07-31 21:29:14 +02:00
}
}
if ( interval )
{
sleep ( interval ) ;
if ( new_line )
puts ( " " ) ;
2002-08-05 18:37:43 +03:00
if ( opt_count_iterations )
nr_iterations - - ;
2000-07-31 21:29:14 +02:00
}
else
break ;
}
mysql_close ( & mysql ) ;
}
2000-11-13 23:55:10 +02:00
my_free ( opt_password , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2000-07-31 21:29:14 +02:00
my_free ( user , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2002-11-15 00:16:30 +05:00
# ifdef HAVE_SMEM
my_free ( shared_memory_base_name , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
# endif
2002-04-04 21:36:24 -07:00
free_defaults ( save_argv ) ;
2000-07-31 21:29:14 +02:00
my_end ( 0 ) ;
2001-10-07 14:18:09 +03:00
exit ( error ? 1 : 0 ) ;
2000-07-31 21:29:14 +02:00
return 0 ;
}
static sig_handler endprog ( int signal_number __attribute__ ( ( unused ) ) )
{
interrupted = 1 ;
}
2002-04-02 20:29:53 +03:00
static my_bool sql_connect ( MYSQL * mysql , uint wait )
2000-07-31 21:29:14 +02:00
{
my_bool info = 0 ;
for ( ; ; )
{
2002-04-02 20:29:53 +03:00
if ( mysql_real_connect ( mysql , host , user , opt_password , NullS , tcp_port ,
unix_port , 0 ) )
2000-07-31 21:29:14 +02:00
{
2004-12-09 14:44:10 +01:00
mysql - > reconnect = 1 ;
2000-07-31 21:29:14 +02:00
if ( info )
{
fputs ( " \n " , stderr ) ;
( void ) fflush ( stderr ) ;
}
return 0 ;
}
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
if ( ! wait )
{
if ( ! option_silent )
{
if ( ! host )
2002-04-02 20:29:53 +03:00
host = ( char * ) LOCAL_HOST ;
2000-07-31 21:29:14 +02:00
my_printf_error ( 0 , " connect to server at '%s' failed \n error: '%s' " ,
MYF ( ME_BELL ) , host , mysql_error ( mysql ) ) ;
if ( mysql_errno ( mysql ) = = CR_CONNECTION_ERROR )
{
fprintf ( stderr ,
" Check that mysqld is running and that the socket: '%s' exists! \n " ,
unix_port ? unix_port : mysql_unix_port ) ;
}
else if ( mysql_errno ( mysql ) = = CR_CONN_HOST_ERROR | |
mysql_errno ( mysql ) = = CR_UNKNOWN_HOST )
{
fprintf ( stderr , " Check that mysqld is running on %s " , host ) ;
fprintf ( stderr , " and that the port is %d. \n " ,
tcp_port ? tcp_port : mysql_port ) ;
fprintf ( stderr , " You can check this by doing 'telnet %s %d' \n " ,
host , tcp_port ? tcp_port : mysql_port ) ;
}
}
return 1 ;
}
if ( wait ! = ( uint ) ~ 0 )
wait - - ; /* One less retry */
if ( ( mysql_errno ( mysql ) ! = CR_CONN_HOST_ERROR ) & &
( mysql_errno ( mysql ) ! = CR_CONNECTION_ERROR ) )
2002-12-05 01:14:51 +03:00
{
2000-07-31 21:29:14 +02:00
fprintf ( stderr , " Got error: %s \n " , mysql_error ( mysql ) ) ;
if ( ! option_force )
return 1 ;
}
2001-01-22 15:33:34 +02:00
else if ( ! option_silent )
2000-07-31 21:29:14 +02:00
{
2001-01-22 15:33:34 +02:00
if ( ! info )
{
info = 1 ;
fputs ( " Waiting for MySQL server to answer " , stderr ) ;
( void ) fflush ( stderr ) ;
}
else
{
2002-12-05 01:14:51 +03:00
putc ( ' . ' , stderr ) ;
2001-01-22 15:33:34 +02:00
( void ) fflush ( stderr ) ;
}
2000-07-31 21:29:14 +02:00
}
2001-01-22 15:33:34 +02:00
sleep ( 5 ) ;
2000-07-31 21:29:14 +02:00
}
}
2004-03-04 10:57:30 +02:00
2001-10-07 14:18:09 +03:00
/*
Execute a command .
Return 0 on ok
- 1 on retryable error
1 on fatal error
*/
2000-07-31 21:29:14 +02:00
2001-10-07 14:18:09 +03:00
static int execute_commands ( MYSQL * mysql , int argc , char * * argv )
2000-07-31 21:29:14 +02:00
{
2002-04-02 17:54:57 +03:00
const char * status ;
2004-03-04 10:57:30 +02:00
/*
MySQL documentation relies on the fact that mysqladmin will
execute commands in the order specified , e . g .
mysqladmin - u root flush - privileges password " newpassword "
to reset a lost root password .
If this behaviour is ever changed , Docs should be notified .
*/
2004-02-25 09:46:43 +10:00
2002-11-24 17:07:53 +03:00
struct rand_struct rand_st ;
2002-12-05 01:14:51 +03:00
2000-07-31 21:29:14 +02:00
for ( ; argc > 0 ; argv + + , argc - - )
{
switch ( find_type ( argv [ 0 ] , & command_typelib , 2 ) ) {
case ADMIN_CREATE :
{
char buff [ FN_REFLEN + 20 ] ;
if ( argc < 2 )
{
my_printf_error ( 0 , " Too few arguments to create " , MYF ( ME_BELL ) ) ;
return 1 ;
}
2001-03-03 03:03:12 +02:00
sprintf ( buff , " create database `%.*s` " , FN_REFLEN , argv [ 1 ] ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_query ( mysql , buff ) )
2001-01-31 04:47:25 +02:00
{
2001-03-03 03:03:12 +02:00
my_printf_error ( 0 , " CREATE DATABASE failed; error: '%-.200s' " ,
MYF ( ME_BELL ) , mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2001-01-31 04:47:25 +02:00
}
2001-03-03 03:03:12 +02:00
argc - - ; argv + + ;
2000-07-31 21:29:14 +02:00
break ;
}
case ADMIN_DROP :
{
if ( argc < 2 )
{
my_printf_error ( 0 , " Too few arguments to drop " , MYF ( ME_BELL ) ) ;
return 1 ;
}
2001-04-25 22:44:27 +03:00
if ( drop_db ( mysql , argv [ 1 ] ) )
2001-10-07 14:18:09 +03:00
return - 1 ;
2001-03-03 03:03:12 +02:00
argc - - ; argv + + ;
2000-07-31 21:29:14 +02:00
break ;
}
case ADMIN_SHUTDOWN :
{
char pidfile [ FN_REFLEN ] ;
2002-08-22 02:22:10 +03:00
my_bool got_pidfile = 0 ;
2002-08-22 17:06:50 +03:00
time_t last_modified = 0 ;
2002-08-22 02:22:10 +03:00
struct stat pidfile_status ;
2002-06-11 11:20:31 +03:00
/*
Only wait for pidfile on local connections
If pidfile doesn ' t exist , continue without pid file checking
*/
2002-08-22 02:37:42 +03:00
if ( mysql - > unix_socket & & ( got_pidfile = ! get_pidfile ( mysql , pidfile ) ) & &
! stat ( pidfile , & pidfile_status ) )
2002-08-22 02:22:10 +03:00
last_modified = pidfile_status . st_mtime ;
2004-06-15 11:35:23 +02:00
if ( mysql_shutdown ( mysql , SHUTDOWN_DEFAULT ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " shutdown failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
mysql_close ( mysql ) ; /* Close connection to avoid error messages */
if ( got_pidfile )
{
if ( opt_verbose )
printf ( " Shutdown signal sent to server; Waiting for pid file to disappear \n " ) ;
2002-08-22 02:22:10 +03:00
/* Wait until pid file is gone */
2003-01-07 16:53:10 +02:00
if ( wait_pidfile ( pidfile , last_modified , & pidfile_status ) )
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
}
case ADMIN_FLUSH_PRIVILEGES :
case ADMIN_RELOAD :
2003-05-15 23:00:58 +02:00
if ( mysql_query ( mysql , " flush privileges " ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " reload failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
case ADMIN_REFRESH :
if ( mysql_refresh ( mysql ,
( uint ) ~ ( REFRESH_GRANT | REFRESH_STATUS |
2000-10-14 03:16:35 +03:00
REFRESH_READ_LOCK | REFRESH_SLAVE |
2003-05-15 23:00:58 +02:00
REFRESH_MASTER ) ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " refresh failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
case ADMIN_FLUSH_THREADS :
2003-05-15 23:00:58 +02:00
if ( mysql_refresh ( mysql , ( uint ) REFRESH_THREADS ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " refresh failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
case ADMIN_VER :
new_line = 1 ;
print_version ( ) ;
2000-08-11 18:41:53 -05:00
puts ( " Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB " ) ;
puts ( " This software comes with ABSOLUTELY NO WARRANTY. This is free software, \n and you are welcome to modify and redistribute it under the GPL license \n " ) ;
2000-07-31 21:29:14 +02:00
printf ( " Server version \t \t %s \n " , mysql_get_server_info ( mysql ) ) ;
printf ( " Protocol version \t %d \n " , mysql_get_proto_info ( mysql ) ) ;
printf ( " Connection \t \t %s \n " , mysql_get_host_info ( mysql ) ) ;
if ( mysql - > unix_socket )
printf ( " UNIX socket \t \t %s \n " , mysql - > unix_socket ) ;
else
printf ( " TCP port \t \t %d \n " , mysql - > port ) ;
status = mysql_stat ( mysql ) ;
{
char * pos , buff [ 40 ] ;
ulong sec ;
2004-11-24 15:24:23 +01:00
pos = ( char * ) strchr ( status , ' ' ) ;
2000-07-31 21:29:14 +02:00
* pos + + = 0 ;
printf ( " %s \t \t \t " , status ) ; /* print label */
if ( ( status = str2int ( pos , 10 , 0 , LONG_MAX , ( long * ) & sec ) ) )
{
nice_time ( sec , buff ) ;
puts ( buff ) ; /* print nice time */
while ( * status = = ' ' ) status + + ; /* to next info */
}
}
putc ( ' \n ' , stdout ) ;
if ( status )
puts ( status ) ;
break ;
case ADMIN_PROCESSLIST :
{
MYSQL_RES * result ;
MYSQL_ROW row ;
if ( mysql_query ( mysql , ( opt_verbose ? " show full processlist " :
" show processlist " ) ) | |
! ( result = mysql_store_result ( mysql ) ) )
{
my_printf_error ( 0 , " process list failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
print_header ( result ) ;
while ( ( row = mysql_fetch_row ( result ) ) )
print_row ( result , row , 0 ) ;
print_top ( result ) ;
mysql_free_result ( result ) ;
new_line = 1 ;
break ;
}
case ADMIN_STATUS :
status = mysql_stat ( mysql ) ;
if ( status )
puts ( status ) ;
break ;
case ADMIN_KILL :
{
uint error = 0 ;
char * pos ;
if ( argc < 2 )
{
my_printf_error ( 0 , " Too few arguments to 'kill' " , MYF ( ME_BELL ) ) ;
return 1 ;
}
pos = argv [ 1 ] ;
for ( ; ; )
{
if ( mysql_kill ( mysql , ( ulong ) atol ( pos ) ) )
{
my_printf_error ( 0 , " kill failed on %ld; error: '%s' " , MYF ( ME_BELL ) ,
atol ( pos ) , mysql_error ( mysql ) ) ;
error = 1 ;
}
if ( ! ( pos = strchr ( pos , ' , ' ) ) )
break ;
pos + + ;
}
argc - - ; argv + + ;
if ( error )
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
break ;
}
case ADMIN_DEBUG :
if ( mysql_dump_debug_info ( mysql ) )
{
my_printf_error ( 0 , " debug failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
case ADMIN_VARIABLES :
{
MYSQL_RES * res ;
MYSQL_ROW row ;
new_line = 1 ;
2002-07-23 18:31:22 +03:00
if ( mysql_query ( mysql , " show /*!40003 GLOBAL */ variables " ) | |
2000-07-31 21:29:14 +02:00
! ( res = mysql_store_result ( mysql ) ) )
{
my_printf_error ( 0 , " unable to show variables; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
print_header ( res ) ;
while ( ( row = mysql_fetch_row ( res ) ) )
print_row ( res , row , 0 ) ;
print_top ( res ) ;
mysql_free_result ( res ) ;
break ;
}
case ADMIN_EXTENDED_STATUS :
{
MYSQL_RES * res ;
MYSQL_ROW row ;
uint rownr = 0 ;
2002-12-05 01:14:51 +03:00
void ( * func ) ( MYSQL_RES * , MYSQL_ROW , uint ) ;
2000-07-31 21:29:14 +02:00
new_line = 1 ;
2005-04-27 17:16:08 +03:00
if ( mysql_query ( mysql , " show /*!50002 GLOBAL */ status " ) | |
2000-07-31 21:29:14 +02:00
! ( res = mysql_store_result ( mysql ) ) )
{
my_printf_error ( 0 , " unable to show status; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
if ( ! opt_vertical )
print_header ( res ) ;
else
{
if ( ! ex_status_printed )
{
store_values ( res ) ;
truncate_names ( ) ; /* Does some printing also */
}
else
{
print_relative_line ( ) ;
print_relative_header ( ) ;
print_relative_line ( ) ;
}
}
/* void (*func) (MYSQL_RES*, MYSQL_ROW, uint); */
if ( opt_relative & & ! opt_vertical )
func = print_relative_row ;
else if ( opt_vertical )
func = print_relative_row_vert ;
else
func = print_row ;
while ( ( row = mysql_fetch_row ( res ) ) )
( * func ) ( res , row , rownr + + ) ;
if ( opt_vertical )
{
if ( ex_status_printed )
{
putchar ( ' \n ' ) ;
print_relative_line ( ) ;
}
}
else
print_top ( res ) ;
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
ex_status_printed = 1 ; /* From now on the output will be relative */
mysql_free_result ( res ) ;
break ;
}
case ADMIN_FLUSH_LOGS :
{
if ( mysql_refresh ( mysql , REFRESH_LOG ) )
{
my_printf_error ( 0 , " refresh failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
}
case ADMIN_FLUSH_HOSTS :
{
2003-05-15 23:00:58 +02:00
if ( mysql_query ( mysql , " flush hosts " ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " refresh failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
}
case ADMIN_FLUSH_TABLES :
{
2003-05-15 23:00:58 +02:00
if ( mysql_query ( mysql , " flush tables " ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " refresh failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
}
case ADMIN_FLUSH_STATUS :
{
2003-05-15 23:00:58 +02:00
if ( mysql_query ( mysql , " flush status " ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " refresh failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
break ;
}
2002-11-24 17:07:53 +03:00
case ADMIN_OLD_PASSWORD :
2000-07-31 21:29:14 +02:00
case ADMIN_PASSWORD :
{
2002-11-24 17:07:53 +03:00
char buff [ 128 ] , crypted_pw [ 64 ] ;
time_t start_time ;
/* Do initialization the same way as we do in mysqld */
2002-12-05 01:14:51 +03:00
start_time = time ( ( time_t * ) 0 ) ;
2002-11-24 17:07:53 +03:00
randominit ( & rand_st , ( ulong ) start_time , ( ulong ) start_time / 2 ) ;
2002-12-05 01:14:51 +03:00
2001-10-07 14:18:09 +03:00
if ( argc < 2 )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " Too few arguments to change password " , MYF ( ME_BELL ) ) ;
return 1 ;
}
if ( argv [ 1 ] [ 0 ] )
2003-07-01 23:40:59 +04:00
{
2004-12-02 14:28:59 +03:00
char * pw = argv [ 1 ] ;
2005-01-11 18:55:53 +02:00
bool old = ( find_type ( argv [ 0 ] , & command_typelib , 2 ) = =
ADMIN_OLD_PASSWORD ) ;
2004-12-02 14:28:59 +03:00
# ifdef __WIN__
uint pw_len = strlen ( pw ) ;
if ( pw_len > 1 & & pw [ 0 ] = = ' \' ' & & pw [ pw_len - 1 ] = = ' \' ' )
printf ( " Warning: single quotes were not trimmed from the password by "
" your command \n line client, as you might have expected. \n " ) ;
# endif
2005-01-05 21:56:18 +01:00
/*
If we don ' t already know to use an old - style password , see what
the server is using
*/
2005-01-11 18:55:53 +02:00
if ( ! old )
{
if ( mysql_query ( mysql , " SHOW VARIABLES LIKE 'old_passwords' " ) )
{
2005-01-05 21:56:18 +01:00
my_printf_error ( 0 , " Could not determine old_passwords setting from server; error: '%s' " ,
MYF ( ME_BELL ) , mysql_error ( mysql ) ) ;
return - 1 ;
2005-01-11 18:55:53 +02:00
}
else
{
2005-01-05 21:56:18 +01:00
MYSQL_RES * res = mysql_store_result ( mysql ) ;
2005-01-11 18:55:53 +02:00
if ( ! res )
{
my_printf_error ( 0 ,
" Could not get old_passwords setting from "
" server; error: '%s' " ,
2005-01-05 21:56:18 +01:00
MYF ( ME_BELL ) , mysql_error ( mysql ) ) ;
return - 1 ;
}
2005-01-11 18:55:53 +02:00
if ( ! mysql_num_rows ( res ) )
2005-01-05 21:56:18 +01:00
old = 1 ;
2005-01-11 18:55:53 +02:00
else
{
2005-01-05 21:56:18 +01:00
MYSQL_ROW row = mysql_fetch_row ( res ) ;
old = ! strncmp ( row [ 1 ] , " ON " , 2 ) ;
}
mysql_free_result ( res ) ;
}
}
if ( old )
2004-12-02 14:28:59 +03:00
make_scrambled_password_323 ( crypted_pw , pw ) ;
2003-07-01 23:40:59 +04:00
else
2004-12-02 14:28:59 +03:00
make_scrambled_password ( crypted_pw , pw ) ;
2003-07-01 23:40:59 +04:00
}
2000-07-31 21:29:14 +02:00
else
crypted_pw [ 0 ] = 0 ; /* No password */
sprintf ( buff , " set password='%s',sql_log_off=0 " , crypted_pw ) ;
if ( mysql_query ( mysql , " set sql_log_off=1 " ) )
{
my_printf_error ( 0 , " Can't turn off logging; error: '%s' " ,
MYF ( ME_BELL ) , mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
if ( mysql_query ( mysql , buff ) )
{
2004-03-16 19:30:22 +04:00
if ( mysql_errno ( mysql ) ! = 1290 )
{
my_printf_error ( 0 , " unable to change password; error: '%s' " ,
MYF ( ME_BELL ) , mysql_error ( mysql ) ) ;
return - 1 ;
}
else
{
2004-03-16 22:54:25 +04:00
/*
We don ' t try to execute ' update mysql . user set . . '
because we can ' t perfectly find out the host
*/
my_printf_error ( 0 , " \n "
" You cannot use 'password' command as mysqld runs \n "
" with grant tables disabled (was started with "
" --skip-grant-tables). \n "
" Use: \" mysqladmin flush-privileges password '*' \" "
" instead " , MYF ( ME_BELL ) ) ;
return - 1 ;
2004-03-16 19:30:22 +04:00
}
2000-07-31 21:29:14 +02:00
}
argc - - ; argv + + ;
break ;
}
case ADMIN_START_SLAVE :
2003-05-15 23:00:58 +02:00
if ( mysql_query ( mysql , " START SLAVE " ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " Error starting slave: %s " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
else
puts ( " Slave started " ) ;
break ;
case ADMIN_STOP_SLAVE :
2003-05-15 23:00:58 +02:00
if ( mysql_query ( mysql , " STOP SLAVE " ) )
2000-07-31 21:29:14 +02:00
{
my_printf_error ( 0 , " Error stopping slave: %s " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
else
puts ( " Slave stopped " ) ;
break ;
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
case ADMIN_PING :
mysql - > reconnect = 0 ; /* We want to know of reconnects */
if ( ! mysql_ping ( mysql ) )
2001-01-22 15:33:34 +02:00
{
if ( option_silent < 2 )
puts ( " mysqld is alive " ) ;
}
2000-07-31 21:29:14 +02:00
else
{
if ( mysql_errno ( mysql ) = = CR_SERVER_GONE_ERROR )
{
mysql - > reconnect = 1 ;
if ( ! mysql_ping ( mysql ) )
puts ( " connection was down, but mysqld is now alive " ) ;
}
else
{
my_printf_error ( 0 , " mysqld doesn't answer to ping, error: '%s' " ,
MYF ( ME_BELL ) , mysql_error ( mysql ) ) ;
2001-10-07 14:18:09 +03:00
return - 1 ;
2000-07-31 21:29:14 +02:00
}
}
mysql - > reconnect = 1 ; /* Automatic reconnect is default */
break ;
2004-11-19 17:21:27 +00:00
# ifdef LATER_HAVE_NDBCLUSTER_DB
2004-11-15 12:40:32 +00:00
case ADMIN_NDB_MGM :
{
if ( argc < 2 )
{
my_printf_error ( 0 , " Too few arguments to ndb-mgm " , MYF ( ME_BELL ) ) ;
return 1 ;
}
{
Ndb_mgmclient_handle cmd =
ndb_mgmclient_handle_create ( opt_ndb_connectstring ) ;
ndb_mgmclient_execute ( cmd , - - argc , + + argv ) ;
ndb_mgmclient_handle_destroy ( cmd ) ;
}
argc = 0 ;
}
break ;
# endif
2000-07-31 21:29:14 +02:00
default :
2001-01-31 04:47:25 +02:00
my_printf_error ( 0 , " Unknown command: '%-.60s' " , MYF ( ME_BELL ) , argv [ 0 ] ) ;
2000-07-31 21:29:14 +02:00
return 1 ;
}
}
return 0 ;
}
2004-05-25 22:00:14 +03:00
# include <help_start.h>
2000-07-31 21:29:14 +02:00
static void print_version ( void )
{
printf ( " %s Ver %s Distrib %s, for %s on %s \n " , my_progname , ADMIN_VERSION ,
MYSQL_SERVER_VERSION , SYSTEM_TYPE , MACHINE_TYPE ) ;
2004-05-25 22:00:14 +03:00
NETWARE_SET_SCREEN_MODE ( 1 ) ;
2000-07-31 21:29:14 +02:00
}
static void usage ( void )
{
print_version ( ) ;
puts ( " Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB " ) ;
puts ( " This software comes with ABSOLUTELY NO WARRANTY. This is free software, \n and you are welcome to modify and redistribute it under the GPL license \n " ) ;
2000-08-11 18:41:53 -05:00
puts ( " Administration program for the mysqld daemon. " ) ;
2000-07-31 21:29:14 +02:00
printf ( " Usage: %s [OPTIONS] command command.... \n " , my_progname ) ;
2002-04-02 20:29:53 +03:00
my_print_help ( my_long_options ) ;
my_print_variables ( my_long_options ) ;
2000-07-31 21:29:14 +02:00
print_defaults ( " my " , load_default_groups ) ;
puts ( " \n Where command is a one or more of: (Commands may be shortened) \n \
create databasename Create a new database \ n \
2005-01-31 19:35:12 -08:00
debug Instruct server to write debug information to log \ n \
2000-07-31 21:29:14 +02:00
drop databasename Delete a database and all its tables \ n \
extended - status Gives an extended status message from the server \ n \
flush - hosts Flush all cached hosts \ n \
flush - logs Flush all logs \ n \
flush - status Clear status variables \ n \
flush - tables Flush all tables \ n \
flush - threads Flush the thread cache \ n \
flush - privileges Reload grant tables ( same as reload ) \ n \
kill id , id , . . . Kill mysql threads " );
# if MYSQL_VERSION_ID >= 32200
puts ( " \
2002-11-24 17:07:53 +03:00
password new - password Change old password to new - password , MySQL 4.1 hashing . \ n \
old - password new - password Change old password to new - password in old format . \ n " );
2000-07-31 21:29:14 +02:00
# endif
puts ( " \
ping Check if mysqld is alive \ n \
processlist Show list of active threads in server \ n \
reload Reload grant tables \ n \
refresh Flush all tables and close and open logfiles \ n \
shutdown Take server down \ n \
status Gives a short status message from the server \ n \
2001-02-15 03:43:14 +02:00
start - slave Start slave \ n \
stop - slave Stop slave \ n \
2000-07-31 21:29:14 +02:00
variables Prints variables available \ n \
version Get version info from server " );
}
2004-05-25 22:00:14 +03:00
# include <help_end.h>
2000-07-31 21:29:14 +02:00
static int drop_db ( MYSQL * mysql , const char * db )
{
char name_buff [ FN_REFLEN + 20 ] , buf [ 10 ] ;
if ( ! option_force )
{
puts ( " Dropping the database is potentially a very bad thing to do. " ) ;
puts ( " Any data stored in the database will be destroyed. \n " ) ;
2001-04-25 22:44:27 +03:00
printf ( " Do you really want to drop the '%s' database [y/N] " , db ) ;
fflush ( stdout ) ;
2000-07-31 21:29:14 +02:00
VOID ( fgets ( buf , sizeof ( buf ) - 1 , stdin ) ) ;
if ( ( * buf ! = ' y ' ) & & ( * buf ! = ' Y ' ) )
{
puts ( " \n OK, aborting database drop! " ) ;
return - 1 ;
}
}
2001-05-29 13:46:17 +03:00
sprintf ( name_buff , " drop database `%.*s` " , FN_REFLEN , db ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_query ( mysql , name_buff ) )
{
2001-04-25 22:44:27 +03:00
my_printf_error ( 0 , " DROP DATABASE %s failed; \n error: '%s' " , MYF ( ME_BELL ) ,
2000-07-31 21:29:14 +02:00
db , mysql_error ( mysql ) ) ;
return 1 ;
}
printf ( " Database \" %s \" dropped \n " , db ) ;
return 0 ;
}
static void nice_time ( ulong sec , char * buff )
{
ulong tmp ;
if ( sec > = 3600L * 24 )
{
tmp = sec / ( 3600L * 24 ) ;
sec - = 3600L * 24 * tmp ;
2004-05-27 17:54:40 +04:00
buff = int10_to_str ( tmp , buff , 10 ) ;
2000-07-31 21:29:14 +02:00
buff = strmov ( buff , tmp > 1 ? " days " : " day " ) ;
}
if ( sec > = 3600L )
{
tmp = sec / 3600L ;
sec - = 3600L * tmp ;
2004-05-27 17:54:40 +04:00
buff = int10_to_str ( tmp , buff , 10 ) ;
2000-07-31 21:29:14 +02:00
buff = strmov ( buff , tmp > 1 ? " hours " : " hour " ) ;
}
if ( sec > = 60 )
{
tmp = sec / 60 ;
sec - = 60 * tmp ;
2004-05-27 17:54:40 +04:00
buff = int10_to_str ( tmp , buff , 10 ) ;
2000-07-31 21:29:14 +02:00
buff = strmov ( buff , " min " ) ;
}
2004-05-27 17:54:40 +04:00
strmov ( int10_to_str ( sec , buff , 10 ) , " sec " ) ;
2000-07-31 21:29:14 +02:00
}
static void print_header ( MYSQL_RES * result )
{
MYSQL_FIELD * field ;
print_top ( result ) ;
mysql_field_seek ( result , 0 ) ;
putchar ( ' | ' ) ;
while ( ( field = mysql_fetch_field ( result ) ) )
{
2001-10-09 03:35:29 +03:00
printf ( " %-*s| " , ( int ) field - > max_length + 1 , field - > name ) ;
2000-07-31 21:29:14 +02:00
}
putchar ( ' \n ' ) ;
print_top ( result ) ;
}
static void print_top ( MYSQL_RES * result )
{
uint i , length ;
MYSQL_FIELD * field ;
putchar ( ' + ' ) ;
mysql_field_seek ( result , 0 ) ;
while ( ( field = mysql_fetch_field ( result ) ) )
{
2000-08-22 00:18:32 +03:00
if ( ( length = ( uint ) strlen ( field - > name ) ) > field - > max_length )
2000-07-31 21:29:14 +02:00
field - > max_length = length ;
else
length = field - > max_length ;
for ( i = length + 2 ; i - - > 0 ; )
putchar ( ' - ' ) ;
putchar ( ' + ' ) ;
}
putchar ( ' \n ' ) ;
}
/* 3.rd argument, uint row, is not in use. Don't remove! */
static void print_row ( MYSQL_RES * result , MYSQL_ROW cur ,
uint row __attribute__ ( ( unused ) ) )
{
uint i , length ;
MYSQL_FIELD * field ;
putchar ( ' | ' ) ;
mysql_field_seek ( result , 0 ) ;
for ( i = 0 ; i < mysql_num_fields ( result ) ; i + + )
{
field = mysql_fetch_field ( result ) ;
length = field - > max_length ;
printf ( " %-*s| " , length + 1 , cur [ i ] ? ( char * ) cur [ i ] : " " ) ;
}
putchar ( ' \n ' ) ;
}
static void print_relative_row ( MYSQL_RES * result , MYSQL_ROW cur , uint row )
{
ulonglong tmp ;
char buff [ 22 ] ;
MYSQL_FIELD * field ;
mysql_field_seek ( result , 0 ) ;
field = mysql_fetch_field ( result ) ;
2001-10-09 03:35:29 +03:00
printf ( " | %-*s| " , ( int ) field - > max_length + 1 , cur [ 0 ] ) ;
2000-07-31 21:29:14 +02:00
field = mysql_fetch_field ( result ) ;
2003-01-24 14:59:36 +02:00
tmp = cur [ 1 ] ? strtoull ( cur [ 1 ] , NULL , 10 ) : ( ulonglong ) 0 ;
2001-10-09 03:35:29 +03:00
printf ( " %-*s| \n " , ( int ) field - > max_length + 1 ,
2000-07-31 21:29:14 +02:00
llstr ( ( tmp - last_values [ row ] ) , buff ) ) ;
last_values [ row ] = tmp ;
}
static void print_relative_row_vert ( MYSQL_RES * result __attribute__ ( ( unused ) ) ,
MYSQL_ROW cur ,
uint row __attribute__ ( ( unused ) ) )
{
uint length ;
ulonglong tmp ;
char buff [ 22 ] ;
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
if ( ! row )
putchar ( ' | ' ) ;
2003-01-25 15:31:07 +02:00
tmp = cur [ 1 ] ? strtoull ( cur [ 1 ] , NULL , 10 ) : ( ulonglong ) 0 ;
2002-12-05 01:14:51 +03:00
printf ( " %-*s| " , ex_val_max_len [ row ] + 1 ,
2000-07-31 21:29:14 +02:00
llstr ( ( tmp - last_values [ row ] ) , buff ) ) ;
/* Find the minimum row length needed to output the relative value */
2000-08-22 00:18:32 +03:00
if ( ( length = ( uint ) strlen ( buff ) > ex_val_max_len [ row ] ) & & ex_status_printed )
2000-07-31 21:29:14 +02:00
ex_val_max_len [ row ] = length ;
last_values [ row ] = tmp ;
}
static void store_values ( MYSQL_RES * result )
{
uint i ;
MYSQL_ROW row ;
MYSQL_FIELD * field ;
field = mysql_fetch_field ( result ) ;
max_var_length = field - > max_length ;
field = mysql_fetch_field ( result ) ;
max_val_length = field - > max_length ;
for ( i = 0 ; ( row = mysql_fetch_row ( result ) ) ; i + + )
{
strmov ( ex_var_names [ i ] , row [ 0 ] ) ;
last_values [ i ] = strtoull ( row [ 1 ] , NULL , 10 ) ;
ex_val_max_len [ i ] = 2 ; /* Default print width for values */
}
ex_var_count = i ;
return ;
}
static void print_relative_header ( )
{
uint i ;
putchar ( ' | ' ) ;
for ( i = 0 ; i < ex_var_count ; i + + )
printf ( " %-*s| " , ex_val_max_len [ i ] + 1 , truncated_var_names [ i ] ) ;
putchar ( ' \n ' ) ;
}
static void print_relative_line ( )
{
uint i ;
putchar ( ' + ' ) ;
for ( i = 0 ; i < ex_var_count ; i + + )
{
uint j ;
for ( j = 0 ; j < ex_val_max_len [ i ] + 2 ; j + + )
putchar ( ' - ' ) ;
putchar ( ' + ' ) ;
}
putchar ( ' \n ' ) ;
}
static void truncate_names ( )
{
uint i ;
char * ptr , top_line [ MAX_TRUNC_LENGTH + 4 + NAME_LEN + 22 + 1 ] , buff [ 22 ] ;
ptr = top_line ;
* ptr + + = ' + ' ;
ptr = strfill ( ptr , max_var_length + 2 , ' - ' ) ;
* ptr + + = ' + ' ;
ptr = strfill ( ptr , MAX_TRUNC_LENGTH + 2 , ' - ' ) ;
* ptr + + = ' + ' ;
ptr = strfill ( ptr , max_val_length + 2 , ' - ' ) ;
* ptr + + = ' + ' ;
* ptr = 0 ;
puts ( top_line ) ;
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
for ( i = 0 ; i < ex_var_count ; i + + )
{
uint sfx = 1 , j ;
printf ( " | %-*s| " , max_var_length + 1 , ex_var_names [ i ] ) ;
ptr = ex_var_names [ i ] ;
/* Make sure no two same truncated names will become */
for ( j = 0 ; j < i ; j + + )
if ( * truncated_var_names [ j ] = = * ptr )
sfx + + ;
truncated_var_names [ i ] [ 0 ] = * ptr ; /* Copy first var char */
int10_to_str ( sfx , truncated_var_names [ i ] + 1 , 10 ) ;
printf ( " %-*s| " , MAX_TRUNC_LENGTH + 1 , truncated_var_names [ i ] ) ;
printf ( " %-*s| \n " , max_val_length + 1 , llstr ( last_values [ i ] , buff ) ) ;
}
puts ( top_line ) ;
return ;
}
static my_bool get_pidfile ( MYSQL * mysql , char * pidfile )
{
MYSQL_RES * result ;
if ( mysql_query ( mysql , " SHOW VARIABLES LIKE 'pid_file' " ) )
{
my_printf_error ( 0 , " query failed; error: '%s' " , MYF ( ME_BELL ) ,
mysql_error ( mysql ) ) ;
}
result = mysql_store_result ( mysql ) ;
if ( result )
{
2000-09-12 03:02:33 +03:00
MYSQL_ROW row = mysql_fetch_row ( result ) ;
if ( row )
strmov ( pidfile , row [ 1 ] ) ;
2000-07-31 21:29:14 +02:00
mysql_free_result ( result ) ;
2000-09-12 03:02:33 +03:00
return row = = 0 ; /* Error if row = 0 */
2000-07-31 21:29:14 +02:00
}
return 1 ; /* Error */
}
2003-01-07 16:53:10 +02:00
/*
Return 1 if pid file didn ' t disappear or change
*/
2000-07-31 21:29:14 +02:00
2003-01-07 16:53:10 +02:00
static my_bool wait_pidfile ( char * pidfile , time_t last_modified ,
struct stat * pidfile_status )
2000-07-31 21:29:14 +02:00
{
char buff [ FN_REFLEN ] ;
2003-01-07 16:53:10 +02:00
int error = 1 ;
uint count = 0 ;
DBUG_ENTER ( " wait_pidfile " ) ;
2000-07-31 21:29:14 +02:00
2002-08-22 02:22:10 +03:00
system_filename ( buff , pidfile ) ;
2003-01-07 16:53:10 +02:00
do
2000-07-31 21:29:14 +02:00
{
2003-01-07 16:53:10 +02:00
int fd ;
if ( ( fd = my_open ( buff , O_RDONLY , MYF ( 0 ) ) ) < 0 )
{
error = 0 ;
break ;
}
( void ) my_close ( fd , MYF ( 0 ) ) ;
if ( last_modified & & ! stat ( pidfile , pidfile_status ) )
{
if ( last_modified ! = pidfile_status - > st_mtime )
{
/* File changed; Let's assume that mysqld did restart */
if ( opt_verbose )
printf ( " pid file '%s' changed while waiting for it to disappear! \n mysqld did probably restart \n " ,
buff ) ;
error = 0 ;
break ;
}
}
if ( count + + = = opt_shutdown_timeout )
break ;
2000-07-31 21:29:14 +02:00
sleep ( 1 ) ;
2003-01-07 16:53:10 +02:00
} while ( ! interrupted ) ;
if ( error )
2000-07-31 21:29:14 +02:00
{
2003-01-07 16:53:10 +02:00
DBUG_PRINT ( " warning " , ( " Pid file didn't disappear " ) ) ;
2000-07-31 21:29:14 +02:00
fprintf ( stderr ,
" Warning; Aborted waiting on pid file: '%s' after %d seconds \n " ,
buff , count - 1 ) ;
}
2003-01-07 16:53:10 +02:00
DBUG_RETURN ( error ) ;
2000-07-31 21:29:14 +02:00
}