mirror of
				https://github.com/MariaDB/server.git
				synced 2025-11-04 04:46:15 +01:00 
			
		
		
		
	Threads can normally exit without a explicit pthread_exit call.
There seem to date to old glibc bugs, many around 2.2.5.
The semi related bug was https://bugs.mysql.com/bug.php?id=82886.
To improve safety in the signal handlers DBUG_* code was removed.
These where also needed to avoid some MSAN unresolved stack issues.
This is effectively a backport of 2719cc4925.
		
	
			
		
			
				
	
	
		
			756 lines
		
	
	
	
		
			23 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			756 lines
		
	
	
	
		
			23 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
   Copyright (c) 2000, 2015, Oracle and/or its affiliates.
 | 
						|
   Copyright (c) 2011, 2024, MariaDB
 | 
						|
 | 
						|
   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; version 2 of the License.
 | 
						|
 | 
						|
   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.
 | 
						|
 | 
						|
   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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1335  USA
 | 
						|
*/
 | 
						|
 | 
						|
/*
 | 
						|
**	   mysqlimport.c  - Imports all given files
 | 
						|
**			    into a table(s).
 | 
						|
**
 | 
						|
**			   *************************
 | 
						|
**			   *			   *
 | 
						|
**			   * AUTHOR: Monty & Jani  *
 | 
						|
**			   * DATE:   June 24, 1997 *
 | 
						|
**			   *			   *
 | 
						|
**			   *************************
 | 
						|
*/
 | 
						|
#define IMPORT_VERSION "3.7"
 | 
						|
 | 
						|
#include "client_priv.h"
 | 
						|
#include <my_sys.h>
 | 
						|
 | 
						|
#include "mysql_version.h"
 | 
						|
 | 
						|
#include <welcome_copyright_notice.h>   /* ORACLE_WELCOME_COPYRIGHT_NOTICE */
 | 
						|
 | 
						|
 | 
						|
/* Global Thread counter */
 | 
						|
uint counter= 0;
 | 
						|
pthread_mutex_t init_mutex;
 | 
						|
pthread_mutex_t counter_mutex;
 | 
						|
pthread_cond_t count_threshhold;
 | 
						|
 | 
						|
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);
 | 
						|
 | 
						|
static my_bool	verbose=0,lock_tables=0,ignore_errors=0,opt_delete=0,
 | 
						|
                replace, silent, ignore, ignore_foreign_keys,
 | 
						|
                opt_compress, opt_low_priority, tty_password;
 | 
						|
static my_bool debug_info_flag= 0, debug_check_flag= 0;
 | 
						|
static uint opt_use_threads=0, opt_local_file=0, my_end_arg= 0;
 | 
						|
static char	*opt_password=0, *current_user=0,
 | 
						|
		*current_host=0, *current_db=0, *fields_terminated=0,
 | 
						|
		*lines_terminated=0, *enclosed=0, *opt_enclosed=0,
 | 
						|
		*escaped=0, *opt_columns=0, 
 | 
						|
		*default_charset= (char*) MYSQL_AUTODETECT_CHARSET_NAME;
 | 
						|
static uint     opt_mysql_port= 0, opt_protocol= 0;
 | 
						|
static char * opt_mysql_unix_port=0;
 | 
						|
static char *opt_plugin_dir= 0, *opt_default_auth= 0;
 | 
						|
static longlong opt_ignore_lines= -1;
 | 
						|
#include <sslopt-vars.h>
 | 
						|
 | 
						|
static char **argv_to_free;
 | 
						|
 | 
						|
static struct my_option my_long_options[] =
 | 
						|
{
 | 
						|
  {"character-sets-dir", 0,
 | 
						|
   "Directory for character set files.", (char**) &charsets_dir,
 | 
						|
   (char**) &charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"default-character-set", 0,
 | 
						|
   "Set the default character set.", &default_charset,
 | 
						|
   &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.",
 | 
						|
   &opt_columns, &opt_columns, 0, GET_STR, REQUIRED_ARG, 0, 0, 0,
 | 
						|
   0, 0, 0},
 | 
						|
  {"compress", 'C', "Use compression in server/client protocol.",
 | 
						|
   &opt_compress, &opt_compress, 0, GET_BOOL, NO_ARG, 0, 0, 0,
 | 
						|
   0, 0, 0},
 | 
						|
  {"debug",'#', "Output debug log. Often this is 'd:t:o,filename'.", 0, 0, 0,
 | 
						|
   GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"debug-check", 0, "Check memory and open file usage at exit.",
 | 
						|
   &debug_check_flag, &debug_check_flag, 0,
 | 
						|
   GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"debug-info", 0, "Print some debug info at exit.",
 | 
						|
   &debug_info_flag, &debug_info_flag,
 | 
						|
   0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"default_auth", 0,
 | 
						|
   "Default authentication client-side plugin to use.",
 | 
						|
   &opt_default_auth, &opt_default_auth, 0,
 | 
						|
   GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"delete", 'd', "First delete all rows from table.", &opt_delete,
 | 
						|
   &opt_delete, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"fields-terminated-by", 0,
 | 
						|
   "Fields in the input file are terminated by the given string.", 
 | 
						|
   &fields_terminated, &fields_terminated, 0, 
 | 
						|
   GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"fields-enclosed-by", 0,
 | 
						|
   "Fields in the import file are enclosed by the given character.", 
 | 
						|
   &enclosed, &enclosed, 0, 
 | 
						|
   GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"fields-optionally-enclosed-by", 0,
 | 
						|
   "Fields in the input file are optionally enclosed by the given character.", 
 | 
						|
   &opt_enclosed, &opt_enclosed, 0, 
 | 
						|
   GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"fields-escaped-by", 0, 
 | 
						|
   "Fields in the input file are escaped by the given character.",
 | 
						|
   &escaped, &escaped, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0,
 | 
						|
   0, 0},
 | 
						|
  {"force", 'f', "Continue even if we get an SQL error.",
 | 
						|
   &ignore_errors, &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.", ¤t_host,
 | 
						|
   ¤t_host, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"ignore", 'i', "If duplicate unique key was found, keep old row.",
 | 
						|
   &ignore, &ignore, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"ignore-foreign-keys", 'k',
 | 
						|
    "Disable foreign key checks while importing the data.",
 | 
						|
    &ignore_foreign_keys, &ignore_foreign_keys, 0, GET_BOOL, NO_ARG,
 | 
						|
    0, 0, 0, 0, 0, 0},
 | 
						|
  {"ignore-lines", 0, "Ignore first n lines of data infile.",
 | 
						|
   &opt_ignore_lines, &opt_ignore_lines, 0, GET_LL,
 | 
						|
   REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"lines-terminated-by", 0, 
 | 
						|
   "Lines in the input file are terminated by the given string.",
 | 
						|
   &lines_terminated, &lines_terminated, 0, GET_STR,
 | 
						|
   REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"local", 'L', "Read all files through the client.", &opt_local_file,
 | 
						|
   &opt_local_file, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"lock-tables", 'l', "Lock all tables for write (this disables threads).",
 | 
						|
    &lock_tables, &lock_tables, 0, GET_BOOL, NO_ARG, 
 | 
						|
    0, 0, 0, 0, 0, 0},
 | 
						|
  {"low-priority", 0,
 | 
						|
   "Use LOW_PRIORITY when updating the table.", &opt_low_priority,
 | 
						|
   &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},
 | 
						|
#ifdef __WIN__
 | 
						|
  {"pipe", 'W', "Use named pipes to connect to server.", 0, 0, 0, GET_NO_ARG,
 | 
						|
   NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
#endif
 | 
						|
  {"plugin_dir", 0, "Directory for client-side plugins.",
 | 
						|
   &opt_plugin_dir, &opt_plugin_dir, 0,
 | 
						|
   GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"port", 'P', "Port number to use for connection or 0 for default to, in "
 | 
						|
   "order of preference, my.cnf, $MYSQL_TCP_PORT, "
 | 
						|
#if MYSQL_PORT_DEFAULT == 0
 | 
						|
   "/etc/services, "
 | 
						|
#endif
 | 
						|
   "built-in default (" STRINGIFY_ARG(MYSQL_PORT) ").",
 | 
						|
   &opt_mysql_port,
 | 
						|
   &opt_mysql_port, 0, GET_UINT, REQUIRED_ARG, 0, 0, 0, 0, 0,
 | 
						|
   0},
 | 
						|
  {"protocol", OPT_MYSQL_PROTOCOL, "The protocol to use for connection (tcp, socket, pipe).",
 | 
						|
   0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"replace", 'r', "If duplicate unique key was found, replace old row.",
 | 
						|
   &replace, &replace, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"silent", 's', "Be more silent.", &silent, &silent, 0,
 | 
						|
   GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
  {"socket", 'S', "The socket file to use for connection.",
 | 
						|
   &opt_mysql_unix_port, &opt_mysql_unix_port, 0, GET_STR,
 | 
						|
   REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
#include <sslopt-longopts.h>
 | 
						|
  {"use-threads", 0,
 | 
						|
   "Load files in parallel. The argument is the number "
 | 
						|
   "of threads to use for loading data.",
 | 
						|
   &opt_use_threads, &opt_use_threads, 0, 
 | 
						|
   GET_UINT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
#ifndef DONT_ALLOW_USER_CHANGE
 | 
						|
  {"user", 'u', "User for login if not current user.", ¤t_user,
 | 
						|
   ¤t_user, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
 | 
						|
#endif
 | 
						|
  {"verbose", 'v', "Print info about the various stages.", &verbose,
 | 
						|
   &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}
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
static const char *load_default_groups[]=
 | 
						|
{ "mysqlimport", "mariadb-import", "client", "client-server", "client-mariadb",
 | 
						|
  0 };
 | 
						|
 | 
						|
 | 
						|
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);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void usage(void)
 | 
						|
{
 | 
						|
  puts("Copyright 2000-2008 MySQL AB, 2008 Sun Microsystems, Inc.");
 | 
						|
  puts("Copyright 2008-2011 Oracle and Monty Program Ab.");
 | 
						|
  puts("Copyright 2012-2019 MariaDB Corporation Ab.");
 | 
						|
  print_version();
 | 
						|
  puts(ORACLE_WELCOME_COPYRIGHT_NOTICE("2000"));
 | 
						|
  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 MariaDB server, the server will open\n\
 | 
						|
and 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("\nUsage: %s [OPTIONS] database textfile...\n",my_progname);
 | 
						|
  print_defaults("my",load_default_groups);
 | 
						|
  puts("");
 | 
						|
  my_print_help(my_long_options);
 | 
						|
  my_print_variables(my_long_options);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static my_bool
 | 
						|
get_one_option(const struct my_option *opt, const char *argument,
 | 
						|
               const char *filename __attribute__((unused)))
 | 
						|
{
 | 
						|
  switch(opt->id) {
 | 
						|
  case 'p':
 | 
						|
    if (argument == disabled_my_option)
 | 
						|
      argument= (char*) "";			/* Don't require password */
 | 
						|
    if (argument)
 | 
						|
    {
 | 
						|
      /*
 | 
						|
        One should not really change the argument, but we make an
 | 
						|
        exception for passwords
 | 
						|
      */
 | 
						|
      char *start= (char*) argument;
 | 
						|
      my_free(opt_password);
 | 
						|
      opt_password=my_strdup(PSI_NOT_INSTRUMENTED, argument,MYF(MY_FAE));
 | 
						|
      while (*argument)
 | 
						|
        *(char*) argument++= 'x';               /* Destroy argument */
 | 
						|
      if (*start)
 | 
						|
	start[1]=0;				/* Cut length of argument */
 | 
						|
      tty_password= 0;
 | 
						|
    }
 | 
						|
    else
 | 
						|
      tty_password= 1;
 | 
						|
    break;
 | 
						|
#ifdef __WIN__
 | 
						|
  case 'W':
 | 
						|
    opt_protocol = MYSQL_PROTOCOL_PIPE;
 | 
						|
    break;
 | 
						|
#endif
 | 
						|
  case OPT_MYSQL_PROTOCOL:
 | 
						|
    if ((opt_protocol= find_type_with_warning(argument, &sql_protocol_typelib,
 | 
						|
                                              opt->name)) <= 0)
 | 
						|
    {
 | 
						|
      sf_leaking_memory= 1; /* no memory leak reports here */
 | 
						|
      exit(1);
 | 
						|
    }
 | 
						|
    break;
 | 
						|
  case '#':
 | 
						|
    DBUG_PUSH(argument ? argument : "d:t:o");
 | 
						|
    debug_check_flag= 1;
 | 
						|
    break;
 | 
						|
#include <sslopt-case.h>
 | 
						|
  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;
 | 
						|
 | 
						|
  if ((ho_error=handle_options(argc, argv, my_long_options, get_one_option)))
 | 
						|
    exit(ho_error);
 | 
						|
  if (debug_info_flag)
 | 
						|
    my_end_arg= MY_CHECK_ERROR | MY_GIVE_INFO;
 | 
						|
  if (debug_check_flag)
 | 
						|
    my_end_arg= MY_CHECK_ERROR;
 | 
						|
 | 
						|
  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);
 | 
						|
  }
 | 
						|
  if (*argc < 2)
 | 
						|
  {
 | 
						|
    usage();
 | 
						|
    return 1;
 | 
						|
  }
 | 
						|
  current_db= *((*argv)++);
 | 
						|
  (*argc)--;
 | 
						|
  if (tty_password)
 | 
						|
    opt_password=my_get_tty_password(NullS);
 | 
						|
  return(0);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
static int write_to_table(char *filename, MYSQL *mysql)
 | 
						|
{
 | 
						|
  char tablename[FN_REFLEN], hard_path[FN_REFLEN],
 | 
						|
       escaped_name[FN_REFLEN * 2 + 1],
 | 
						|
       sql_statement[FN_REFLEN*16+256], *end, *pos;
 | 
						|
  DBUG_ENTER("write_to_table");
 | 
						|
  DBUG_PRINT("enter",("filename: %s",filename));
 | 
						|
 | 
						|
  fn_format(tablename, filename, "", "", 1 | 2); /* removes path & ext. */
 | 
						|
  if (!opt_local_file)
 | 
						|
    strmov(hard_path,filename);
 | 
						|
  else
 | 
						|
    my_load_path(hard_path, filename, NULL); /* filename includes the path */
 | 
						|
 | 
						|
  if (opt_delete)
 | 
						|
  {
 | 
						|
    if (verbose)
 | 
						|
      fprintf(stdout, "Deleting the old data from table %s\n", tablename);
 | 
						|
    snprintf(sql_statement, FN_REFLEN*16+256, "DELETE FROM %s", tablename);
 | 
						|
    if (mysql_query(mysql, sql_statement))
 | 
						|
    {
 | 
						|
      db_error_with_table(mysql, tablename);
 | 
						|
      DBUG_RETURN(1);
 | 
						|
    }
 | 
						|
  }
 | 
						|
  to_unix_path(hard_path);
 | 
						|
  if (verbose)
 | 
						|
  {
 | 
						|
    if (opt_local_file)
 | 
						|
      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);
 | 
						|
  }
 | 
						|
  mysql_real_escape_string(mysql, escaped_name, hard_path,
 | 
						|
                           (unsigned long) strlen(hard_path));
 | 
						|
  sprintf(sql_statement, "LOAD DATA %s %s INFILE '%s'",
 | 
						|
	  opt_low_priority ? "LOW_PRIORITY" : "",
 | 
						|
	  opt_local_file ? "LOCAL" : "", escaped_name);
 | 
						|
  end= strend(sql_statement);
 | 
						|
  if (replace)
 | 
						|
    end= strmov(end, " REPLACE");
 | 
						|
  if (ignore)
 | 
						|
    end= strmov(end, " IGNORE");
 | 
						|
  end= strmov(end, " INTO TABLE `");
 | 
						|
  /* Turn any ` into `` in table name. */
 | 
						|
  for (pos= tablename; *pos; pos++)
 | 
						|
  {
 | 
						|
    if (*pos == '`')
 | 
						|
      *end++= '`';
 | 
						|
    *end++= *pos;
 | 
						|
  }
 | 
						|
  end= strmov(end, "`");
 | 
						|
 | 
						|
  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");
 | 
						|
  if (opt_ignore_lines >= 0)
 | 
						|
    end= strmov(longlong10_to_str(opt_ignore_lines, 
 | 
						|
				  strmov(end, " IGNORE "),10), " LINES");
 | 
						|
  if (opt_columns)
 | 
						|
    end= strmov(strmov(strmov(end, " ("), opt_columns), ")");
 | 
						|
  *end= '\0';
 | 
						|
 | 
						|
  if (mysql_query(mysql, sql_statement))
 | 
						|
  {
 | 
						|
    db_error_with_table(mysql, tablename);
 | 
						|
    DBUG_RETURN(1);
 | 
						|
  }
 | 
						|
  if (!silent)
 | 
						|
  {
 | 
						|
    if (mysql_info(mysql)) /* If NULL-pointer, print nothing */
 | 
						|
    {
 | 
						|
      fprintf(stdout, "%s.%s: %s\n", current_db, tablename,
 | 
						|
	      mysql_info(mysql));
 | 
						|
    }
 | 
						|
  }
 | 
						|
  DBUG_RETURN(0);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
static void lock_table(MYSQL *mysql, int tablecount, char **raw_tablename)
 | 
						|
{
 | 
						|
  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,");
 | 
						|
  }
 | 
						|
  if (mysql_real_query(mysql, query.str, (ulong)query.length-1))
 | 
						|
    db_error(mysql); /* We shall continue here, if --force was given */
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
static MYSQL *db_connect(char *host, char *database,
 | 
						|
                         char *user, char *passwd)
 | 
						|
{
 | 
						|
  MYSQL *mysql;
 | 
						|
  my_bool reconnect;
 | 
						|
  if (verbose)
 | 
						|
    fprintf(stdout, "Connecting to %s\n", host ? host : "localhost");
 | 
						|
  if (opt_use_threads && !lock_tables)
 | 
						|
  {
 | 
						|
    pthread_mutex_lock(&init_mutex);
 | 
						|
    if (!(mysql= mysql_init(NULL)))
 | 
						|
    {
 | 
						|
      pthread_mutex_unlock(&init_mutex);
 | 
						|
      return 0;
 | 
						|
    }
 | 
						|
    pthread_mutex_unlock(&init_mutex);
 | 
						|
  }
 | 
						|
  else
 | 
						|
    if (!(mysql= mysql_init(NULL)))
 | 
						|
      return 0;
 | 
						|
  if (opt_compress)
 | 
						|
    mysql_options(mysql,MYSQL_OPT_COMPRESS,NullS);
 | 
						|
  if (opt_local_file)
 | 
						|
    mysql_options(mysql,MYSQL_OPT_LOCAL_INFILE,
 | 
						|
		  (char*) &opt_local_file);
 | 
						|
#ifdef HAVE_OPENSSL
 | 
						|
  if (opt_use_ssl)
 | 
						|
  {
 | 
						|
    mysql_ssl_set(mysql, opt_ssl_key, opt_ssl_cert, opt_ssl_ca,
 | 
						|
		  opt_ssl_capath, opt_ssl_cipher);
 | 
						|
    mysql_options(mysql, MYSQL_OPT_SSL_CRL, opt_ssl_crl);
 | 
						|
    mysql_options(mysql, MYSQL_OPT_SSL_CRLPATH, opt_ssl_crlpath);
 | 
						|
    mysql_options(mysql, MARIADB_OPT_TLS_VERSION, opt_tls_version);
 | 
						|
  }
 | 
						|
  mysql_options(mysql,MYSQL_OPT_SSL_VERIFY_SERVER_CERT,
 | 
						|
                (char*)&opt_ssl_verify_server_cert);
 | 
						|
#endif
 | 
						|
  if (opt_protocol)
 | 
						|
    mysql_options(mysql,MYSQL_OPT_PROTOCOL,(char*)&opt_protocol);
 | 
						|
 | 
						|
  if (opt_plugin_dir && *opt_plugin_dir)
 | 
						|
    mysql_options(mysql, MYSQL_PLUGIN_DIR, opt_plugin_dir);
 | 
						|
 | 
						|
  if (opt_default_auth && *opt_default_auth)
 | 
						|
    mysql_options(mysql, MYSQL_DEFAULT_AUTH, opt_default_auth);
 | 
						|
  if (!strcmp(default_charset,MYSQL_AUTODETECT_CHARSET_NAME))
 | 
						|
    default_charset= (char *)my_default_csname();
 | 
						|
  mysql_options(mysql, MYSQL_SET_CHARSET_NAME, my_default_csname());
 | 
						|
  mysql_options(mysql, MYSQL_OPT_CONNECT_ATTR_RESET, 0);
 | 
						|
  mysql_options4(mysql, MYSQL_OPT_CONNECT_ATTR_ADD,
 | 
						|
                 "program_name", "mysqlimport");
 | 
						|
  if (!(mysql_real_connect(mysql,host,user,passwd,
 | 
						|
                           database,opt_mysql_port,opt_mysql_unix_port,
 | 
						|
                           0)))
 | 
						|
  {
 | 
						|
    ignore_errors=0;	  /* NO RETURN FROM db_error */
 | 
						|
    db_error(mysql);
 | 
						|
  }
 | 
						|
  reconnect= 0;
 | 
						|
  mysql_options(mysql, MYSQL_OPT_RECONNECT, &reconnect);
 | 
						|
  if (verbose)
 | 
						|
    fprintf(stdout, "Selecting database %s\n", database);
 | 
						|
  if (mysql_select_db(mysql, database))
 | 
						|
  {
 | 
						|
    ignore_errors=0;
 | 
						|
    db_error(mysql);
 | 
						|
  }
 | 
						|
  if (ignore_foreign_keys)
 | 
						|
    mysql_query(mysql, "set foreign_key_checks= 0;");
 | 
						|
 | 
						|
  return mysql;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
static void db_disconnect(char *host, MYSQL *mysql)
 | 
						|
{
 | 
						|
  if (verbose)
 | 
						|
    fprintf(stdout, "Disconnecting from %s\n", host ? host : "localhost");
 | 
						|
  mysql_close(mysql);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void safe_exit(int error, MYSQL *mysql)
 | 
						|
{
 | 
						|
  if (error && ignore_errors)
 | 
						|
    return;
 | 
						|
 | 
						|
  /* in multi-threaded mode protect from concurrent safe_exit's */
 | 
						|
  if (counter)
 | 
						|
    pthread_mutex_lock(&counter_mutex);
 | 
						|
 | 
						|
  if (mysql)
 | 
						|
    mysql_close(mysql);
 | 
						|
 | 
						|
  if (error)
 | 
						|
    sf_leaking_memory= 1; /* dirty exit, some threads are still running */
 | 
						|
  else
 | 
						|
  {
 | 
						|
    mysql_library_end();
 | 
						|
    free_defaults(argv_to_free);
 | 
						|
    my_free(opt_password);
 | 
						|
    my_end(my_end_arg); /* clean exit */
 | 
						|
  }
 | 
						|
  exit(error);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
static void db_error_with_table(MYSQL *mysql, char *table)
 | 
						|
{
 | 
						|
  my_printf_error(0,"Error: %d, %s, when using table: %s",
 | 
						|
		  MYF(0), mysql_errno(mysql), mysql_error(mysql), table);
 | 
						|
  safe_exit(1, mysql);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
static void db_error(MYSQL *mysql)
 | 
						|
{
 | 
						|
  my_printf_error(0,"Error: %d %s", MYF(0), mysql_errno(mysql), mysql_error(mysql));
 | 
						|
  safe_exit(1, mysql);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static char *add_load_option(char *ptr, const char *object,
 | 
						|
			     const char *statement)
 | 
						|
{
 | 
						|
  if (object)
 | 
						|
  {
 | 
						|
    /* 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++= '\'';
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return ptr;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
** Allow the user to specify field terminator strings like:
 | 
						|
** "'", "\", "\\" (escaped backslash), "\t" (tab), "\n" (newline)
 | 
						|
** This is done by doubling ' and add a end -\ if needed to avoid
 | 
						|
** syntax errors from the SQL parser.
 | 
						|
*/ 
 | 
						|
 | 
						|
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 duplicate of "'" for MySQL */
 | 
						|
      end_backslashes=0;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  /* Add missing backslashes if user has specified odd number of backs.*/
 | 
						|
  if (end_backslashes)
 | 
						|
    *to++= '\\';          
 | 
						|
  return to;
 | 
						|
}
 | 
						|
 | 
						|
int exitcode= 0;
 | 
						|
 | 
						|
pthread_handler_t worker_thread(void *arg)
 | 
						|
{
 | 
						|
  int error;
 | 
						|
  char *raw_table_name= (char *)arg;
 | 
						|
  MYSQL *mysql= 0;
 | 
						|
 | 
						|
  if (mysql_thread_init())
 | 
						|
    goto error;
 | 
						|
  
 | 
						|
  if (!(mysql= db_connect(current_host,current_db,current_user,opt_password)))
 | 
						|
  {
 | 
						|
    goto error;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mysql_query(mysql, "/*!40101 set @@character_set_database=binary */;"))
 | 
						|
  {
 | 
						|
    db_error(mysql); /* We shall continue here, if --force was given */
 | 
						|
    goto error;
 | 
						|
  }
 | 
						|
 | 
						|
  /*
 | 
						|
    We are not currently catching the error here.
 | 
						|
  */
 | 
						|
  if((error= write_to_table(raw_table_name, mysql)))
 | 
						|
    if (exitcode == 0)
 | 
						|
      exitcode= error;
 | 
						|
 | 
						|
error:
 | 
						|
  if (mysql)
 | 
						|
    db_disconnect(current_host, mysql);
 | 
						|
 | 
						|
  pthread_mutex_lock(&counter_mutex);
 | 
						|
  counter--;
 | 
						|
  pthread_cond_signal(&count_threshhold);
 | 
						|
  pthread_mutex_unlock(&counter_mutex);
 | 
						|
  mysql_thread_end();
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int main(int argc, char **argv)
 | 
						|
{
 | 
						|
  int error=0;
 | 
						|
  MY_INIT(argv[0]);
 | 
						|
  sf_leaking_memory=1; /* don't report memory leaks on early exits */
 | 
						|
 | 
						|
  load_defaults_or_exit("my", load_default_groups, &argc, &argv);
 | 
						|
  /* argv is changed in the program */
 | 
						|
  argv_to_free= argv;
 | 
						|
  if (get_options(&argc, &argv))
 | 
						|
  {
 | 
						|
    free_defaults(argv_to_free);
 | 
						|
    return(1);
 | 
						|
  }
 | 
						|
  sf_leaking_memory=0; /* from now on we cleanup properly */
 | 
						|
 | 
						|
  if (opt_use_threads && !lock_tables)
 | 
						|
  {
 | 
						|
    char **save_argv;
 | 
						|
    uint worker_thread_count= 0, table_count= 0, i= 0;
 | 
						|
    pthread_t *worker_threads;       /* Thread descriptor */
 | 
						|
    pthread_attr_t attr;             /* Thread attributes */
 | 
						|
    pthread_attr_init(&attr);
 | 
						|
    pthread_attr_setdetachstate(&attr,
 | 
						|
                                PTHREAD_CREATE_JOINABLE);
 | 
						|
 | 
						|
    pthread_mutex_init(&init_mutex, NULL);
 | 
						|
    pthread_mutex_init(&counter_mutex, NULL);
 | 
						|
    pthread_cond_init(&count_threshhold, NULL);
 | 
						|
 | 
						|
    /* Count the number of tables. This number denotes the total number
 | 
						|
       of threads spawn.
 | 
						|
    */
 | 
						|
    save_argv= argv;
 | 
						|
    for (table_count= 0; *argv != NULL; argv++)
 | 
						|
      table_count++;
 | 
						|
    argv= save_argv;
 | 
						|
 | 
						|
    if (!(worker_threads= (pthread_t*) my_malloc(PSI_NOT_INSTRUMENTED,
 | 
						|
                               table_count * sizeof(*worker_threads), MYF(0))))
 | 
						|
      return -2;
 | 
						|
 | 
						|
    for (; *argv != NULL; argv++) /* Loop through tables */
 | 
						|
    {
 | 
						|
      pthread_mutex_lock(&counter_mutex);
 | 
						|
      while (counter == opt_use_threads)
 | 
						|
      {
 | 
						|
        struct timespec abstime;
 | 
						|
 | 
						|
        set_timespec(abstime, 3);
 | 
						|
        pthread_cond_timedwait(&count_threshhold, &counter_mutex, &abstime);
 | 
						|
      }
 | 
						|
      /* Before exiting the lock we set ourselves up for the next thread */
 | 
						|
      counter++;
 | 
						|
      pthread_mutex_unlock(&counter_mutex);
 | 
						|
      /* now create the thread */
 | 
						|
      if (pthread_create(&worker_threads[worker_thread_count], &attr,
 | 
						|
                         worker_thread, (void *)*argv) != 0)
 | 
						|
      {
 | 
						|
        pthread_mutex_lock(&counter_mutex);
 | 
						|
        counter--;
 | 
						|
        pthread_mutex_unlock(&counter_mutex);
 | 
						|
        fprintf(stderr,"%s: Could not create thread\n", my_progname);
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      worker_thread_count++;
 | 
						|
    }
 | 
						|
 | 
						|
    /*
 | 
						|
      We loop until we know that all children have cleaned up.
 | 
						|
    */
 | 
						|
    pthread_mutex_lock(&counter_mutex);
 | 
						|
    while (counter)
 | 
						|
    {
 | 
						|
      struct timespec abstime;
 | 
						|
 | 
						|
      set_timespec(abstime, 3);
 | 
						|
      pthread_cond_timedwait(&count_threshhold, &counter_mutex, &abstime);
 | 
						|
    }
 | 
						|
    pthread_mutex_unlock(&counter_mutex);
 | 
						|
    pthread_mutex_destroy(&init_mutex);
 | 
						|
    pthread_mutex_destroy(&counter_mutex);
 | 
						|
    pthread_cond_destroy(&count_threshhold);
 | 
						|
    pthread_attr_destroy(&attr);
 | 
						|
 | 
						|
    for(i= 0; i < worker_thread_count; i++)
 | 
						|
    {
 | 
						|
      if (pthread_join(worker_threads[i], NULL))
 | 
						|
        fprintf(stderr,"%s: Could not join worker thread.\n", my_progname);
 | 
						|
    }
 | 
						|
 | 
						|
    my_free(worker_threads);
 | 
						|
  }
 | 
						|
  else
 | 
						|
  {
 | 
						|
    MYSQL *mysql= 0;
 | 
						|
    if (!(mysql= db_connect(current_host,current_db,current_user,opt_password)))
 | 
						|
    {
 | 
						|
      free_defaults(argv_to_free);
 | 
						|
      return(1); /* purecov: dead code */
 | 
						|
    }
 | 
						|
 | 
						|
    if (mysql_query(mysql, "/*!40101 set @@character_set_database=binary */;"))
 | 
						|
    {
 | 
						|
      db_error(mysql); /* We shall continue here, if --force was given */
 | 
						|
      return(1);
 | 
						|
    }
 | 
						|
 | 
						|
    if (lock_tables)
 | 
						|
      lock_table(mysql, argc, argv);
 | 
						|
    for (; *argv != NULL; argv++)
 | 
						|
      if ((error= write_to_table(*argv, mysql)))
 | 
						|
        if (exitcode == 0)
 | 
						|
          exitcode= error;
 | 
						|
    db_disconnect(current_host, mysql);
 | 
						|
  }
 | 
						|
  safe_exit(0, 0);
 | 
						|
  return(exitcode);
 | 
						|
}
 |