2003-01-21 21:07:59 +02:00
/* Copyright (C) 2000-2003 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 */
/* mysql command tool
* Commands compatible with mSQL by David J . Hughes
*
* Written by :
* Michael ' Monty ' Widenius
2002-04-03 13:36:01 +03:00
* Andi Gutmans < andi @ zend . com >
* Zeev Suraski < zeev @ zend . com >
* Jani Tolonen < jani @ mysql . com >
* Matt Wagner < matt @ mysql . com >
* Jeremy Cole < jcole @ mysql . com >
* Tonu Samuel < tonu @ mysql . com >
2004-03-03 15:54:37 -05:00
* Harrison Fisk < harrison @ mysql . com >
2000-07-31 21:29:14 +02:00
*
* */
2001-09-12 23:53:31 +03:00
# include "client_priv.h"
2000-07-31 21:29:14 +02:00
# include <m_ctype.h>
2002-04-02 20:29:53 +03:00
# include <stdarg.h>
2000-07-31 21:29:14 +02:00
# include <my_dir.h>
# ifndef __GNU_LIBRARY__
2000-11-17 14:33:29 +02:00
# define __GNU_LIBRARY__ // Skip warnings in getopt.h
2000-07-31 21:29:14 +02:00
# endif
# include "my_readline.h"
# include <signal.h>
2001-08-27 03:34:52 +08:00
# include <violite.h>
2000-07-31 21:29:14 +02:00
2003-04-06 20:14:36 -04:00
# if defined(USE_LIBEDIT_INTERFACE) && defined(HAVE_LOCALE_H)
# include <locale.h>
# endif
2005-01-25 18:32:09 +02:00
const char * VER = " 14.8 " ;
2000-11-28 04:47:47 +02:00
2001-02-02 03:47:06 +02:00
/* Don't try to make a nice table if the data is too big */
# define MAX_COLUMN_LENGTH 1024
2000-11-17 14:33:29 +02:00
gptr sql_alloc ( unsigned size ) ; // Don't use mysqld alloc for these
2000-07-31 21:29:14 +02:00
void sql_element_free ( void * ptr ) ;
# include "sql_string.h"
extern " C " {
# if defined(HAVE_CURSES_H) && defined(HAVE_TERM_H)
# include <curses.h>
# include <term.h>
# else
# if defined(HAVE_TERMIOS_H)
# include <termios.h>
# include <unistd.h>
# elif defined(HAVE_TERMBITS_H)
# include <termbits.h>
# elif defined(HAVE_ASM_TERMBITS_H) && (!defined __GLIBC__ || !(__GLIBC__ > 2 || __GLIBC__ == 2 && __GLIBC_MINOR__ > 0))
2000-11-17 14:33:29 +02:00
# include <asm/termbits.h> // Standard linux
2000-07-31 21:29:14 +02:00
# endif
# undef VOID
# if defined(HAVE_TERMCAP_H)
# include <termcap.h>
# else
# ifdef HAVE_CURSES_H
# include <curses.h>
# endif
2000-11-17 14:33:29 +02:00
# undef SYSV // hack to avoid syntax error
2000-07-31 21:29:14 +02:00
# ifdef HAVE_TERM_H
# include <term.h>
# endif
# endif
# endif
2000-11-17 14:33:29 +02:00
# undef bcmp // Fix problem with new readline
2001-08-22 01:45:07 +03:00
# if defined( __WIN__) || defined(OS2)
2000-07-31 21:29:14 +02:00
# include <conio.h>
2003-01-28 08:38:28 +02:00
# elif !defined(__NETWARE__)
2000-07-31 21:29:14 +02:00
# include <readline/readline.h>
# define HAVE_READLINE
# endif
//int vidattr(long unsigned int attrs); // Was missing in sun curses
}
# if !defined(HAVE_VIDATTR)
# undef vidattr
2000-11-17 14:33:29 +02:00
# define vidattr(A) {} // Can't get this to work
2000-07-31 21:29:14 +02:00
# endif
2002-10-08 22:28:24 +03:00
# ifdef FN_NO_CASE_SENCE
2003-03-16 11:20:45 +04:00
# define cmp_database(cs,A,B) my_strcasecmp((cs), (A), (B))
2000-07-31 21:29:14 +02:00
# else
2003-03-16 11:20:45 +04:00
# define cmp_database(cs,A,B) strcmp((A),(B))
2000-07-31 21:29:14 +02:00
# endif
2003-01-28 08:38:28 +02:00
# if !defined( __WIN__) && !defined( OS2) && !defined(__NETWARE__) && (!defined(HAVE_mit_thread) || !defined(THREAD))
# define USE_POPEN
# endif
2000-07-31 21:29:14 +02:00
# include "completion_hash.h"
2002-04-03 13:36:01 +03:00
# define PROMPT_CHAR '\\'
2003-04-23 03:29:03 +03:00
# define DEFAULT_DELIMITER ";"
2002-04-03 13:36:01 +03:00
2000-07-31 21:29:14 +02:00
typedef struct st_status
{
int exit_status ;
ulong query_start_line ;
char * file_name ;
LINE_BUFFER * line_buff ;
bool batch , add_to_history ;
} STATUS ;
static HashTable ht ;
2002-08-24 02:08:10 +03:00
static char * * defaults_argv ;
2000-07-31 21:29:14 +02:00
enum enum_info_type { INFO_INFO , INFO_ERROR , INFO_RESULT } ;
typedef enum enum_info_type INFO_TYPE ;
static MYSQL mysql ; /* The connection */
2002-04-02 20:29:53 +03:00
static my_bool info_flag = 0 , ignore_errors = 0 , wait_flag = 0 , quick = 0 ,
connected = 0 , opt_raw_data = 0 , unbuffered = 0 , output_tables = 0 ,
2002-04-16 11:39:03 +03:00
rehash = 1 , skip_updates = 0 , safe_updates = 0 , one_database = 0 ,
2002-04-03 13:36:01 +03:00
opt_compress = 0 , using_opt_local_infile = 0 ,
2002-04-02 20:29:53 +03:00
vertical = 0 , line_numbers = 1 , column_names = 1 , opt_html = 0 ,
opt_xml = 0 , opt_nopager = 1 , opt_outfile = 0 , named_cmds = 0 ,
2003-08-19 00:08:08 +03:00
tty_password = 0 , opt_nobeep = 0 , opt_reconnect = 1 ,
2004-01-14 04:58:37 +02:00
default_charset_used = 0 , opt_secure_auth = 0 ,
2004-10-07 22:47:11 +03:00
default_pager_set = 0 , opt_sigint_ignore = 0 ;
2004-05-26 21:40:27 +05:00
static ulong opt_max_allowed_packet , opt_net_buffer_length ;
2002-04-03 13:36:01 +03:00
static uint verbose = 0 , opt_silent = 0 , opt_mysql_port = 0 , opt_local_infile = 0 ;
2000-07-31 21:29:14 +02:00
static my_string opt_mysql_unix_port = 0 ;
static int connect_flag = CLIENT_INTERACTIVE ;
static char * current_host , * current_db , * current_user = 0 , * opt_password = 0 ,
2003-04-23 03:29:03 +03:00
* current_prompt = 0 , * delimiter_str = 0 ,
2003-05-28 16:23:04 +05:00
* default_charset = ( char * ) MYSQL_DEFAULT_CHARSET_NAME ;
2000-07-31 21:29:14 +02:00
static char * histfile ;
2005-01-25 18:32:09 +02:00
static char * histfile_tmp ;
2000-07-31 21:29:14 +02:00
static String glob_buffer , old_buffer ;
2002-04-03 13:36:01 +03:00
static String processed_prompt ;
static char * full_username = 0 , * part_username = 0 , * default_prompt = 0 ;
2001-01-17 05:47:33 -07:00
static int wait_time = 5 ;
2000-07-31 21:29:14 +02:00
static STATUS status ;
2000-11-27 02:28:41 +02:00
static ulong select_limit , max_join_size , opt_connect_timeout = 0 ;
2001-11-07 02:30:34 +02:00
static char mysql_charsets_dir [ FN_REFLEN + 1 ] ;
2001-04-25 01:11:29 +03:00
static const char * xmlmeta [ ] = {
2001-04-11 13:04:03 +02:00
" & " , " & " ,
" < " , " < " ,
2005-01-25 14:25:40 -08:00
" > " , " > " ,
" \" " , " " " ,
2001-04-11 13:04:03 +02:00
0 , 0
} ;
2002-04-03 13:36:01 +03:00
static const char * day_names [ ] = { " Sun " , " Mon " , " Tue " , " Wed " , " Thu " , " Fri " , " Sat " } ;
static const char * month_names [ ] = { " Jan " , " Feb " , " Mar " , " Apr " , " May " , " Jun " , " Jul " ,
" Aug " , " Sep " , " Oct " , " Nov " , " Dec " } ;
static char default_pager [ FN_REFLEN ] ;
static char pager [ FN_REFLEN ] , outfile [ FN_REFLEN ] ;
static FILE * PAGER , * OUTFILE ;
static MEM_ROOT hash_mem_root ;
static uint prompt_counter ;
2003-04-23 03:29:03 +03:00
static char delimiter [ 16 ] = DEFAULT_DELIMITER ;
static uint delimiter_length = 1 ;
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 ;
2003-03-16 11:20:45 +04:00
static CHARSET_INFO * charset_info = & my_charset_latin1 ;
2005-01-14 22:46:04 +01:00
2000-07-31 21:29:14 +02:00
# include "sslopt-vars.h"
const char * default_dbug_option = " d:t:o,/tmp/mysql.trace " ;
2000-11-17 14:33:29 +02:00
void tee_fprintf ( FILE * file , const char * fmt , . . . ) ;
void tee_fputs ( const char * s , FILE * file ) ;
void tee_puts ( const char * s , FILE * file ) ;
2000-11-18 17:34:15 +02:00
void tee_putc ( int c , FILE * file ) ;
2000-07-31 21:29:14 +02:00
/* The names of functions that actually do the manipulation. */
static int get_options ( int argc , char * * argv ) ;
static int com_quit ( String * str , char * ) ,
com_go ( String * str , char * ) , com_ego ( String * str , char * ) ,
2000-11-22 03:45:02 +02:00
com_print ( String * str , char * ) ,
2000-07-31 21:29:14 +02:00
com_help ( String * str , char * ) , com_clear ( String * str , char * ) ,
com_connect ( String * str , char * ) , com_status ( String * str , char * ) ,
com_use ( String * str , char * ) , com_source ( String * str , char * ) ,
2000-11-22 03:45:02 +02:00
com_rehash ( String * str , char * ) , com_tee ( String * str , char * ) ,
2003-01-28 08:38:28 +02:00
com_notee ( String * str , char * ) ,
2003-04-23 03:29:03 +03:00
com_prompt ( String * str , char * ) , com_delimiter ( String * str , char * ) ;
2000-07-31 21:29:14 +02:00
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-22 03:45:02 +02:00
static int com_nopager ( String * str , char * ) , com_pager ( String * str , char * ) ,
2003-01-28 08:38:28 +02:00
com_edit ( String * str , char * ) , com_shell ( String * str , char * ) ;
2000-11-22 03:45:02 +02:00
# endif
2000-07-31 21:29:14 +02:00
static int read_lines ( bool execute_commands ) ;
static int sql_connect ( char * host , char * database , char * user , char * password ,
uint silent ) ;
2003-06-04 18:28:51 +03:00
static int put_info ( const char * str , INFO_TYPE info , uint error = 0 ,
const char * sql_state = 0 ) ;
static int put_error ( MYSQL * mysql ) ;
2000-07-31 21:29:14 +02:00
static void safe_put_field ( const char * pos , ulong length ) ;
2001-12-14 17:43:18 +02:00
static void xmlencode_print ( const char * src , uint length ) ;
2000-11-17 14:33:29 +02:00
static void init_pager ( ) ;
static void end_pager ( ) ;
2003-01-28 08:38:28 +02:00
static void init_tee ( const char * ) ;
2000-11-17 14:33:29 +02:00
static void end_tee ( ) ;
2002-04-03 13:36:01 +03:00
static const char * construct_prompt ( ) ;
2002-12-25 23:57:01 +02:00
static char * get_arg ( char * line , my_bool get_next_arg ) ;
2002-04-03 13:36:01 +03:00
static void init_username ( ) ;
static void add_int_to_prompt ( int toadd ) ;
2000-07-31 21:29:14 +02:00
/* A structure which contains information on the commands this program
can understand . */
typedef struct {
const char * name ; /* User printable name of the function. */
char cmd_char ; /* msql command character */
int ( * func ) ( String * str , char * ) ; /* Function to call to do the job. */
bool takes_params ; /* Max parameters for command */
const char * doc ; /* Documentation for this function. */
} COMMANDS ;
static COMMANDS commands [ ] = {
2002-10-28 17:44:19 +04:00
{ " ? " , ' ? ' , com_help , 1 , " Synonym for `help'. " } ,
2000-11-17 14:33:29 +02:00
{ " clear " , ' c ' , com_clear , 0 , " Clear command. " } ,
2000-08-11 10:17:38 -05:00
{ " connect " , ' r ' , com_connect , 1 ,
2000-11-17 14:33:29 +02:00
" Reconnect to the server. Optional arguments are db and host. " } ,
2003-12-30 17:58:39 +02:00
{ " delimiter " , ' d ' , com_delimiter , 1 ,
" Set query delimiter. " } ,
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-17 14:33:29 +02:00
{ " edit " , ' e ' , com_edit , 0 , " Edit command with $EDITOR. " } ,
2000-11-22 03:45:02 +02:00
# endif
2000-08-11 10:17:38 -05:00
{ " ego " , ' G ' , com_ego , 0 ,
2000-11-17 14:33:29 +02:00
" Send command to mysql server, display result vertically. " } ,
{ " exit " , ' q ' , com_quit , 0 , " Exit mysql. Same as quit. " } ,
{ " go " , ' g ' , com_go , 0 , " Send command to mysql server. " } ,
2003-12-30 17:58:39 +02:00
{ " help " , ' h ' , com_help , 1 , " Display this help. " } ,
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-17 14:33:29 +02:00
{ " nopager " , ' n ' , com_nopager , 0 , " Disable pager, print to stdout. " } ,
# endif
{ " notee " , ' t ' , com_notee , 0 , " Don't write into outfile. " } ,
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-17 14:33:29 +02:00
{ " pager " , ' P ' , com_pager , 1 ,
" Set PAGER [to_pager]. Print the query results via PAGER. " } ,
# endif
{ " print " , ' p ' , com_print , 0 , " Print current command. " } ,
2002-04-03 13:36:01 +03:00
{ " prompt " , ' R ' , com_prompt , 1 , " Change your mysql prompt. " } ,
2000-11-17 14:33:29 +02:00
{ " quit " , ' q ' , com_quit , 0 , " Quit mysql. " } ,
{ " rehash " , ' # ' , com_rehash , 0 , " Rebuild completion hash. " } ,
2000-07-31 21:29:14 +02:00
{ " source " , ' . ' , com_source , 1 ,
2000-11-17 14:33:29 +02:00
" Execute a SQL script file. Takes a file name as an argument. " } ,
{ " status " , ' s ' , com_status , 0 , " Get status information from the server. " } ,
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2001-11-06 23:00:03 +02:00
{ " system " , ' ! ' , com_shell , 1 , " Execute a system shell command. " } ,
# endif
2000-11-17 14:33:29 +02:00
{ " tee " , ' T ' , com_tee , 1 ,
" Set outfile [to_outfile]. Append everything into given outfile. " } ,
2000-08-11 10:17:38 -05:00
{ " use " , ' u ' , com_use , 1 ,
2000-11-17 14:33:29 +02:00
" Use another database. Takes database name as argument. " } ,
/* Get bash-like expansion for some commands */
2000-08-11 10:17:38 -05:00
{ " create table " , 0 , 0 , 0 , " " } ,
{ " create database " , 0 , 0 , 0 , " " } ,
{ " drop " , 0 , 0 , 0 , " " } ,
{ " select " , 0 , 0 , 0 , " " } ,
{ " insert " , 0 , 0 , 0 , " " } ,
{ " replace " , 0 , 0 , 0 , " " } ,
{ " update " , 0 , 0 , 0 , " " } ,
{ " delete " , 0 , 0 , 0 , " " } ,
{ " explain " , 0 , 0 , 0 , " " } ,
{ " show databases " , 0 , 0 , 0 , " " } ,
{ " show fields from " , 0 , 0 , 0 , " " } ,
{ " show keys from " , 0 , 0 , 0 , " " } ,
{ " show tables " , 0 , 0 , 0 , " " } ,
{ " load data from " , 0 , 0 , 0 , " " } ,
{ " alter table " , 0 , 0 , 0 , " " } ,
{ " set option " , 0 , 0 , 0 , " " } ,
{ " lock tables " , 0 , 0 , 0 , " " } ,
{ " unlock tables " , 0 , 0 , 0 , " " } ,
{ ( char * ) NULL , 0 , 0 , 0 , " " }
2000-07-31 21:29:14 +02:00
} ;
static const char * load_default_groups [ ] = { " mysql " , " client " , 0 } ;
2001-10-02 05:53:00 +03:00
static const char * server_default_groups [ ] =
{ " server " , " embedded " , " mysql_SERVER " , 0 } ;
2000-07-31 21:29:14 +02:00
# ifdef HAVE_READLINE
2004-03-03 15:54:37 -05:00
/*
HIST_ENTRY is defined for libedit , but not for the real readline
Need to redefine it for real readline to find it
*/
2004-08-24 19:23:23 +05:00
# if !defined(HAVE_HIST_ENTRY)
2004-03-03 15:54:37 -05:00
typedef struct _hist_entry {
const char * line ;
const char * data ;
} HIST_ENTRY ;
# endif
2003-01-18 20:28:48 +04:00
extern " C " int add_history ( const char * command ) ; /* From readline directory */
extern " C " int read_history ( const char * command ) ;
extern " C " int write_history ( const char * command ) ;
2004-03-03 15:54:37 -05:00
extern " C " HIST_ENTRY * history_get ( int num ) ;
extern " C " int history_length ;
static int not_in_history ( const char * line ) ;
2000-07-31 21:29:14 +02:00
static void initialize_readline ( char * name ) ;
2004-03-03 15:54:37 -05:00
static void fix_history ( String * final_command ) ;
2000-07-31 21:29:14 +02:00
# endif
static COMMANDS * find_command ( char * name , char cmd_name ) ;
2002-12-24 00:53:07 +02:00
static bool add_line ( String & buffer , char * line , char * in_string ,
bool * ml_comment ) ;
2000-07-31 21:29:14 +02:00
static void remove_cntrl ( String & buffer ) ;
static void print_table_data ( MYSQL_RES * result ) ;
static void print_table_data_html ( MYSQL_RES * result ) ;
2001-04-11 13:04:03 +02:00
static void print_table_data_xml ( MYSQL_RES * result ) ;
2000-07-31 21:29:14 +02:00
static void print_tab_data ( MYSQL_RES * result ) ;
static void print_table_data_vertically ( MYSQL_RES * result ) ;
static ulong start_timer ( void ) ;
static void end_timer ( ulong start_time , char * buff ) ;
static void mysql_end_timer ( ulong start_time , char * buff ) ;
static void nice_time ( double sec , char * buff , bool part_second ) ;
static sig_handler mysql_end ( int sig ) ;
int main ( int argc , char * argv [ ] )
{
char buff [ 80 ] ;
2005-01-03 19:21:54 +04:00
char * defaults , * extra_defaults ;
char * emb_argv [ 3 ] ;
int emb_argc = 1 ;
emb_argv [ 0 ] = argv [ 0 ] ;
get_defaults_files ( argc , argv , & defaults , & extra_defaults ) ;
if ( defaults )
emb_argv [ emb_argc + + ] = defaults ;
if ( extra_defaults )
emb_argv [ emb_argc + + ] = extra_defaults ;
2000-07-31 21:29:14 +02:00
MY_INIT ( argv [ 0 ] ) ;
DBUG_ENTER ( " main " ) ;
DBUG_PROCESS ( argv [ 0 ] ) ;
2003-04-23 03:29:03 +03:00
delimiter_str = delimiter ;
2002-04-03 13:36:01 +03:00
default_prompt = my_strdup ( getenv ( " MYSQL_PS1 " ) ?
getenv ( " MYSQL_PS1 " ) :
" mysql> " , MYF ( MY_WME ) ) ;
current_prompt = my_strdup ( default_prompt , MYF ( MY_WME ) ) ;
prompt_counter = 0 ;
2003-01-28 08:38:28 +02:00
outfile [ 0 ] = 0 ; // no (default) outfile
strmov ( pager , " stdout " ) ; // the default, if --pager wasn't given
2001-04-17 01:45:30 +03:00
{
char * tmp = getenv ( " PAGER " ) ;
2004-01-14 04:58:37 +02:00
if ( tmp & & strlen ( tmp ) )
{
default_pager_set = 1 ;
strmov ( default_pager , tmp ) ;
}
2001-04-17 01:45:30 +03:00
}
2000-07-31 21:29:14 +02:00
if ( ! isatty ( 0 ) | | ! isatty ( 1 ) )
{
status . batch = 1 ; opt_silent = 1 ;
ignore_errors = 0 ;
}
else
status . add_to_history = 1 ;
status . exit_status = 1 ;
load_defaults ( " my " , load_default_groups , & argc , & argv ) ;
2002-08-24 02:08:10 +03:00
defaults_argv = argv ;
2002-04-02 20:29:53 +03:00
if ( get_options ( argc , ( char * * ) argv ) )
2000-07-31 21:29:14 +02:00
{
2002-08-24 02:08:10 +03:00
free_defaults ( defaults_argv ) ;
2000-07-31 21:29:14 +02:00
my_end ( 0 ) ;
exit ( 1 ) ;
}
if ( status . batch & & ! status . line_buff & &
2004-05-26 21:40:27 +05:00
! ( status . line_buff = batch_readline_init ( opt_max_allowed_packet + 512 , stdin ) ) )
2002-08-24 02:08:10 +03:00
{
free_defaults ( defaults_argv ) ;
2004-05-21 17:23:27 +05:00
my_end ( 0 ) ;
2000-07-31 21:29:14 +02:00
exit ( 1 ) ;
2002-08-24 02:08:10 +03:00
}
2005-01-03 19:21:54 +04:00
if ( mysql_server_init ( emb_argc , emb_argv , ( char * * ) server_default_groups ) )
2004-05-17 12:07:58 +05:00
{
free_defaults ( defaults_argv ) ;
2004-05-21 17:23:27 +05:00
my_end ( 0 ) ;
2000-07-31 21:29:14 +02:00
exit ( 1 ) ;
2002-08-24 02:08:10 +03:00
}
2000-07-31 21:29:14 +02:00
glob_buffer . realloc ( 512 ) ;
2001-10-08 04:58:07 +03:00
completion_hash_init ( & ht , 128 ) ;
init_alloc_root ( & hash_mem_root , 16384 , 0 ) ;
2001-09-22 17:40:57 +03:00
bzero ( ( char * ) & mysql , sizeof ( mysql ) ) ;
2000-07-31 21:29:14 +02:00
if ( sql_connect ( current_host , current_db , current_user , opt_password ,
opt_silent ) )
{
2000-09-22 01:46:26 +03:00
quick = 1 ; // Avoid history
status . exit_status = 1 ;
mysql_end ( - 1 ) ;
2000-07-31 21:29:14 +02:00
}
if ( ! status . batch )
ignore_errors = 1 ; // Don't abort monitor
2004-10-07 22:47:11 +03:00
if ( opt_sigint_ignore )
signal ( SIGINT , SIG_IGN ) ;
else
signal ( SIGINT , mysql_end ) ; // Catch SIGINT to clean up
2002-03-30 17:50:20 +02:00
signal ( SIGQUIT , mysql_end ) ; // Catch SIGQUIT to clean up
2000-07-31 21:29:14 +02:00
2000-11-17 14:33:29 +02:00
/*
2002-06-11 11:20:31 +03:00
Run in interactive mode like the ingres / postgres monitor
2000-07-31 21:29:14 +02:00
*/
put_info ( " Welcome to the MySQL monitor. Commands end with ; or \\ g. " ,
INFO_INFO ) ;
sprintf ( ( char * ) glob_buffer . ptr ( ) ,
2003-12-12 17:18:48 -04:00
" Your MySQL connection id is %lu to server version: %s \n " ,
2000-07-31 21:29:14 +02:00
mysql_thread_id ( & mysql ) , mysql_get_server_info ( & mysql ) ) ;
put_info ( ( char * ) glob_buffer . ptr ( ) , INFO_INFO ) ;
# ifdef HAVE_READLINE
initialize_readline ( my_progname ) ;
2001-04-11 13:04:03 +02:00
if ( ! status . batch & & ! quick & & ! opt_html & & ! opt_xml )
2000-07-31 21:29:14 +02:00
{
2002-06-11 11:20:31 +03:00
/* read-history from file, default ~/.mysql_history*/
2000-07-31 21:29:14 +02:00
if ( getenv ( " MYSQL_HISTFILE " ) )
histfile = my_strdup ( getenv ( " MYSQL_HISTFILE " ) , MYF ( MY_WME ) ) ;
else if ( getenv ( " HOME " ) )
{
2000-08-22 00:18:32 +03:00
histfile = ( char * ) my_malloc ( ( uint ) strlen ( getenv ( " HOME " ) )
+ ( uint ) strlen ( " /.mysql_history " ) + 2 ,
2000-07-31 21:29:14 +02:00
MYF ( MY_WME ) ) ;
if ( histfile )
sprintf ( histfile , " %s/.mysql_history " , getenv ( " HOME " ) ) ;
}
if ( histfile )
{
if ( verbose )
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " Reading history-file %s \n " , histfile ) ;
2000-07-31 21:29:14 +02:00
read_history ( histfile ) ;
2005-01-25 18:32:09 +02:00
if ( ! ( histfile_tmp = ( char * ) my_malloc ( ( uint ) strlen ( histfile ) + 5 ,
MYF ( MY_WME ) ) ) )
{
fprintf ( stderr , " Couldn't allocate memory for temp histfile! \n " ) ;
exit ( 1 ) ;
}
sprintf ( histfile_tmp , " %s.TMP " , histfile ) ;
2000-07-31 21:29:14 +02:00
}
}
# endif
2003-01-06 01:48:59 +02:00
sprintf ( buff , " %s " ,
2004-04-05 16:28:16 +03:00
# ifndef NOT_YET
2003-01-06 01:48:59 +02:00
" Type 'help;' or ' \\ h' for help. Type ' \\ c' to clear the buffer. \n " ) ;
2004-04-05 16:28:16 +03:00
# else
2002-10-28 17:44:19 +04:00
" Type 'help [[%]function name[%]]' to get help on usage of function. \n " ) ;
2003-01-06 01:48:59 +02:00
# endif
2000-07-31 21:29:14 +02:00
put_info ( buff , INFO_INFO ) ;
status . exit_status = read_lines ( 1 ) ; // read lines and execute them
2000-11-17 14:33:29 +02:00
if ( opt_outfile )
end_tee ( ) ;
2000-07-31 21:29:14 +02:00
mysql_end ( 0 ) ;
# ifndef _lint
DBUG_RETURN ( 0 ) ; // Keep compiler happy
# endif
}
sig_handler mysql_end ( int sig )
{
2001-09-22 17:40:57 +03:00
mysql_close ( & mysql ) ;
2000-07-31 21:29:14 +02:00
# ifdef HAVE_READLINE
2001-04-11 13:04:03 +02:00
if ( ! status . batch & & ! quick & & ! opt_html & & ! opt_xml )
2000-07-31 21:29:14 +02:00
{
/* write-history */
if ( verbose )
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " Writing history-file %s \n " , histfile ) ;
2005-01-25 18:32:09 +02:00
if ( ! write_history ( histfile_tmp ) )
my_rename ( histfile_tmp , histfile , MYF ( MY_WME ) ) ;
2000-07-31 21:29:14 +02:00
}
batch_readline_end ( status . line_buff ) ;
completion_hash_free ( & ht ) ;
2001-10-08 04:58:07 +03:00
free_root ( & hash_mem_root , MYF ( 0 ) ) ;
2000-07-31 21:29:14 +02:00
# endif
2000-09-22 01:46:26 +03:00
if ( sig > = 0 )
put_info ( sig ? " Aborted " : " Bye " , INFO_RESULT ) ;
2000-07-31 21:29:14 +02:00
glob_buffer . free ( ) ;
old_buffer . free ( ) ;
2002-04-03 13:36:01 +03:00
processed_prompt . free ( ) ;
2000-07-31 21:29:14 +02:00
my_free ( opt_password , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
my_free ( opt_mysql_unix_port , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
my_free ( histfile , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2005-01-25 18:32:09 +02:00
my_free ( histfile_tmp , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2000-07-31 21:29:14 +02:00
my_free ( current_db , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
my_free ( current_host , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
my_free ( current_user , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2002-04-03 13:36:01 +03:00
my_free ( full_username , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
my_free ( part_username , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
my_free ( default_prompt , 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-03 13:36:01 +03:00
my_free ( current_prompt , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2001-10-02 05:53:00 +03:00
mysql_server_end ( ) ;
2002-08-24 02:08:10 +03:00
free_defaults ( defaults_argv ) ;
2000-07-31 21:29:14 +02:00
my_end ( info_flag ? MY_CHECK_ERROR | MY_GIVE_INFO : 0 ) ;
exit ( status . exit_status ) ;
}
2002-04-02 20:29:53 +03:00
static struct my_option my_long_options [ ] =
2000-07-31 21:29:14 +02:00
{
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 } ,
2004-01-14 04:58:37 +02:00
{ " help " , ' I ' , " Synonym for -? " , 0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 ,
0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " auto-rehash " , OPT_AUTO_REHASH ,
" Enable automatic rehashing. One doesn't need to use 'rehash' to get table and field completion, but startup and reconnecting may take a longer time. Disable with --disable-auto-rehash. " ,
2002-06-11 11:20:31 +03:00
( gptr * ) & rehash , ( gptr * ) & rehash , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " no-auto-rehash " , ' A ' ,
2002-06-01 11:46:06 +03:00
" No automatic rehashing. One has to use 'rehash' to get table and field completion. This gives a quicker start of mysql and disables rehashing on reconnect. WARNING: options deprecated; use --disable-auto-rehash instead. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " batch " , ' B ' ,
2004-05-05 22:47:09 +02:00
" Don't use history file. Disable interactive behavior. (Enables --silent) " , 0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " 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 } ,
2002-04-18 13:53:28 +03: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
{ " compress " , ' C ' , " Use compression in server/client protocol. " ,
( gptr * ) & opt_compress , ( gptr * ) & opt_compress , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 ,
0 , 0 , 0 } ,
2004-01-14 04:58:37 +02:00
# ifdef DBUG_OFF
{ " debug " , ' # ' , " This is a non-debug version. Catch this and exit " ,
2004-01-14 16:42:08 +01:00
0 , 0 , 0 , GET_DISABLED , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2004-01-14 04:58:37 +02:00
# else
{ " debug " , ' # ' , " Output debug log " , ( gptr * ) & default_dbug_option ,
2002-04-02 20:29:53 +03:00
( gptr * ) & default_dbug_option , 0 , GET_STR , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2002-05-14 21:41:55 +03:00
{ " database " , ' D ' , " Database to use. " , ( gptr * ) & current_db ,
2002-05-15 20:24:00 +03:00
( gptr * ) & current_db , 0 , GET_STR_ALLOC , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-04-23 03:29:03 +03:00
{ " delimiter " , OPT_DELIMITER , " Delimiter to be used. " , ( gptr * ) & delimiter_str ,
( gptr * ) & delimiter_str , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2004-05-05 22:47:09 +02:00
{ " execute " , ' e ' , " Execute command and quit. (Disables --force and history file) " , 0 ,
2002-04-02 20:29:53 +03:00
0 , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " vertical " , ' E ' , " Print the output of a query (rows) vertically. " ,
( gptr * ) & vertical , ( gptr * ) & vertical , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 ,
0 } ,
{ " force " , ' f ' , " Continue even if we get an sql error. " ,
( gptr * ) & ignore_errors , ( gptr * ) & ignore_errors , 0 , GET_BOOL , NO_ARG , 0 , 0 ,
0 , 0 , 0 , 0 } ,
2002-05-24 14:06:58 +03:00
{ " no-named-commands " , ' g ' ,
2002-06-01 11:46:06 +03:00
" Named commands are disabled. Use \\ * form only, or use named commands only in the beginning of a line ending with a semicolon (;) Since version 10.9 the client now starts with this option ENABLED by default! Disable with '-G'. Long format commands still work from the first line. WARNING: option deprecated; use --disable-named-commands instead. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " named-commands " , ' G ' ,
2002-05-24 14:06:58 +03:00
" Enable named commands. Named commands mean this program's internal commands; see mysql> help . When enabled, the named commands can be used from any line of the query, otherwise only from the first line, before an enter. Disable with --disable-named-commands. This option is disabled by default. " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & named_cmds , ( gptr * ) & named_cmds , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 ,
0 , 0 } ,
2002-09-30 20:35:21 +03:00
{ " ignore-spaces " , ' i ' , " Ignore space after function names. " , 0 , 0 , 0 ,
2002-04-02 20:29:53 +03:00
GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-03 13:36:01 +03:00
{ " local-infile " , OPT_LOCAL_INFILE , " Enable/disable LOAD DATA LOCAL INFILE. " ,
2002-09-22 10:35:15 +03:00
( gptr * ) & opt_local_infile ,
( gptr * ) & opt_local_infile , 0 , GET_BOOL , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-05-14 21:41:55 +03:00
{ " no-beep " , ' b ' , " Turn off beep on error. " , ( gptr * ) & opt_nobeep ,
( gptr * ) & opt_nobeep , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-05-11 14:36:34 +03:00
{ " host " , ' h ' , " Connect to host. " , ( gptr * ) & current_host ,
2002-05-15 20:24:00 +03:00
( gptr * ) & current_host , 0 , GET_STR_ALLOC , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " html " , ' H ' , " Produce HTML output. " , ( gptr * ) & opt_html , ( gptr * ) & opt_html ,
0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " xml " , ' X ' , " Produce XML output " , ( gptr * ) & opt_xml , ( gptr * ) & opt_xml , 0 ,
GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " line-numbers " , OPT_LINE_NUMBERS , " Write line numbers for errors. " ,
( gptr * ) & line_numbers , ( gptr * ) & line_numbers , 0 , GET_BOOL ,
2002-06-11 11:20:31 +03:00
NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
2002-06-01 11:46:06 +03:00
{ " skip-line-numbers " , ' L ' , " Don't write line number for errors. WARNING: -L is deprecated, use long version of this option instead. " , 0 , 0 , 0 , GET_NO_ARG ,
2002-04-02 20:29:53 +03:00
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2002-04-02 20:29:53 +03:00
{ " no-pager " , OPT_NOPAGER ,
2002-06-01 11:46:06 +03:00
" Disable pager and print to stdout. See interactive help ( \\ h) also. WARNING: option deprecated; use --disable-pager instead. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# endif
2002-06-01 11:46:06 +03:00
{ " no-tee " , OPT_NOTEE , " Disable outfile. See interactive help ( \\ h) also. WARNING: option deprecated; use --disable-tee instead " , 0 , 0 , 0 , GET_NO_ARG ,
2002-04-02 20:29:53 +03:00
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " unbuffered " , ' n ' , " Flush buffer after each query. " , ( gptr * ) & unbuffered ,
( gptr * ) & unbuffered , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " column-names " , OPT_COLUMN_NAMES , " Write column names in results. " ,
( gptr * ) & column_names , ( gptr * ) & column_names , 0 , GET_BOOL ,
2002-05-28 14:47:51 +03:00
NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " skip-column-names " , ' N ' ,
2002-06-01 11:46:06 +03:00
" Don't write column names in results. WARNING: -N is deprecated, use long version of this options instead. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_NO_ARG , 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 } ,
2004-10-07 22:47:11 +03:00
{ " sigint-ignore " , OPT_SIGINT_IGNORE , " Ignore SIGINT (CTRL-C) " ,
( gptr * ) & opt_sigint_ignore , ( gptr * ) & opt_sigint_ignore , 0 , GET_BOOL ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " one-database " , ' o ' ,
" Only update the default database. This is useful for skipping updates to other database in the update log. " ,
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2002-04-02 20:29:53 +03:00
{ " pager " , OPT_PAGER ,
" Pager to use to display results. If you don't supply an option the default pager is taken from your ENV variable PAGER. Valid pagers are less, more, cat [> filename], etc. See interactive help ( \\ h) also. This option does not work in batch mode. " ,
0 , 0 , 0 , GET_STR , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-11-17 14:33:29 +02:00
# endif
2002-04-02 20:29:53 +03:00
{ " 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 * ) & opt_mysql_port ,
( gptr * ) & opt_mysql_port , 0 , GET_UINT , REQUIRED_ARG , MYSQL_PORT , 0 , 0 , 0 , 0 ,
0 } ,
2002-05-14 21:41:55 +03:00
{ " prompt " , OPT_PROMPT , " Set the mysql prompt to this value. " ,
2002-05-15 20:24:00 +03:00
( gptr * ) & current_prompt , ( gptr * ) & current_prompt , 0 , GET_STR_ALLOC ,
2002-05-14 21:41:55 +03:00
REQUIRED_ARG , 0 , 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
{ " quick " , ' q ' ,
2003-06-13 10:59:02 +02:00
" Don't cache result, print it row by row. This may slow down the server if the output is suspended. Doesn't use history file. " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & quick , ( gptr * ) & quick , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " raw " , ' r ' , " Write fields without conversion. Used with --batch. " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & opt_raw_data , ( gptr * ) & opt_raw_data , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 ,
0 , 0 , 0 } ,
2003-03-01 23:34:34 +01:00
{ " reconnect " , OPT_RECONNECT , " Reconnect if the connection is lost. Disable with --disable-reconnect. This option is enabled by default. " ,
( gptr * ) & opt_reconnect , ( gptr * ) & opt_reconnect , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
2004-05-05 22:47:09 +02:00
{ " silent " , ' s ' , " Be more silent. Print results with a tab as separator, each row on new line. " , 0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 ,
2002-04-02 20:29:53 +03:00
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
2002-04-02 20:29:53 +03:00
{ " socket " , ' S ' , " Socket file to use for connection. " ,
2002-05-15 20:24:00 +03:00
( gptr * ) & opt_mysql_unix_port , ( gptr * ) & opt_mysql_unix_port , 0 , GET_STR_ALLOC ,
2002-05-14 21:41:55 +03:00
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# include "sslopt-longopts.h"
2002-04-02 20:29:53 +03:00
{ " table " , ' t ' , " Output in table format. " , ( gptr * ) & output_tables ,
( gptr * ) & output_tables , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " debug-info " , ' T ' , " Print some debug info at exit. " , ( gptr * ) & info_flag ,
( gptr * ) & info_flag , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " tee " , OPT_TEE ,
" Append everything into outfile. See interactive help ( \\ h) also. Does not work in batch mode. " ,
0 , 0 , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
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 * ) & current_user ,
2002-05-15 20:24:00 +03:00
( gptr * ) & current_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
{ " safe-updates " , ' U ' , " Only allow UPDATE and DELETE that uses keys. " ,
2004-03-15 19:47:51 +04:00
( gptr * ) & safe_updates , ( gptr * ) & safe_updates , 0 , GET_BOOL , NO_ARG , 0 , 0 ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , 0 } ,
2002-10-04 17:18:41 +03:00
{ " i-am-a-dummy " , ' U ' , " Synonym for option --safe-updates, -U. " ,
2004-03-15 19:47:51 +04:00
( gptr * ) & safe_updates , ( gptr * ) & safe_updates , 0 , GET_BOOL , NO_ARG , 0 , 0 ,
2002-10-04 17:18:41 +03:00
0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " verbose " , ' v ' , " Write more. (-v -v -v gives the table output format). " , 0 ,
2002-04-02 20:29:53 +03:00
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 } ,
{ " wait " , ' w ' , " Wait and retry if connection is down. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2004-01-14 04:58:37 +02:00
{ " connect_timeout " , OPT_CONNECT_TIMEOUT ,
" Number of seconds before connection timeout. " ,
( gptr * ) & opt_connect_timeout ,
2002-05-11 14:36:34 +03:00
( gptr * ) & opt_connect_timeout , 0 , GET_ULONG , REQUIRED_ARG , 0 , 0 , 3600 * 12 , 0 ,
2002-04-02 20:29:53 +03:00
0 , 1 } ,
2004-01-14 04:58:37 +02:00
{ " max_allowed_packet " , OPT_MAX_ALLOWED_PACKET ,
" Max packet length to send to, or receive from server " ,
2004-05-26 21:40:27 +05:00
( gptr * ) & opt_max_allowed_packet , ( gptr * ) & opt_max_allowed_packet , 0 , GET_ULONG ,
2003-12-14 06:39:52 +02:00
REQUIRED_ARG , 16 * 1024L * 1024L , 4096 , ( longlong ) 2 * 1024L * 1024L * 1024L ,
MALLOC_OVERHEAD , 1024 , 0 } ,
2004-01-14 04:58:37 +02:00
{ " net_buffer_length " , OPT_NET_BUFFER_LENGTH ,
" Buffer for TCP/IP and socket communication " ,
2004-05-26 21:40:27 +05:00
( gptr * ) & opt_net_buffer_length , ( gptr * ) & opt_net_buffer_length , 0 , GET_ULONG ,
2002-04-02 20:29:53 +03:00
REQUIRED_ARG , 16384 , 1024 , 512 * 1024 * 1024L , MALLOC_OVERHEAD , 1024 , 0 } ,
2004-01-14 04:58:37 +02:00
{ " select_limit " , OPT_SELECT_LIMIT ,
" Automatic limit for SELECT when using --safe-updates " ,
( gptr * ) & select_limit ,
2002-05-11 14:36:34 +03:00
( gptr * ) & select_limit , 0 , GET_ULONG , REQUIRED_ARG , 1000L , 1 , ~ 0L , 0 , 1 , 0 } ,
2004-01-14 04:58:37 +02:00
{ " max_join_size " , OPT_MAX_JOIN_SIZE ,
" Automatic limit for rows in a join when using --safe-updates " ,
( gptr * ) & max_join_size ,
2002-05-11 14:36:34 +03:00
( gptr * ) & max_join_size , 0 , GET_ULONG , REQUIRED_ARG , 1000000L , 1 , ~ 0L , 0 , 1 ,
2002-04-02 20:29:53 +03:00
0 } ,
2003-11-28 13:11:44 +03:00
{ " secure-auth " , OPT_SECURE_AUTH , " Refuse client connecting to server if it "
" uses old (pre-4.1.1) protocol " , ( gptr * ) & opt_secure_auth ,
( gptr * ) & opt_secure_auth , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
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
} ;
static void usage ( int version )
{
2004-05-25 22:00:14 +03:00
/* Divert all help information on NetWare to logger screen. */
# ifdef __NETWARE__
# define printf consoleprintf
# endif
2000-11-18 17:34:15 +02:00
printf ( " %s Ver %s Distrib %s, for %s (%s) \n " ,
my_progname , VER , MYSQL_SERVER_VERSION , SYSTEM_TYPE , MACHINE_TYPE ) ;
2000-07-31 21:29:14 +02:00
if ( version )
return ;
2002-04-03 13:36:01 +03:00
printf ( " \
2002-06-11 11:20:31 +03:00
Copyright ( C ) 2002 MySQL AB \ n \
2002-04-03 13:36:01 +03:00
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 ( " Usage: %s [OPTIONS] [database] \n " , my_progname ) ;
2002-04-02 20:29:53 +03:00
my_print_help ( my_long_options ) ;
2002-04-03 15:44:20 +03:00
print_defaults ( " my " , load_default_groups ) ;
2002-04-02 20:29:53 +03:00
my_print_variables ( my_long_options ) ;
2004-05-25 22:00:14 +03:00
NETWARE_SET_SCREEN_MODE ( 1 ) ;
# ifdef __NETWARE__
# undef printf
# endif
2000-07-31 21:29:14 +02:00
}
2004-05-25 22:00:14 +03:00
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 ) {
2002-05-31 15:43:21 +03:00
case OPT_CHARSETS_DIR :
strmov ( mysql_charsets_dir , argument ) ;
charsets_dir = mysql_charsets_dir ;
break ;
2003-08-19 00:08:08 +03:00
case OPT_DEFAULT_CHARSET :
default_charset_used = 1 ;
break ;
2003-04-23 03:29:03 +03:00
case OPT_DELIMITER :
if ( argument = = disabled_my_option )
strmov ( delimiter , DEFAULT_DELIMITER ) ;
else
strmake ( delimiter , argument , sizeof ( delimiter ) - 1 ) ;
delimiter_length = strlen ( delimiter ) ;
delimiter_str = delimiter ;
break ;
2002-05-31 15:43:21 +03:00
case OPT_LOCAL_INFILE :
using_opt_local_infile = 1 ;
break ;
case OPT_TEE :
if ( argument = = disabled_my_option )
{
2000-11-17 14:33:29 +02:00
if ( opt_outfile )
end_tee ( ) ;
2002-05-31 15:43:21 +03:00
}
else
2003-01-28 08:38:28 +02:00
init_tee ( argument ) ;
2002-05-31 15:43:21 +03:00
break ;
case OPT_NOTEE :
2002-06-01 11:46:06 +03:00
printf ( " WARNING: option deprecated; use --disable-tee instead. \n " ) ;
2002-05-31 15:43:21 +03:00
if ( opt_outfile )
end_tee ( ) ;
break ;
case OPT_PAGER :
2002-10-15 23:34:55 +03:00
if ( argument = = disabled_my_option )
opt_nopager = 1 ;
2002-05-31 15:43:21 +03:00
else
2002-10-15 23:34:55 +03:00
{
opt_nopager = 0 ;
2004-01-14 04:58:37 +02:00
if ( argument & & strlen ( argument ) )
{
default_pager_set = 1 ;
2002-10-15 23:34:55 +03:00
strmov ( pager , argument ) ;
2004-01-14 04:58:37 +02:00
strmov ( default_pager , pager ) ;
}
else if ( default_pager_set )
2002-10-15 23:34:55 +03:00
strmov ( pager , default_pager ) ;
2004-01-14 04:58:37 +02:00
else
opt_nopager = 1 ;
2002-10-15 23:34:55 +03:00
}
2002-05-31 15:43:21 +03:00
break ;
case OPT_NOPAGER :
2002-06-01 11:46:06 +03:00
printf ( " WARNING: option deprecated; use --disable-pager instead. \n " ) ;
2002-05-31 15:43:21 +03:00
opt_nopager = 1 ;
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 ;
}
2003-12-01 13:13:16 +02:00
break ;
2002-05-31 15:43:21 +03:00
case ' A ' :
rehash = 0 ;
break ;
case ' N ' :
column_names = 0 ;
break ;
case ' e ' :
status . batch = 1 ;
status . add_to_history = 0 ;
2004-05-05 22:47:09 +02:00
if ( ! status . line_buff )
ignore_errors = 0 ; // do it for the first -e only
if ( ! ( status . line_buff = batch_readline_command ( status . line_buff , argument ) ) )
2002-05-31 15:43:21 +03:00
return 1 ;
break ;
case ' o ' :
if ( argument = = disabled_my_option )
one_database = 0 ;
else
one_database = skip_updates = 1 ;
break ;
case ' p ' :
if ( argument = = disabled_my_option )
argument = ( char * ) " " ; // Don't require password
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 ;
2004-08-26 16:15:36 +03:00
tty_password = 0 ;
2002-05-31 15:43:21 +03:00
}
else
tty_password = 1 ;
break ;
case ' # ' :
DBUG_PUSH ( argument ? argument : default_dbug_option ) ;
info_flag = 1 ;
break ;
case ' s ' :
if ( argument = = disabled_my_option )
opt_silent = 0 ;
else
opt_silent + + ;
break ;
case ' v ' :
if ( argument = = disabled_my_option )
verbose = 0 ;
else
verbose + + ;
break ;
case ' B ' :
2004-05-05 22:47:09 +02:00
status . batch = 1 ;
status . add_to_history = 0 ;
2004-05-06 15:13:23 +02:00
set_if_bigger ( opt_silent , 1 ) ; // more silent
2002-05-31 15:43:21 +03:00
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-05-31 15:43:21 +03:00
break ;
2002-09-20 14:05:18 +03:00
# include <sslopt-case.h>
2002-05-31 15:43:21 +03:00
case ' V ' :
usage ( 1 ) ;
exit ( 0 ) ;
case ' I ' :
case ' ? ' :
usage ( 0 ) ;
exit ( 0 ) ;
2000-07-31 21:29:14 +02:00
}
2002-04-02 20:29:53 +03:00
return 0 ;
}
static int get_options ( int argc , char * * argv )
{
char * tmp , * pagpoint ;
int ho_error ;
2004-05-26 21:40:27 +05:00
MYSQL_PARAMETERS * mysql_params = mysql_get_parameters ( ) ;
2002-04-02 20:29:53 +03:00
tmp = ( char * ) getenv ( " MYSQL_HOST " ) ;
if ( tmp )
current_host = my_strdup ( tmp , MYF ( MY_WME ) ) ;
pagpoint = getenv ( " PAGER " ) ;
if ( ! ( ( char * ) ( pagpoint ) ) )
{
strmov ( pager , " stdout " ) ;
opt_nopager = 1 ;
}
else
strmov ( pager , pagpoint ) ;
strmov ( default_pager , pager ) ;
2004-05-26 21:40:27 +05:00
opt_max_allowed_packet = * mysql_params - > p_max_allowed_packet ;
opt_net_buffer_length = * mysql_params - > p_net_buffer_length ;
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
2004-05-26 21:40:27 +05:00
* mysql_params - > p_max_allowed_packet = opt_max_allowed_packet ;
* mysql_params - > p_net_buffer_length = opt_net_buffer_length ;
2000-11-17 14:33:29 +02:00
if ( status . batch ) /* disable pager and outfile in this case */
{
strmov ( default_pager , " stdout " ) ;
strmov ( pager , " stdout " ) ;
2002-04-02 20:29:53 +03:00
opt_nopager = 1 ;
2004-01-14 04:58:37 +02:00
default_pager_set = 0 ;
2002-04-02 20:29:53 +03:00
opt_outfile = 0 ;
2003-03-01 23:34:34 +01:00
opt_reconnect = 0 ;
2002-08-06 22:51:43 +03:00
connect_flag = 0 ; /* Not in interactive mode */
2000-11-17 14:33:29 +02:00
}
2004-02-11 15:15:39 +04:00
if ( strcmp ( default_charset , charset_info - > csname ) & &
! ( charset_info = get_charset_by_csname ( default_charset ,
2003-05-22 14:37:01 +05:00
MY_CS_PRIMARY , MYF ( MY_WME ) ) ) )
2003-03-16 11:20:45 +04:00
exit ( 1 ) ;
2000-07-31 21:29:14 +02:00
if ( argc > 1 )
{
usage ( 0 ) ;
exit ( 1 ) ;
}
if ( argc = = 1 )
{
2003-03-17 15:50:56 +02:00
skip_updates = 0 ;
2002-04-02 20:29:53 +03:00
my_free ( current_db , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
current_db = my_strdup ( * argv , MYF ( MY_WME ) ) ;
2000-07-31 21:29:14 +02:00
}
if ( tty_password )
2002-04-02 20:29:53 +03:00
opt_password = get_tty_password ( NullS ) ;
2000-07-31 21:29:14 +02:00
return ( 0 ) ;
}
static int read_lines ( bool execute_commands )
{
2003-01-28 08:38:28 +02:00
# if defined( __WIN__) || defined(OS2) || defined(__NETWARE__)
2000-07-31 21:29:14 +02:00
char linebuffer [ 254 ] ;
# endif
char * line ;
char in_string = 0 ;
ulong line_number = 0 ;
2002-12-24 00:53:07 +02:00
bool ml_comment = 0 ;
2000-07-31 21:29:14 +02:00
COMMANDS * com ;
status . exit_status = 1 ;
2002-12-24 00:53:07 +02:00
2000-07-31 21:29:14 +02:00
for ( ; ; )
{
if ( status . batch | | ! execute_commands )
{
line = batch_readline ( status . line_buff ) ;
line_number + + ;
if ( ! glob_buffer . length ( ) )
status . query_start_line = line_number ;
}
else
2000-11-17 14:33:29 +02:00
{
2003-01-28 08:38:28 +02:00
char * prompt = ( char * ) ( glob_buffer . is_empty ( ) ? construct_prompt ( ) :
! in_string ? " -> " :
in_string = = ' \' ' ?
2003-09-17 21:41:36 +03:00
" '> " : ( in_string = = ' ` ' ?
" `> " :
" \" > " ) ) ;
2000-11-22 03:45:02 +02:00
if ( opt_outfile & & glob_buffer . is_empty ( ) )
fflush ( OUTFILE ) ;
2003-01-28 08:38:28 +02:00
# if defined( __WIN__) || defined(OS2) || defined(__NETWARE__)
tee_fputs ( prompt , stdout ) ;
# ifdef __NETWARE__
line = fgets ( linebuffer , sizeof ( linebuffer ) - 1 , stdin ) ;
/* Remove the '\n' */
2004-05-25 22:00:14 +03:00
if ( line )
2000-11-17 21:36:47 +02:00
{
2003-01-28 08:38:28 +02:00
char * p = strrchr ( line , ' \n ' ) ;
2003-03-21 16:45:39 -05:00
if ( p ! = NULL )
* p = ' \0 ' ;
2000-11-17 21:36:47 +02:00
}
2003-01-28 08:38:28 +02:00
# else
linebuffer [ 0 ] = ( char ) sizeof ( linebuffer ) ;
line = _cgets ( linebuffer ) ;
# endif /* __NETWARE__ */
# else
if ( opt_outfile )
fputs ( prompt , OUTFILE ) ;
line = readline ( prompt ) ;
# endif /* defined( __WIN__) || defined(OS2) || defined(__NETWARE__) */
2004-05-25 22:00:14 +03:00
/*
When Ctrl + d or Ctrl + z is pressed , the line may be NULL on some OS
which may cause coredump .
*/
if ( opt_outfile & & line )
2000-11-17 21:36:47 +02:00
fprintf ( OUTFILE , " %s \n " , line ) ;
2000-11-20 19:50:41 +02:00
}
2000-07-31 21:29:14 +02:00
if ( ! line ) // End of file
{
status . exit_status = 0 ;
break ;
}
if ( ! in_string & & ( line [ 0 ] = = ' # ' | |
( line [ 0 ] = = ' - ' & & line [ 1 ] = = ' - ' ) | |
line [ 0 ] = = 0 ) )
2000-08-29 19:38:32 +03:00
continue ; // Skip comment lines
2000-07-31 21:29:14 +02:00
2002-06-11 11:20:31 +03:00
/*
Check if line is a mysql command line
( We want to allow help , print and clear anywhere at line start
*/
2002-04-02 20:29:53 +03:00
if ( execute_commands & & ( named_cmds | | glob_buffer . is_empty ( ) )
2000-08-29 19:38:32 +03:00
& & ! in_string & & ( com = find_command ( line , 0 ) ) )
2000-07-31 21:29:14 +02:00
{
if ( ( * com - > func ) ( & glob_buffer , line ) > 0 )
break ;
if ( glob_buffer . is_empty ( ) ) // If buffer was emptied
in_string = 0 ;
# ifdef HAVE_READLINE
2004-03-03 15:54:37 -05:00
if ( status . add_to_history & & not_in_history ( line ) )
2000-07-31 21:29:14 +02:00
add_history ( line ) ;
# endif
continue ;
}
2002-12-24 00:53:07 +02:00
if ( add_line ( glob_buffer , line , & in_string , & ml_comment ) )
2000-07-31 21:29:14 +02:00
break ;
}
/* if in batch mode, send last query even if it doesn't end with \g or go */
if ( ( status . batch | | ! execute_commands ) & & ! status . exit_status )
{
remove_cntrl ( glob_buffer ) ;
if ( ! glob_buffer . is_empty ( ) )
{
status . exit_status = 1 ;
if ( com_go ( & glob_buffer , line ) < = 0 )
status . exit_status = 0 ;
}
}
return status . exit_status ;
}
static COMMANDS * find_command ( char * name , char cmd_char )
{
uint len ;
char * end ;
if ( ! name )
{
len = 0 ;
end = 0 ;
}
else
{
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * name ) )
2000-07-31 21:29:14 +02:00
name + + ;
2003-04-23 03:29:03 +03:00
if ( strstr ( name , delimiter ) | | strstr ( name , " \\ g " ) )
2000-07-31 21:29:14 +02:00
return ( ( COMMANDS * ) 0 ) ;
if ( ( end = strcont ( name , " \t " ) ) )
{
len = ( uint ) ( end - name ) ;
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * end ) )
2000-07-31 21:29:14 +02:00
end + + ;
if ( ! * end )
end = 0 ; // no arguments to function
}
else
2000-08-22 00:18:32 +03:00
len = ( uint ) strlen ( name ) ;
2000-07-31 21:29:14 +02:00
}
for ( uint i = 0 ; commands [ i ] . name ; i + + )
{
if ( commands [ i ] . func & &
2002-03-12 21:37:58 +04:00
( ( name & &
2003-04-01 14:17:28 +05:00
! my_strnncoll ( charset_info , ( uchar * ) name , len ,
( uchar * ) commands [ i ] . name , len ) & &
2000-07-31 21:29:14 +02:00
! commands [ i ] . name [ len ] & &
( ! end | | ( end & & commands [ i ] . takes_params ) ) ) | |
! name & & commands [ i ] . cmd_char = = cmd_char ) )
return ( & commands [ i ] ) ;
}
return ( ( COMMANDS * ) 0 ) ;
}
2002-12-24 00:53:07 +02:00
static bool add_line ( String & buffer , char * line , char * in_string ,
bool * ml_comment )
2000-07-31 21:29:14 +02:00
{
uchar inchar ;
2003-04-23 03:29:03 +03:00
char buff [ 80 ] , * pos , * out ;
2000-07-31 21:29:14 +02:00
COMMANDS * com ;
if ( ! line [ 0 ] & & buffer . is_empty ( ) )
return 0 ;
# ifdef HAVE_READLINE
2004-03-03 15:54:37 -05:00
if ( status . add_to_history & & line [ 0 ] & & not_in_history ( line ) )
2000-07-31 21:29:14 +02:00
add_history ( line ) ;
# endif
# ifdef USE_MB
2000-08-22 00:18:32 +03:00
char * strend = line + ( uint ) strlen ( line ) ;
2000-07-31 21:29:14 +02:00
# endif
for ( pos = out = line ; ( inchar = ( uchar ) * pos ) ; pos + + )
{
2003-03-16 11:20:45 +04:00
if ( my_isspace ( charset_info , inchar ) & & out = = line & &
2002-03-12 21:37:58 +04:00
buffer . is_empty ( ) )
2000-07-31 21:29:14 +02:00
continue ;
# ifdef USE_MB
int l ;
2003-03-16 11:20:45 +04:00
if ( use_mb ( charset_info ) & &
( l = my_ismbchar ( charset_info , pos , strend ) ) ) {
2000-07-31 21:29:14 +02:00
while ( l - - )
* out + + = * pos + + ;
pos - - ;
continue ;
}
# endif
2002-12-24 00:53:07 +02:00
if ( ! * ml_comment & & inchar = = ' \\ ' )
2003-04-23 03:29:03 +03:00
{
// Found possbile one character command like \c
2000-07-31 21:29:14 +02:00
if ( ! ( inchar = ( uchar ) * + + pos ) )
break ; // readline adds one '\'
2001-10-02 21:08:08 +03:00
if ( * in_string | | inchar = = ' N ' ) // \N is short for NULL
2000-07-31 21:29:14 +02:00
{ // Don't allow commands in string
* out + + = ' \\ ' ;
* out + + = ( char ) inchar ;
continue ;
}
if ( ( com = find_command ( NullS , ( char ) inchar ) ) )
{
2003-03-16 11:20:45 +04:00
const String tmp ( line , ( uint ) ( out - line ) , charset_info ) ;
2000-07-31 21:29:14 +02:00
buffer . append ( tmp ) ;
if ( ( * com - > func ) ( & buffer , pos - 1 ) > 0 )
return 1 ; // Quit
if ( com - > takes_params )
{
2003-04-23 03:29:03 +03:00
for ( pos + + ;
* pos & & ( * pos ! = * delimiter | |
! is_prefix ( pos + 1 , delimiter + 1 ) ) ; pos + + )
; // Remove parameters
2000-07-31 21:29:14 +02:00
if ( ! * pos )
pos - - ;
2003-04-23 03:29:03 +03:00
else
pos + = delimiter_length - 1 ; // Point at last delim char
2000-07-31 21:29:14 +02:00
}
out = line ;
}
else
{
sprintf ( buff , " Unknown command ' \\ %c'. " , inchar ) ;
if ( put_info ( buff , INFO_ERROR ) > 0 )
return 1 ;
* out + + = ' \\ ' ;
* out + + = ( char ) inchar ;
continue ;
}
}
2003-04-23 03:29:03 +03:00
else if ( ! * ml_comment & & ( * pos = = * delimiter & &
is_prefix ( pos + 1 , delimiter + 1 ) ) & &
! * in_string )
{
uint old_delimiter_length = delimiter_length ;
2000-07-31 21:29:14 +02:00
if ( out ! = line )
2003-04-23 03:29:03 +03:00
buffer . append ( line , ( uint ) ( out - line ) ) ; // Add this line
if ( ( com = find_command ( buffer . c_ptr ( ) , 0 ) ) )
2000-07-31 21:29:14 +02:00
{
2003-04-23 03:29:03 +03:00
if ( ( * com - > func ) ( & buffer , buffer . c_ptr ( ) ) > 0 )
2000-07-31 21:29:14 +02:00
return 1 ; // Quit
}
else
{
2004-05-05 22:47:09 +02:00
if ( com_go ( & buffer , 0 ) > 0 ) // < 0 is not fatal
return 1 ;
2000-07-31 21:29:14 +02:00
}
buffer . length ( 0 ) ;
2003-04-23 03:29:03 +03:00
out = line ;
pos + = old_delimiter_length - 1 ;
2000-07-31 21:29:14 +02:00
}
2002-12-24 00:53:07 +02:00
else if ( ! * ml_comment & & ( ! * in_string & & ( inchar = = ' # ' | |
inchar = = ' - ' & & pos [ 1 ] = = ' - ' & &
2003-03-16 11:20:45 +04:00
my_isspace ( charset_info , pos [ 2 ] ) ) ) )
2000-07-31 21:29:14 +02:00
break ; // comment to end of line
2003-04-23 03:29:03 +03:00
else if ( ! * in_string & & inchar = = ' / ' & & * ( pos + 1 ) = = ' * ' & &
* ( pos + 2 ) ! = ' ! ' )
2002-12-24 00:53:07 +02:00
{
pos + + ;
* ml_comment = 1 ;
if ( out ! = line )
{
buffer . append ( line , ( uint ) ( out - line ) ) ;
out = line ;
}
}
2003-07-15 14:38:48 +05:00
else if ( * ml_comment & & inchar = = ' * ' & & * ( pos + 1 ) = = ' / ' )
2002-12-24 00:53:07 +02:00
{
pos + + ;
* ml_comment = 0 ;
}
2000-07-31 21:29:14 +02:00
else
{ // Add found char to buffer
if ( inchar = = * in_string )
2003-07-15 14:38:48 +05:00
* in_string = 0 ;
2003-07-15 17:38:01 +05:00
else if ( ! * ml_comment & & ! * in_string & &
( inchar = = ' \' ' | | inchar = = ' " ' | | inchar = = ' ` ' ) )
* in_string = ( char ) inchar ;
if ( ! * ml_comment )
2003-07-15 14:38:48 +05:00
* out + + = ( char ) inchar ;
2000-07-31 21:29:14 +02:00
}
}
if ( out ! = line | | ! buffer . is_empty ( ) )
{
* out + + = ' \n ' ;
uint length = ( uint ) ( out - line ) ;
if ( buffer . length ( ) + length > = buffer . alloced_length ( ) )
buffer . realloc ( buffer . length ( ) + length + IO_SIZE ) ;
2002-12-24 00:53:07 +02:00
if ( ! ( * ml_comment ) & & buffer . append ( line , length ) )
2000-07-31 21:29:14 +02:00
return 1 ;
}
return 0 ;
}
2002-06-11 11:20:31 +03:00
/*****************************************************************
Interface to Readline Completion
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-07-31 21:29:14 +02:00
# ifdef HAVE_READLINE
2002-11-19 18:26:53 +04:00
static char * new_command_generator ( const char * text , int ) ;
static char * * new_mysql_completion ( const char * text , int start , int end ) ;
2000-07-31 21:29:14 +02:00
2002-06-11 11:20:31 +03:00
/*
Tell the GNU Readline library how to complete . We want to try to complete
on command names if this is the first word in the line , or on filenames
if not .
*/
2000-07-31 21:29:14 +02:00
2003-01-18 20:28:48 +04:00
# if defined(USE_NEW_READLINE_INTERFACE) || defined(USE_LIBEDIT_INTERFACE)
char * no_completion ( const char * , int )
# else
int no_completion ( )
# endif
2000-07-31 21:29:14 +02:00
{
return 0 ; /* No filename completion */
}
2004-03-03 15:54:37 -05:00
/* glues pieces of history back together if in pieces */
static void fix_history ( String * final_command )
{
int total_lines = 1 ;
char * ptr = final_command - > c_ptr ( ) ;
String fixed_buffer ; /* Converted buffer */
char str_char = ' \0 ' ; /* Character if we are in a string or not */
/* find out how many lines we have and remove newlines */
while ( * ptr ! = ' \0 ' )
{
switch ( * ptr ) {
/* string character */
case ' " ' :
case ' \' ' :
case ' ` ' :
if ( str_char = = ' \0 ' ) /* open string */
str_char = * ptr ;
else if ( str_char = = * ptr ) /* close string */
str_char = ' \0 ' ;
fixed_buffer . append ( ptr , 1 ) ;
break ;
case ' \n ' :
/*
not in string , change to space
if in string , leave it alone
*/
fixed_buffer . append ( str_char = = ' \0 ' ? " " : " \n " ) ;
total_lines + + ;
break ;
case ' \\ ' :
fixed_buffer . append ( ' \\ ' ) ;
/* need to see if the backslash is escaping anything */
if ( str_char )
{
ptr + + ;
/* special characters that need escaping */
if ( * ptr = = ' \' ' | | * ptr = = ' " ' | | * ptr = = ' \\ ' )
fixed_buffer . append ( ptr , 1 ) ;
else
ptr - - ;
}
break ;
default :
fixed_buffer . append ( ptr , 1 ) ;
}
ptr + + ;
}
if ( total_lines > 1 )
add_history ( fixed_buffer . ptr ( ) ) ;
}
/*
returns 0 if line matches the previous history entry
returns 1 if the line doesn ' t match the previous history entry
*/
static int not_in_history ( const char * line )
{
HIST_ENTRY * oldhist = history_get ( history_length ) ;
if ( oldhist = = 0 )
return 1 ;
if ( strcmp ( oldhist - > line , line ) = = 0 )
return 0 ;
return 1 ;
}
2000-07-31 21:29:14 +02:00
static void initialize_readline ( char * name )
{
/* Allow conditional parsing of the ~/.inputrc file. */
rl_readline_name = name ;
/* Tell the completer that we want a crack first. */
2003-01-18 20:28:48 +04:00
# if defined(USE_NEW_READLINE_INTERFACE)
rl_attempted_completion_function = ( rl_completion_func_t * ) & new_mysql_completion ;
rl_completion_entry_function = ( rl_compentry_func_t * ) & no_completion ;
# elif defined(USE_LIBEDIT_INTERFACE)
2003-04-06 20:14:36 -04:00
# ifdef HAVE_LOCALE_H
setlocale ( LC_ALL , " " ) ; /* so as libedit use isprint */
# endif
2003-01-18 20:28:48 +04:00
rl_attempted_completion_function = ( CPPFunction * ) & new_mysql_completion ;
rl_completion_entry_function = ( CPFunction * ) & no_completion ;
2002-11-26 14:03:25 +01:00
# else
2003-01-18 20:28:48 +04:00
rl_attempted_completion_function = ( CPPFunction * ) & new_mysql_completion ;
rl_completion_entry_function = ( Function * ) & no_completion ;
2002-11-26 14:03:25 +01:00
# endif
2000-07-31 21:29:14 +02:00
}
2002-06-11 11:20:31 +03:00
/*
Attempt to complete on the contents of TEXT . START and END show the
region of TEXT that contains the word to complete . We can use the
entire line in case we want to do some simple parsing . Return the
array of matches , or NULL if there aren ' t any .
*/
2000-07-31 21:29:14 +02:00
2002-11-19 18:26:53 +04:00
static char * * new_mysql_completion ( const char * text ,
2000-07-31 21:29:14 +02:00
int start __attribute__ ( ( unused ) ) ,
int end __attribute__ ( ( unused ) ) )
{
if ( ! status . batch & & ! quick )
2003-01-18 20:28:48 +04:00
# if defined(USE_NEW_READLINE_INTERFACE)
2002-11-19 18:26:53 +04:00
return rl_completion_matches ( text , new_command_generator ) ;
2002-11-26 14:03:25 +01:00
# else
return completion_matches ( ( char * ) text , ( CPFunction * ) new_command_generator ) ;
# endif
2000-07-31 21:29:14 +02:00
else
return ( char * * ) 0 ;
}
2002-11-19 18:26:53 +04:00
static char * new_command_generator ( const char * text , int state )
2000-07-31 21:29:14 +02:00
{
static int textlen ;
char * ptr ;
static Bucket * b ;
static entry * e ;
static uint i ;
2002-06-11 11:20:31 +03:00
if ( ! state )
2000-08-22 00:18:32 +03:00
textlen = ( uint ) strlen ( text ) ;
2000-07-31 21:29:14 +02:00
2002-06-11 11:20:31 +03:00
if ( textlen > 0 )
{ /* lookup in the hash */
if ( ! state )
{
2000-07-31 21:29:14 +02:00
uint len ;
2000-08-22 00:18:32 +03:00
b = find_all_matches ( & ht , text , ( uint ) strlen ( text ) , & len ) ;
2002-06-11 11:20:31 +03:00
if ( ! b )
2000-07-31 21:29:14 +02:00
return NullS ;
e = b - > pData ;
}
2002-06-11 11:20:31 +03:00
if ( e )
{
2000-07-31 21:29:14 +02:00
ptr = strdup ( e - > str ) ;
e = e - > pNext ;
return ptr ;
}
2002-06-11 11:20:31 +03:00
}
else
{ /* traverse the entire hash, ugly but works */
2000-07-31 21:29:14 +02:00
2002-06-11 11:20:31 +03:00
if ( ! state )
{
2000-07-31 21:29:14 +02:00
/* find the first used bucket */
2002-06-11 11:20:31 +03:00
for ( i = 0 ; i < ht . nTableSize ; i + + )
{
if ( ht . arBuckets [ i ] )
{
2000-07-31 21:29:14 +02:00
b = ht . arBuckets [ i ] ;
e = b - > pData ;
break ;
}
}
}
ptr = NullS ;
2002-06-11 11:20:31 +03:00
while ( e & & ! ptr )
{ /* find valid entry in bucket */
if ( ( uint ) strlen ( e - > str ) = = b - > nKeyLength )
2000-07-31 21:29:14 +02:00
ptr = strdup ( e - > str ) ;
/* find the next used entry */
e = e - > pNext ;
2002-06-11 11:20:31 +03:00
if ( ! e )
{ /* find the next used bucket */
2000-07-31 21:29:14 +02:00
b = b - > pNext ;
2002-06-11 11:20:31 +03:00
if ( ! b )
{
for ( i + + ; i < ht . nTableSize ; i + + )
{
if ( ht . arBuckets [ i ] )
{
2000-07-31 21:29:14 +02:00
b = ht . arBuckets [ i ] ;
e = b - > pData ;
break ;
}
}
}
2002-06-11 11:20:31 +03:00
else
e = b - > pData ;
2000-07-31 21:29:14 +02:00
}
}
2002-06-11 11:20:31 +03:00
if ( ptr )
2000-07-31 21:29:14 +02:00
return ptr ;
}
return NullS ;
}
/* Build up the completion hash */
2002-04-16 11:39:03 +03:00
static void build_completion_hash ( bool rehash , bool write_info )
2000-07-31 21:29:14 +02:00
{
COMMANDS * cmd = commands ;
2001-10-08 04:58:07 +03:00
MYSQL_RES * databases = 0 , * tables = 0 ;
MYSQL_RES * fields ;
2000-07-31 21:29:14 +02:00
static char * * * field_names = 0 ;
MYSQL_ROW database_row , table_row ;
MYSQL_FIELD * sql_field ;
char buf [ NAME_LEN * 2 + 2 ] ; // table name plus field name plus 2
int i , j , num_fields ;
DBUG_ENTER ( " build_completion_hash " ) ;
2000-09-22 01:46:26 +03:00
if ( status . batch | | quick | | ! current_db )
2000-07-31 21:29:14 +02:00
DBUG_VOID_RETURN ; // We don't need completion in batches
/* hash SQL commands */
while ( cmd - > name ) {
add_word ( & ht , ( char * ) cmd - > name ) ;
cmd + + ;
}
2002-04-16 11:39:03 +03:00
if ( ! rehash )
2000-07-31 21:29:14 +02:00
DBUG_VOID_RETURN ;
2001-10-08 04:58:07 +03:00
/* Free old used memory */
if ( field_names )
field_names = 0 ;
completion_hash_clean ( & ht ) ;
free_root ( & hash_mem_root , MYF ( 0 ) ) ;
2000-07-31 21:29:14 +02:00
/* hash MySQL functions (to be implemented) */
/* hash all database names */
2001-10-08 04:58:07 +03:00
if ( mysql_query ( & mysql , " show databases " ) = = 0 )
{
2000-07-31 21:29:14 +02:00
if ( ! ( databases = mysql_store_result ( & mysql ) ) )
put_info ( mysql_error ( & mysql ) , INFO_INFO ) ;
else
{
while ( ( database_row = mysql_fetch_row ( databases ) ) )
2001-10-08 04:58:07 +03:00
{
char * str = strdup_root ( & hash_mem_root , ( char * ) database_row [ 0 ] ) ;
if ( str )
add_word ( & ht , ( char * ) str ) ;
}
mysql_free_result ( databases ) ;
2000-07-31 21:29:14 +02:00
}
}
/* hash all table names */
if ( mysql_query ( & mysql , " show tables " ) = = 0 )
{
if ( ! ( tables = mysql_store_result ( & mysql ) ) )
put_info ( mysql_error ( & mysql ) , INFO_INFO ) ;
else
{
if ( mysql_num_rows ( tables ) > 0 & & ! opt_silent & & write_info )
{
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " \
2000-07-31 21:29:14 +02:00
Reading table information for completion of table and column names \ n \
You can turn off this feature to get a quicker startup with - A \ n \ n " );
}
while ( ( table_row = mysql_fetch_row ( tables ) ) )
{
2001-10-08 04:58:07 +03:00
char * str = strdup_root ( & hash_mem_root , ( char * ) table_row [ 0 ] ) ;
if ( str & &
! completion_hash_exists ( & ht , ( char * ) str , ( uint ) strlen ( str ) ) )
add_word ( & ht , str ) ;
2000-07-31 21:29:14 +02:00
}
}
}
/* hash all field names, both with the table prefix and without it */
2001-01-12 13:52:32 +02:00
if ( ! tables ) /* no tables */
{
2000-07-31 21:29:14 +02:00
DBUG_VOID_RETURN ;
}
mysql_data_seek ( tables , 0 ) ;
2001-10-08 04:58:07 +03:00
if ( ! ( field_names = ( char * * * ) alloc_root ( & hash_mem_root , sizeof ( char * * ) *
( uint ) ( mysql_num_rows ( tables ) + 1 ) ) ) )
{
mysql_free_result ( tables ) ;
2000-07-31 21:29:14 +02:00
DBUG_VOID_RETURN ;
2001-10-08 04:58:07 +03:00
}
2000-07-31 21:29:14 +02:00
i = 0 ;
while ( ( table_row = mysql_fetch_row ( tables ) ) )
{
if ( ( fields = mysql_list_fields ( & mysql , ( const char * ) table_row [ 0 ] , NullS ) ) )
{
num_fields = mysql_num_fields ( fields ) ;
2001-10-08 04:58:07 +03:00
if ( ! ( field_names [ i ] = ( char * * ) alloc_root ( & hash_mem_root ,
sizeof ( char * ) *
( num_fields * 2 + 1 ) ) ) )
2005-01-08 21:25:31 +01:00
{
mysql_free_result ( fields ) ;
break ;
}
2003-01-06 01:48:59 +02:00
field_names [ i ] [ num_fields * 2 ] = ' \0 ' ;
2000-07-31 21:29:14 +02:00
j = 0 ;
while ( ( sql_field = mysql_fetch_field ( fields ) ) )
{
2005-01-14 22:46:04 +01:00
sprintf ( buf , " %.64s.%.64s " , table_row [ 0 ] , sql_field - > name ) ;
2001-10-08 04:58:07 +03:00
field_names [ i ] [ j ] = strdup_root ( & hash_mem_root , buf ) ;
2000-07-31 21:29:14 +02:00
add_word ( & ht , field_names [ i ] [ j ] ) ;
2001-10-08 04:58:07 +03:00
field_names [ i ] [ num_fields + j ] = strdup_root ( & hash_mem_root ,
sql_field - > name ) ;
2000-07-31 21:29:14 +02:00
if ( ! completion_hash_exists ( & ht , field_names [ i ] [ num_fields + j ] ,
2000-08-22 00:18:32 +03:00
( uint ) strlen ( field_names [ i ] [ num_fields + j ] ) ) )
2000-07-31 21:29:14 +02:00
add_word ( & ht , field_names [ i ] [ num_fields + j ] ) ;
j + + ;
}
2001-10-08 04:58:07 +03:00
mysql_free_result ( fields ) ;
2000-07-31 21:29:14 +02:00
}
else
2001-01-12 13:52:32 +02:00
{
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout ,
" Didn't find any fields in table '%s' \n " , table_row [ 0 ] ) ;
2003-01-06 01:48:59 +02:00
field_names [ i ] = 0 ;
2001-01-12 13:52:32 +02:00
}
2000-07-31 21:29:14 +02:00
i + + ;
}
2001-10-08 04:58:07 +03:00
mysql_free_result ( tables ) ;
2001-01-12 13:52:32 +02:00
field_names [ i ] = 0 ; // End pointer
2000-07-31 21:29:14 +02:00
DBUG_VOID_RETURN ;
}
/* for gnu readline */
# ifndef HAVE_INDEX
extern " C " {
2002-06-30 18:57:21 +03:00
extern char * index ( const char * , int c ) , * rindex ( const char * , int ) ;
2000-07-31 21:29:14 +02:00
2002-06-30 18:57:21 +03:00
char * index ( const char * s , int c )
2000-07-31 21:29:14 +02:00
{
for ( ; ; )
{
if ( * s = = ( char ) c ) return ( char * ) s ;
if ( ! * s + + ) return NullS ;
}
}
2002-06-30 18:57:21 +03:00
char * rindex ( const char * s , int c )
2000-07-31 21:29:14 +02:00
{
reg3 char * t ;
t = NullS ;
do if ( * s = = ( char ) c ) t = ( char * ) s ; while ( * s + + ) ;
return ( char * ) t ;
}
}
# endif
# endif /* HAVE_READLINE */
2002-06-11 11:20:31 +03:00
2000-07-31 21:29:14 +02:00
static int reconnect ( void )
{
2003-03-01 23:34:34 +01:00
if ( opt_reconnect )
2000-07-31 21:29:14 +02:00
{
put_info ( " No connection. Trying to reconnect... " , INFO_INFO ) ;
( void ) com_connect ( ( String * ) 0 , 0 ) ;
2002-06-11 11:20:31 +03:00
if ( rehash )
com_rehash ( NULL , NULL ) ;
2000-07-31 21:29:14 +02:00
}
if ( ! connected )
return put_info ( " Can't connect to the server \n " , INFO_ERROR ) ;
return 0 ;
}
/***************************************************************************
The different commands
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-10-28 17:44:19 +04:00
int mysql_real_query_for_lazy ( const char * buf , int length )
{
for ( uint retry = 0 ; ; retry + + )
{
if ( ! mysql_real_query ( & mysql , buf , length ) )
2005-01-14 22:46:04 +01:00
return 0 ;
2003-06-04 18:28:51 +03:00
int error = put_error ( & mysql ) ;
2003-02-12 21:55:37 +02:00
if ( mysql_errno ( & mysql ) ! = CR_SERVER_GONE_ERROR | | retry > 1 | |
2003-03-01 23:34:34 +01:00
! opt_reconnect )
2002-10-28 17:44:19 +04:00
return error ;
if ( reconnect ( ) )
return error ;
}
}
int mysql_store_result_for_lazy ( MYSQL_RES * * result )
{
if ( ( * result = mysql_store_result ( & mysql ) ) )
return 0 ;
if ( mysql_error ( & mysql ) [ 0 ] )
2003-06-04 18:28:51 +03:00
return put_error ( & mysql ) ;
2002-10-28 17:44:19 +04:00
return 0 ;
}
2003-05-29 17:47:31 -04:00
static void print_help_item ( MYSQL_ROW * cur , int num_name , int num_cat , char * last_char )
{
char ccat = ( * cur ) [ num_cat ] [ 0 ] ;
if ( * last_char ! = ccat )
{
2004-11-19 14:53:40 -06:00
put_info ( ccat = = ' Y ' ? " categories: " : " topics: " , INFO_INFO ) ;
2003-05-29 17:47:31 -04:00
* last_char = ccat ;
}
tee_fprintf ( PAGER , " %s \n " , ( * cur ) [ num_name ] ) ;
}
2003-02-12 21:55:37 +02:00
2003-10-06 22:56:34 +03:00
2002-10-28 17:44:19 +04:00
static int com_server_help ( String * buffer __attribute__ ( ( unused ) ) ,
2003-02-12 21:55:37 +02:00
char * line __attribute__ ( ( unused ) ) , char * help_arg )
2002-10-28 17:44:19 +04:00
{
MYSQL_ROW cur ;
const char * server_cmd = buffer - > ptr ( ) ;
char cmd_buf [ 100 ] ;
2003-01-06 01:48:59 +02:00
MYSQL_RES * result ;
int error ;
2003-05-29 17:47:31 -04:00
2003-01-06 01:48:59 +02:00
if ( help_arg [ 0 ] ! = ' \' ' )
2002-10-28 17:44:19 +04:00
{
2003-01-06 01:48:59 +02:00
( void ) strxnmov ( cmd_buf , sizeof ( cmd_buf ) , " help ' " , help_arg , " ' " , NullS ) ;
2002-10-28 17:44:19 +04:00
server_cmd = cmd_buf ;
}
2003-05-29 17:47:31 -04:00
2002-10-28 17:44:19 +04:00
if ( ! status . batch )
{
old_buffer = * buffer ;
old_buffer . copy ( ) ;
}
if ( ! connected & & reconnect ( ) )
return 1 ;
2003-05-29 17:47:31 -04:00
if ( ( error = mysql_real_query_for_lazy ( server_cmd , strlen ( server_cmd ) ) ) | |
( error = mysql_store_result_for_lazy ( & result ) ) )
2002-10-28 17:44:19 +04:00
return error ;
if ( result )
{
2003-05-29 17:47:31 -04:00
unsigned int num_fields = mysql_num_fields ( result ) ;
my_ulonglong num_rows = mysql_num_rows ( result ) ;
2004-05-19 05:09:10 +03:00
mysql_fetch_fields ( result ) ;
2003-05-29 17:47:31 -04:00
if ( num_fields = = 3 & & num_rows = = 1 )
2002-10-28 17:44:19 +04:00
{
if ( ! ( cur = mysql_fetch_row ( result ) ) )
2003-01-06 01:48:59 +02:00
{
error = - 1 ;
goto err ;
}
2002-10-28 17:44:19 +04:00
init_pager ( ) ;
2003-05-29 17:47:31 -04:00
tee_fprintf ( PAGER , " Name: \' %s \' \n " , cur [ 0 ] ) ;
tee_fprintf ( PAGER , " Description: \n %s " , cur [ 1 ] ) ;
if ( cur [ 2 ] & & * ( ( char * ) cur [ 2 ] ) )
tee_fprintf ( PAGER , " Examples: \n %s " , cur [ 2 ] ) ;
tee_fprintf ( PAGER , " \n " ) ;
2002-10-28 17:44:19 +04:00
end_pager ( ) ;
}
2003-05-29 17:47:31 -04:00
else if ( num_fields > = 2 & & num_rows )
2002-10-28 17:44:19 +04:00
{
init_pager ( ) ;
2005-01-24 15:48:25 +01:00
char last_char = 0 ;
2004-01-14 17:48:38 +02:00
int num_name = 0 , num_cat = 0 ;
2003-06-14 11:37:42 +03:00
LINT_INIT ( num_name ) ;
LINT_INIT ( num_cat ) ;
2003-05-29 17:47:31 -04:00
if ( num_fields = = 2 )
2003-01-06 01:48:59 +02:00
{
2004-11-19 14:53:40 -06:00
put_info ( " Many help items for your request exist. " , INFO_INFO ) ;
2004-12-18 14:30:00 -06:00
put_info ( " To make a more specific request, please type 'help <item>', \n where <item> is one of the following " , INFO_INFO ) ;
2003-05-29 17:47:31 -04:00
num_name = 0 ;
num_cat = 1 ;
2002-10-28 17:44:19 +04:00
}
2003-05-29 17:47:31 -04:00
else if ( ( cur = mysql_fetch_row ( result ) ) )
{
2004-08-27 12:27:57 -05:00
tee_fprintf ( PAGER , " You asked for help about help category: \" %s \" \n " , cur [ 0 ] ) ;
2004-12-18 14:30:00 -06:00
put_info ( " For more information, type 'help <item>', where <item> is one of the following " , INFO_INFO ) ;
2003-05-29 17:47:31 -04:00
num_name = 1 ;
num_cat = 2 ;
print_help_item ( & cur , 1 , 2 , & last_char ) ;
}
2005-01-24 15:48:25 +01:00
2003-05-29 17:47:31 -04:00
while ( ( cur = mysql_fetch_row ( result ) ) )
print_help_item ( & cur , num_name , num_cat , & last_char ) ;
2002-10-28 17:44:19 +04:00
tee_fprintf ( PAGER , " \n " ) ;
end_pager ( ) ;
}
else
{
2003-05-29 17:47:31 -04:00
put_info ( " \n Nothing found " , INFO_INFO ) ;
2004-08-27 12:27:57 -05:00
put_info ( " Please try to run 'help contents' for a list of all accessible topics \n " , INFO_INFO ) ;
2002-10-28 17:44:19 +04:00
}
}
2003-01-06 01:48:59 +02:00
err :
2002-10-28 17:44:19 +04:00
mysql_free_result ( result ) ;
return error ;
}
2000-07-31 21:29:14 +02:00
static int
2003-02-12 21:55:37 +02:00
com_help ( String * buffer __attribute__ ( ( unused ) ) ,
char * line __attribute__ ( ( unused ) ) )
2000-07-31 21:29:14 +02:00
{
2003-12-30 17:58:39 +02:00
reg1 int i , j ;
char * help_arg = strchr ( line , ' ' ) , buff [ 32 ] , * end ;
2000-07-31 21:29:14 +02:00
2002-10-28 17:44:19 +04:00
if ( help_arg )
return com_server_help ( buffer , line , help_arg + 1 ) ;
2003-10-06 22:56:34 +03:00
2004-08-27 12:27:57 -05:00
put_info ( " \n For the complete MySQL Manual online, visit: \n http://www.mysql.com/documentation \n " , INFO_INFO ) ;
put_info ( " For info on technical support from MySQL developers, visit: \n http://www.mysql.com/support \n " , INFO_INFO ) ;
put_info ( " For info on MySQL books, utilities, consultants, etc., visit: \n http://www.mysql.com/portal \n " , INFO_INFO ) ;
2003-10-06 22:56:34 +03:00
put_info ( " List of all MySQL commands: " , INFO_INFO ) ;
if ( ! named_cmds )
put_info ( " Note that all text commands must be first on line and end with ';' " , INFO_INFO ) ;
for ( i = 0 ; commands [ i ] . name ; i + + )
2002-10-28 17:44:19 +04:00
{
2003-12-30 17:58:39 +02:00
end = strmov ( buff , commands [ i ] . name ) ;
for ( j = strlen ( commands [ i ] . name ) ; j < 10 ; j + + )
end = strmov ( end , " " ) ;
2003-10-06 22:56:34 +03:00
if ( commands [ i ] . func )
2003-12-30 17:58:39 +02:00
tee_fprintf ( stdout , " %s( \\ %c) %s \n " , buff ,
2003-10-06 22:56:34 +03:00
commands [ i ] . cmd_char , commands [ i ] . doc ) ;
2002-10-28 17:44:19 +04:00
}
2003-02-12 21:55:37 +02:00
if ( connected & & mysql_get_server_version ( & mysql ) > = 40100 )
2004-07-13 08:57:55 +02:00
put_info ( " \n For server side help, type 'help contents' \n " , INFO_INFO ) ;
2000-07-31 21:29:14 +02:00
return 0 ;
}
/* ARGSUSED */
static int
com_clear ( String * buffer , char * line __attribute__ ( ( unused ) ) )
{
2004-03-03 15:54:37 -05:00
# ifdef HAVE_READLINE
if ( status . add_to_history )
fix_history ( buffer ) ;
# endif
2000-07-31 21:29:14 +02:00
buffer - > length ( 0 ) ;
return 0 ;
}
/*
2002-06-11 11:20:31 +03:00
Execute command
Returns : 0 if ok
- 1 if not fatal error
1 if fatal error
2000-07-31 21:29:14 +02:00
*/
static int
com_go ( String * buffer , char * line __attribute__ ( ( unused ) ) )
{
2002-10-02 13:33:08 +03:00
char buff [ 200 ] , time_buff [ 32 ] , * pos ;
2000-07-31 21:29:14 +02:00
MYSQL_RES * result ;
2002-10-02 13:33:08 +03:00
ulong timer , warnings ;
2003-11-20 20:06:51 +02:00
uint error = 0 ;
int err = 0 ;
2000-07-31 21:29:14 +02:00
if ( ! status . batch )
{
old_buffer = * buffer ; // Save for edit command
old_buffer . copy ( ) ;
}
2002-06-11 11:20:31 +03:00
/* Remove garbage for nicer messages */
2000-07-31 21:29:14 +02:00
LINT_INIT ( buff [ 0 ] ) ;
remove_cntrl ( * buffer ) ;
if ( buffer - > is_empty ( ) )
{
if ( status . batch ) // Ignore empty quries
return 0 ;
return put_info ( " No query specified \n " , INFO_ERROR ) ;
}
if ( ! connected & & reconnect ( ) )
{
buffer - > length ( 0 ) ; // Remove query on error
2003-03-01 23:34:34 +01:00
return opt_reconnect ? - 1 : 1 ; // Fatal error
2000-07-31 21:29:14 +02:00
}
if ( verbose )
( void ) com_print ( buffer , 0 ) ;
if ( skip_updates & &
2003-03-16 11:20:45 +04:00
( buffer - > length ( ) < 4 | | my_strnncoll ( charset_info ,
2002-10-09 16:33:35 +05:00
( const uchar * ) buffer - > ptr ( ) , 4 ,
( const uchar * ) " SET " , 4 ) ) )
2000-07-31 21:29:14 +02:00
{
( void ) put_info ( " Ignoring query to other database " , INFO_INFO ) ;
return 0 ;
}
timer = start_timer ( ) ;
2002-10-28 17:44:19 +04:00
error = mysql_real_query_for_lazy ( buffer - > ptr ( ) , buffer - > length ( ) ) ;
2004-03-03 15:54:37 -05:00
# ifdef HAVE_READLINE
if ( status . add_to_history )
{
buffer - > append ( vertical ? " \\ G " : delimiter ) ;
/* Append final command onto history */
fix_history ( buffer ) ;
}
# endif
2002-10-28 17:44:19 +04:00
if ( error )
2000-07-31 21:29:14 +02:00
{
2002-10-28 17:44:19 +04:00
buffer - > length ( 0 ) ; // Remove query on error
return error ;
2000-07-31 21:29:14 +02:00
}
error = 0 ;
buffer - > length ( 0 ) ;
2003-11-20 14:33:07 +02:00
do
2000-07-31 21:29:14 +02:00
{
2003-11-20 14:33:07 +02:00
if ( quick )
2000-07-31 21:29:14 +02:00
{
2003-11-20 14:33:07 +02:00
if ( ! ( result = mysql_use_result ( & mysql ) ) & & mysql_field_count ( & mysql ) )
return put_error ( & mysql ) ;
2000-07-31 21:29:14 +02:00
}
else
{
2003-11-20 14:33:07 +02:00
error = mysql_store_result_for_lazy ( & result ) ;
if ( error )
return error ;
}
if ( verbose > = 3 | | ! opt_silent )
mysql_end_timer ( timer , time_buff ) ;
else
time_buff [ 0 ] = 0 ;
if ( result )
{
if ( ! mysql_num_rows ( result ) & & ! quick )
{
strmov ( buff , " Empty set " ) ;
}
2000-07-31 21:29:14 +02:00
else
2003-11-20 14:33:07 +02:00
{
init_pager ( ) ;
if ( opt_html )
print_table_data_html ( result ) ;
else if ( opt_xml )
print_table_data_xml ( result ) ;
else if ( vertical )
print_table_data_vertically ( result ) ;
else if ( opt_silent & & verbose < = 2 & & ! output_tables )
print_tab_data ( result ) ;
else
print_table_data ( result ) ;
sprintf ( buff , " %ld %s in set " ,
( long ) mysql_num_rows ( result ) ,
( long ) mysql_num_rows ( result ) = = 1 ? " row " : " rows " ) ;
end_pager ( ) ;
}
2000-07-31 21:29:14 +02:00
}
2003-11-20 14:33:07 +02:00
else if ( mysql_affected_rows ( & mysql ) = = ~ ( ulonglong ) 0 )
strmov ( buff , " Query OK " ) ;
else
sprintf ( buff , " Query OK, %ld %s affected " ,
( long ) mysql_affected_rows ( & mysql ) ,
( long ) mysql_affected_rows ( & mysql ) = = 1 ? " row " : " rows " ) ;
pos = strend ( buff ) ;
if ( ( warnings = mysql_warning_count ( & mysql ) ) )
{
* pos + + = ' , ' ;
* pos + + = ' ' ;
2004-05-27 17:54:40 +04:00
pos = int10_to_str ( warnings , pos , 10 ) ;
2003-11-20 14:33:07 +02:00
pos = strmov ( pos , " warning " ) ;
if ( warnings ! = 1 )
* pos + + = ' s ' ;
}
strmov ( pos , time_buff ) ;
put_info ( buff , INFO_RESULT ) ;
if ( mysql_info ( & mysql ) )
put_info ( mysql_info ( & mysql ) , INFO_RESULT ) ;
put_info ( " " , INFO_RESULT ) ; // Empty row
if ( result & & ! mysql_eof ( result ) ) /* Something wrong when using quick */
error = put_error ( & mysql ) ;
else if ( unbuffered )
fflush ( stdout ) ;
mysql_free_result ( result ) ;
2003-11-20 20:06:51 +02:00
} while ( ! ( err = mysql_next_result ( & mysql ) ) ) ;
if ( err > = 1 )
error = put_error ( & mysql ) ;
2003-11-20 14:33:07 +02:00
2000-07-31 21:29:14 +02:00
return error ; /* New command follows */
}
2000-11-17 14:33:29 +02:00
static void init_pager ( )
{
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-17 14:33:29 +02:00
if ( ! opt_nopager )
{
if ( ! ( PAGER = popen ( pager , " w " ) ) )
{
tee_fprintf ( stdout , " popen() failed! defaulting PAGER to stdout! \n " ) ;
PAGER = stdout ;
}
}
else
# endif
PAGER = stdout ;
}
static void end_pager ( )
{
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-17 14:33:29 +02:00
if ( ! opt_nopager )
pclose ( PAGER ) ;
# endif
}
2002-06-12 17:35:04 +03:00
2003-01-28 08:38:28 +02:00
static void init_tee ( const char * file_name )
2000-11-17 14:33:29 +02:00
{
2002-06-12 17:35:04 +03:00
FILE * new_outfile ;
2002-05-31 12:34:47 +03:00
if ( opt_outfile )
2002-06-12 17:35:04 +03:00
end_tee ( ) ;
2003-01-28 08:38:28 +02:00
if ( ! ( new_outfile = my_fopen ( file_name , O_APPEND | O_WRONLY , MYF ( MY_WME ) ) ) )
{
tee_fprintf ( stdout , " Error logging to file '%s' \n " , file_name ) ;
return ;
}
2002-06-12 17:35:04 +03:00
OUTFILE = new_outfile ;
2003-01-28 08:38:28 +02:00
strmake ( outfile , file_name , FN_REFLEN - 1 ) ;
tee_fprintf ( stdout , " Logging to file '%s' \n " , file_name ) ;
opt_outfile = 1 ;
return ;
2000-11-17 14:33:29 +02:00
}
2002-06-12 17:35:04 +03:00
2000-11-17 14:33:29 +02:00
static void end_tee ( )
{
my_fclose ( OUTFILE , MYF ( 0 ) ) ;
2003-01-28 08:38:28 +02:00
OUTFILE = 0 ;
2002-05-31 12:34:47 +03:00
opt_outfile = 0 ;
2000-11-17 14:33:29 +02:00
return ;
}
2000-07-31 21:29:14 +02:00
static int
com_ego ( String * buffer , char * line )
{
int result ;
bool oldvertical = vertical ;
vertical = 1 ;
result = com_go ( buffer , line ) ;
vertical = oldvertical ;
return result ;
}
2001-03-11 21:20:15 +02:00
static void
print_field_types ( MYSQL_RES * result )
{
MYSQL_FIELD * field ;
while ( ( field = mysql_fetch_field ( result ) ) )
{
2004-10-22 18:44:51 +03:00
tee_fprintf ( PAGER , " Catalog: '%s' \n Database: '%s' \n Table: '%s' \n Name: '%s' \n Type: %d \n Length: %ld \n Max length: %ld \n Is_null: %d \n Flags: %u \n Decimals: %u \n \n " ,
2003-05-27 18:40:37 +03:00
field - > catalog , field - > db , field - > table , field - > name ,
2001-03-11 21:20:15 +02:00
( int ) field - > type ,
2003-08-27 10:26:03 +03:00
field - > length , field - > max_length ,
! IS_NOT_NULL ( field - > flags ) ,
2001-03-11 21:20:15 +02:00
field - > flags , field - > decimals ) ;
}
tee_puts ( " " , PAGER ) ;
}
2000-07-31 21:29:14 +02:00
static void
print_table_data ( MYSQL_RES * result )
{
String separator ( 256 ) ;
MYSQL_ROW cur ;
MYSQL_FIELD * field ;
bool * num_flag ;
num_flag = ( bool * ) my_alloca ( sizeof ( bool ) * mysql_num_fields ( result ) ) ;
2001-03-11 21:20:15 +02:00
if ( info_flag )
{
print_field_types ( result ) ;
mysql_field_seek ( result , 0 ) ;
}
2003-03-16 11:20:45 +04:00
separator . copy ( " + " , 1 , charset_info ) ;
2000-07-31 21:29:14 +02:00
while ( ( field = mysql_fetch_field ( result ) ) )
{
2003-02-08 01:00:35 +02:00
uint length = column_names ? field - > name_length : 0 ;
2000-07-31 21:29:14 +02:00
if ( quick )
length = max ( length , field - > length ) ;
else
length = max ( length , field - > max_length ) ;
if ( length < 4 & & ! IS_NOT_NULL ( field - > flags ) )
length = 4 ; // Room for "NULL"
field - > max_length = length + 1 ;
separator . fill ( separator . length ( ) + length + 2 , ' - ' ) ;
separator . append ( ' + ' ) ;
}
2003-06-14 11:37:42 +03:00
tee_puts ( separator . c_ptr_safe ( ) , PAGER ) ;
2002-04-02 20:29:53 +03:00
if ( column_names )
2000-07-31 21:29:14 +02:00
{
mysql_field_seek ( result , 0 ) ;
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " | " , PAGER ) ;
2000-07-31 21:29:14 +02:00
for ( uint off = 0 ; ( field = mysql_fetch_field ( result ) ) ; off + + )
{
2004-10-22 18:44:51 +03:00
tee_fprintf ( PAGER , " %-*s| " , ( int ) min ( field - > max_length ,
MAX_COLUMN_LENGTH ) ,
2001-02-02 03:47:06 +02:00
field - > name ) ;
2000-07-31 21:29:14 +02:00
num_flag [ off ] = IS_NUM ( field - > type ) ;
}
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " \n " , PAGER ) ;
tee_puts ( separator . c_ptr ( ) , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
2002-05-31 15:43:21 +03:00
while ( ( cur = mysql_fetch_row ( result ) ) )
2000-07-31 21:29:14 +02:00
{
2004-08-25 13:03:34 +05:00
ulong * lengths = mysql_fetch_lengths ( result ) ;
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " | " , PAGER ) ;
2002-05-31 15:43:21 +03:00
mysql_field_seek ( result , 0 ) ;
for ( uint off = 0 ; off < mysql_num_fields ( result ) ; off + + )
2000-07-31 21:29:14 +02:00
{
2002-05-31 15:43:21 +03:00
const char * str = cur [ off ] ? cur [ off ] : " NULL " ;
field = mysql_fetch_field ( result ) ;
2004-08-25 13:03:34 +05:00
uint maxlength = field - > max_length ;
if ( maxlength > MAX_COLUMN_LENGTH )
2001-02-02 03:47:06 +02:00
{
2002-05-31 15:43:21 +03:00
tee_fputs ( str , PAGER ) ;
tee_fputs ( " | " , PAGER ) ;
2001-02-02 03:47:06 +02:00
}
else
2004-08-25 13:03:34 +05:00
{
uint currlength = ( uint ) lengths [ off ] ;
uint numcells = charset_info - > cset - > numcells ( charset_info ,
str , str + currlength ) ;
tee_fprintf ( PAGER , num_flag [ off ] ? " %*s | " : " %-*s| " ,
maxlength + currlength - numcells , str ) ;
}
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " \n " , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
tee_puts ( separator . c_ptr ( ) , PAGER ) ;
2000-07-31 21:29:14 +02:00
my_afree ( ( gptr ) num_flag ) ;
}
2003-01-28 08:38:28 +02:00
2000-07-31 21:29:14 +02:00
static void
print_table_data_html ( MYSQL_RES * result )
{
2003-01-28 08:38:28 +02:00
MYSQL_ROW cur ;
MYSQL_FIELD * field ;
2000-07-31 21:29:14 +02:00
mysql_field_seek ( result , 0 ) ;
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " <TABLE BORDER=1><TR> " , PAGER ) ;
2002-04-02 20:29:53 +03:00
if ( column_names )
2000-07-31 21:29:14 +02:00
{
while ( ( field = mysql_fetch_field ( result ) ) )
{
2000-11-17 14:33:29 +02:00
tee_fprintf ( PAGER , " <TH>%s</TH> " , ( field - > name ?
( field - > name [ 0 ] ? field - > name :
" " ) : " NULL " ) ) ;
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " </TR> " , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
while ( ( cur = mysql_fetch_row ( result ) ) )
{
2005-01-08 21:25:31 +01:00
ulong * lengths = mysql_fetch_lengths ( result ) ;
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " <TR> " , PAGER ) ;
2000-07-31 21:29:14 +02:00
for ( uint i = 0 ; i < mysql_num_fields ( result ) ; i + + )
{
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " <TD> " , PAGER ) ;
2000-07-31 21:29:14 +02:00
safe_put_field ( cur [ i ] , lengths [ i ] ) ;
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " </TD> " , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " </TR> " , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " </TABLE> " , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
2001-04-11 13:04:03 +02:00
static void
print_table_data_xml ( MYSQL_RES * result )
{
MYSQL_ROW cur ;
MYSQL_FIELD * fields ;
mysql_field_seek ( result , 0 ) ;
2001-12-14 17:43:18 +02:00
tee_fputs ( " <?xml version= \" 1.0 \" ?> \n \n <resultset statement= \" " , PAGER ) ;
2002-04-03 13:36:01 +03:00
xmlencode_print ( glob_buffer . ptr ( ) , strlen ( glob_buffer . ptr ( ) ) ) ;
2001-12-14 17:43:18 +02:00
tee_fputs ( " \" > " , PAGER ) ;
2001-04-11 13:04:03 +02:00
fields = mysql_fetch_fields ( result ) ;
while ( ( cur = mysql_fetch_row ( result ) ) )
{
2005-01-08 21:25:31 +01:00
ulong * lengths = mysql_fetch_lengths ( result ) ;
2001-04-11 13:04:03 +02:00
( void ) tee_fputs ( " \n <row> \n " , PAGER ) ;
for ( uint i = 0 ; i < mysql_num_fields ( result ) ; i + + )
{
2005-01-25 14:25:40 -08:00
tee_fprintf ( PAGER , " \t <field name= \" " ) ;
xmlencode_print ( fields [ i ] . name , strlen ( fields [ i ] . name ) ) ;
tee_fprintf ( PAGER , " \" > " ) ;
2001-12-14 17:43:18 +02:00
xmlencode_print ( cur [ i ] , lengths [ i ] ) ;
2005-01-25 14:25:40 -08:00
tee_fprintf ( PAGER , " </field> \n " ) ;
2001-04-11 13:04:03 +02:00
}
( void ) tee_fputs ( " </row> \n " , PAGER ) ;
}
( void ) tee_fputs ( " </resultset> \n " , PAGER ) ;
}
2000-07-31 21:29:14 +02:00
static void
print_table_data_vertically ( MYSQL_RES * result )
{
MYSQL_ROW cur ;
uint max_length = 0 ;
MYSQL_FIELD * field ;
while ( ( field = mysql_fetch_field ( result ) ) )
{
2003-02-08 01:00:35 +02:00
uint length = field - > name_length ;
2000-07-31 21:29:14 +02:00
if ( length > max_length )
max_length = length ;
field - > max_length = length ;
}
mysql_field_seek ( result , 0 ) ;
for ( uint row_count = 1 ; ( cur = mysql_fetch_row ( result ) ) ; row_count + + )
{
mysql_field_seek ( result , 0 ) ;
2000-11-17 14:33:29 +02:00
tee_fprintf ( PAGER ,
" *************************** %d. row *************************** \n " , row_count ) ;
2000-07-31 21:29:14 +02:00
for ( uint off = 0 ; off < mysql_num_fields ( result ) ; off + + )
{
field = mysql_fetch_field ( result ) ;
2000-11-17 14:33:29 +02:00
tee_fprintf ( PAGER , " %*s: " , ( int ) max_length , field - > name ) ;
tee_fprintf ( PAGER , " %s \n " , cur [ off ] ? ( char * ) cur [ off ] : " NULL " ) ;
2000-07-31 21:29:14 +02:00
}
}
}
2001-12-14 17:43:18 +02:00
2001-04-25 01:11:29 +03:00
static const char
2001-12-14 17:43:18 +02:00
* array_value ( const char * * array , char key )
{
2001-04-11 13:04:03 +02:00
int x ;
2001-12-14 17:43:18 +02:00
for ( x = 0 ; array [ x ] ; x + = 2 )
if ( * array [ x ] = = key )
return array [ x + 1 ] ;
2001-04-11 13:04:03 +02:00
return 0 ;
}
2001-12-14 17:43:18 +02:00
2001-04-25 01:11:29 +03:00
static void
2001-12-14 17:43:18 +02:00
xmlencode_print ( const char * src , uint length )
2001-04-25 01:11:29 +03:00
{
2001-12-14 17:43:18 +02:00
if ( ! src )
tee_fputs ( " NULL " , PAGER ) ;
else
2001-04-25 01:11:29 +03:00
{
2001-12-14 17:43:18 +02:00
for ( const char * p = src ; * p & & length ; * p + + , length - - )
{
const char * t ;
if ( ( t = array_value ( xmlmeta , * p ) ) )
tee_fputs ( t , PAGER ) ;
else
tee_putc ( * p , PAGER ) ;
}
}
2001-04-11 13:04:03 +02:00
}
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
static void
safe_put_field ( const char * pos , ulong length )
{
if ( ! pos )
2000-11-17 14:33:29 +02:00
tee_fputs ( " NULL " , PAGER ) ;
2000-07-31 21:29:14 +02:00
else
{
if ( opt_raw_data )
2000-11-17 14:33:29 +02:00
tee_fputs ( pos , PAGER ) ;
2000-07-31 21:29:14 +02:00
else for ( const char * end = pos + length ; pos ! = end ; pos + + )
{
# ifdef USE_MB
int l ;
2003-03-16 11:20:45 +04:00
if ( use_mb ( charset_info ) & &
( l = my_ismbchar ( charset_info , pos , end ) ) )
2002-08-30 12:40:40 +03:00
{
2000-07-31 21:29:14 +02:00
while ( l - - )
2000-11-18 17:34:15 +02:00
tee_putc ( * pos + + , PAGER ) ;
2000-07-31 21:29:14 +02:00
pos - - ;
continue ;
}
# endif
if ( ! * pos )
2000-11-17 14:33:29 +02:00
tee_fputs ( " \\ 0 " , PAGER ) ; // This makes everything hard
2000-07-31 21:29:14 +02:00
else if ( * pos = = ' \t ' )
2000-11-17 14:33:29 +02:00
tee_fputs ( " \\ t " , PAGER ) ; // This would destroy tab format
2000-07-31 21:29:14 +02:00
else if ( * pos = = ' \n ' )
2000-11-17 14:33:29 +02:00
tee_fputs ( " \\ n " , PAGER ) ; // This too
2000-07-31 21:29:14 +02:00
else if ( * pos = = ' \\ ' )
2000-11-17 14:33:29 +02:00
tee_fputs ( " \\ \\ " , PAGER ) ;
2001-04-11 13:04:03 +02:00
else
2000-11-18 17:34:15 +02:00
tee_putc ( * pos , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
}
}
static void
print_tab_data ( MYSQL_RES * result )
{
MYSQL_ROW cur ;
MYSQL_FIELD * field ;
ulong * lengths ;
2002-04-02 20:29:53 +03:00
if ( opt_silent < 2 & & column_names )
2000-07-31 21:29:14 +02:00
{
int first = 0 ;
while ( ( field = mysql_fetch_field ( result ) ) )
{
if ( first + + )
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " \t " , PAGER ) ;
( void ) tee_fputs ( field - > name , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " \n " , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
while ( ( cur = mysql_fetch_row ( result ) ) )
{
lengths = mysql_fetch_lengths ( result ) ;
safe_put_field ( cur [ 0 ] , lengths [ 0 ] ) ;
for ( uint off = 1 ; off < mysql_num_fields ( result ) ; off + + )
{
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " \t " , PAGER ) ;
2002-05-31 12:34:47 +03:00
safe_put_field ( cur [ off ] , lengths [ off ] ) ;
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
( void ) tee_fputs ( " \n " , PAGER ) ;
2000-07-31 21:29:14 +02:00
}
}
2000-11-17 14:33:29 +02:00
static int
com_tee ( String * buffer , char * line __attribute__ ( ( unused ) ) )
{
char file_name [ FN_REFLEN ] , * end , * param ;
if ( status . batch )
return 0 ;
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * line ) )
2000-11-17 14:33:29 +02:00
line + + ;
if ( ! ( param = strchr ( line , ' ' ) ) ) // if outfile wasn't given, use the default
{
if ( ! strlen ( outfile ) )
{
2002-05-31 12:34:47 +03:00
printf ( " No previous outfile available, you must give a filename! \n " ) ;
2000-11-17 14:33:29 +02:00
return 0 ;
}
2002-06-12 17:35:04 +03:00
else if ( opt_outfile )
{
tee_fprintf ( stdout , " Currently logging to file '%s' \n " , outfile ) ;
return 0 ;
}
else
param = outfile ; //resume using the old outfile
2000-11-17 14:33:29 +02:00
}
2002-06-12 17:35:04 +03:00
/* eliminate the spaces before the parameters */
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * param ) )
2002-06-12 17:35:04 +03:00
param + + ;
end = strmake ( file_name , param , sizeof ( file_name ) - 1 ) ;
/* remove end space from command line */
2003-03-16 11:20:45 +04:00
while ( end > file_name & & ( my_isspace ( charset_info , end [ - 1 ] ) | |
my_iscntrl ( charset_info , end [ - 1 ] ) ) )
2002-06-12 17:35:04 +03:00
end - - ;
end [ 0 ] = 0 ;
2002-09-03 15:44:25 +03:00
if ( end = = file_name )
2000-11-17 14:33:29 +02:00
{
printf ( " No outfile specified! \n " ) ;
return 0 ;
}
2003-01-28 08:38:28 +02:00
init_tee ( file_name ) ;
2000-11-17 14:33:29 +02:00
return 0 ;
}
2002-06-12 17:35:04 +03:00
2000-11-17 14:33:29 +02:00
static int
com_notee ( String * buffer __attribute__ ( ( unused ) ) ,
char * line __attribute__ ( ( unused ) ) )
{
if ( opt_outfile )
end_tee ( ) ;
tee_fprintf ( stdout , " Outfile disabled. \n " ) ;
return 0 ;
}
2000-11-22 03:45:02 +02:00
/*
2002-06-11 11:20:31 +03:00
Sorry , this command is not available in Windows .
2000-11-22 03:45:02 +02:00
*/
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-17 14:33:29 +02:00
static int
com_pager ( String * buffer , char * line __attribute__ ( ( unused ) ) )
{
char pager_name [ FN_REFLEN ] , * end , * param ;
if ( status . batch )
return 0 ;
2004-01-14 04:58:37 +02:00
/* Skip spaces in front of the pager command */
while ( my_isspace ( charset_info , * line ) )
2000-11-17 14:33:29 +02:00
line + + ;
2004-01-14 04:58:37 +02:00
/* Skip the pager command */
param = strchr ( line , ' ' ) ;
/* Skip the spaces between the command and the argument */
while ( param & & my_isspace ( charset_info , * param ) )
param + + ;
if ( ! param | | ! strlen ( param ) ) // if pager was not given, use the default
2000-11-17 14:33:29 +02:00
{
2004-01-14 04:58:37 +02:00
if ( ! default_pager_set )
2000-11-17 14:33:29 +02:00
{
2000-11-20 21:22:02 +02:00
tee_fprintf ( stdout , " Default pager wasn't set, using stdout. \n " ) ;
2000-11-17 14:33:29 +02:00
opt_nopager = 1 ;
strmov ( pager , " stdout " ) ;
PAGER = stdout ;
return 0 ;
}
strmov ( pager , default_pager ) ;
}
else
{
2004-01-14 04:58:37 +02:00
end = strmake ( pager_name , param , sizeof ( pager_name ) - 1 ) ;
2003-03-16 11:20:45 +04:00
while ( end > pager_name & & ( my_isspace ( charset_info , end [ - 1 ] ) | |
my_iscntrl ( charset_info , end [ - 1 ] ) ) )
2000-11-17 14:33:29 +02:00
end - - ;
end [ 0 ] = 0 ;
strmov ( pager , pager_name ) ;
2001-04-11 06:17:20 +03:00
strmov ( default_pager , pager_name ) ;
2000-11-17 14:33:29 +02:00
}
opt_nopager = 0 ;
2004-01-14 04:58:37 +02:00
tee_fprintf ( stdout , " PAGER set to '%s' \n " , pager ) ;
2000-11-17 14:33:29 +02:00
return 0 ;
}
2000-11-22 03:45:02 +02:00
2000-11-17 14:33:29 +02:00
static int
com_nopager ( String * buffer __attribute__ ( ( unused ) ) ,
char * line __attribute__ ( ( unused ) ) )
{
strmov ( pager , " stdout " ) ;
opt_nopager = 1 ;
2004-01-14 04:58:37 +02:00
PAGER = stdout ;
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " PAGER set to stdout \n " ) ;
return 0 ;
}
2000-11-22 03:45:02 +02:00
# endif
2000-11-17 14:33:29 +02:00
2000-07-31 21:29:14 +02:00
2000-11-22 03:45:02 +02:00
/*
2002-06-11 11:20:31 +03:00
Sorry , you can ' t send the result to an editor in Win32
2000-11-22 03:45:02 +02:00
*/
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-07-31 21:29:14 +02:00
static int
com_edit ( String * buffer , char * line __attribute__ ( ( unused ) ) )
{
2000-10-03 14:18:03 +03:00
char filename [ FN_REFLEN ] , buff [ 160 ] ;
2000-07-31 21:29:14 +02:00
int fd , tmp ;
const char * editor ;
2000-10-03 14:18:03 +03:00
if ( ( fd = create_temp_file ( filename , NullS , " sql " , O_CREAT | O_WRONLY ,
MYF ( MY_WME ) ) ) < 0 )
2000-07-31 21:29:14 +02:00
goto err ;
if ( buffer - > is_empty ( ) & & ! old_buffer . is_empty ( ) )
( void ) my_write ( fd , ( byte * ) old_buffer . ptr ( ) , old_buffer . length ( ) ,
MYF ( MY_WME ) ) ;
else
( void ) my_write ( fd , ( byte * ) buffer - > ptr ( ) , buffer - > length ( ) , MYF ( MY_WME ) ) ;
( void ) my_close ( fd , MYF ( 0 ) ) ;
if ( ! ( editor = ( char * ) getenv ( " EDITOR " ) ) & &
! ( editor = ( char * ) getenv ( " VISUAL " ) ) )
editor = " vi " ;
strxmov ( buff , editor , " " , filename , NullS ) ;
( void ) system ( buff ) ;
MY_STAT stat_arg ;
if ( ! my_stat ( filename , & stat_arg , MYF ( MY_WME ) ) )
goto err ;
if ( ( fd = my_open ( filename , O_RDONLY , MYF ( MY_WME ) ) ) < 0 )
goto err ;
( void ) buffer - > alloc ( ( uint ) stat_arg . st_size ) ;
if ( ( tmp = read ( fd , ( char * ) buffer - > ptr ( ) , buffer - > alloced_length ( ) ) ) > = 0L )
buffer - > length ( ( uint ) tmp ) ;
else
buffer - > length ( 0 ) ;
( void ) my_close ( fd , MYF ( 0 ) ) ;
( void ) my_delete ( filename , MYF ( MY_WME ) ) ;
err :
return 0 ;
}
2000-11-22 03:45:02 +02:00
# endif
2000-07-31 21:29:14 +02:00
/* If arg is given, exit without errors. This happens on command 'quit' */
static int
com_quit ( String * buffer __attribute__ ( ( unused ) ) ,
char * line __attribute__ ( ( unused ) ) )
{
2004-05-25 22:00:14 +03:00
/* let the screen auto close on a normal shutdown */
NETWARE_SET_SCREEN_MODE ( SCR_AUTOCLOSE_ON_EXIT ) ;
2000-07-31 21:29:14 +02:00
status . exit_status = 0 ;
return 1 ;
}
static int
com_rehash ( String * buffer __attribute__ ( ( unused ) ) ,
char * line __attribute__ ( ( unused ) ) )
{
# ifdef HAVE_READLINE
2002-04-16 11:39:03 +03:00
build_completion_hash ( 1 , 0 ) ;
2000-07-31 21:29:14 +02:00
# endif
return 0 ;
}
2001-11-06 23:00:03 +02:00
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2001-11-06 23:00:03 +02:00
static int
com_shell ( String * buffer , char * line __attribute__ ( ( unused ) ) )
{
char * shell_cmd ;
2004-05-06 16:15:46 +02:00
/* Skip space from line begin */
2004-05-18 09:56:45 +02:00
while ( my_isspace ( charset_info , * line ) )
2004-05-06 16:15:46 +02:00
line + + ;
2001-11-06 23:00:03 +02:00
if ( ! ( shell_cmd = strchr ( line , ' ' ) ) )
{
put_info ( " Usage: \\ ! shell-command " , INFO_ERROR ) ;
return - 1 ;
}
2002-06-11 11:20:31 +03:00
/*
The output of the shell command does not
get directed to the pager or the outfile
*/
if ( system ( shell_cmd ) = = - 1 )
2001-11-06 23:00:03 +02:00
{
put_info ( strerror ( errno ) , INFO_ERROR , errno ) ;
return - 1 ;
}
return 0 ;
}
# endif
2000-07-31 21:29:14 +02:00
static int
com_print ( String * buffer , char * line __attribute__ ( ( unused ) ) )
{
2000-11-17 14:33:29 +02:00
tee_puts ( " -------------- " , stdout ) ;
( void ) tee_fputs ( buffer - > c_ptr ( ) , stdout ) ;
2000-07-31 21:29:14 +02:00
if ( ! buffer - > length ( ) | | ( * buffer ) [ buffer - > length ( ) - 1 ] ! = ' \n ' )
2000-11-18 17:34:15 +02:00
tee_putc ( ' \n ' , stdout ) ;
2000-11-17 14:33:29 +02:00
tee_puts ( " -------------- \n " , stdout ) ;
2000-07-31 21:29:14 +02:00
return 0 ; /* If empty buffer */
}
/* ARGSUSED */
static int
com_connect ( String * buffer , char * line )
{
2002-12-25 23:57:01 +02:00
char * tmp , buff [ 256 ] ;
2002-04-16 11:39:03 +03:00
bool save_rehash = rehash ;
2000-07-31 21:29:14 +02:00
int error ;
2002-12-25 23:57:01 +02:00
bzero ( buff , sizeof ( buff ) ) ;
2000-07-31 21:29:14 +02:00
if ( buffer )
{
2002-12-25 23:57:01 +02:00
strmov ( buff , line ) ;
tmp = get_arg ( buff , 0 ) ;
if ( tmp & & * tmp )
2000-07-31 21:29:14 +02:00
{
2002-12-25 23:57:01 +02:00
my_free ( current_db , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
current_db = my_strdup ( tmp , MYF ( MY_WME ) ) ;
tmp = get_arg ( buff , 1 ) ;
if ( tmp )
2000-07-31 21:29:14 +02:00
{
my_free ( current_host , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
current_host = my_strdup ( tmp , MYF ( MY_WME ) ) ;
}
}
else
2002-04-16 11:39:03 +03:00
rehash = 0 ; // Quick re-connect
2000-07-31 21:29:14 +02:00
buffer - > length ( 0 ) ; // command used
}
else
2002-04-16 11:39:03 +03:00
rehash = 0 ;
2000-07-31 21:29:14 +02:00
error = sql_connect ( current_host , current_db , current_user , opt_password , 0 ) ;
2002-04-16 11:39:03 +03:00
rehash = save_rehash ;
2000-07-31 21:29:14 +02:00
if ( connected )
{
2003-12-12 17:18:48 -04:00
sprintf ( buff , " Connection id: %lu " , mysql_thread_id ( & mysql ) ) ;
2000-07-31 21:29:14 +02:00
put_info ( buff , INFO_INFO ) ;
2005-01-14 22:46:04 +01:00
sprintf ( buff , " Current database: %.128s \n " ,
2000-08-11 10:17:38 -05:00
current_db ? current_db : " *** NONE *** " ) ;
2000-07-31 21:29:14 +02:00
put_info ( buff , INFO_INFO ) ;
}
return error ;
}
static int com_source ( String * buffer , char * line )
{
char source_name [ FN_REFLEN ] , * end , * param ;
LINE_BUFFER * line_buff ;
int error ;
STATUS old_status ;
FILE * sql_file ;
/* Skip space from file name */
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * line ) )
2000-07-31 21:29:14 +02:00
line + + ;
2000-11-17 14:33:29 +02:00
if ( ! ( param = strchr ( line , ' ' ) ) ) // Skip command name
return put_info ( " Usage: \\ . <filename> | source <filename> " ,
INFO_ERROR , 0 ) ;
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * param ) )
2000-07-31 21:29:14 +02:00
param + + ;
end = strmake ( source_name , param , sizeof ( source_name ) - 1 ) ;
2003-03-16 11:20:45 +04:00
while ( end > source_name & & ( my_isspace ( charset_info , end [ - 1 ] ) | |
my_iscntrl ( charset_info , end [ - 1 ] ) ) )
2000-07-31 21:29:14 +02:00
end - - ;
end [ 0 ] = 0 ;
2000-09-07 04:55:17 +03:00
unpack_filename ( source_name , source_name ) ;
2000-07-31 21:29:14 +02:00
/* open file name */
2000-09-07 04:55:17 +03:00
if ( ! ( sql_file = my_fopen ( source_name , O_RDONLY | O_BINARY , MYF ( 0 ) ) ) )
2000-07-31 21:29:14 +02:00
{
char buff [ FN_REFLEN + 60 ] ;
sprintf ( buff , " Failed to open file '%s', error: %d " , source_name , errno ) ;
return put_info ( buff , INFO_ERROR , 0 ) ;
}
2004-05-26 21:40:27 +05:00
if ( ! ( line_buff = batch_readline_init ( opt_max_allowed_packet + 512 , sql_file ) ) )
2000-07-31 21:29:14 +02:00
{
my_fclose ( sql_file , MYF ( 0 ) ) ;
return put_info ( " Can't initialize batch_readline " , INFO_ERROR , 0 ) ;
}
/* Save old status */
old_status = status ;
bfill ( ( char * ) & status , sizeof ( status ) , ( char ) 0 ) ;
status . batch = old_status . batch ; // Run in batch mode
status . line_buff = line_buff ;
status . file_name = source_name ;
glob_buffer . length ( 0 ) ; // Empty command buffer
error = read_lines ( 0 ) ; // Read lines from file
status = old_status ; // Continue as before
my_fclose ( sql_file , MYF ( 0 ) ) ;
batch_readline_end ( line_buff ) ;
return error ;
}
2003-04-23 03:29:03 +03:00
/* ARGSUSED */
static int
com_delimiter ( String * buffer __attribute__ ( ( unused ) ) , char * line )
{
char buff [ 256 ] , * tmp ;
strmake ( buff , line , sizeof ( buff ) - 1 ) ;
tmp = get_arg ( buff , 0 ) ;
if ( ! tmp | | ! * tmp )
{
put_info ( " DELIMITER must be followed by a 'delimiter' character or string " ,
INFO_ERROR ) ;
return 0 ;
}
strmake ( delimiter , tmp , sizeof ( delimiter ) - 1 ) ;
delimiter_length = strlen ( delimiter ) ;
2003-11-20 14:33:07 +02:00
delimiter_str = delimiter ;
2003-04-23 03:29:03 +03:00
return 0 ;
}
2000-07-31 21:29:14 +02:00
/* ARGSUSED */
static int
com_use ( String * buffer __attribute__ ( ( unused ) ) , char * line )
{
2004-01-05 21:45:14 +02:00
char * tmp , buff [ FN_REFLEN + 1 ] ;
MYSQL_RES * res ;
MYSQL_ROW row ;
2000-07-31 21:29:14 +02:00
2002-12-25 23:57:01 +02:00
bzero ( buff , sizeof ( buff ) ) ;
2002-12-11 16:03:50 +02:00
strmov ( buff , line ) ;
2002-12-25 23:57:01 +02:00
tmp = get_arg ( buff , 0 ) ;
2002-12-11 16:03:50 +02:00
if ( ! tmp | | ! * tmp )
2000-07-31 21:29:14 +02:00
{
2002-12-11 16:03:50 +02:00
put_info ( " USE must be followed by a database name " , INFO_ERROR ) ;
2000-07-31 21:29:14 +02:00
return 0 ;
}
2004-09-06 16:48:11 +05:00
/*
We need to recheck the current database , because it may change
under our feet , for example if DROP DATABASE or RENAME DATABASE
( latter one not yet available by the time the comment was written )
2004-01-05 21:45:14 +02:00
*/
2004-09-07 14:24:47 +05:00
/* Let's reset current_db, assume it's gone */
my_free ( current_db , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
current_db = 0 ;
2004-09-06 16:48:11 +05:00
/*
We don ' t care about in case of an error below because current_db
was just set to 0.
2004-01-05 21:45:14 +02:00
*/
if ( ! mysql_query ( & mysql , " SELECT DATABASE() " ) & &
( res = mysql_use_result ( & mysql ) ) )
{
row = mysql_fetch_row ( res ) ;
2004-09-07 14:24:47 +05:00
if ( row [ 0 ] )
2004-01-05 21:45:14 +02:00
{
current_db = my_strdup ( row [ 0 ] , MYF ( MY_WME ) ) ;
}
2004-09-06 16:48:11 +05:00
( void ) mysql_fetch_row ( res ) ; // Read eof
2004-01-05 21:45:14 +02:00
mysql_free_result ( res ) ;
}
2004-09-06 16:48:11 +05:00
2004-09-07 22:10:15 +02:00
if ( ! current_db | | cmp_database ( charset_info , current_db , tmp ) )
2000-07-31 21:29:14 +02:00
{
if ( one_database )
2002-04-02 20:29:53 +03:00
skip_updates = 1 ;
2000-07-31 21:29:14 +02:00
else
{
/*
reconnect once if connection is down or if connection was found to
be down during query
*/
if ( ! connected & & reconnect ( ) )
2003-03-01 23:34:34 +01:00
return opt_reconnect ? - 1 : 1 ; // Fatal error
2000-07-31 21:29:14 +02:00
if ( mysql_select_db ( & mysql , tmp ) )
{
if ( mysql_errno ( & mysql ) ! = CR_SERVER_GONE_ERROR )
2003-06-04 18:28:51 +03:00
return put_error ( & mysql ) ;
2000-07-31 21:29:14 +02:00
if ( reconnect ( ) )
2003-03-01 23:34:34 +01:00
return opt_reconnect ? - 1 : 1 ; // Fatal error
2000-07-31 21:29:14 +02:00
if ( mysql_select_db ( & mysql , tmp ) )
2003-06-04 18:28:51 +03:00
return put_error ( & mysql ) ;
2000-07-31 21:29:14 +02:00
}
2001-01-12 13:52:32 +02:00
my_free ( current_db , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
current_db = my_strdup ( tmp , MYF ( MY_WME ) ) ;
2000-07-31 21:29:14 +02:00
# ifdef HAVE_READLINE
2002-04-16 11:39:03 +03:00
build_completion_hash ( rehash , 1 ) ;
2000-07-31 21:29:14 +02:00
# endif
}
}
else
2002-04-02 20:29:53 +03:00
skip_updates = 0 ;
2000-07-31 21:29:14 +02:00
put_info ( " Database changed " , INFO_INFO ) ;
return 0 ;
}
2002-12-25 23:57:01 +02:00
/*
Gets argument from a command on the command line . If get_next_arg is
not defined , skips the command and returns the first argument . The
line is modified by adding zero to the end of the argument . If
get_next_arg is defined , then the function searches for end of string
first , after found , returns the next argument and adds zero to the
end . If you ever wish to use this feature , remember to initialize all
items in the array to zero first .
*/
char * get_arg ( char * line , my_bool get_next_arg )
2002-12-11 16:03:50 +02:00
{
2003-04-23 03:29:03 +03:00
char * ptr , * start ;
2002-12-11 16:03:50 +02:00
my_bool quoted = 0 , valid_arg = 0 ;
2003-01-18 16:39:21 +02:00
char qtype = 0 ;
2002-12-11 16:03:50 +02:00
ptr = line ;
2002-12-25 23:57:01 +02:00
if ( get_next_arg )
{
2003-04-23 03:29:03 +03:00
for ( ; * ptr ; ptr + + ) ;
if ( * ( ptr + 1 ) )
2002-12-25 23:57:01 +02:00
ptr + + ;
}
else
{
/* skip leading white spaces */
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * ptr ) )
2002-12-25 23:57:01 +02:00
ptr + + ;
if ( * ptr = = ' \\ ' ) // short command was used
ptr + = 2 ;
2004-09-03 19:56:24 +04:00
else
while ( * ptr & & ! my_isspace ( charset_info , * ptr ) ) // skip command
ptr + + ;
2002-12-25 23:57:01 +02:00
}
2003-03-12 21:43:40 +02:00
if ( ! * ptr )
return NullS ;
2003-03-16 11:20:45 +04:00
while ( my_isspace ( charset_info , * ptr ) )
2002-12-11 16:03:50 +02:00
ptr + + ;
2003-01-18 16:39:21 +02:00
if ( * ptr = = ' \' ' | | * ptr = = ' \" ' | | * ptr = = ' ` ' )
2002-12-11 16:03:50 +02:00
{
2003-01-18 16:39:21 +02:00
qtype = * ptr ;
2002-12-11 16:03:50 +02:00
quoted = 1 ;
ptr + + ;
}
2003-04-23 03:29:03 +03:00
for ( start = ptr ; * ptr ; ptr + + )
2002-12-11 16:03:50 +02:00
{
2003-04-23 03:29:03 +03:00
if ( * ptr = = ' \\ ' & & ptr [ 1 ] ) // escaped character
2002-12-11 16:03:50 +02:00
{
2003-04-23 03:29:03 +03:00
// Remove the backslash
strmov ( ptr , ptr + 1 ) ;
2002-12-11 16:03:50 +02:00
}
2003-01-18 16:39:21 +02:00
else if ( ( ! quoted & & * ptr = = ' ' ) | | ( quoted & & * ptr = = qtype ) )
2002-12-11 16:03:50 +02:00
{
* ptr = 0 ;
break ;
}
}
2003-04-23 03:29:03 +03:00
valid_arg = ptr ! = start ;
return valid_arg ? start : NullS ;
2002-12-11 16:03:50 +02:00
}
2000-07-31 21:29:14 +02:00
static int
sql_real_connect ( char * host , char * database , char * user , char * password ,
uint silent )
{
2003-12-18 01:13:52 +02:00
if ( connected )
{
connected = 0 ;
mysql_close ( & mysql ) ;
}
2000-07-31 21:29:14 +02:00
mysql_init ( & mysql ) ;
2000-11-18 02:15:06 +02:00
if ( opt_connect_timeout )
2000-11-27 02:28:41 +02:00
{
uint timeout = opt_connect_timeout ;
2000-11-18 02:15:06 +02:00
mysql_options ( & mysql , MYSQL_OPT_CONNECT_TIMEOUT ,
2000-11-27 02:28:41 +02:00
( char * ) & timeout ) ;
}
2000-07-31 21:29:14 +02:00
if ( opt_compress )
mysql_options ( & mysql , MYSQL_OPT_COMPRESS , NullS ) ;
2003-11-28 13:11:44 +03:00
if ( opt_secure_auth )
mysql_options ( & mysql , MYSQL_SECURE_AUTH , ( char * ) & opt_secure_auth ) ;
2002-04-03 13:36:01 +03:00
if ( using_opt_local_infile )
mysql_options ( & mysql , MYSQL_OPT_LOCAL_INFILE , ( char * ) & opt_local_infile ) ;
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-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
if ( safe_updates )
{
char init_command [ 100 ] ;
sprintf ( init_command ,
" SET SQL_SAFE_UPDATES=1,SQL_SELECT_LIMIT=%lu,SQL_MAX_JOIN_SIZE=%lu " ,
select_limit , max_join_size ) ;
mysql_options ( & mysql , MYSQL_INIT_COMMAND , init_command ) ;
}
2003-08-19 00:08:08 +03:00
if ( default_charset_used )
mysql_options ( & mysql , MYSQL_SET_CHARSET_NAME , default_charset ) ;
2002-04-02 20:29:53 +03:00
if ( ! mysql_real_connect ( & mysql , host , user , password ,
database , opt_mysql_port , opt_mysql_unix_port ,
2003-11-18 13:47:27 +02:00
connect_flag | CLIENT_MULTI_STATEMENTS ) )
2000-07-31 21:29:14 +02:00
{
if ( ! silent | |
( mysql_errno ( & mysql ) ! = CR_CONN_HOST_ERROR & &
mysql_errno ( & mysql ) ! = CR_CONNECTION_ERROR ) )
{
2003-06-04 18:28:51 +03:00
( void ) put_error ( & mysql ) ;
2000-07-31 21:29:14 +02:00
( void ) fflush ( stdout ) ;
return ignore_errors ? - 1 : 1 ; // Abort
}
return - 1 ; // Retryable
}
connected = 1 ;
2002-12-16 17:33:29 +04:00
# ifndef EMBEDDED_LIBRARY
2000-07-31 21:29:14 +02:00
mysql . reconnect = info_flag ? 1 : 0 ; // We want to know if this happens
2004-12-09 14:44:10 +01:00
# else
mysql . reconnect = 1 ;
2002-12-16 17:33:29 +04:00
# endif
2000-07-31 21:29:14 +02:00
# ifdef HAVE_READLINE
2002-04-16 11:39:03 +03:00
build_completion_hash ( rehash , 1 ) ;
2000-07-31 21:29:14 +02:00
# endif
return 0 ;
}
static int
sql_connect ( char * host , char * database , char * user , char * password , uint silent )
{
bool message = 0 ;
uint count = 0 ;
int error ;
for ( ; ; )
{
if ( ( error = sql_real_connect ( host , database , user , password , wait_flag ) ) > = 0 )
{
if ( count )
{
2000-11-17 14:33:29 +02:00
tee_fputs ( " \n " , stderr ) ;
2000-07-31 21:29:14 +02:00
( void ) fflush ( stderr ) ;
}
return error ;
}
if ( ! wait_flag )
return ignore_errors ? - 1 : 1 ;
if ( ! message & & ! silent )
{
message = 1 ;
2000-11-17 14:33:29 +02:00
tee_fputs ( " Waiting " , stderr ) ; ( void ) fflush ( stderr ) ;
2000-07-31 21:29:14 +02:00
}
2001-01-17 05:47:33 -07:00
( void ) sleep ( wait_time ) ;
2000-07-31 21:29:14 +02:00
if ( ! silent )
{
putc ( ' . ' , stderr ) ; ( void ) fflush ( stderr ) ;
count + + ;
}
}
}
static int
com_status ( String * buffer __attribute__ ( ( unused ) ) ,
char * line __attribute__ ( ( unused ) ) )
{
2002-04-03 13:36:01 +03:00
const char * status ;
2004-02-05 10:22:08 +01:00
char buff [ 22 ] ;
ulonglong id ;
2004-09-13 12:51:08 +05:00
MYSQL_RES * result ;
LINT_INIT ( result ) ;
2004-02-05 10:22:08 +01:00
2000-11-17 14:33:29 +02:00
tee_puts ( " -------------- " , stdout ) ;
2000-07-31 21:29:14 +02:00
usage ( 1 ) ; /* Print version */
if ( connected )
{
2004-09-15 10:06:30 +05:00
tee_fprintf ( stdout , " \n Connection id: \t \t %lu \n " , mysql_thread_id ( & mysql ) ) ;
2004-09-15 13:26:47 +05:00
/*
Don ' t remove " limit 1 " ,
it is protection againts SQL_SELECT_LIMIT = 0
*/
2004-09-15 10:06:30 +05:00
if ( ! mysql_query ( & mysql , " select DATABASE(), USER() limit 1 " ) & &
( result = mysql_use_result ( & mysql ) ) )
{
MYSQL_ROW cur = mysql_fetch_row ( result ) ;
if ( cur )
{
tee_fprintf ( stdout , " Current database: \t %s \n " , cur [ 0 ] ? cur [ 0 ] : " " ) ;
tee_fprintf ( stdout , " Current user: \t \t %s \n " , cur [ 1 ] ) ;
}
mysql_free_result ( result ) ;
}
2002-04-22 23:36:34 +03:00
# ifdef HAVE_OPENSSL
2003-08-27 02:51:39 +03:00
if ( mysql . net . vio & & mysql . net . vio - > ssl_arg & &
SSL_get_cipher ( ( SSL * ) mysql . net . vio - > ssl_arg ) )
2002-05-31 12:34:47 +03:00
tee_fprintf ( stdout , " SSL: \t \t \t Cipher in use is %s \n " ,
2003-08-27 02:51:39 +03:00
SSL_get_cipher ( ( SSL * ) mysql . net . vio - > ssl_arg ) ) ;
2002-04-22 23:36:34 +03:00
else
# endif /* HAVE_OPENSSL */
2002-05-31 12:34:47 +03:00
tee_puts ( " SSL: \t \t \t Not in use " , stdout ) ;
2000-07-31 21:29:14 +02:00
}
else
{
vidattr ( A_BOLD ) ;
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " \n No connection \n " ) ;
2000-07-31 21:29:14 +02:00
vidattr ( A_NORMAL ) ;
return 0 ;
}
if ( skip_updates )
{
vidattr ( A_BOLD ) ;
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " \n All updates ignored to this database \n " ) ;
2000-07-31 21:29:14 +02:00
vidattr ( A_NORMAL ) ;
}
2003-01-28 08:38:28 +02:00
# ifdef USE_POPEN
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " Current pager: \t \t %s \n " , pager ) ;
2000-11-28 04:47:47 +02:00
tee_fprintf ( stdout , " Using outfile: \t \t '%s' \n " , opt_outfile ? outfile : " " ) ;
2000-11-17 14:33:29 +02:00
# endif
2003-04-23 03:29:03 +03:00
tee_fprintf ( stdout , " Using delimiter: \t %s \n " , delimiter ) ;
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " Server version: \t \t %s \n " , mysql_get_server_info ( & mysql ) ) ;
tee_fprintf ( stdout , " Protocol version: \t %d \n " , mysql_get_proto_info ( & mysql ) ) ;
tee_fprintf ( stdout , " Connection: \t \t %s \n " , mysql_get_host_info ( & mysql ) ) ;
2004-02-05 10:22:08 +01:00
if ( ( id = mysql_insert_id ( & mysql ) ) )
tee_fprintf ( stdout , " Insert id: \t \t %s \n " , llstr ( id , buff ) ) ;
2004-09-15 13:26:47 +05:00
/*
Don ' t remove " limit 1 " ,
it is protection againts SQL_SELECT_LIMIT = 0
*/
if ( ! mysql_query ( & mysql , " select @@character_set_client, @@character_set_connection, @@character_set_server, @@character_set_database limit 1 " ) & &
2004-09-13 12:51:08 +05:00
( result = mysql_use_result ( & mysql ) ) )
{
MYSQL_ROW cur = mysql_fetch_row ( result ) ;
if ( cur )
{
2005-02-22 12:51:23 +02:00
tee_fprintf ( stdout , " Server characterset: \t %s \n " , cur [ 2 ] ? cur [ 2 ] : " " ) ;
2004-09-13 12:51:08 +05:00
tee_fprintf ( stdout , " Db characterset: \t %s \n " , cur [ 3 ] ? cur [ 3 ] : " " ) ;
2005-02-22 12:51:23 +02:00
tee_fprintf ( stdout , " Client characterset: \t %s \n " , cur [ 0 ] ? cur [ 0 ] : " " ) ;
2004-09-13 12:51:08 +05:00
tee_fprintf ( stdout , " Conn. characterset: \t %s \n " , cur [ 1 ] ? cur [ 1 ] : " " ) ;
}
mysql_free_result ( result ) ;
}
2004-09-15 10:06:30 +05:00
else
{
/* Probably pre-4.1 server */
tee_fprintf ( stdout , " Client characterset: \t %s \n " , charset_info - > csname ) ;
tee_fprintf ( stdout , " Server characterset: \t %s \n " , mysql . charset - > csname ) ;
}
2004-09-13 12:51:08 +05:00
2002-12-16 17:33:29 +04:00
# ifndef EMBEDDED_LIBRARY
2000-07-31 21:29:14 +02:00
if ( strstr ( mysql_get_host_info ( & mysql ) , " TCP/IP " ) | | ! mysql . unix_socket )
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " TCP port: \t \t %d \n " , mysql . port ) ;
2000-07-31 21:29:14 +02:00
else
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " UNIX socket: \t \t %s \n " , mysql . unix_socket ) ;
2000-11-28 04:47:47 +02:00
if ( mysql . net . compress )
tee_fprintf ( stdout , " Protocol: \t \t Compressed \n " ) ;
2002-12-16 17:33:29 +04:00
# endif
2000-11-28 04:47:47 +02:00
2000-07-31 21:29:14 +02:00
if ( ( status = mysql_stat ( & mysql ) ) & & ! mysql_error ( & mysql ) [ 0 ] )
{
ulong sec ;
2002-07-23 18:31:22 +03:00
char buff [ 40 ] ;
const char * pos = strchr ( status , ' ' ) ;
/* print label */
tee_fprintf ( stdout , " %.*s \t \t \t " , ( int ) ( pos - status ) , status ) ;
2000-07-31 21:29:14 +02:00
if ( ( status = str2int ( pos , 10 , 0 , LONG_MAX , ( long * ) & sec ) ) )
{
nice_time ( ( double ) sec , buff , 0 ) ;
2000-11-17 14:33:29 +02:00
tee_puts ( buff , stdout ) ; /* print nice time */
2000-07-31 21:29:14 +02:00
while ( * status = = ' ' ) status + + ; /* to next info */
}
if ( status )
{
2000-11-18 17:34:15 +02:00
tee_putc ( ' \n ' , stdout ) ;
2000-11-17 14:33:29 +02:00
tee_puts ( status , stdout ) ;
2000-07-31 21:29:14 +02:00
}
}
if ( safe_updates )
{
vidattr ( A_BOLD ) ;
2001-02-20 20:39:48 -06:00
tee_fprintf ( stdout , " \n Note that you are running in safe_update_mode: \n " ) ;
2000-07-31 21:29:14 +02:00
vidattr ( A_NORMAL ) ;
2000-11-17 14:33:29 +02:00
tee_fprintf ( stdout , " \
2001-02-20 20:39:48 -06:00
UPDATEs and DELETEs that don ' t use a key in the WHERE clause are not allowed . \ n \
( One can force an UPDATE / DELETE by adding LIMIT # at the end of the command . ) \ n \
SELECT has an automatic ' LIMIT % lu ' if LIMIT is not used . \ n \
2000-07-31 21:29:14 +02:00
Max number of examined row combination in a join is set to : % lu \ n \ n " ,
2001-02-20 20:39:48 -06:00
select_limit , max_join_size ) ;
2000-07-31 21:29:14 +02:00
}
2000-11-17 14:33:29 +02:00
tee_puts ( " -------------- \n " , stdout ) ;
2000-07-31 21:29:14 +02:00
return 0 ;
}
static int
2003-06-04 18:28:51 +03:00
put_info ( const char * str , INFO_TYPE info_type , uint error , const char * sqlstate )
2000-07-31 21:29:14 +02:00
{
2003-04-03 19:55:37 +03:00
FILE * file = ( info_type = = INFO_ERROR ? stderr : stdout ) ;
2000-07-31 21:29:14 +02:00
static int inited = 0 ;
2001-12-06 14:10:51 +02:00
2000-07-31 21:29:14 +02:00
if ( status . batch )
{
if ( info_type = = INFO_ERROR )
{
2003-04-03 19:55:37 +03:00
( void ) fflush ( file ) ;
fprintf ( file , " ERROR " ) ;
2000-07-31 21:29:14 +02:00
if ( error )
2004-09-08 13:28:11 -05:00
{
if ( sqlstate )
( void ) fprintf ( file , " %d (%s) " , error , sqlstate ) ;
else
( void ) fprintf ( file , " %d " , error ) ;
}
2002-04-02 20:29:53 +03:00
if ( status . query_start_line & & line_numbers )
2000-07-31 21:29:14 +02:00
{
2003-04-03 19:55:37 +03:00
( void ) fprintf ( file , " at line %lu " , status . query_start_line ) ;
2000-07-31 21:29:14 +02:00
if ( status . file_name )
2003-04-03 19:55:37 +03:00
( void ) fprintf ( file , " in file: '%s' " , status . file_name ) ;
2000-07-31 21:29:14 +02:00
}
2003-04-03 19:55:37 +03:00
( void ) fprintf ( file , " : %s \n " , str ) ;
( void ) fflush ( file ) ;
2000-07-31 21:29:14 +02:00
if ( ! ignore_errors )
return 1 ;
}
else if ( info_type = = INFO_RESULT & & verbose > 1 )
2003-04-03 19:55:37 +03:00
tee_puts ( str , file ) ;
2000-07-31 21:29:14 +02:00
if ( unbuffered )
2003-04-03 19:55:37 +03:00
fflush ( file ) ;
2000-07-31 21:29:14 +02:00
return info_type = = INFO_ERROR ? - 1 : 0 ;
}
if ( ! opt_silent | | info_type = = INFO_ERROR )
{
if ( ! inited )
{
inited = 1 ;
# ifdef HAVE_SETUPTERM
( void ) setupterm ( ( char * ) 0 , 1 , ( int * ) 0 ) ;
# endif
}
if ( info_type = = INFO_ERROR )
{
2002-06-11 11:20:31 +03:00
if ( ! opt_nobeep )
2002-04-03 13:36:01 +03:00
putchar ( ' \007 ' ) ; /* This should make a bell */
2000-07-31 21:29:14 +02:00
vidattr ( A_STANDOUT ) ;
if ( error )
2003-06-04 18:28:51 +03:00
{
if ( sqlstate )
( void ) tee_fprintf ( file , " ERROR %d (%s): " , error , sqlstate ) ;
else
( void ) tee_fprintf ( file , " ERROR %d: " , error ) ;
}
2000-07-31 21:29:14 +02:00
else
2003-04-03 19:55:37 +03:00
tee_puts ( " ERROR: " , file ) ;
2000-07-31 21:29:14 +02:00
}
else
vidattr ( A_BOLD ) ;
2003-04-03 19:55:37 +03:00
( void ) tee_puts ( str , file ) ;
2000-07-31 21:29:14 +02:00
vidattr ( A_NORMAL ) ;
}
if ( unbuffered )
2003-04-03 19:55:37 +03:00
fflush ( file ) ;
2000-07-31 21:29:14 +02:00
return info_type = = INFO_ERROR ? - 1 : 0 ;
}
2000-11-17 14:33:29 +02:00
2003-06-04 18:28:51 +03:00
static int
put_error ( MYSQL * mysql )
{
return put_info ( mysql_error ( mysql ) , INFO_ERROR , mysql_errno ( mysql ) ,
mysql_sqlstate ( mysql ) ) ;
}
2000-07-31 21:29:14 +02:00
static void remove_cntrl ( String & buffer )
{
char * start , * end ;
end = ( start = ( char * ) buffer . ptr ( ) ) + buffer . length ( ) ;
2003-03-16 11:20:45 +04:00
while ( start < end & & ! my_isgraph ( charset_info , end [ - 1 ] ) )
2000-07-31 21:29:14 +02:00
end - - ;
buffer . length ( ( uint ) ( end - start ) ) ;
}
2000-11-17 14:33:29 +02:00
void tee_fprintf ( FILE * file , const char * fmt , . . . )
{
va_list args ;
2004-05-25 22:00:14 +03:00
NETWARE_YIELD ;
2000-11-17 14:33:29 +02:00
va_start ( args , fmt ) ;
2000-11-18 02:15:06 +02:00
( void ) vfprintf ( file , fmt , args ) ;
2001-08-22 01:45:07 +03:00
# ifdef OS2
fflush ( file ) ;
# endif
2000-11-17 14:33:29 +02:00
if ( opt_outfile )
2000-11-18 02:15:06 +02:00
( void ) vfprintf ( OUTFILE , fmt , args ) ;
2000-11-17 14:33:29 +02:00
va_end ( args ) ;
}
void tee_fputs ( const char * s , FILE * file )
{
2004-05-25 22:00:14 +03:00
NETWARE_YIELD ;
2000-11-17 14:33:29 +02:00
fputs ( s , file ) ;
2001-08-22 01:45:07 +03:00
# ifdef OS2
fflush ( file ) ;
# endif
2000-11-17 14:33:29 +02:00
if ( opt_outfile )
fputs ( s , OUTFILE ) ;
}
void tee_puts ( const char * s , FILE * file )
{
2004-05-25 22:00:14 +03:00
NETWARE_YIELD ;
2000-11-17 14:33:29 +02:00
fputs ( s , file ) ;
fputs ( " \n " , file ) ;
2001-08-22 01:45:07 +03:00
# ifdef OS2
fflush ( file ) ;
# endif
2000-11-17 14:33:29 +02:00
if ( opt_outfile )
{
fputs ( s , OUTFILE ) ;
fputs ( " \n " , OUTFILE ) ;
}
}
2000-11-18 17:34:15 +02:00
void tee_putc ( int c , FILE * file )
{
putc ( c , file ) ;
2001-08-22 01:45:07 +03:00
# ifdef OS2
fflush ( file ) ;
# endif
2000-11-18 17:34:15 +02:00
if ( opt_outfile )
putc ( c , OUTFILE ) ;
}
2003-01-28 08:38:28 +02:00
# if defined( __WIN__) || defined( OS2) || defined(__NETWARE__)
2000-07-31 21:29:14 +02:00
# include <time.h>
# else
# include <sys/times.h>
2002-06-20 23:26:39 +03:00
# ifdef _SC_CLK_TCK // For mit-pthreads
2000-07-31 21:29:14 +02:00
# undef CLOCKS_PER_SEC
# define CLOCKS_PER_SEC (sysconf(_SC_CLK_TCK))
# endif
2002-06-20 23:26:39 +03:00
# endif
2000-07-31 21:29:14 +02:00
static ulong start_timer ( void )
{
2003-01-28 08:38:28 +02:00
# if defined( __WIN__) || defined( OS2) || defined(__NETWARE__)
2000-07-31 21:29:14 +02:00
return clock ( ) ;
# else
struct tms tms_tmp ;
return times ( & tms_tmp ) ;
# endif
}
static void nice_time ( double sec , char * buff , bool part_second )
{
ulong tmp ;
if ( sec > = 3600.0 * 24 )
{
tmp = ( ulong ) floor ( sec / ( 3600.0 * 24 ) ) ;
sec - = 3600.0 * 24 * tmp ;
2004-05-27 17:54:40 +04:00
buff = int10_to_str ( ( long ) tmp , buff , 10 ) ;
2000-07-31 21:29:14 +02:00
buff = strmov ( buff , tmp > 1 ? " days " : " day " ) ;
}
if ( sec > = 3600.0 )
{
tmp = ( ulong ) floor ( sec / 3600.0 ) ;
sec - = 3600.0 * tmp ;
2004-05-27 17:54:40 +04:00
buff = int10_to_str ( ( long ) tmp , buff , 10 ) ;
2000-07-31 21:29:14 +02:00
buff = strmov ( buff , tmp > 1 ? " hours " : " hour " ) ;
}
if ( sec > = 60.0 )
{
tmp = ( ulong ) floor ( sec / 60.0 ) ;
sec - = 60.0 * tmp ;
2004-05-27 17:54:40 +04:00
buff = int10_to_str ( ( long ) tmp , buff , 10 ) ;
2000-07-31 21:29:14 +02:00
buff = strmov ( buff , " min " ) ;
}
if ( part_second )
sprintf ( buff , " %.2f sec " , sec ) ;
else
sprintf ( buff , " %d sec " , ( int ) sec ) ;
}
static void end_timer ( ulong start_time , char * buff )
{
nice_time ( ( double ) ( start_timer ( ) - start_time ) /
CLOCKS_PER_SEC , buff , 1 ) ;
}
static void mysql_end_timer ( ulong start_time , char * buff )
{
buff [ 0 ] = ' ' ;
buff [ 1 ] = ' ( ' ;
end_timer ( start_time , buff + 2 ) ;
strmov ( strend ( buff ) , " ) " ) ;
}
2002-06-11 11:20:31 +03:00
static const char * construct_prompt ( )
{
2003-07-03 11:55:36 +03:00
processed_prompt . free ( ) ; // Erase the old prompt
time_t lclock = time ( NULL ) ; // Get the date struct
2002-04-03 13:36:01 +03:00
struct tm * t = localtime ( & lclock ) ;
2003-07-03 11:55:36 +03:00
/* parse thru the settings for the prompt */
2002-06-11 11:20:31 +03:00
for ( char * c = current_prompt ; * c ; * c + + )
{
if ( * c ! = PROMPT_CHAR )
2002-04-03 13:36:01 +03:00
processed_prompt . append ( * c ) ;
2002-06-11 11:20:31 +03:00
else
{
2002-04-03 13:36:01 +03:00
switch ( * + + c ) {
case ' \0 ' :
2003-07-03 11:55:36 +03:00
c - - ; // stop it from going beyond if ends with %
2002-04-03 13:36:01 +03:00
break ;
case ' c ' :
add_int_to_prompt ( + + prompt_counter ) ;
break ;
case ' v ' :
2003-05-27 16:40:14 +03:00
if ( connected )
processed_prompt . append ( mysql_get_server_info ( & mysql ) ) ;
else
processed_prompt . append ( " not_connected " ) ;
2002-04-03 13:36:01 +03:00
break ;
case ' d ' :
processed_prompt . append ( current_db ? current_db : " (none) " ) ;
break ;
case ' h ' :
{
2003-05-27 16:40:14 +03:00
const char * prompt ;
prompt = connected ? mysql_get_host_info ( & mysql ) : " not_connected " ;
2002-04-03 13:36:01 +03:00
if ( strstr ( prompt , " Localhost " ) )
processed_prompt . append ( " localhost " ) ;
else
{
const char * end = strcend ( prompt , ' ' ) ;
processed_prompt . append ( prompt , ( uint ) ( end - prompt ) ) ;
}
break ;
}
case ' p ' :
2005-01-15 23:19:34 -06:00
{
2002-12-16 17:33:29 +04:00
# ifndef EMBEDDED_LIBRARY
2003-05-27 16:40:14 +03:00
if ( ! connected )
{
processed_prompt . append ( " not_connected " ) ;
break ;
}
2005-01-15 23:19:34 -06:00
const char * host_info = mysql_get_host_info ( & mysql ) ;
if ( strstr ( host_info , " memory " ) )
{
processed_prompt . append ( mysql . host ) ;
}
else if ( strstr ( host_info , " TCP/IP " ) | |
2003-06-04 19:21:51 +03:00
! mysql . unix_socket )
2002-04-03 13:36:01 +03:00
add_int_to_prompt ( mysql . port ) ;
else
2003-02-02 05:13:09 +02:00
{
char * pos = strrchr ( mysql . unix_socket , ' / ' ) ;
2003-02-04 21:52:14 +02:00
processed_prompt . append ( pos ? pos + 1 : mysql . unix_socket ) ;
2003-02-02 05:13:09 +02:00
}
2002-12-16 17:33:29 +04:00
# endif
2005-01-15 23:19:34 -06:00
}
2002-04-03 13:36:01 +03:00
break ;
case ' U ' :
if ( ! full_username )
init_username ( ) ;
processed_prompt . append ( full_username ) ;
break ;
case ' u ' :
if ( ! full_username )
init_username ( ) ;
processed_prompt . append ( part_username ) ;
break ;
case PROMPT_CHAR :
processed_prompt . append ( PROMPT_CHAR ) ;
break ;
case ' n ' :
processed_prompt . append ( ' \n ' ) ;
break ;
case ' ' :
case ' _ ' :
processed_prompt . append ( ' ' ) ;
break ;
case ' R ' :
2004-03-08 15:52:06 -06:00
if ( t - > tm_hour < 10 )
processed_prompt . append ( ' 0 ' ) ;
2002-04-03 13:36:01 +03:00
add_int_to_prompt ( t - > tm_hour ) ;
break ;
case ' r ' :
int getHour ;
getHour = t - > tm_hour % 12 ;
if ( getHour = = 0 )
getHour = 12 ;
2004-03-08 15:52:06 -06:00
if ( getHour < 10 )
processed_prompt . append ( ' 0 ' ) ;
2002-04-03 13:36:01 +03:00
add_int_to_prompt ( getHour ) ;
break ;
case ' m ' :
if ( t - > tm_min < 10 )
processed_prompt . append ( ' 0 ' ) ;
add_int_to_prompt ( t - > tm_min ) ;
break ;
case ' y ' :
int getYear ;
getYear = t - > tm_year % 100 ;
if ( getYear < 10 )
processed_prompt . append ( ' 0 ' ) ;
add_int_to_prompt ( getYear ) ;
break ;
case ' Y ' :
add_int_to_prompt ( t - > tm_year + 1900 ) ;
break ;
case ' D ' :
char * dateTime ;
time_t lclock ;
lclock = time ( NULL ) ;
dateTime = ctime ( & lclock ) ;
processed_prompt . append ( strtok ( dateTime , " \n " ) ) ;
break ;
case ' s ' :
2004-03-08 15:52:06 -06:00
if ( t - > tm_sec < 10 )
processed_prompt . append ( ' 0 ' ) ;
2002-04-03 13:36:01 +03:00
add_int_to_prompt ( t - > tm_sec ) ;
break ;
case ' w ' :
processed_prompt . append ( day_names [ t - > tm_wday ] ) ;
break ;
case ' P ' :
processed_prompt . append ( t - > tm_hour < 12 ? " am " : " pm " ) ;
break ;
case ' o ' :
add_int_to_prompt ( t - > tm_mon + 1 ) ;
break ;
case ' O ' :
processed_prompt . append ( month_names [ t - > tm_mon ] ) ;
break ;
case ' \' ' :
processed_prompt . append ( " ' " ) ;
break ;
case ' " ' :
processed_prompt . append ( ' " ' ) ;
break ;
case ' S ' :
processed_prompt . append ( ' ; ' ) ;
break ;
case ' t ' :
processed_prompt . append ( ' \t ' ) ;
break ;
default :
processed_prompt . append ( c ) ;
}
}
}
processed_prompt . append ( ' \0 ' ) ;
return processed_prompt . ptr ( ) ;
}
2002-06-11 11:20:31 +03:00
static void add_int_to_prompt ( int toadd )
{
2002-04-03 13:36:01 +03:00
char buffer [ 16 ] ;
int10_to_str ( toadd , buffer , 10 ) ;
processed_prompt . append ( buffer ) ;
}
2002-06-11 11:20:31 +03:00
static void init_username ( )
{
2002-04-03 13:36:01 +03:00
my_free ( full_username , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
my_free ( part_username , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
MYSQL_RES * result ;
LINT_INIT ( result ) ;
if ( ! mysql_query ( & mysql , " select USER() " ) & &
( result = mysql_use_result ( & mysql ) ) )
2002-06-11 11:20:31 +03:00
{
MYSQL_ROW cur = mysql_fetch_row ( result ) ;
full_username = my_strdup ( cur [ 0 ] , MYF ( MY_WME ) ) ;
part_username = my_strdup ( strtok ( cur [ 0 ] , " @ " ) , MYF ( MY_WME ) ) ;
( void ) mysql_fetch_row ( result ) ; // Read eof
}
2002-04-03 13:36:01 +03:00
}
2002-06-11 11:20:31 +03:00
static int com_prompt ( String * buffer , char * line )
{
char * ptr = strchr ( line , ' ' ) ;
2002-04-03 13:36:01 +03:00
prompt_counter = 0 ;
my_free ( current_prompt , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2002-06-11 11:20:31 +03:00
current_prompt = my_strdup ( ptr ? ptr + 1 : default_prompt , MYF ( MY_WME ) ) ;
if ( ! ptr )
2002-04-03 13:36:01 +03:00
tee_fprintf ( stdout , " Returning to default PROMPT of %s \n " , default_prompt ) ;
else
tee_fprintf ( stdout , " PROMPT set to '%s' \n " , current_prompt ) ;
return 0 ;
}
2001-09-22 17:40:57 +03:00
# ifndef EMBEDDED_LIBRARY
2000-07-31 21:29:14 +02:00
/* Keep sql_string library happy */
gptr sql_alloc ( unsigned int Size )
{
return my_malloc ( Size , MYF ( MY_WME ) ) ;
}
void sql_element_free ( void * ptr )
{
my_free ( ( gptr ) ptr , MYF ( 0 ) ) ;
}
2001-09-22 17:40:57 +03:00
# endif /* EMBEDDED_LIBRARY */