2006-12-31 01:02:27 +01:00
/* Copyright (C) 2000-2006 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
2006-12-23 20:17:15 +01:00
the Free Software Foundation ; version 2 of the License .
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 */
/* Show databases, tables or columns */
2004-10-07 22:08:17 +03:00
# define SHOW_VERSION "9.5"
2000-07-31 21:29:14 +02:00
2002-04-02 20:29:53 +03:00
# include "client_priv.h"
2000-07-31 21:29:14 +02:00
# include <my_sys.h>
# include <m_string.h>
2002-09-17 23:46:53 +03:00
# include <mysql.h>
# include <mysqld_error.h>
2000-07-31 21:29:14 +02:00
# include <signal.h>
# include <stdarg.h>
2002-09-17 23:46:53 +03:00
# include <sslopt-vars.h>
2000-07-31 21:29:14 +02:00
2000-11-13 23:55:10 +02:00
static my_string host = 0 , opt_password = 0 , user = 0 ;
2005-05-06 11:39:30 +03:00
static my_bool opt_show_keys = 0 , opt_compress = 0 , opt_count = 0 , opt_status = 0 ,
2005-03-15 10:46:43 -08:00
tty_password = 0 , opt_table_type = 0 ;
2001-06-12 11:59:14 +03:00
static uint opt_verbose = 0 ;
2003-05-30 23:09:35 +05:00
static char * default_charset = ( char * ) MYSQL_DEFAULT_CHARSET_NAME ;
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 ;
2000-07-31 21:29:14 +02:00
static void get_options ( int * argc , char * * * argv ) ;
static uint opt_mysql_port = 0 ;
static int list_dbs ( MYSQL * mysql , const char * wild ) ;
static int list_tables ( MYSQL * mysql , const char * db , const char * table ) ;
static int list_table_status ( MYSQL * mysql , const char * db , const char * table ) ;
static int list_fields ( MYSQL * mysql , const char * db , const char * table ,
const char * field ) ;
static void print_header ( const char * header , uint head_length , . . . ) ;
static void print_row ( const char * header , uint head_length , . . . ) ;
static void print_trailer ( uint length , . . . ) ;
static void print_res_header ( MYSQL_RES * result ) ;
static void print_res_top ( MYSQL_RES * result ) ;
static void print_res_row ( MYSQL_RES * result , MYSQL_ROW cur ) ;
static const char * load_default_groups [ ] = { " mysqlshow " , " client " , 0 } ;
static my_string opt_mysql_unix_port = 0 ;
int main ( int argc , char * * argv )
{
int error ;
2002-11-08 17:02:33 +02:00
my_bool first_argument_uses_wildcards = 0 ;
2000-07-31 21:29:14 +02:00
char * wild ;
MYSQL mysql ;
MY_INIT ( argv [ 0 ] ) ;
load_defaults ( " my " , load_default_groups , & argc , & argv ) ;
get_options ( & argc , & argv ) ;
wild = 0 ;
2002-11-08 17:02:33 +02:00
if ( argc )
2000-07-31 21:29:14 +02:00
{
2002-11-08 17:02:33 +02:00
char * pos = argv [ argc - 1 ] , * to ;
for ( to = pos ; * pos ; pos + + , to + + )
{
2005-05-06 11:39:30 +03:00
switch ( * pos ) {
2002-11-08 17:02:33 +02:00
case ' * ' :
* pos = ' % ' ;
first_argument_uses_wildcards = 1 ;
break ;
case ' ? ' :
* pos = ' _ ' ;
first_argument_uses_wildcards = 1 ;
break ;
case ' % ' :
case ' _ ' :
first_argument_uses_wildcards = 1 ;
break ;
case ' \\ ' :
pos + + ;
default : break ;
}
* to = * pos ;
}
2003-03-06 13:51:37 +01:00
* to = * pos ; /* just to copy a '\0' if '\\' was used */
2000-07-31 21:29:14 +02:00
}
2002-11-08 17:02:33 +02:00
if ( first_argument_uses_wildcards )
wild = argv [ - - argc ] ;
2000-07-31 21:29:14 +02:00
else if ( argc = = 3 ) /* We only want one field */
2002-11-08 17:02:33 +02:00
wild = argv [ - - argc ] ;
2000-07-31 21:29:14 +02:00
if ( argc > 2 )
{
fprintf ( stderr , " %s: Too many arguments \n " , my_progname ) ;
exit ( 1 ) ;
}
mysql_init ( & mysql ) ;
if ( opt_compress )
mysql_options ( & mysql , MYSQL_OPT_COMPRESS , NullS ) ;
# 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 ) ;
2006-04-18 17:58:27 +02:00
mysql_options ( & mysql , MYSQL_OPT_SSL_VERIFY_SERVER_CERT ,
( char * ) & opt_ssl_verify_server_cert ) ;
2002-11-15 00:16:30 +05:00
# endif
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 ) ;
2000-07-31 21:29:14 +02:00
# endif
2003-05-30 23:09:35 +05:00
mysql_options ( & mysql , MYSQL_SET_CHARSET_NAME , default_charset ) ;
2000-11-13 23:55:10 +02:00
if ( ! ( mysql_real_connect ( & mysql , host , user , opt_password ,
2002-11-08 17:02:33 +02:00
( first_argument_uses_wildcards ) ? " " : argv [ 0 ] , opt_mysql_port , opt_mysql_unix_port ,
2000-07-31 21:29:14 +02:00
0 ) ) )
{
fprintf ( stderr , " %s: %s \n " , my_progname , mysql_error ( & mysql ) ) ;
exit ( 1 ) ;
}
2004-12-09 14:44:10 +01:00
mysql . reconnect = 1 ;
2000-07-31 21:29:14 +02:00
switch ( argc )
{
case 0 : error = list_dbs ( & mysql , wild ) ; break ;
case 1 :
if ( opt_status )
error = list_table_status ( & mysql , argv [ 0 ] , wild ) ;
else
error = list_tables ( & mysql , argv [ 0 ] , wild ) ;
break ;
default :
if ( opt_status & & ! wild )
error = list_table_status ( & mysql , argv [ 0 ] , argv [ 1 ] ) ;
else
2000-11-13 23:55:10 +02:00
error = list_fields ( & mysql , argv [ 0 ] , argv [ 1 ] , wild ) ;
break ;
2000-07-31 21:29:14 +02:00
}
mysql_close ( & mysql ) ; /* Close & free connection */
2000-11-13 23:55:10 +02:00
if ( opt_password )
my_free ( opt_password , MYF ( 0 ) ) ;
2002-11-15 00:16:30 +05:00
# ifdef HAVE_SMEM
my_free ( shared_memory_base_name , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
# endif
2000-07-31 21:29:14 +02:00
my_end ( 0 ) ;
exit ( error ? 1 : 0 ) ;
return 0 ; /* No compiler warnings */
}
2002-04-02 20:29:53 +03:00
static struct my_option my_long_options [ ] =
2000-07-31 21:29:14 +02:00
{
2005-08-24 22:03:34 +03:00
# ifdef __NETWARE__
2005-11-18 18:25:46 +01:00
{ " autoclose " , OPT_AUTO_CLOSE , " Auto close the screen on exit for Netware. " ,
2005-08-24 22:03:34 +03:00
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2003-06-13 10:59:02 +02:00
{ " character-sets-dir " , ' c ' , " Directory where character sets are. " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & charsets_dir , ( gptr * ) & charsets_dir , 0 , GET_STR , REQUIRED_ARG , 0 ,
0 , 0 , 0 , 0 , 0 } ,
2003-05-30 23:09:35 +05: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 } ,
2005-05-06 11:39:30 +03:00
{ " count " , OPT_COUNT ,
" Show number of rows per table (may be slow for not MyISAM tables) " ,
( gptr * ) & opt_count , ( gptr * ) & opt_count , 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 } ,
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 } ,
{ " help " , ' ? ' , " Display this help and exit. " , 0 , 0 , 0 , GET_NO_ARG , NO_ARG ,
0 , 0 , 0 , 0 , 0 , 0 } ,
{ " host " , ' h ' , " Connect to host. " , ( gptr * ) & host , ( gptr * ) & host , 0 , GET_STR ,
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " status " , ' i ' , " Shows a lot of extra information about each table. " ,
( gptr * ) & opt_status , ( gptr * ) & opt_status , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 ,
0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " keys " , ' k ' , " Show keys for table. " , ( gptr * ) & opt_show_keys ,
2002-04-02 20:29:53 +03:00
( gptr * ) & opt_show_keys , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " password " , ' p ' ,
2003-06-13 10:59:02 +02:00
" Password to use when connecting to server. If password is not given it's asked from the tty. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_STR , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-05-11 14:36:34 +03:00
{ " port " , ' P ' , " Port number to use for connection. " , ( gptr * ) & opt_mysql_port ,
2005-10-13 12:28:43 -07:00
( gptr * ) & opt_mysql_port , 0 , GET_UINT , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 ,
2002-05-11 14:36:34 +03:00
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 } ,
2002-11-15 00:16:30 +05:00
# endif
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 } ,
# 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 } ,
2000-07-31 21:29:14 +02:00
# endif
2005-03-15 10:46:43 -08:00
{ " show-table-type " , ' t ' , " Show table type column. " ,
( gptr * ) & opt_table_type , ( gptr * ) & opt_table_type , 0 , GET_BOOL ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " socket " , ' S ' , " Socket file to use for connection. " ,
( gptr * ) & opt_mysql_unix_port , ( gptr * ) & opt_mysql_unix_port , 0 , GET_STR ,
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 ,
( gptr * ) & user , 0 , GET_STR , 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 ' ,
" More verbose output; You can use this multiple times to get even more verbose output. " ,
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " version " , ' V ' , " Output version information and exit. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 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
} ;
2003-06-13 10:59:02 +02:00
2002-04-02 20:29:53 +03:00
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 (%s) \n " , my_progname , SHOW_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
}
2004-05-25 22:00:14 +03:00
2000-07-31 21:29:14 +02:00
static void usage ( void )
{
print_version ( ) ;
2006-12-31 01:02:27 +01:00
puts ( " Copyright (C) 2000-2006 MySQL AB " ) ;
2000-07-31 21:29:14 +02:00
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 " ) ;
puts ( " Shows the structure of a mysql database (databases,tables and columns) \n " ) ;
printf ( " Usage: %s [OPTIONS] [database [table [column]]] \n " , my_progname ) ;
puts ( " \n \
If last argument contains a shell or SQL wildcard ( * , ? , % or _ ) then only \ n \
what \ ' s matched by the wildcard is shown . \ n \
If no database is given then all matching databases are shown . \ n \
If no table is given then all matching tables in database are shown \ n \
If no column is given then all matching columns and columntypes in table \ n \
are shown " );
print_defaults ( " my " , load_default_groups ) ;
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
}
2004-05-25 22:00:14 +03:00
# include <help_end.h>
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
switch ( optid ) {
2005-08-24 22:03:34 +03:00
# ifdef __NETWARE__
case OPT_AUTO_CLOSE :
setscreenmode ( SCR_AUTOCLOSE_ON_EXIT ) ;
break ;
# endif
2002-04-02 20:29:53 +03:00
case ' v ' :
opt_verbose + + ;
break ;
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 ' 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 ;
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 ;
}
2002-04-02 20:29:53 +03:00
case ' # ' :
DBUG_PUSH ( argument ? argument : " d:t:o " ) ;
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 ' ? ' :
case ' I ' : /* Info */
usage ( ) ;
exit ( 0 ) ;
}
return 0 ;
}
static void
get_options ( int * argc , char * * * argv )
{
int ho_error ;
2004-08-31 21:27:58 +05:00
if ( ( ho_error = handle_options ( argc , argv , my_long_options , get_one_option ) ) )
2002-05-29 15:07:30 +03:00
exit ( ho_error ) ;
2002-04-02 20:29:53 +03:00
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 ) ;
2005-05-06 11:39:30 +03:00
if ( opt_count )
{
/*
We need to set verbose to 2 as we need to change the output to include
the number - of - rows column
*/
opt_verbose = 2 ;
}
2000-07-31 21:29:14 +02:00
return ;
}
static int
list_dbs ( MYSQL * mysql , const char * wild )
{
const char * header ;
2001-06-12 11:59:14 +03:00
uint length , counter = 0 ;
ulong rowcount = 0L ;
char tables [ NAME_LEN + 1 ] , rows [ NAME_LEN + 1 ] ;
char query [ 255 ] ;
2000-07-31 21:29:14 +02:00
MYSQL_FIELD * field ;
MYSQL_RES * result ;
2006-07-21 20:29:25 -07:00
MYSQL_ROW row = NULL , rrow ;
2000-07-31 21:29:14 +02:00
if ( ! ( result = mysql_list_dbs ( mysql , wild ) ) )
{
fprintf ( stderr , " %s: Cannot list databases: %s \n " , my_progname ,
mysql_error ( mysql ) ) ;
return 1 ;
}
2006-07-21 20:29:25 -07:00
/*
If a wildcard was used , but there was only one row and it ' s name is an
exact match , we ' ll assume they really wanted to see the contents of that
database . This is because it is fairly common for database names to
contain the underscore ( _ ) , like INFORMATION_SCHEMA .
*/
if ( wild & & mysql_num_rows ( result ) = = 1 )
{
row = mysql_fetch_row ( result ) ;
if ( ! my_strcasecmp ( & my_charset_latin1 , row [ 0 ] , wild ) )
{
mysql_free_result ( result ) ;
if ( opt_status )
return list_table_status ( mysql , wild , NULL ) ;
else
return list_tables ( mysql , wild , NULL ) ;
}
}
2000-07-31 21:29:14 +02:00
if ( wild )
printf ( " Wildcard: %s \n " , wild ) ;
header = " Databases " ;
2000-08-22 00:18:32 +03:00
length = ( uint ) strlen ( header ) ;
2000-07-31 21:29:14 +02:00
field = mysql_fetch_field ( result ) ;
if ( length < field - > max_length )
length = field - > max_length ;
2001-06-12 11:59:14 +03:00
if ( ! opt_verbose )
print_header ( header , length , NullS ) ;
else if ( opt_verbose = = 1 )
print_header ( header , length , " Tables " , 6 , NullS ) ;
else
print_header ( header , length , " Tables " , 6 , " Total Rows " , 12 , NullS ) ;
2006-07-21 20:29:25 -07:00
/* The first row may have already been read up above. */
while ( row | | ( row = mysql_fetch_row ( result ) ) )
2001-06-12 11:59:14 +03:00
{
counter + + ;
if ( opt_verbose )
{
if ( ! ( mysql_select_db ( mysql , row [ 0 ] ) ) )
{
MYSQL_RES * tresult = mysql_list_tables ( mysql , ( char * ) NULL ) ;
if ( mysql_affected_rows ( mysql ) > 0 )
{
sprintf ( tables , " %6lu " , ( ulong ) mysql_affected_rows ( mysql ) ) ;
rowcount = 0 ;
if ( opt_verbose > 1 )
{
2005-05-06 11:39:30 +03:00
/* Print the count of tables and rows for each database */
MYSQL_ROW trow ;
2001-06-12 11:59:14 +03:00
while ( ( trow = mysql_fetch_row ( tresult ) ) )
{
sprintf ( query , " SELECT COUNT(*) FROM `%s` " , trow [ 0 ] ) ;
if ( ! ( mysql_query ( mysql , query ) ) )
{
MYSQL_RES * rresult ;
if ( ( rresult = mysql_store_result ( mysql ) ) )
{
rrow = mysql_fetch_row ( rresult ) ;
rowcount + = ( ulong ) strtoull ( rrow [ 0 ] , ( char * * ) 0 , 10 ) ;
mysql_free_result ( rresult ) ;
}
}
}
sprintf ( rows , " %12lu " , rowcount ) ;
}
}
else
{
sprintf ( tables , " %6d " , 0 ) ;
sprintf ( rows , " %12d " , 0 ) ;
}
mysql_free_result ( tresult ) ;
}
else
{
strmov ( tables , " N/A " ) ;
strmov ( rows , " N/A " ) ;
}
}
if ( ! opt_verbose )
print_row ( row [ 0 ] , length , 0 ) ;
else if ( opt_verbose = = 1 )
print_row ( row [ 0 ] , length , tables , 6 , NullS ) ;
else
print_row ( row [ 0 ] , length , tables , 6 , rows , 12 , NullS ) ;
2006-07-21 20:29:25 -07:00
row = NULL ;
2001-06-12 11:59:14 +03:00
}
print_trailer ( length ,
( opt_verbose > 0 ? 6 : 0 ) ,
( opt_verbose > 1 ? 12 : 0 ) ,
0 ) ;
if ( counter & & opt_verbose )
printf ( " %u row%s in set. \n " , counter , ( counter > 1 ) ? " s " : " " ) ;
2000-07-31 21:29:14 +02:00
mysql_free_result ( result ) ;
return 0 ;
}
static int
list_tables ( MYSQL * mysql , const char * db , const char * table )
{
const char * header ;
2001-06-12 11:59:14 +03:00
uint head_length , counter = 0 ;
2005-03-15 10:46:43 -08:00
char query [ 255 ] , rows [ NAME_LEN ] , fields [ 16 ] ;
2000-07-31 21:29:14 +02:00
MYSQL_FIELD * field ;
MYSQL_RES * result ;
2001-06-12 11:59:14 +03:00
MYSQL_ROW row , rrow ;
2000-07-31 21:29:14 +02:00
if ( mysql_select_db ( mysql , db ) )
{
fprintf ( stderr , " %s: Cannot connect to db %s: %s \n " , my_progname , db ,
mysql_error ( mysql ) ) ;
return 1 ;
}
2005-03-15 10:46:43 -08:00
if ( table )
{
/*
We just hijack the ' rows ' variable for a bit to store the escaped
table name
*/
2005-06-10 14:03:22 -05:00
mysql_real_escape_string ( mysql , rows , table , ( unsigned long ) strlen ( table ) ) ;
2005-03-15 10:46:43 -08:00
my_snprintf ( query , sizeof ( query ) , " show%s tables like '%s' " ,
opt_table_type ? " full " : " " , rows ) ;
}
else
my_snprintf ( query , sizeof ( query ) , " show%s tables " ,
opt_table_type ? " full " : " " ) ;
if ( mysql_query ( mysql , query ) | | ! ( result = mysql_store_result ( mysql ) ) )
2000-07-31 21:29:14 +02:00
{
fprintf ( stderr , " %s: Cannot list tables in %s: %s \n " , my_progname , db ,
mysql_error ( mysql ) ) ;
exit ( 1 ) ;
}
printf ( " Database: %s " , db ) ;
if ( table )
printf ( " Wildcard: %s " , table ) ;
putchar ( ' \n ' ) ;
header = " Tables " ;
2000-08-22 00:18:32 +03:00
head_length = ( uint ) strlen ( header ) ;
2000-07-31 21:29:14 +02:00
field = mysql_fetch_field ( result ) ;
if ( head_length < field - > max_length )
head_length = field - > max_length ;
2005-03-15 10:46:43 -08:00
if ( opt_table_type )
{
if ( ! opt_verbose )
print_header ( header , head_length , " table_type " , 10 , NullS ) ;
else if ( opt_verbose = = 1 )
print_header ( header , head_length , " table_type " , 10 , " Columns " , 8 , NullS ) ;
else
{
print_header ( header , head_length , " table_type " , 10 , " Columns " , 8 ,
" Total Rows " , 10 , NullS ) ;
}
}
2001-06-12 11:59:14 +03:00
else
2005-03-15 10:46:43 -08:00
{
if ( ! opt_verbose )
print_header ( header , head_length , NullS ) ;
else if ( opt_verbose = = 1 )
print_header ( header , head_length , " Columns " , 8 , NullS ) ;
else
print_header ( header , head_length , " Columns " , 8 , " Total Rows " , 10 , NullS ) ;
}
2001-06-12 11:59:14 +03:00
2000-07-31 21:29:14 +02:00
while ( ( row = mysql_fetch_row ( result ) ) )
2001-06-12 11:59:14 +03:00
{
counter + + ;
if ( opt_verbose > 0 )
{
if ( ! ( mysql_select_db ( mysql , db ) ) )
{
MYSQL_RES * rresult = mysql_list_fields ( mysql , row [ 0 ] , NULL ) ;
ulong rowcount = 0L ;
if ( ! rresult )
{
strmov ( fields , " N/A " ) ;
strmov ( rows , " N/A " ) ;
}
else
{
sprintf ( fields , " %8u " , ( uint ) mysql_num_fields ( rresult ) ) ;
mysql_free_result ( rresult ) ;
if ( opt_verbose > 1 )
{
2005-05-06 11:39:30 +03:00
/* Print the count of rows for each table */
2001-06-12 11:59:14 +03:00
sprintf ( query , " SELECT COUNT(*) FROM `%s` " , row [ 0 ] ) ;
if ( ! ( mysql_query ( mysql , query ) ) )
{
if ( ( rresult = mysql_store_result ( mysql ) ) )
{
rrow = mysql_fetch_row ( rresult ) ;
rowcount + = ( unsigned long ) strtoull ( rrow [ 0 ] , ( char * * ) 0 , 10 ) ;
mysql_free_result ( rresult ) ;
}
sprintf ( rows , " %10lu " , rowcount ) ;
}
else
sprintf ( rows , " %10d " , 0 ) ;
}
}
}
else
{
strmov ( fields , " N/A " ) ;
strmov ( rows , " N/A " ) ;
}
}
2005-03-15 10:46:43 -08:00
if ( opt_table_type )
{
if ( ! opt_verbose )
print_row ( row [ 0 ] , head_length , row [ 1 ] , 10 , NullS ) ;
else if ( opt_verbose = = 1 )
print_row ( row [ 0 ] , head_length , row [ 1 ] , 10 , fields , 8 , NullS ) ;
else
print_row ( row [ 0 ] , head_length , row [ 1 ] , 10 , fields , 8 , rows , 10 , NullS ) ;
}
else
{
if ( ! opt_verbose )
print_row ( row [ 0 ] , head_length , NullS ) ;
else if ( opt_verbose = = 1 )
print_row ( row [ 0 ] , head_length , fields , 8 , NullS ) ;
else
print_row ( row [ 0 ] , head_length , fields , 8 , rows , 10 , NullS ) ;
}
2001-06-12 11:59:14 +03:00
}
print_trailer ( head_length ,
2005-03-15 10:46:43 -08:00
( opt_table_type ? 10 : opt_verbose > 0 ? 8 : 0 ) ,
( opt_table_type ? ( opt_verbose > 0 ? 8 : 0 )
: ( opt_verbose > 1 ? 10 : 0 ) ) ,
! opt_table_type ? 0 : opt_verbose > 1 ? 10 : 0 ,
2001-06-12 11:59:14 +03:00
0 ) ;
if ( counter & & opt_verbose )
printf ( " %u row%s in set. \n \n " , counter , ( counter > 1 ) ? " s " : " " ) ;
2000-07-31 21:29:14 +02:00
mysql_free_result ( result ) ;
return 0 ;
}
2001-06-12 11:59:14 +03:00
2000-07-31 21:29:14 +02:00
static int
list_table_status ( MYSQL * mysql , const char * db , const char * wild )
{
char query [ 1024 ] , * end ;
MYSQL_RES * result ;
MYSQL_ROW row ;
2005-05-06 11:39:30 +03:00
end = strxmov ( query , " show table status from ` " , db , " ` " , NullS ) ;
2000-07-31 21:29:14 +02:00
if ( wild & & wild [ 0 ] )
strxmov ( end , " like ' " , wild , " ' " , NullS ) ;
if ( mysql_query ( mysql , query ) | | ! ( result = mysql_store_result ( mysql ) ) )
{
fprintf ( stderr , " %s: Cannot get status for db: %s, table: %s: %s \n " ,
my_progname , db , wild ? wild : " " , mysql_error ( mysql ) ) ;
if ( mysql_errno ( mysql ) = = ER_PARSE_ERROR )
fprintf ( stderr , " This error probably means that your MySQL server doesn't support the \n \' show table status' command. \n " ) ;
return 1 ;
}
printf ( " Database: %s " , db ) ;
if ( wild )
printf ( " Wildcard: %s " , wild ) ;
putchar ( ' \n ' ) ;
print_res_header ( result ) ;
while ( ( row = mysql_fetch_row ( result ) ) )
print_res_row ( result , row ) ;
print_res_top ( result ) ;
mysql_free_result ( result ) ;
return 0 ;
}
/*
2005-05-06 11:39:30 +03:00
list fields uses field interface as an example of how to parse
a MYSQL FIELD
2000-07-31 21:29:14 +02:00
*/
static int
list_fields ( MYSQL * mysql , const char * db , const char * table ,
const char * wild )
{
char query [ 1024 ] , * end ;
MYSQL_RES * result ;
MYSQL_ROW row ;
2005-04-13 12:22:20 +02:00
ulong rows ;
2005-05-06 11:39:30 +03:00
LINT_INIT ( rows ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_select_db ( mysql , db ) )
{
fprintf ( stderr , " %s: Cannot connect to db: %s: %s \n " , my_progname , db ,
mysql_error ( mysql ) ) ;
return 1 ;
}
2005-05-06 11:39:30 +03:00
if ( opt_count )
2005-04-13 12:22:20 +02:00
{
2005-05-06 11:39:30 +03:00
sprintf ( query , " select count(*) from `%s` " , table ) ;
if ( mysql_query ( mysql , query ) | | ! ( result = mysql_store_result ( mysql ) ) )
{
fprintf ( stderr , " %s: Cannot get record count for db: %s, table: %s: %s \n " ,
my_progname , db , table , mysql_error ( mysql ) ) ;
return 1 ;
}
row = mysql_fetch_row ( result ) ;
rows = ( ulong ) strtoull ( row [ 0 ] , ( char * * ) 0 , 10 ) ;
mysql_free_result ( result ) ;
2005-04-13 12:22:20 +02:00
}
2003-05-30 23:09:35 +05:00
end = strmov ( strmov ( strmov ( query , " show /*!32332 FULL */ columns from ` " ) , table ) , " ` " ) ;
2000-07-31 21:29:14 +02:00
if ( wild & & wild [ 0 ] )
strxmov ( end , " like ' " , wild , " ' " , NullS ) ;
if ( mysql_query ( mysql , query ) | | ! ( result = mysql_store_result ( mysql ) ) )
{
fprintf ( stderr , " %s: Cannot list columns in db: %s, table: %s: %s \n " ,
my_progname , db , table , mysql_error ( mysql ) ) ;
return 1 ;
}
2005-05-06 11:39:30 +03:00
printf ( " Database: %s Table: %s " , db , table ) ;
if ( opt_count )
printf ( " Rows: %lu " , rows ) ;
2000-07-31 21:29:14 +02:00
if ( wild & & wild [ 0 ] )
printf ( " Wildcard: %s " , wild ) ;
putchar ( ' \n ' ) ;
print_res_header ( result ) ;
while ( ( row = mysql_fetch_row ( result ) ) )
print_res_row ( result , row ) ;
print_res_top ( result ) ;
if ( opt_show_keys )
{
2003-05-30 23:09:35 +05:00
end = strmov ( strmov ( strmov ( query , " show keys from ` " ) , table ) , " ` " ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_query ( mysql , query ) | | ! ( result = mysql_store_result ( mysql ) ) )
{
fprintf ( stderr , " %s: Cannot list keys in db: %s, table: %s: %s \n " ,
my_progname , db , table , mysql_error ( mysql ) ) ;
return 1 ;
}
if ( mysql_num_rows ( result ) )
{
print_res_header ( result ) ;
while ( ( row = mysql_fetch_row ( result ) ) )
print_res_row ( result , row ) ;
print_res_top ( result ) ;
}
else
puts ( " Table has no keys " ) ;
}
mysql_free_result ( result ) ;
return 0 ;
}
/*****************************************************************************
2005-05-06 11:39:30 +03:00
General functions to print a nice ascii - table from data
2000-07-31 21:29:14 +02:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void
print_header ( const char * header , uint head_length , . . . )
{
va_list args ;
uint length , i , str_length , pre_space ;
const char * field ;
va_start ( args , head_length ) ;
putchar ( ' + ' ) ;
field = header ; length = head_length ;
for ( ; ; )
{
for ( i = 0 ; i < length + 2 ; i + + )
putchar ( ' - ' ) ;
putchar ( ' + ' ) ;
if ( ! ( field = va_arg ( args , my_string ) ) )
break ;
length = va_arg ( args , uint ) ;
}
va_end ( args ) ;
putchar ( ' \n ' ) ;
va_start ( args , head_length ) ;
field = header ; length = head_length ;
putchar ( ' | ' ) ;
for ( ; ; )
{
2000-08-22 00:18:32 +03:00
str_length = ( uint ) strlen ( field ) ;
2000-07-31 21:29:14 +02:00
if ( str_length > length )
str_length = length + 1 ;
pre_space = ( uint ) ( ( ( int ) length - ( int ) str_length ) / 2 ) + 1 ;
for ( i = 0 ; i < pre_space ; i + + )
putchar ( ' ' ) ;
for ( i = 0 ; i < str_length ; i + + )
putchar ( field [ i ] ) ;
length = length + 2 - str_length - pre_space ;
for ( i = 0 ; i < length ; i + + )
putchar ( ' ' ) ;
putchar ( ' | ' ) ;
if ( ! ( field = va_arg ( args , my_string ) ) )
break ;
length = va_arg ( args , uint ) ;
}
va_end ( args ) ;
putchar ( ' \n ' ) ;
va_start ( args , head_length ) ;
putchar ( ' + ' ) ;
field = header ; length = head_length ;
for ( ; ; )
{
for ( i = 0 ; i < length + 2 ; i + + )
putchar ( ' - ' ) ;
putchar ( ' + ' ) ;
if ( ! ( field = va_arg ( args , my_string ) ) )
break ;
length = va_arg ( args , uint ) ;
}
va_end ( args ) ;
putchar ( ' \n ' ) ;
}
static void
print_row ( const char * header , uint head_length , . . . )
{
va_list args ;
const char * field ;
uint i , length , field_length ;
va_start ( args , head_length ) ;
field = header ; length = head_length ;
for ( ; ; )
{
putchar ( ' | ' ) ;
putchar ( ' ' ) ;
fputs ( field , stdout ) ;
2000-08-22 00:18:32 +03:00
field_length = ( uint ) strlen ( field ) ;
2000-07-31 21:29:14 +02:00
for ( i = field_length ; i < = length ; i + + )
putchar ( ' ' ) ;
if ( ! ( field = va_arg ( args , my_string ) ) )
break ;
length = va_arg ( args , uint ) ;
}
va_end ( args ) ;
putchar ( ' | ' ) ;
putchar ( ' \n ' ) ;
}
static void
print_trailer ( uint head_length , . . . )
{
va_list args ;
uint length , i ;
va_start ( args , head_length ) ;
length = head_length ;
putchar ( ' + ' ) ;
for ( ; ; )
{
for ( i = 0 ; i < length + 2 ; i + + )
putchar ( ' - ' ) ;
putchar ( ' + ' ) ;
if ( ! ( length = va_arg ( args , uint ) ) )
break ;
}
va_end ( args ) ;
putchar ( ' \n ' ) ;
}
static void print_res_header ( MYSQL_RES * result )
{
MYSQL_FIELD * field ;
print_res_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_res_top ( result ) ;
}
static void print_res_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 ' ) ;
}
static void print_res_row ( MYSQL_RES * result , MYSQL_ROW cur )
{
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 ' ) ;
}