2001-12-06 13:10:51 +01:00
/* Copyright (C) 2000 MySQL AB
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 13:10:51 +01: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 13:10:51 +01: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 */
/*
* * mysqlimport . c - Imports all given files
* * into a table ( s ) .
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * *
* * * AUTHOR : Monty & Jani *
* * * DATE : June 24 , 1997 *
* * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * *
*/
2004-10-07 21:08:17 +02:00
# define IMPORT_VERSION "3.5"
2000-07-31 21:29:14 +02:00
2001-09-12 22:53:31 +02:00
# include "client_priv.h"
2000-07-31 21:29:14 +02:00
# include "mysql_version.h"
2006-02-26 03:03:11 +01:00
# ifdef HAVE_LIBPTHREAD
2006-02-13 09:45:59 +01:00
# include <my_pthread.h>
2006-02-26 03:03:11 +01:00
# endif
2006-02-13 09:45:59 +01:00
2006-02-13 13:04:53 +01:00
2006-02-13 09:45:59 +01:00
/* Global Thread counter */
int counter = 0 ;
2006-02-26 03:03:11 +01:00
# ifdef HAVE_LIBPTHREAD
2006-02-13 13:04:53 +01:00
pthread_mutex_t counter_mutex ;
2006-02-26 03:03:11 +01:00
# endif
2000-07-31 21:29:14 +02:00
static void db_error_with_table ( MYSQL * mysql , char * table ) ;
static void db_error ( MYSQL * mysql ) ;
static char * field_escape ( char * to , const char * from , uint length ) ;
static char * add_load_option ( char * ptr , const char * object ,
const char * statement ) ;
2001-08-22 00:45:07 +02:00
static my_bool verbose = 0 , lock_tables = 0 , ignore_errors = 0 , opt_delete = 0 ,
2005-09-02 22:52:52 +02:00
replace = 0 , silent = 0 , ignore = 0 , opt_compress = 0 ,
2002-04-02 19:29:53 +02:00
opt_low_priority = 0 , tty_password = 0 ;
2006-02-13 09:45:59 +01:00
static my_bool opt_use_threads = 0 ;
2005-09-02 22:52:52 +02:00
static uint opt_local_file = 0 ;
2000-11-13 22:55:10 +01:00
static char * opt_password = 0 , * current_user = 0 ,
2000-07-31 21:29:14 +02:00
* current_host = 0 , * current_db = 0 , * fields_terminated = 0 ,
* lines_terminated = 0 , * enclosed = 0 , * opt_enclosed = 0 ,
2003-03-16 08:20:45 +01:00
* escaped = 0 , * opt_columns = 0 ,
2003-05-28 13:23:04 +02:00
* default_charset = ( char * ) MYSQL_DEFAULT_CHARSET_NAME ;
2003-09-11 19:24:14 +02:00
static uint opt_mysql_port = 0 , opt_protocol = 0 ;
2000-07-31 21:29:14 +02:00
static my_string opt_mysql_unix_port = 0 ;
2003-09-07 00:59:04 +02:00
static longlong opt_ignore_lines = - 1 ;
2003-03-16 08:20:45 +01:00
static CHARSET_INFO * charset_info = & my_charset_latin1 ;
2002-09-17 22:46:53 +02:00
# include <sslopt-vars.h>
2000-07-31 21:29:14 +02:00
2002-11-14 20:16:30 +01:00
# ifdef HAVE_SMEM
static char * shared_memory_base_name = 0 ;
# endif
2002-04-02 19:29:53 +02:00
static struct my_option my_long_options [ ] =
2000-07-31 21:29:14 +02:00
{
2005-08-24 21:03:34 +02:00
# ifdef __NETWARE__
2005-11-18 18:25:46 +01:00
{ " autoclose " , OPT_AUTO_CLOSE , " Auto close the screen on exit for Netware. " ,
2005-08-24 21:03:34 +02:00
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2002-04-02 19:29:53 +02:00
{ " character-sets-dir " , OPT_CHARSETS_DIR ,
2003-06-13 10:59:02 +02:00
" Directory where character sets are. " , ( gptr * ) & charsets_dir ,
2002-04-02 19:29:53 +02:00
( gptr * ) & charsets_dir , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " 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 } ,
{ " columns " , ' c ' ,
" Use only these columns to import the data to. Give the column names in a comma separated list. This is same as giving columns to LOAD DATA INFILE. " ,
( gptr * ) & opt_columns , ( gptr * ) & opt_columns , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 ,
0 , 0 , 0 } ,
{ " 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 } ,
2003-06-13 10:59:02 +02:00
{ " debug " , ' # ' , " Output debug log. Often this is 'd:t:o,filename'. " , 0 , 0 , 0 ,
GET_STR , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 19:29:53 +02:00
{ " delete " , ' d ' , " First delete all rows from table. " , ( gptr * ) & opt_delete ,
( gptr * ) & opt_delete , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " fields-terminated-by " , OPT_FTB ,
" Fields in the textfile are terminated by ... " , ( gptr * ) & fields_terminated ,
( gptr * ) & fields_terminated , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " fields-enclosed-by " , OPT_ENC ,
" Fields in the importfile are enclosed by ... " , ( gptr * ) & enclosed ,
( gptr * ) & enclosed , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " fields-optionally-enclosed-by " , OPT_O_ENC ,
" Fields in the i.file are opt. enclosed by ... " , ( gptr * ) & opt_enclosed ,
( gptr * ) & opt_enclosed , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " fields-escaped-by " , OPT_ESC , " Fields in the i.file are escaped by ... " ,
( gptr * ) & escaped , ( gptr * ) & escaped , 0 , GET_STR , REQUIRED_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 } ,
{ " help " , ' ? ' , " Displays this help and exits. " , 0 , 0 , 0 , GET_NO_ARG , NO_ARG ,
0 , 0 , 0 , 0 , 0 , 0 } ,
{ " host " , ' h ' , " Connect to host. " , ( gptr * ) & current_host ,
( gptr * ) & current_host , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " ignore " , ' i ' , " If duplicate unique key was found, keep old row. " ,
( gptr * ) & ignore , ( gptr * ) & ignore , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-05-14 20:41:55 +02:00
{ " ignore-lines " , OPT_IGN_LINES , " Ignore first n lines of data infile. " ,
2003-09-07 00:59:04 +02:00
( gptr * ) & opt_ignore_lines , ( gptr * ) & opt_ignore_lines , 0 , GET_LL ,
2002-05-14 20:41:55 +02:00
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 19:29:53 +02:00
{ " lines-terminated-by " , OPT_LTB , " Lines in the i.file are terminated by ... " ,
( gptr * ) & lines_terminated , ( gptr * ) & lines_terminated , 0 , GET_STR ,
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " local " , ' L ' , " Read all files through the client. " , ( gptr * ) & opt_local_file ,
2002-04-02 19:29:53 +02:00
( gptr * ) & opt_local_file , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2006-02-13 13:04:53 +01:00
{ " lock-tables " , ' l ' , " Lock all tables for write (this disables threads). " ,
( gptr * ) & lock_tables , ( gptr * ) & lock_tables , 0 , GET_BOOL , NO_ARG ,
0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 19:29:53 +02:00
{ " low-priority " , OPT_LOW_PRIORITY ,
2003-06-13 10:59:02 +02:00
" Use LOW_PRIORITY when updating the table. " , ( gptr * ) & opt_low_priority ,
2002-04-02 19:29:53 +02:00
( gptr * ) & opt_low_priority , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " password " , ' p ' ,
" Password to use when connecting to server. If password is not given it's asked from the tty. " ,
0 , 0 , 0 , GET_STR , OPT_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# ifdef __WIN__
2002-04-02 19:29:53 +02: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-04-02 19:29:53 +02:00
{ " port " , ' P ' , " Port number to use for connection. " , ( gptr * ) & opt_mysql_port ,
2005-10-13 21:28:43 +02:00
( gptr * ) & opt_mysql_port , 0 , GET_UINT , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 ,
2002-05-11 13:36:34 +02:00
0 } ,
2003-06-13 10:59:02 +02:00
{ " protocol " , OPT_MYSQL_PROTOCOL , " The protocol of connection (tcp,socket,pipe,memory). " ,
2002-11-14 20:16:30 +01:00
0 , 0 , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 19:29:53 +02:00
{ " replace " , ' r ' , " If duplicate unique key was found, replace old row. " ,
( gptr * ) & replace , ( gptr * ) & replace , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-11-14 20:16:30 +01:00
# ifdef HAVE_SMEM
2004-08-30 18:11:10 +02: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-14 20:16:30 +01:00
0 , GET_STR_ALLOC , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2002-04-02 19:29:53 +02:00
{ " silent " , ' s ' , " Be more silent. " , ( gptr * ) & silent , ( gptr * ) & silent , 0 ,
GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " socket " , ' S ' , " Socket file to use for connection. " ,
( gptr * ) & opt_mysql_unix_port , ( gptr * ) & opt_mysql_unix_port , 0 , GET_STR ,
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-09-17 22:46:53 +02:00
# include <sslopt-longopts.h>
2006-02-13 09:45:59 +01:00
{ " use-threads " , OPT_USE_THREADS ,
2006-02-14 19:11:26 +01:00
" Load files in parallel. The argument is the number "
" of threads to use for loading data. " ,
( gptr * ) & opt_use_threads , ( gptr * ) & opt_use_threads , 0 ,
GET_UINT , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# ifndef DONT_ALLOW_USER_CHANGE
2002-04-02 19:29:53 +02:00
{ " user " , ' u ' , " User for login if not current user. " , ( gptr * ) & current_user ,
( gptr * ) & current_user , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2000-07-31 21:29:14 +02:00
# endif
2002-04-02 19:29:53 +02:00
{ " verbose " , ' v ' , " Print info about the various stages. " , ( gptr * ) & verbose ,
( gptr * ) & verbose , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " version " , ' V ' , " Output version information and exit. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 }
2000-07-31 21:29:14 +02:00
} ;
static const char * load_default_groups [ ] = { " mysqlimport " , " client " , 0 } ;
2004-05-25 21:00:14 +02:00
# include <help_start.h>
2000-07-31 21:29:14 +02:00
static void print_version ( void )
{
printf ( " %s Ver %s Distrib %s, for %s (%s) \n " , my_progname ,
IMPORT_VERSION , MYSQL_SERVER_VERSION , SYSTEM_TYPE , MACHINE_TYPE ) ;
2004-05-25 21:00:14 +02:00
NETWARE_SET_SCREEN_MODE ( 1 ) ;
2000-07-31 21:29:14 +02:00
}
static void usage ( void )
{
print_version ( ) ;
puts ( " Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB " ) ;
puts ( " This software comes with ABSOLUTELY NO WARRANTY. This is free software, \n and you are welcome to modify and redistribute it under the GPL license \n " ) ;
printf ( " \
Loads tables from text files in various formats . The base name of the \ n \
text file must be the name of the table that should be used . \ n \
If one uses sockets to connect to the MySQL server , the server will open and \ n \
read the text file directly . In other cases the client will open the text \ n \
file . The SQL command ' LOAD DATA INFILE ' is used to import the rows . \ n " );
printf ( " \n Usage: %s [OPTIONS] database textfile... " , my_progname ) ;
print_defaults ( " my " , load_default_groups ) ;
2002-04-02 19:29:53 +02:00
my_print_help ( my_long_options ) ;
my_print_variables ( my_long_options ) ;
2000-07-31 21:29:14 +02:00
}
2004-05-25 21:00:14 +02:00
# include <help_end.h>
2000-07-31 21:29:14 +02:00
2002-04-02 19:29:53 +02:00
static my_bool
get_one_option ( int optid , const struct my_option * opt __attribute__ ( ( unused ) ) ,
char * argument )
{
switch ( optid ) {
2005-08-24 21:03:34 +02:00
# ifdef __NETWARE__
case OPT_AUTO_CLOSE :
setscreenmode ( SCR_AUTOCLOSE_ON_EXIT ) ;
break ;
# endif
2002-04-02 19:29:53 +02:00
case ' p ' :
if ( argument )
{
char * start = argument ;
my_free ( opt_password , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
opt_password = my_strdup ( argument , MYF ( MY_FAE ) ) ;
while ( * argument ) * argument + + = ' x ' ; /* Destroy argument */
if ( * start )
start [ 1 ] = 0 ; /* Cut length of argument */
2004-10-07 21:08:17 +02:00
tty_password = 0 ;
2002-04-02 19:29:53 +02:00
}
else
tty_password = 1 ;
break ;
2000-07-31 21:29:14 +02:00
# ifdef __WIN__
2002-04-02 19:29:53 +02:00
case ' W ' :
2002-11-14 20:16:30 +01:00
opt_protocol = MYSQL_PROTOCOL_PIPE ;
2002-04-02 19:29:53 +02:00
opt_local_file = 1 ;
break ;
2000-07-31 21:29:14 +02:00
# endif
2002-11-14 20:16:30 +01:00
case OPT_MYSQL_PROTOCOL :
{
2004-08-24 17:13:31 +02:00
if ( ( opt_protocol = find_type ( argument , & sql_protocol_typelib , 0 ) ) < = 0 )
2002-11-14 20:16:30 +01:00
{
fprintf ( stderr , " Unknown option to protocol: %s \n " , argument ) ;
exit ( 1 ) ;
}
break ;
}
2002-04-02 19:29:53 +02:00
case ' # ' :
DBUG_PUSH ( argument ? argument : " d:t:o " ) ;
break ;
2002-09-17 22:46:53 +02:00
# include <sslopt-case.h>
2002-04-02 19:29:53 +02:00
case ' V ' : print_version ( ) ; exit ( 0 ) ;
case ' I ' :
case ' ? ' :
usage ( ) ;
exit ( 0 ) ;
}
return 0 ;
}
static int get_options ( int * argc , char * * * argv )
{
int ho_error ;
2004-08-31 18:27:58 +02:00
if ( ( ho_error = handle_options ( argc , argv , my_long_options , get_one_option ) ) )
2002-05-29 14:07:30 +02:00
exit ( ho_error ) ;
2000-07-31 21:29:14 +02:00
if ( enclosed & & opt_enclosed )
{
fprintf ( stderr , " You can't use ..enclosed.. and ..optionally-enclosed.. at the same time. \n " ) ;
return ( 1 ) ;
}
if ( replace & & ignore )
{
fprintf ( stderr , " You can't use --ignore (-i) and --replace (-r) at the same time. \n " ) ;
return ( 1 ) ;
}
2004-02-11 12:15:39 +01:00
if ( strcmp ( default_charset , charset_info - > csname ) & &
! ( charset_info = get_charset_by_csname ( default_charset ,
2003-05-22 11:37:01 +02:00
MY_CS_PRIMARY , MYF ( MY_WME ) ) ) )
2003-03-16 08:20:45 +01:00
exit ( 1 ) ;
2000-07-31 21:29:14 +02:00
if ( * argc < 2 )
{
usage ( ) ;
return 1 ;
}
current_db = * ( ( * argv ) + + ) ;
( * argc ) - - ;
if ( tty_password )
2000-11-13 22:55:10 +01:00
opt_password = get_tty_password ( NullS ) ;
2000-07-31 21:29:14 +02:00
return ( 0 ) ;
}
2006-02-17 15:52:05 +01:00
static int write_to_table ( char * filename , MYSQL * mysql )
2000-07-31 21:29:14 +02:00
{
char tablename [ FN_REFLEN ] , hard_path [ FN_REFLEN ] ,
2003-08-16 16:03:48 +02:00
sql_statement [ FN_REFLEN * 16 + 256 ] , * end ;
2000-07-31 21:29:14 +02:00
DBUG_ENTER ( " write_to_table " ) ;
DBUG_PRINT ( " enter " , ( " filename: %s " , filename ) ) ;
fn_format ( tablename , filename , " " , " " , 1 | 2 ) ; /* removes path & ext. */
2006-02-20 13:04:40 +01:00
if ( ! opt_local_file )
2000-07-31 21:29:14 +02:00
strmov ( hard_path , filename ) ;
else
my_load_path ( hard_path , filename , NULL ) ; /* filename includes the path */
2001-08-22 00:45:07 +02:00
if ( opt_delete )
2000-07-31 21:29:14 +02:00
{
if ( verbose )
fprintf ( stdout , " Deleting the old data from table %s \n " , tablename ) ;
2006-02-18 08:46:18 +01:00
# ifdef HAVE_SNPRINTF
2006-02-13 09:45:59 +01:00
snprintf ( sql_statement , FN_REFLEN * 16 + 256 , " DELETE FROM %s " , tablename ) ;
2006-02-18 08:46:18 +01:00
# else
sprintf ( sql_statement , " DELETE FROM %s " , tablename ) ;
# endif
2006-02-17 15:52:05 +01:00
if ( mysql_query ( mysql , sql_statement ) )
2000-07-31 21:29:14 +02:00
{
2006-02-17 15:52:05 +01:00
db_error_with_table ( mysql , tablename ) ;
2000-07-31 21:29:14 +02:00
DBUG_RETURN ( 1 ) ;
}
}
to_unix_path ( hard_path ) ;
if ( verbose )
{
2004-10-04 20:26:43 +02:00
if ( opt_local_file )
2000-07-31 21:29:14 +02:00
fprintf ( stdout , " Loading data from LOCAL file: %s into %s \n " ,
hard_path , tablename ) ;
else
fprintf ( stdout , " Loading data from SERVER file: %s into %s \n " ,
hard_path , tablename ) ;
}
sprintf ( sql_statement , " LOAD DATA %s %s INFILE '%s' " ,
opt_low_priority ? " LOW_PRIORITY " : " " ,
2004-10-04 20:26:43 +02:00
opt_local_file ? " LOCAL " : " " , hard_path ) ;
2000-07-31 21:29:14 +02:00
end = strend ( sql_statement ) ;
if ( replace )
end = strmov ( end , " REPLACE " ) ;
if ( ignore )
end = strmov ( end , " IGNORE " ) ;
end = strmov ( strmov ( end , " INTO TABLE " ) , tablename ) ;
2001-12-06 13:10:51 +01:00
2000-07-31 21:29:14 +02:00
if ( fields_terminated | | enclosed | | opt_enclosed | | escaped )
end = strmov ( end , " FIELDS " ) ;
end = add_load_option ( end , fields_terminated , " TERMINATED BY " ) ;
end = add_load_option ( end , enclosed , " ENCLOSED BY " ) ;
end = add_load_option ( end , opt_enclosed ,
" OPTIONALLY ENCLOSED BY " ) ;
end = add_load_option ( end , escaped , " ESCAPED BY " ) ;
end = add_load_option ( end , lines_terminated , " LINES TERMINATED BY " ) ;
2003-09-07 00:59:04 +02:00
if ( opt_ignore_lines > = 0 )
end = strmov ( longlong10_to_str ( opt_ignore_lines ,
strmov ( end , " IGNORE " ) , 10 ) , " LINES " ) ;
2000-07-31 21:29:14 +02:00
if ( opt_columns )
end = strmov ( strmov ( strmov ( end , " ( " ) , opt_columns ) , " ) " ) ;
* end = ' \0 ' ;
2006-02-17 15:52:05 +01:00
if ( mysql_query ( mysql , sql_statement ) )
2000-07-31 21:29:14 +02:00
{
2006-02-17 15:52:05 +01:00
db_error_with_table ( mysql , tablename ) ;
2000-07-31 21:29:14 +02:00
DBUG_RETURN ( 1 ) ;
}
if ( ! silent )
{
2006-02-17 15:52:05 +01:00
if ( mysql_info ( mysql ) ) /* If NULL-pointer, print nothing */
2000-07-31 21:29:14 +02:00
{
fprintf ( stdout , " %s.%s: %s \n " , current_db , tablename ,
2006-02-17 15:52:05 +01:00
mysql_info ( mysql ) ) ;
2000-07-31 21:29:14 +02:00
}
}
DBUG_RETURN ( 0 ) ;
}
2006-02-17 15:52:05 +01:00
static void lock_table ( MYSQL * mysql , int tablecount , char * * raw_tablename )
2000-07-31 21:29:14 +02:00
{
DYNAMIC_STRING query ;
int i ;
char tablename [ FN_REFLEN ] ;
if ( verbose )
fprintf ( stdout , " Locking tables for write \n " ) ;
init_dynamic_string ( & query , " LOCK TABLES " , 256 , 1024 ) ;
for ( i = 0 ; i < tablecount ; i + + )
{
fn_format ( tablename , raw_tablename [ i ] , " " , " " , 1 | 2 ) ;
dynstr_append ( & query , tablename ) ;
dynstr_append ( & query , " WRITE, " ) ;
}
2006-02-17 15:52:05 +01:00
if ( mysql_real_query ( mysql , query . str , query . length - 1 ) )
db_error ( mysql ) ; /* We shall countinue here, if --force was given */
2000-07-31 21:29:14 +02:00
}
2006-02-17 17:16:18 +01:00
static MYSQL * db_connect ( char * host , char * database ,
2006-02-17 15:52:05 +01:00
char * user , char * passwd )
2000-07-31 21:29:14 +02:00
{
2006-02-17 17:16:18 +01:00
MYSQL * mysql ;
2000-07-31 21:29:14 +02:00
if ( verbose )
fprintf ( stdout , " Connecting to %s \n " , host ? host : " localhost " ) ;
2006-02-17 15:52:05 +01:00
if ( ! ( mysql = mysql_init ( NULL ) ) )
return 0 ;
2000-07-31 21:29:14 +02:00
if ( opt_compress )
2006-02-17 15:52:05 +01:00
mysql_options ( mysql , MYSQL_OPT_COMPRESS , NullS ) ;
2002-04-03 12:36:01 +02:00
if ( opt_local_file )
2006-02-17 15:52:05 +01:00
mysql_options ( mysql , MYSQL_OPT_LOCAL_INFILE ,
2002-04-03 12:36:01 +02:00
( char * ) & opt_local_file ) ;
2000-07-31 21:29:14 +02:00
# ifdef HAVE_OPENSSL
if ( opt_use_ssl )
2006-02-17 15:52:05 +01:00
mysql_ssl_set ( mysql , opt_ssl_key , opt_ssl_cert , opt_ssl_ca ,
2001-09-30 04:46:20 +02:00
opt_ssl_capath , opt_ssl_cipher ) ;
2006-05-11 03:11:50 +02:00
mysql_options ( mysql , MYSQL_OPT_SSL_VERIFY_SERVER_CERT ,
2006-04-18 17:58:27 +02:00
( char * ) & opt_ssl_verify_server_cert ) ;
2002-11-14 20:16:30 +01:00
# endif
if ( opt_protocol )
2006-02-17 15:52:05 +01:00
mysql_options ( mysql , MYSQL_OPT_PROTOCOL , ( char * ) & opt_protocol ) ;
2002-11-14 20:16:30 +01:00
# ifdef HAVE_SMEM
if ( shared_memory_base_name )
2006-02-17 15:52:05 +01:00
mysql_options ( mysql , MYSQL_SHARED_MEMORY_BASE_NAME , shared_memory_base_name ) ;
2000-07-31 21:29:14 +02:00
# endif
2006-02-17 15:52:05 +01:00
if ( ! ( mysql_real_connect ( mysql , host , user , passwd ,
database , opt_mysql_port , opt_mysql_unix_port ,
0 ) ) )
2000-07-31 21:29:14 +02:00
{
ignore_errors = 0 ; /* NO RETURN FROM db_error */
2006-02-17 15:52:05 +01:00
db_error ( mysql ) ;
2000-07-31 21:29:14 +02:00
}
2006-02-17 15:52:05 +01:00
mysql - > reconnect = 0 ;
2000-07-31 21:29:14 +02:00
if ( verbose )
fprintf ( stdout , " Selecting database %s \n " , database ) ;
2006-02-17 15:52:05 +01:00
if ( mysql_select_db ( mysql , database ) )
2000-07-31 21:29:14 +02:00
{
ignore_errors = 0 ;
2006-02-17 15:52:05 +01:00
db_error ( mysql ) ;
2000-07-31 21:29:14 +02:00
}
2006-02-17 15:52:05 +01:00
return mysql ;
2000-07-31 21:29:14 +02:00
}
2006-02-17 15:52:05 +01:00
static void db_disconnect ( char * host , MYSQL * mysql )
2000-07-31 21:29:14 +02:00
{
if ( verbose )
fprintf ( stdout , " Disconnecting from %s \n " , host ? host : " localhost " ) ;
2006-02-17 15:52:05 +01:00
mysql_close ( mysql ) ;
2000-07-31 21:29:14 +02:00
}
2006-02-17 15:52:05 +01:00
static void safe_exit ( int error , MYSQL * mysql )
2000-07-31 21:29:14 +02:00
{
if ( ignore_errors )
return ;
2006-02-17 15:52:05 +01:00
if ( mysql )
mysql_close ( mysql ) ;
2000-07-31 21:29:14 +02:00
exit ( error ) ;
}
static void db_error_with_table ( MYSQL * mysql , char * table )
{
2006-02-17 15:52:05 +01:00
my_printf_error ( 0 , " Error: %d, %s, when using table: %s " ,
MYF ( 0 ) , mysql_errno ( mysql ) , mysql_error ( mysql ) , table ) ;
2000-07-31 21:29:14 +02:00
safe_exit ( 1 , mysql ) ;
}
static void db_error ( MYSQL * mysql )
{
2006-02-17 15:52:05 +01:00
my_printf_error ( 0 , " Error: %d %s " , MYF ( 0 ) , mysql_errno ( mysql ) , mysql_error ( mysql ) ) ;
2000-07-31 21:29:14 +02:00
safe_exit ( 1 , mysql ) ;
}
2001-01-02 13:29:47 +01:00
static char * add_load_option ( char * ptr , const char * object ,
const char * statement )
2000-07-31 21:29:14 +02:00
{
if ( object )
{
2001-01-02 13:29:47 +01:00
/* Don't escape hex constants */
if ( object [ 0 ] = = ' 0 ' & & ( object [ 1 ] = = ' x ' | | object [ 1 ] = = ' X ' ) )
ptr = strxmov ( ptr , " " , statement , " " , object , NullS ) ;
else
{
/* char constant; escape */
ptr = strxmov ( ptr , " " , statement , " ' " , NullS ) ;
ptr = field_escape ( ptr , object , ( uint ) strlen ( object ) ) ;
* ptr + + = ' \' ' ;
}
2000-07-31 21:29:14 +02:00
}
return ptr ;
}
/*
* * Allow the user to specify field terminator strings like :
* * " ' " , " \" , " \ \ " (escaped backslash), " \ t " (tab), " \ n " (newline)
* * This is done by doubleing ' and add a end - \ if needed to avoid
* * syntax errors from the SQL parser .
*/
2001-12-06 13:10:51 +01:00
2000-07-31 21:29:14 +02:00
static char * field_escape ( char * to , const char * from , uint length )
{
const char * end ;
uint end_backslashes = 0 ;
for ( end = from + length ; from ! = end ; from + + )
{
* to + + = * from ;
if ( * from = = ' \\ ' )
end_backslashes ^ = 1 ; /* find odd number of backslashes */
else
{
if ( * from = = ' \' ' & & ! end_backslashes )
* to + + = * from ; /* We want a dublicate of "'" for MySQL */
end_backslashes = 0 ;
}
}
/* Add missing backslashes if user has specified odd number of backs.*/
if ( end_backslashes )
* to + + = ' \\ ' ;
return to ;
}
2001-12-06 13:10:51 +01:00
2006-02-17 15:52:05 +01:00
int exitcode = 0 ;
2000-07-31 21:29:14 +02:00
2006-02-26 03:03:11 +01:00
# ifdef HAVE_LIBPTHREAD
2006-02-13 20:21:15 +01:00
pthread_handler_t worker_thread ( void * arg )
2006-02-13 09:45:59 +01:00
{
2006-02-17 15:52:05 +01:00
int error ;
2006-02-13 20:21:15 +01:00
char * raw_table_name = ( char * ) arg ;
2006-02-19 18:41:59 +01:00
MYSQL * mysql = 0 ;
if ( mysql_thread_init ( ) )
goto error ;
2006-02-17 15:52:05 +01:00
if ( ! ( mysql = db_connect ( current_host , current_db , current_user , opt_password ) ) )
2006-02-13 09:45:59 +01:00
{
goto error ;
}
2006-02-17 15:52:05 +01:00
if ( mysql_query ( mysql , " set @@character_set_database=binary; " ) )
2006-02-13 09:45:59 +01:00
{
2006-02-17 15:52:05 +01:00
db_error ( mysql ) ; /* We shall countinue here, if --force was given */
2006-02-13 09:45:59 +01:00
goto error ;
}
2006-02-19 18:41:59 +01:00
/*
We are not currently catching the error here .
*/
2006-02-17 15:52:05 +01:00
if ( ( error = write_to_table ( raw_table_name , mysql ) ) )
if ( exitcode = = 0 )
exitcode = error ;
2006-02-13 09:45:59 +01:00
error :
2006-02-17 15:52:05 +01:00
if ( mysql )
db_disconnect ( current_host , mysql ) ;
2006-02-13 09:45:59 +01:00
2006-02-13 13:04:53 +01:00
pthread_mutex_lock ( & counter_mutex ) ;
2006-02-13 09:45:59 +01:00
counter - - ;
2006-02-13 13:04:53 +01:00
pthread_mutex_unlock ( & counter_mutex ) ;
2006-02-19 18:41:59 +01:00
my_thread_end ( ) ;
2006-02-18 16:18:51 +01:00
return 0 ;
2006-02-13 09:45:59 +01:00
}
2006-02-26 03:03:11 +01:00
# endif
2006-02-13 09:45:59 +01:00
2000-07-31 21:29:14 +02:00
int main ( int argc , char * * argv )
{
2006-02-17 15:52:05 +01:00
int error = 0 ;
2000-07-31 21:29:14 +02:00
char * * argv_to_free ;
MY_INIT ( argv [ 0 ] ) ;
load_defaults ( " my " , load_default_groups , & argc , & argv ) ;
/* argv is changed in the program */
argv_to_free = argv ;
if ( get_options ( & argc , & argv ) )
2002-08-24 01:08:10 +02:00
{
free_defaults ( argv_to_free ) ;
2000-07-31 21:29:14 +02:00
return ( 1 ) ;
2002-08-24 01:08:10 +02:00
}
2005-10-25 23:50:08 +02:00
2006-02-26 03:03:11 +01:00
# ifdef HAVE_LIBPTHREAD
2006-02-13 13:04:53 +01:00
if ( opt_use_threads & & ! lock_tables )
2005-10-25 23:50:08 +02:00
{
2006-02-13 09:45:59 +01:00
pthread_t mainthread ; /* Thread descriptor */
pthread_attr_t attr ; /* Thread attributes */
2006-02-13 13:04:53 +01:00
VOID ( pthread_mutex_init ( & counter_mutex , NULL ) ) ;
2006-02-13 09:45:59 +01:00
2006-02-13 20:21:15 +01:00
for ( ; * argv ! = NULL ; argv + + ) /* Loop through tables */
2006-02-13 09:45:59 +01:00
{
/*
If we hit thread count limit we loop until some threads exit .
We sleep for a second , so that we don ' t chew up a lot of
CPU in the loop .
*/
sanity_label :
if ( counter = = opt_use_threads )
{
sleep ( 1 ) ;
goto sanity_label ;
}
2006-02-13 13:04:53 +01:00
pthread_mutex_lock ( & counter_mutex ) ;
2006-02-13 09:45:59 +01:00
counter + + ;
2006-02-13 13:04:53 +01:00
pthread_mutex_unlock ( & counter_mutex ) ;
2006-02-13 09:45:59 +01:00
pthread_attr_init ( & attr ) ;
pthread_attr_setdetachstate ( & attr ,
PTHREAD_CREATE_DETACHED ) ;
/* now create the thread */
2006-02-13 20:21:15 +01:00
if ( pthread_create ( & mainthread , & attr , worker_thread ,
2006-02-13 09:45:59 +01:00
( void * ) * argv ) ! = 0 )
{
2006-02-13 13:04:53 +01:00
pthread_mutex_lock ( & counter_mutex ) ;
2006-02-13 09:45:59 +01:00
counter - - ;
2006-02-13 13:04:53 +01:00
pthread_mutex_unlock ( & counter_mutex ) ;
2006-02-13 09:45:59 +01:00
fprintf ( stderr , " %s: Could not create thread \n " ,
my_progname ) ;
}
}
/*
We loop until we know that all children have cleaned up .
*/
loop_label :
if ( counter )
{
sleep ( 1 ) ;
goto loop_label ;
}
2006-02-13 13:04:53 +01:00
VOID ( pthread_mutex_destroy ( & counter_mutex ) ) ;
2005-10-25 23:50:08 +02:00
}
2006-02-13 09:45:59 +01:00
else
2006-02-26 03:03:11 +01:00
# endif
2006-02-13 09:45:59 +01:00
{
2006-02-17 17:16:18 +01:00
MYSQL * mysql = 0 ;
2006-02-17 15:52:05 +01:00
if ( ! ( mysql = db_connect ( current_host , current_db , current_user , opt_password ) ) )
2006-02-13 09:45:59 +01:00
{
free_defaults ( argv_to_free ) ;
return ( 1 ) ; /* purecov: deadcode */
}
2005-10-25 23:50:08 +02:00
2006-02-17 15:52:05 +01:00
if ( mysql_query ( mysql , " set @@character_set_database=binary; " ) )
2006-02-13 09:45:59 +01:00
{
2006-02-17 15:52:05 +01:00
db_error ( mysql ) ; /* We shall countinue here, if --force was given */
2006-02-13 09:45:59 +01:00
return ( 1 ) ;
}
if ( lock_tables )
2006-02-17 15:52:05 +01:00
lock_table ( mysql , argc , argv ) ;
2006-02-13 09:45:59 +01:00
for ( ; * argv ! = NULL ; argv + + )
2006-02-20 13:04:40 +01:00
if ( ( error = write_to_table ( * argv , mysql ) ) )
2006-02-13 09:45:59 +01:00
if ( exitcode = = 0 )
2006-02-20 13:04:40 +01:00
exitcode = error ;
2006-02-17 15:52:05 +01:00
db_disconnect ( current_host , mysql ) ;
2006-02-13 09:45:59 +01:00
}
2000-11-13 22:55:10 +01:00
my_free ( opt_password , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2002-11-14 20:16:30 +01:00
# ifdef HAVE_SMEM
my_free ( shared_memory_base_name , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
# endif
2000-07-31 21:29:14 +02:00
free_defaults ( argv_to_free ) ;
my_end ( 0 ) ;
return ( exitcode ) ;
}