2001-12-06 14:10:51 +02:00
/* Copyright (C) 2000 MySQL AB
2000-08-30 21:40:59 +04: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 .
2000-08-30 21:40:59 +04: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 .
2000-08-30 21:40:59 +04: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 */
/* mysqldump.c - Dump a tables contents and format to an ASCII file
* *
* * The author ' s original notes follow : -
* *
2001-10-04 20:43:06 +03:00
* * AUTHOR : Igor Romanenko ( igor @ frog . kiev . ua )
* * DATE : December 3 , 1994
* * WARRANTY : None , expressed , impressed , implied
* * or other
* * STATUS : Public domain
* * Adapted and optimized for MySQL by
* * Michael Widenius , Sinisa Milivojevic , Jani Tolonen
* * - w - - where added 9 / 10 / 98 by Jim Faucette
* * slave code by David Saez Padros < david @ ols . es >
* * master / autocommit code by Brian Aker < brian @ tangent . org >
* * SSL by
* * Andrei Errapart < andreie @ no . spam . ee >
2003-03-28 14:24:32 -05:00
* * Tõnu Samuel < tonu @ please . do . not . remove . this . spam . ee >
2001-11-05 23:48:03 +02:00
* * XML by Gary Huntress < ghuntress @ mediaone . net > 10 / 10 / 01 , cleaned up
* * and adapted to mysqldump 05 / 11 / 01 by Jani Tolonen
2002-06-11 13:37:48 +04:00
* * Added - - single - transaction option 06 / 06 / 2002 by Peter Zaitsev
2003-06-10 11:34:00 +05:00
* * 10 Jun 2003 : SET NAMES and - - no - set - names by Alexander Barkov
2001-11-05 23:48:03 +02:00
*/
2000-07-31 21:29:14 +02:00
2003-12-17 17:35:34 +02:00
# define DUMP_VERSION "10.4"
2000-07-31 21:29:14 +02:00
2001-09-14 02:54:33 +03:00
# include <my_global.h>
2000-07-31 21:29:14 +02:00
# include <my_sys.h>
# include <m_string.h>
# include <m_ctype.h>
2001-10-31 03:22:31 +02:00
# include "client_priv.h"
2000-07-31 21:29:14 +02:00
# include "mysql.h"
# include "mysql_version.h"
# include "mysqld_error.h"
/* Exit codes */
# define EX_USAGE 1
# define EX_MYSQLERR 2
# define EX_CONSCHECK 3
# define EX_EOM 4
/* index into 'show fields from table' */
# define SHOW_FIELDNAME 0
# define SHOW_TYPE 1
# define SHOW_NULL 2
# define SHOW_DEFAULT 4
# define SHOW_EXTRA 5
# define QUOTE_CHAR '`'
2002-04-27 12:09:59 +03:00
/* Size of buffer for dump's select query */
# define QUERY_LENGTH 1536
2000-07-31 21:29:14 +02:00
static char * add_load_option ( char * ptr , const char * object ,
const char * statement ) ;
2003-01-16 05:35:59 +02:00
static ulong find_set ( TYPELIB * lib , const char * x , uint length ,
char * * err_pos , uint * err_len ) ;
2000-07-31 21:29:14 +02:00
static char * field_escape ( char * to , const char * from , uint length ) ;
2003-01-16 05:35:59 +02:00
static my_bool verbose = 0 , tFlag = 0 , cFlag = 0 , dFlag = 0 , quick = 1 , extended_insert = 1 ,
lock_tables = 1 , ignore_errors = 0 , flush_logs = 0 , replace = 0 ,
ignore = 0 , opt_drop = 1 , opt_keywords = 0 , opt_lock = 1 , opt_compress = 0 ,
opt_delayed = 0 , create_options = 1 , opt_quoted = 0 , opt_databases = 0 ,
2003-05-30 15:21:03 +05:00
opt_alldbs = 0 , opt_create_db = 0 , opt_first_slave = 0 , opt_set_names = 0 ,
2003-01-16 05:35:59 +02:00
opt_autocommit = 0 , opt_master_data , opt_disable_keys = 1 , opt_xml = 0 ,
2003-04-03 12:33:13 +03:00
opt_delete_master_logs = 0 , tty_password = 0 ,
2003-12-11 17:37:45 +04:00
opt_single_transaction = 0 , opt_comments = 0 ;
2000-07-31 21:29:14 +02:00
static MYSQL mysql_connection , * sock = 0 ;
2000-11-13 23:55:10 +02:00
static char insert_pat [ 12 * 1024 ] , * opt_password = 0 , * current_user = 0 ,
2000-07-31 21:29:14 +02:00
* current_host = 0 , * path = 0 , * fields_terminated = 0 ,
* lines_terminated = 0 , * enclosed = 0 , * opt_enclosed = 0 , * escaped = 0 ,
2003-07-31 09:00:22 +05:00
* where = 0 ,
2003-03-16 11:20:45 +04:00
* opt_compatible_mode_str = 0 ,
2003-01-16 05:35:59 +02:00
* err_ptr = 0 ;
2003-12-19 14:12:29 +04:00
static char * default_charset = ( char * ) MYSQL_UNIVERSAL_CLIENT_CHARSET ;
2003-01-16 05:35:59 +02:00
static ulong opt_compatible_mode = 0 ;
static uint opt_mysql_port = 0 , err_len = 0 ;
2000-07-31 21:29:14 +02:00
static my_string opt_mysql_unix_port = 0 ;
static int first_error = 0 ;
extern ulong net_buffer_length ;
static DYNAMIC_STRING extended_row ;
2002-09-17 23:46:53 +03:00
# include <sslopt-vars.h>
2001-08-22 01:45:07 +03:00
FILE * md_result_file ;
2002-11-15 00:16:30 +05:00
# ifdef HAVE_SMEM
static char * shared_memory_base_name = 0 ;
# endif
2003-01-16 05:35:59 +02:00
static uint opt_protocol = 0 ;
2003-03-16 11:20:45 +04:00
static CHARSET_INFO * charset_info = & my_charset_latin1 ;
2003-01-16 05:35:59 +02:00
const char * compatible_mode_names [ ] =
{
" MYSQL323 " , " MYSQL40 " , " POSTGRESQL " , " ORACLE " , " MSSQL " , " DB2 " ,
" SAPDB " , " NO_KEY_OPTIONS " , " NO_TABLE_OPTIONS " , " NO_FIELD_OPTIONS " ,
NullS
} ;
TYPELIB compatible_mode_typelib = { array_elements ( compatible_mode_names ) - 1 ,
" " , compatible_mode_names } ;
2000-07-31 21:29:14 +02:00
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
{ " all-databases " , ' A ' ,
" Dump all the databases. This will be same as --databases with all databases selected. " ,
( gptr * ) & opt_alldbs , ( gptr * ) & opt_alldbs , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 ,
0 , 0 } ,
2002-05-14 21:41:55 +03:00
{ " all " , ' a ' , " Include all MySQL specific create options. " ,
2003-01-16 05:35:59 +02:00
( gptr * ) & create_options , ( gptr * ) & create_options , 0 , GET_BOOL , NO_ARG , 1 ,
2002-05-14 21:41:55 +03:00
0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " add-drop-table " , OPT_DROP , " Add a 'drop table' before each create. " ,
2003-01-16 05:35:59 +02:00
( gptr * ) & opt_drop , ( gptr * ) & opt_drop , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 ,
2002-04-02 20:29:53 +03:00
0 } ,
{ " add-locks " , OPT_LOCKS , " Add locks around insert statements. " ,
2003-01-16 05:35:59 +02:00
( gptr * ) & opt_lock , ( gptr * ) & opt_lock , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 ,
2002-04-02 20:29:53 +03:00
0 } ,
{ " allow-keywords " , OPT_KEYWORDS ,
" Allow creation of column names that are keywords. " , ( gptr * ) & opt_keywords ,
( gptr * ) & opt_keywords , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " 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 20:29:53 +03:00
( gptr * ) & charsets_dir , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-01-16 05:35:59 +02:00
{ " compatible " , OPT_COMPATIBLE ,
" Change the dump to be compatible with a given mode. By default tables are dumped without any restrictions. Legal modes are: mysql323, mysql40, postgresql, oracle, mssql, db2, sapdb, no_key_options, no_table_options, no_field_options. One can use several modes separated by commas. Note: Requires MySQL server version 4.1.0 or higher. This option does a no operation on earlier server versions. " ,
( gptr * ) & opt_compatible_mode_str , ( gptr * ) & opt_compatible_mode_str , 0 ,
GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-05-14 21:41:55 +03:00
{ " complete-insert " , ' c ' , " Use complete insert statements. " , ( gptr * ) & cFlag ,
( gptr * ) & cFlag , 0 , GET_BOOL , NO_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 } ,
{ " databases " , ' B ' ,
" To dump several databases. Note the difference in usage; In this case no tables are given. All name arguments are regarded as databasenames. 'USE db_name;' will be included in the output. " ,
( gptr * ) & opt_databases , ( gptr * ) & opt_databases , 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 } ,
{ " 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 } ,
{ " delayed-insert " , OPT_DELAYED , " Insert rows with INSERT DELAYED. " ,
( gptr * ) & opt_delayed , ( gptr * ) & opt_delayed , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 ,
0 , 0 } ,
2003-04-03 12:33:13 +03:00
{ " delete-master-logs " , OPT_DELETE_MASTER_LOGS ,
" Delete logs on master after backup. This will automagically enable --first-slave. " ,
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " disable-keys " , ' K ' ,
" '/*!40000 ALTER TABLE tb_name DISABLE KEYS */; and '/*!40000 ALTER TABLE tb_name ENABLE KEYS */; will be put in the output. " , ( gptr * ) & opt_disable_keys ,
2003-01-16 05:35:59 +02:00
( gptr * ) & opt_disable_keys , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " extended-insert " , ' e ' ,
" Allows utilization of the new, much faster INSERT syntax. " ,
( gptr * ) & extended_insert , ( gptr * ) & extended_insert , 0 , GET_BOOL , NO_ARG ,
2003-01-16 05:35:59 +02:00
1 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " 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 , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " first-slave " , ' x ' , " Locks all tables across all databases. " ,
( gptr * ) & opt_first_slave , ( gptr * ) & opt_first_slave , 0 , GET_BOOL , NO_ARG ,
0 , 0 , 0 , 0 , 0 , 0 } ,
{ " flush-logs " , ' F ' , " Flush logs file in server before starting dump. " ,
( gptr * ) & flush_logs , ( gptr * ) & flush_logs , 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 } ,
{ " help " , ' ? ' , " Display this help message and exit. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " 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
{ " 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 } ,
{ " lock-tables " , ' l ' , " Lock all tables for read. " , ( gptr * ) & lock_tables ,
2003-01-16 05:35:59 +02:00
( gptr * ) & lock_tables , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " master-data " , OPT_MASTER_DATA ,
" This will cause the master position and filename to be appended to your output. This will automagically enable --first-slave. " ,
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " no-autocommit " , OPT_AUTOCOMMIT ,
" Wrap tables with autocommit/commit statements. " ,
( gptr * ) & opt_autocommit , ( gptr * ) & opt_autocommit , 0 , GET_BOOL , NO_ARG ,
0 , 0 , 0 , 0 , 0 , 0 } ,
2002-06-11 13:37:48 +04:00
{ " single-transaction " , OPT_TRANSACTION ,
" Dump all tables in single transaction to get consistent snapshot. Mutually exclusive with --lock-tables. " ,
2003-01-16 05:35:59 +02:00
( gptr * ) & opt_single_transaction , ( gptr * ) & opt_single_transaction , 0 ,
2003-06-13 10:59:02 +02:00
GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " no-create-db " , ' n ' ,
2003-06-13 10:59:02 +02:00
" 'CREATE DATABASE /*!32312 IF NOT EXISTS*/ db_name;' will not be put in the output. The above line will be added otherwise, if --databases or --all-databases option was given.}. " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & opt_create_db , ( gptr * ) & opt_create_db , 0 , GET_BOOL , NO_ARG , 0 , 0 ,
0 , 0 , 0 , 0 } ,
{ " no-create-info " , ' t ' , " Don't write table creation info. " ,
2002-05-14 21:41:55 +03:00
( gptr * ) & tFlag , ( gptr * ) & tFlag , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " no-data " , ' d ' , " No row information. " , ( gptr * ) & dFlag , ( gptr * ) & dFlag , 0 ,
GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-05-30 15:21:03 +05:00
{ " no-set-names " , ' N ' ,
" 'SET NAMES charset_name' will not be put in the output. " ,
( gptr * ) & opt_set_names , ( gptr * ) & opt_set_names , 0 , GET_BOOL , NO_ARG , 0 , 0 ,
0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " 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 } ,
{ " opt " , OPT_OPTIMIZE ,
2003-01-16 05:35:59 +02:00
" Same as --add-drop-table --add-locks --all --quick --extended-insert --lock-tables --disable-keys. Enabled by default, disable with --skip-opt. " ,
2002-04-02 20:29:53 +03:00
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " password " , ' p ' ,
" Password to use when connecting to server. If password is not given it's solicited on 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__
2003-06-13 10:59:02 +02:00
{ " pipe " , ' W ' , " Use named pipes to connect to server. " , 0 , 0 , 0 , GET_NO_ARG ,
2002-04-02 20:29:53 +03:00
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 } ,
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 ' , " Don't buffer query, dump directly to stdout. " ,
2003-01-16 05:35:59 +02:00
( gptr * ) & quick , ( gptr * ) & quick , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " quote-names " , ' Q ' , " Quote table and column names with backticks (`). " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & opt_quoted , ( gptr * ) & opt_quoted , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 ,
0 , 0 } ,
{ " result-file " , ' r ' ,
2002-08-05 09:00:38 +02:00
" Direct output to a given file. This option should be used in MSDOS, because it prevents new line ' \\ n' from being converted to ' \\ r \\ n' (carriage return + line feed). " ,
2002-04-30 12:32:45 +03:00
0 , 0 , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-11-15 00:16:30 +05:00
# ifdef HAVE_SMEM
{ " shared_memory_base_name " , OPT_SHARED_MEMORY_BASE_NAME ,
2003-06-13 10:59:02 +02:00
" Base name of shared memory. " , ( gptr * ) & shared_memory_base_name , ( gptr * ) & shared_memory_base_name ,
2002-11-15 00:16:30 +05:00
0 , GET_STR_ALLOC , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2003-01-16 05:35:59 +02:00
{ " skip-opt " , OPT_SKIP_OPTIMIZATION ,
" Disable --opt. Disables --add-locks, --all, --quick, --extended-insert, --lock-tables and --disable-keys. " ,
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-04-02 20:29:53 +03:00
{ " socket " , ' S ' , " Socket file to use for connection. " ,
( gptr * ) & opt_mysql_unix_port , ( gptr * ) & opt_mysql_unix_port , 0 , GET_STR ,
REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-09-17 23:46:53 +03:00
# include <sslopt-longopts.h>
2002-04-02 20:29:53 +03:00
{ " tab " , ' T ' ,
" Creates tab separated textfile for each table to given path. (creates .sql and .txt files). NOTE: This only works if mysqldump is run on the same machine as the mysqld daemon. " ,
( gptr * ) & path , ( gptr * ) & path , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " tables " , OPT_TABLES , " Overrides option --databases (-B). " ,
0 , 0 , 0 , GET_NO_ARG , NO_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 , ( 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 20:29:53 +03: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 } ,
{ " where " , ' w ' , " Dump only selected records; QUOTES mandatory! " ,
( gptr * ) & where , ( gptr * ) & where , 0 , GET_STR , REQUIRED_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ " xml " , ' X ' , " Dump a database as well formed XML. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2002-05-11 14:36:34 +03:00
{ " max_allowed_packet " , OPT_MAX_ALLOWED_PACKET , " " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & max_allowed_packet , ( gptr * ) & max_allowed_packet , 0 ,
2003-12-14 06:39:52 +02:00
GET_ULONG , REQUIRED_ARG , 24 * 1024 * 1024 , 4096 ,
( longlong ) 2L * 1024L * 1024L * 1024L , MALLOC_OVERHEAD , 1024 , 0 } ,
2002-05-11 14:36:34 +03:00
{ " net_buffer_length " , OPT_NET_BUFFER_LENGTH , " " ,
2002-04-02 20:29:53 +03:00
( gptr * ) & net_buffer_length , ( gptr * ) & net_buffer_length , 0 ,
2002-05-11 14:36:34 +03:00
GET_ULONG , REQUIRED_ARG , 1024 * 1024L - 1025 , 4096 , 16 * 1024L * 1024L ,
2002-04-03 13:36:01 +03:00
MALLOC_OVERHEAD - 1024 , 1024 , 0 } ,
2003-12-11 17:37:45 +04:00
{ " comments " , ' i ' , " Write additional information. " ,
( gptr * ) & opt_comments , ( gptr * ) & opt_comments , 0 , GET_BOOL , NO_ARG ,
1 , 0 , 0 , 0 , 0 , 0 } ,
2002-05-11 14:36:34 +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 const char * load_default_groups [ ] = { " mysqldump " , " client " , 0 } ;
static void safe_exit ( int error ) ;
2002-01-02 21:29:41 +02:00
static void write_header ( FILE * sql_file , char * db_name ) ;
2000-07-31 21:29:14 +02:00
static void print_value ( FILE * file , MYSQL_RES * result , MYSQL_ROW row ,
const char * prefix , const char * name ,
int string_value ) ;
static int dump_selected_tables ( char * db , char * * table_names , int tables ) ;
static int dump_all_tables_in_db ( char * db ) ;
static int init_dumping ( char * ) ;
static int dump_databases ( char * * ) ;
static int dump_all_databases ( ) ;
2003-09-03 18:48:10 +03:00
static char * quote_name ( const char * name , char * buff , my_bool force ) ;
2003-12-14 06:39:52 +02:00
static const char * check_if_ignore_table ( const char * table_name ) ;
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 , DUMP_VERSION ,
MYSQL_SERVER_VERSION , SYSTEM_TYPE , MACHINE_TYPE ) ;
} /* print_version */
2002-09-24 17:11:59 +03:00
static void short_usage_sub ( void )
{
printf ( " Usage: %s [OPTIONS] database [tables] \n " , my_progname ) ;
printf ( " OR %s [OPTIONS] --databases [OPTIONS] DB1 [DB2 DB3...] \n " ,
my_progname ) ;
printf ( " OR %s [OPTIONS] --all-databases [OPTIONS] \n " , my_progname ) ;
}
2000-07-31 21:29:14 +02:00
static void usage ( void )
{
print_version ( ) ;
puts ( " By Igor Romanenko, Monty, Jani & Sinisa " ) ;
puts ( " This software comes with ABSOLUTELY NO WARRANTY. This is free software, \n and you are welcome to modify and redistribute it under the GPL license \n " ) ;
puts ( " Dumping definition and data mysql database or table " ) ;
2002-09-24 17:11:59 +03:00
short_usage_sub ( ) ;
2000-07-31 21:29:14 +02:00
print_defaults ( " my " , load_default_groups ) ;
2002-04-02 20:29:53 +03:00
my_print_help ( my_long_options ) ;
my_print_variables ( my_long_options ) ;
2000-07-31 21:29:14 +02:00
} /* usage */
2002-09-24 17:11:59 +03:00
static void short_usage ( void )
{
short_usage_sub ( ) ;
printf ( " For more options, use %s --help \n " , my_progname ) ;
}
2002-01-02 21:29:41 +02:00
static void write_header ( FILE * sql_file , char * db_name )
2000-07-31 21:29:14 +02:00
{
2001-11-05 23:48:03 +02:00
if ( opt_xml )
2002-11-18 20:50:42 +04:00
{
2003-10-30 14:58:30 +04:00
fputs ( " <?xml version= \" 1.0 \" ?> \n " , sql_file ) ;
fputs ( " <mysqldump> \n " , sql_file ) ;
2002-11-18 20:50:42 +04:00
}
2003-12-11 17:37:45 +04:00
else if ( opt_comments )
2001-11-05 23:48:03 +02:00
{
fprintf ( sql_file , " -- MySQL dump %s \n -- \n " , DUMP_VERSION ) ;
fprintf ( sql_file , " -- Host: %s Database: %s \n " ,
current_host ? current_host : " localhost " , db_name ? db_name : " " ) ;
2003-09-06 15:06:08 -04:00
fputs ( " -- ------------------------------------------------------ \n " ,
2001-11-05 23:48:03 +02:00
sql_file ) ;
fprintf ( sql_file , " -- Server version \t %s \n " ,
mysql_get_server_info ( & mysql_connection ) ) ;
2003-05-30 15:21:03 +05:00
if ( ! opt_set_names )
2003-12-09 15:28:58 +04:00
fprintf ( sql_file , " \n /*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT, CHARACTER_SET_CLIENT=%s */; \n " , default_charset ) ;
2003-07-22 18:58:30 -04:00
fprintf ( md_result_file , " \
/*!40014 SET @OLD_UNIQUE_CHECKS=@@UNIQUE_CHECKS, UNIQUE_CHECKS=0 */ ; \ n \
/*!40014 SET @OLD_FOREIGN_KEY_CHECKS=@@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS=0 */ ; \ n \
/*!40101 SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE=NO_AUTO_VALUE_ON_ZERO */ ; \ n \
" );
2001-11-05 23:48:03 +02:00
}
2000-07-31 21:29:14 +02:00
return ;
2002-01-02 21:29:41 +02:00
} /* write_header */
2000-07-31 21:29:14 +02:00
2003-01-16 05:35:59 +02:00
2002-11-18 20:50:42 +04:00
static void write_footer ( FILE * sql_file )
{
if ( opt_xml )
2003-10-30 14:58:30 +04:00
fputs ( " </mysqldump> " , sql_file ) ;
2003-07-22 18:58:30 -04:00
else
{
2003-08-07 14:18:43 -04:00
fprintf ( md_result_file , " \n \
2003-07-22 18:58:30 -04:00
/*!40101 SET SQL_MODE=@OLD_SQL_MODE */ ; \ n \
/*!40014 SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS */ ; \ n \
/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */ ; \ n \
2003-12-09 15:28:58 +04:00
/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */ ; \ n \
2003-07-22 18:58:30 -04:00
" );
}
2002-11-18 20:50:42 +04:00
fputs ( " \n " , sql_file ) ;
} /* write_footer */
2000-07-31 21:29:14 +02:00
2003-01-16 05:35:59 +02:00
2003-04-03 21:19:12 +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
{
2003-04-03 12:33:13 +03:00
switch ( optid ) {
2002-04-02 20:29:53 +03:00
case OPT_MASTER_DATA :
opt_master_data = 1 ;
opt_first_slave = 1 ;
break ;
2003-04-03 12:33:13 +03:00
case OPT_DELETE_MASTER_LOGS :
opt_delete_master_logs = 1 ;
opt_first_slave = 1 ;
break ;
2002-04-02 20:29:53 +03:00
case ' p ' :
if ( argument )
{
char * start = argument ;
my_free ( opt_password , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
opt_password = my_strdup ( argument , MYF ( MY_FAE ) ) ;
while ( * argument ) * argument + + = ' x ' ; /* Destroy argument */
if ( * start )
start [ 1 ] = 0 ; /* Cut length of argument */
}
else
tty_password = 1 ;
break ;
case ' r ' :
2003-01-28 08:38:28 +02:00
if ( ! ( md_result_file = my_fopen ( argument , O_WRONLY | FILE_BINARY ,
2002-04-02 20:29:53 +03:00
MYF ( MY_WME ) ) ) )
exit ( 1 ) ;
break ;
case ' W ' :
2000-07-31 21:29:14 +02:00
# ifdef __WIN__
2002-11-15 00:16:30 +05:00
opt_protocol = MYSQL_PROTOCOL_PIPE ;
2000-07-31 21:29:14 +02:00
# endif
2002-04-02 20:29:53 +03:00
break ;
case ' T ' :
opt_disable_keys = 0 ;
break ;
case ' # ' :
DBUG_PUSH ( argument ? argument : " d:t:o " ) ;
break ;
2002-09-17 23:46:53 +03:00
# include <sslopt-case.h>
2002-04-02 20:29:53 +03:00
case ' V ' : print_version ( ) ; exit ( 0 ) ;
case ' X ' :
opt_xml = 1 ;
2003-10-14 12:19:54 +02:00
extended_insert = opt_drop = opt_lock =
opt_disable_keys = opt_autocommit = opt_create_db = 0 ;
2002-04-02 20:29:53 +03:00
break ;
case ' I ' :
case ' ? ' :
usage ( ) ;
exit ( 0 ) ;
case ( int ) OPT_OPTIMIZE :
2003-01-16 05:35:59 +02:00
extended_insert = opt_drop = opt_lock = quick = create_options =
opt_disable_keys = lock_tables = 1 ;
2002-06-11 13:37:48 +04:00
if ( opt_single_transaction ) lock_tables = 0 ;
2002-04-02 20:29:53 +03:00
break ;
2003-01-16 05:35:59 +02:00
case ( int ) OPT_SKIP_OPTIMIZATION :
extended_insert = opt_drop = opt_lock = quick = create_options =
opt_disable_keys = lock_tables = 0 ;
break ;
2002-04-02 20:29:53 +03:00
case ( int ) OPT_TABLES :
opt_databases = 0 ;
break ;
2003-01-16 05:35:59 +02:00
case ( int ) OPT_COMPATIBLE :
{
char buff [ 255 ] ;
opt_quoted = 1 ;
2003-06-10 11:34:00 +05:00
opt_set_names = 1 ;
2003-01-16 05:35:59 +02:00
opt_compatible_mode_str = argument ;
opt_compatible_mode = find_set ( & compatible_mode_typelib ,
argument , strlen ( argument ) ,
& err_ptr , & err_len ) ;
if ( err_len )
{
strmake ( buff , err_ptr , min ( sizeof ( buff ) , err_len ) ) ;
fprintf ( stderr , " Invalid mode to --compatible: %s \n " , buff ) ;
exit ( 1 ) ;
}
break ;
}
case ( int ) OPT_MYSQL_PROTOCOL :
2002-11-15 00:16:30 +05:00
{
2003-01-16 05:35:59 +02:00
if ( ( opt_protocol = find_type ( argument , & sql_protocol_typelib , 0 ) )
= = ~ ( ulong ) 0 )
{
fprintf ( stderr , " Unknown option to protocol: %s \n " , argument ) ;
exit ( 1 ) ;
}
break ;
2002-11-15 00:16:30 +05:00
}
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 )
{
int ho_error ;
md_result_file = stdout ;
load_defaults ( " my " , load_default_groups , argc , argv ) ;
if ( ( ho_error = handle_options ( argc , argv , my_long_options , get_one_option ) ) )
2002-05-29 15:07:30 +03:00
exit ( ho_error ) ;
2002-04-02 20:29:53 +03:00
2000-07-31 21:29:14 +02:00
if ( opt_delayed )
opt_lock = 0 ; /* Can't have lock with delayed */
if ( ! path & & ( enclosed | | opt_enclosed | | escaped | | lines_terminated | |
fields_terminated ) )
{
2002-04-02 20:29:53 +03:00
fprintf ( stderr ,
" %s: You must use option --tab with --fields-... \n " , my_progname ) ;
2000-07-31 21:29:14 +02:00
return ( 1 ) ;
}
2003-01-16 05:35:59 +02:00
if ( opt_single_transaction )
lock_tables = 0 ;
2000-07-31 21:29:14 +02:00
if ( enclosed & & opt_enclosed )
{
fprintf ( stderr , " %s: You can't use ..enclosed.. and ..optionally-enclosed.. at the same time. \n " , my_progname ) ;
return ( 1 ) ;
}
if ( replace & & ignore )
{
fprintf ( stderr , " %s: You can't use --ignore (-i) and --replace (-r) at the same time. \n " , my_progname ) ;
return ( 1 ) ;
}
if ( ( opt_databases | | opt_alldbs ) & & path )
{
fprintf ( stderr ,
" %s: --databases or --all-databases can't be used with --tab. \n " ,
my_progname ) ;
return ( 1 ) ;
}
2003-05-22 14:37:01 +05:00
if ( ! ( charset_info = get_charset_by_csname ( default_charset ,
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 & & ! opt_alldbs ) | | ( * argc > 0 & & opt_alldbs ) )
{
2002-09-24 17:11:59 +03:00
short_usage ( ) ;
2000-07-31 21:29:14 +02:00
return 1 ;
}
if ( tty_password )
2000-11-13 23:55:10 +02:00
opt_password = get_tty_password ( NullS ) ;
2000-07-31 21:29:14 +02:00
return ( 0 ) ;
} /* get_options */
/*
* * DBerror - - prints mysql error message and exits the program .
*/
static void DBerror ( MYSQL * mysql , const char * when )
{
DBUG_ENTER ( " DBerror " ) ;
2000-08-30 21:40:59 +04:00
my_printf_error ( 0 , " Got error: %d: %s %s " , MYF ( 0 ) ,
2000-07-31 21:29:14 +02:00
mysql_errno ( mysql ) , mysql_error ( mysql ) , when ) ;
safe_exit ( EX_MYSQLERR ) ;
DBUG_VOID_RETURN ;
} /* DBerror */
static void safe_exit ( int error )
{
if ( ! first_error )
first_error = error ;
if ( ignore_errors )
return ;
if ( sock )
mysql_close ( sock ) ;
exit ( error ) ;
}
/* safe_exit */
/*
* * dbConnect - - connects to the host and selects DB .
* * Also checks whether the tablename is a valid table name .
*/
static int dbConnect ( char * host , char * user , char * passwd )
{
DBUG_ENTER ( " dbConnect " ) ;
if ( verbose )
{
2001-07-04 09:39:58 +03:00
fprintf ( stderr , " -- Connecting to %s... \n " , host ? host : " localhost " ) ;
2000-07-31 21:29:14 +02:00
}
mysql_init ( & mysql_connection ) ;
if ( opt_compress )
mysql_options ( & mysql_connection , MYSQL_OPT_COMPRESS , NullS ) ;
# ifdef HAVE_OPENSSL
if ( opt_use_ssl )
mysql_ssl_set ( & mysql_connection , 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_connection , MYSQL_OPT_PROTOCOL , ( char * ) & opt_protocol ) ;
# ifdef HAVE_SMEM
if ( shared_memory_base_name )
mysql_options ( & mysql_connection , MYSQL_SHARED_MEMORY_BASE_NAME , shared_memory_base_name ) ;
2000-07-31 21:29:14 +02:00
# endif
2003-06-10 12:07:02 +05:00
if ( ! opt_set_names )
mysql_options ( & mysql_connection , MYSQL_SET_CHARSET_NAME , default_charset ) ;
2000-07-31 21:29:14 +02:00
if ( ! ( sock = mysql_real_connect ( & mysql_connection , host , user , passwd ,
NULL , opt_mysql_port , opt_mysql_unix_port ,
0 ) ) )
{
DBerror ( & mysql_connection , " when trying to connect " ) ;
return 1 ;
}
return 0 ;
} /* dbConnect */
/*
* * dbDisconnect - - disconnects from the host .
*/
static void dbDisconnect ( char * host )
{
if ( verbose )
2001-07-04 09:39:58 +03:00
fprintf ( stderr , " -- Disconnecting from %s... \n " , host ? host : " localhost " ) ;
2000-07-31 21:29:14 +02:00
mysql_close ( sock ) ;
} /* dbDisconnect */
static void unescape ( FILE * file , char * pos , uint length )
{
char * tmp ;
DBUG_ENTER ( " unescape " ) ;
if ( ! ( tmp = ( char * ) my_malloc ( length * 2 + 1 , MYF ( MY_WME ) ) ) )
{
ignore_errors = 0 ; /* Fatal error */
safe_exit ( EX_MYSQLERR ) ; /* Force exit */
}
2001-11-05 23:48:03 +02:00
mysql_real_escape_string ( & mysql_connection , tmp , pos , length ) ;
2000-07-31 21:29:14 +02:00
fputc ( ' \' ' , file ) ;
fputs ( tmp , file ) ;
fputc ( ' \' ' , file ) ;
my_free ( tmp , MYF ( MY_WME ) ) ;
DBUG_VOID_RETURN ;
} /* unescape */
static my_bool test_if_special_chars ( const char * str )
{
# if MYSQL_VERSION_ID >= 32300
for ( ; * str ; str + + )
2003-03-16 11:20:45 +04:00
if ( ! my_isvar ( charset_info , * str ) & & * str ! = ' $ ' )
2000-07-31 21:29:14 +02:00
return 1 ;
# endif
return 0 ;
} /* test_if_special_chars */
2003-09-03 18:48:10 +03:00
2003-11-04 09:40:36 +02:00
2003-09-03 18:48:10 +03:00
static char * quote_name ( const char * name , char * buff , my_bool force )
2000-07-31 21:29:14 +02:00
{
2003-09-03 18:48:10 +03:00
char * to = buff ;
if ( ! force & & ! opt_quoted & & ! test_if_special_chars ( name ) )
return ( char * ) name ;
* to + + = QUOTE_CHAR ;
while ( * name )
{
if ( * name = = QUOTE_CHAR )
* to = QUOTE_CHAR ;
* to + + = * name + + ;
}
to [ 0 ] = QUOTE_CHAR ;
to [ 1 ] = 0 ;
2000-07-31 21:29:14 +02:00
return buff ;
} /* quote_name */
2003-12-14 06:39:52 +02:00
static char * quote_for_like ( const char * name , char * buff )
{
char * to = buff ;
* to + + = ' \' ' ;
while ( * name )
{
if ( * name = = ' \' ' | | * name = = ' _ ' | | * name = = ' \\ ' | | * name = = ' % ' )
* to + + = ' \\ ' ;
* to + + = * name + + ;
}
to [ 0 ] = ' \' ' ;
to [ 1 ] = 0 ;
return buff ;
}
2003-11-03 16:49:39 +04:00
/*
Quote and print a string .
SYNOPSIS
print_quoted_xml ( )
output - output file
str - string to print
len - its length
DESCRIPTION
Quote ' < ' ' > ' ' & ' ' \" ' singns and print a string to the xml_file .
*/
static void print_quoted_xml ( FILE * xml_file , const char * str , ulong len )
{
const char * end ;
for ( end = str + len ; str ! = end ; str + + )
{
switch ( * str ) {
case ' < ' :
fputs ( " < " , xml_file ) ;
break ;
case ' > ' :
fputs ( " > " , xml_file ) ;
break ;
case ' & ' :
fputs ( " & " , xml_file ) ;
break ;
case ' \" ' :
fputs ( " " " , xml_file ) ;
break ;
default :
fputc ( * str , xml_file ) ;
break ;
}
}
}
/*
Print xml tag with one attribute .
SYNOPSIS
print_xml_tag1 ( )
xml_file - output file
sbeg - line beginning
stag_atr - tag and attribute
sval - value of attribute
send - line ending
DESCRIPTION
Print tag with one attribute to the xml_file . Format is :
sbeg < stag_atr = " sval " > send
NOTE
sval MUST be a NULL terminated string .
sval string will be qouted before output .
*/
static void print_xml_tag1 ( FILE * xml_file , const char * sbeg ,
const char * stag_atr , const char * sval ,
const char * send )
{
fputs ( sbeg , xml_file ) ;
fputs ( " < " , xml_file ) ;
fputs ( stag_atr , xml_file ) ;
fputs ( " \" " , xml_file ) ;
print_quoted_xml ( xml_file , sval , strlen ( sval ) ) ;
fputs ( " \" > " , xml_file ) ;
fputs ( send , xml_file ) ;
}
/*
Print xml tag with many attributes .
SYNOPSIS
print_xml_row ( )
xml_file - output file
row_name - xml tag name
tableRes - query result
row - result row
DESCRIPTION
Print tag with many attribute to the xml_file . Format is :
\ t \ t < row_name Atr1 = " Val1 " Atr2 = " Val2 " . . . / >
NOTE
All atributes and values will be quoted before output .
*/
static void print_xml_row ( FILE * xml_file , const char * row_name ,
MYSQL_RES * tableRes , MYSQL_ROW * row )
2003-10-30 14:58:30 +04:00
{
uint i ;
MYSQL_FIELD * field ;
ulong * lengths = mysql_fetch_lengths ( tableRes ) ;
fprintf ( xml_file , " \t \t <%s " , row_name ) ;
mysql_field_seek ( tableRes , 0 ) ;
for ( i = 0 ; ( field = mysql_fetch_field ( tableRes ) ) ; i + + )
{
2003-11-03 16:49:39 +04:00
if ( ( * row ) [ i ] )
2003-10-30 14:58:30 +04:00
{
fputs ( " " , xml_file ) ;
2003-11-03 16:49:39 +04:00
print_quoted_xml ( xml_file , field - > name , field - > name_length ) ;
2003-10-30 14:58:30 +04:00
fputs ( " = \" " , xml_file ) ;
print_quoted_xml ( xml_file , ( * row ) [ i ] , lengths [ i ] ) ;
fputs ( " \" " , xml_file ) ;
}
}
fputs ( " /> \n " , xml_file ) ;
}
2000-07-31 21:29:14 +02:00
/*
2003-09-03 18:48:10 +03:00
getStructure - - retrievs database structure , prints out corresponding
CREATE statement and fills out insert_pat .
RETURN
number of fields in table , 0 if error
2000-07-31 21:29:14 +02:00
*/
2003-09-03 18:48:10 +03:00
2000-07-31 21:29:14 +02:00
static uint getTableStructure ( char * table , char * db )
{
MYSQL_RES * tableRes ;
MYSQL_ROW row ;
my_bool init = 0 ;
uint numFields ;
2003-09-03 18:48:10 +03:00
char * strpos , * result_table , * opt_quoted_table ;
2000-07-31 21:29:14 +02:00
const char * delayed ;
2003-09-03 18:48:10 +03:00
char name_buff [ NAME_LEN + 3 ] , table_buff [ NAME_LEN * 2 + 3 ] ;
char table_buff2 [ NAME_LEN * 2 + 3 ] ;
2001-08-22 01:45:07 +03:00
FILE * sql_file = md_result_file ;
2000-07-31 21:29:14 +02:00
DBUG_ENTER ( " getTableStructure " ) ;
delayed = opt_delayed ? " DELAYED " : " " ;
if ( verbose )
2001-07-04 09:39:58 +03:00
fprintf ( stderr , " -- Retrieving table structure for table %s... \n " , table ) ;
2000-07-31 21:29:14 +02:00
2003-11-04 09:40:36 +02:00
sprintf ( insert_pat , " SET OPTION SQL_QUOTE_SHOW_CREATE=%d " ,
( opt_quoted | | opt_keywords ) ) ;
2003-09-03 18:48:10 +03:00
result_table = quote_name ( table , table_buff , 1 ) ;
opt_quoted_table = quote_name ( table , table_buff2 , 0 ) ;
2003-10-30 14:58:30 +04:00
if ( ! opt_xml & & ! mysql_query ( sock , insert_pat ) )
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
/* using SHOW CREATE statement */
if ( ! tFlag )
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
/* Make an sql-file, if path was given iow. option -T was given */
char buff [ 20 + FN_REFLEN ] ;
2003-01-16 05:35:59 +02:00
if ( opt_compatible_mode )
{
char * end ;
uint i ;
sprintf ( buff , " /*!41000 SET @@sql_mode= \" " ) ;
end = strend ( buff ) ;
for ( i = 0 ; opt_compatible_mode ; opt_compatible_mode > > = 1 , i + + )
{
if ( opt_compatible_mode & 1 )
{
end = strmov ( end , compatible_mode_names [ i ] ) ;
end = strmov ( end , " , " ) ;
}
}
2003-09-11 20:24:14 +03:00
end = strmov ( end - 1 , " \" */ " ) ;
2003-01-16 05:35:59 +02:00
if ( mysql_query ( sock , buff ) )
{
fprintf ( stderr , " %s: Can't set the compatible mode '%s' (%s) \n " ,
my_progname , table , mysql_error ( sock ) ) ;
safe_exit ( EX_MYSQLERR ) ;
DBUG_RETURN ( 0 ) ;
}
}
2003-09-03 18:48:10 +03:00
sprintf ( buff , " show create table %s " , result_table ) ;
2000-10-29 13:49:42 +01:00
if ( mysql_query ( sock , buff ) )
2000-07-31 21:29:14 +02:00
{
2003-09-03 18:48:10 +03:00
fprintf ( stderr , " %s: Can't get CREATE TABLE for table %s (%s) \n " ,
my_progname , result_table , mysql_error ( sock ) ) ;
2000-10-29 13:49:42 +01:00
safe_exit ( EX_MYSQLERR ) ;
DBUG_RETURN ( 0 ) ;
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
if ( path )
{
char filename [ FN_REFLEN ] , tmp_path [ FN_REFLEN ] ;
2001-10-08 04:58:07 +03:00
convert_dirname ( tmp_path , path , NullS ) ;
2000-10-29 13:49:42 +01:00
sql_file = my_fopen ( fn_format ( filename , table , tmp_path , " .sql " , 4 ) ,
2001-12-20 18:14:55 +00:00
O_WRONLY , MYF ( MY_WME ) ) ;
2000-10-29 13:49:42 +01:00
if ( ! sql_file ) /* If file couldn't be opened */
{
2001-04-24 00:31:34 +03:00
safe_exit ( EX_MYSQLERR ) ;
DBUG_RETURN ( 0 ) ;
2000-10-29 13:49:42 +01:00
}
2002-01-02 21:29:41 +02:00
write_header ( sql_file , db ) ;
2000-10-29 13:49:42 +01:00
}
2003-12-11 17:37:45 +04:00
if ( ! opt_xml & & opt_comments )
2003-12-17 17:35:34 +02:00
fprintf ( sql_file , " \n -- \n -- Table structure for table %s \n -- \n \n " ,
2003-09-03 18:48:10 +03:00
result_table ) ;
2000-10-29 13:49:42 +01:00
if ( opt_drop )
2003-09-03 18:48:10 +03:00
fprintf ( sql_file , " DROP TABLE IF EXISTS %s; \n " , opt_quoted_table ) ;
2000-10-29 13:49:42 +01:00
tableRes = mysql_store_result ( sock ) ;
row = mysql_fetch_row ( tableRes ) ;
2003-10-30 14:58:30 +04:00
fprintf ( sql_file , " %s; \n " , row [ 1 ] ) ;
2000-10-29 13:49:42 +01:00
mysql_free_result ( tableRes ) ;
}
2003-09-03 18:48:10 +03:00
sprintf ( insert_pat , " show fields from %s " , result_table ) ;
2000-10-29 13:49:42 +01:00
if ( mysql_query ( sock , insert_pat ) | | ! ( tableRes = mysql_store_result ( sock ) ) )
2000-07-31 21:29:14 +02:00
{
2003-09-03 18:48:10 +03:00
fprintf ( stderr , " %s: Can't get info about table: %s \n error: %s \n " ,
my_progname , result_table , mysql_error ( sock ) ) ;
2002-01-02 21:29:41 +02:00
if ( path )
my_fclose ( sql_file , MYF ( MY_WME ) ) ;
2000-10-29 13:49:42 +01:00
safe_exit ( EX_MYSQLERR ) ;
DBUG_RETURN ( 0 ) ;
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
2000-07-31 21:29:14 +02:00
if ( cFlag )
2003-09-03 18:48:10 +03:00
sprintf ( insert_pat , " INSERT %sINTO %s ( " , delayed , opt_quoted_table ) ;
2000-10-29 13:49:42 +01:00
else
2000-07-31 21:29:14 +02:00
{
2003-09-03 18:48:10 +03:00
sprintf ( insert_pat , " INSERT %sINTO %s VALUES " , delayed ,
opt_quoted_table ) ;
2000-10-29 13:49:42 +01:00
if ( ! extended_insert )
strcat ( insert_pat , " ( " ) ;
}
strpos = strend ( insert_pat ) ;
while ( ( row = mysql_fetch_row ( tableRes ) ) )
{
if ( init )
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
if ( cFlag )
2000-12-15 10:58:17 +02:00
strpos = strmov ( strpos , " , " ) ;
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
init = 1 ;
if ( cFlag )
2003-09-03 18:48:10 +03:00
strpos = strmov ( strpos , quote_name ( row [ SHOW_FIELDNAME ] , name_buff , 0 ) ) ;
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
numFields = ( uint ) mysql_num_rows ( tableRes ) ;
mysql_free_result ( tableRes ) ;
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
else
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
/* fprintf(stderr, "%s: Can't set SQL_QUOTE_SHOW_CREATE option (%s)\n",
my_progname , mysql_error ( sock ) ) ; */
2003-09-03 18:48:10 +03:00
sprintf ( insert_pat , " show fields from %s " , result_table ) ;
2000-10-29 13:49:42 +01:00
if ( mysql_query ( sock , insert_pat ) | | ! ( tableRes = mysql_store_result ( sock ) ) )
2000-07-31 21:29:14 +02:00
{
2003-09-03 18:48:10 +03:00
fprintf ( stderr , " %s: Can't get info about table: %s \n error: %s \n " ,
my_progname , result_table , mysql_error ( sock ) ) ;
2000-07-31 21:29:14 +02:00
safe_exit ( EX_MYSQLERR ) ;
DBUG_RETURN ( 0 ) ;
}
2000-10-29 13:49:42 +01:00
/* Make an sql-file, if path was given iow. option -T was given */
if ( ! tFlag )
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
if ( path )
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
char filename [ FN_REFLEN ] , tmp_path [ FN_REFLEN ] ;
2001-10-08 04:58:07 +03:00
convert_dirname ( tmp_path , path , NullS ) ;
2000-10-29 13:49:42 +01:00
sql_file = my_fopen ( fn_format ( filename , table , tmp_path , " .sql " , 4 ) ,
O_WRONLY , MYF ( MY_WME ) ) ;
if ( ! sql_file ) /* If file couldn't be opened */
{
2002-01-02 21:29:41 +02:00
safe_exit ( EX_MYSQLERR ) ;
DBUG_RETURN ( 0 ) ;
2000-10-29 13:49:42 +01:00
}
2002-01-02 21:29:41 +02:00
write_header ( sql_file , db ) ;
2000-07-31 21:29:14 +02:00
}
2003-12-11 17:37:45 +04:00
if ( ! opt_xml & & opt_comments )
2003-09-03 18:48:10 +03:00
fprintf ( sql_file , " \n -- \n -- Table structure for table %s \n -- \n \n " ,
result_table ) ;
2000-10-29 13:49:42 +01:00
if ( opt_drop )
2003-09-02 18:30:34 +02:00
fprintf ( sql_file , " DROP TABLE IF EXISTS %s; \n " , result_table ) ;
2003-10-30 14:58:30 +04:00
if ( ! opt_xml )
fprintf ( sql_file , " CREATE TABLE %s ( \n " , result_table ) ;
else
2003-11-03 16:49:39 +04:00
print_xml_tag1 ( sql_file , " \t " , " table_structure name= " , table , " \n " ) ;
2000-10-29 13:49:42 +01:00
}
if ( cFlag )
2003-09-02 18:30:34 +02:00
sprintf ( insert_pat , " INSERT %sINTO %s ( " , delayed , result_table ) ;
2000-10-29 13:49:42 +01:00
else
{
2003-09-02 18:30:34 +02:00
sprintf ( insert_pat , " INSERT %sINTO %s VALUES " , delayed , result_table ) ;
2000-10-29 13:49:42 +01:00
if ( ! extended_insert )
strcat ( insert_pat , " ( " ) ;
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
strpos = strend ( insert_pat ) ;
2000-07-31 21:29:14 +02:00
while ( ( row = mysql_fetch_row ( tableRes ) ) )
{
2000-10-29 13:49:42 +01:00
ulong * lengths = mysql_fetch_lengths ( tableRes ) ;
if ( init )
2000-07-31 21:29:14 +02:00
{
2003-10-30 14:58:30 +04:00
if ( ! opt_xml & & ! tFlag )
2001-04-26 00:10:43 +03:00
fputs ( " , \n " , sql_file ) ;
2000-10-29 13:49:42 +01:00
if ( cFlag )
2001-04-26 00:10:43 +03:00
strpos = strmov ( strpos , " , " ) ;
2000-10-29 13:49:42 +01:00
}
init = 1 ;
if ( cFlag )
2003-09-03 18:48:10 +03:00
strpos = strmov ( strpos , quote_name ( row [ SHOW_FIELDNAME ] , name_buff , 0 ) ) ;
2000-10-29 13:49:42 +01:00
if ( ! tFlag )
{
2003-10-30 14:58:30 +04:00
if ( opt_xml )
{
print_xml_row ( sql_file , " field " , tableRes , & row ) ;
continue ;
}
2000-10-29 13:49:42 +01:00
if ( opt_keywords )
2003-09-02 18:30:34 +02:00
fprintf ( sql_file , " %s.%s %s " , result_table ,
2003-09-03 18:48:10 +03:00
quote_name ( row [ SHOW_FIELDNAME ] , name_buff , 0 ) ,
row [ SHOW_TYPE ] ) ;
2000-10-29 13:49:42 +01:00
else
2001-04-24 00:31:34 +03:00
fprintf ( sql_file , " %s %s " , quote_name ( row [ SHOW_FIELDNAME ] ,
2003-09-03 18:48:10 +03:00
name_buff , 0 ) ,
row [ SHOW_TYPE ] ) ;
2000-10-29 13:49:42 +01:00
if ( row [ SHOW_DEFAULT ] )
{
2001-04-24 00:31:34 +03:00
fputs ( " DEFAULT " , sql_file ) ;
2001-11-05 23:48:03 +02:00
unescape ( sql_file , row [ SHOW_DEFAULT ] , lengths [ SHOW_DEFAULT ] ) ;
2000-10-29 13:49:42 +01:00
}
if ( ! row [ SHOW_NULL ] [ 0 ] )
2001-04-24 00:31:34 +03:00
fputs ( " NOT NULL " , sql_file ) ;
2000-10-29 13:49:42 +01:00
if ( row [ SHOW_EXTRA ] [ 0 ] )
2001-04-24 00:31:34 +03:00
fprintf ( sql_file , " %s " , row [ SHOW_EXTRA ] ) ;
2000-07-31 21:29:14 +02:00
}
}
2000-10-29 13:49:42 +01:00
numFields = ( uint ) mysql_num_rows ( tableRes ) ;
mysql_free_result ( tableRes ) ;
if ( ! tFlag )
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
/* Make an sql-file, if path was given iow. option -T was given */
char buff [ 20 + FN_REFLEN ] ;
uint keynr , primary_key ;
2003-09-03 18:48:10 +03:00
sprintf ( buff , " show keys from %s " , result_table ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_query ( sock , buff ) )
{
2003-09-03 18:48:10 +03:00
fprintf ( stderr , " %s: Can't get keys for table %s (%s) \n " ,
my_progname , result_table , mysql_error ( sock ) ) ;
2002-01-02 21:29:41 +02:00
if ( path )
2001-04-24 00:31:34 +03:00
my_fclose ( sql_file , MYF ( MY_WME ) ) ;
2000-10-29 13:49:42 +01:00
safe_exit ( EX_MYSQLERR ) ;
DBUG_RETURN ( 0 ) ;
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
tableRes = mysql_store_result ( sock ) ;
/* Find first which key is primary key */
keynr = 0 ;
primary_key = INT_MAX ;
while ( ( row = mysql_fetch_row ( tableRes ) ) )
2000-07-31 21:29:14 +02:00
{
2000-10-29 13:49:42 +01:00
if ( atoi ( row [ 3 ] ) = = 1 )
{
2001-04-24 00:31:34 +03:00
keynr + + ;
# ifdef FORCE_PRIMARY_KEY
if ( atoi ( row [ 1 ] ) = = 0 & & primary_key = = INT_MAX )
primary_key = keynr ;
# endif
if ( ! strcmp ( row [ 2 ] , " PRIMARY " ) )
{
primary_key = keynr ;
break ;
}
2000-10-29 13:49:42 +01:00
}
2000-07-31 21:29:14 +02:00
}
2000-10-29 13:49:42 +01:00
mysql_data_seek ( tableRes , 0 ) ;
keynr = 0 ;
while ( ( row = mysql_fetch_row ( tableRes ) ) )
{
2003-10-30 14:58:30 +04:00
if ( opt_xml )
{
print_xml_row ( sql_file , " key " , tableRes , & row ) ;
continue ;
}
2000-10-29 13:49:42 +01:00
if ( atoi ( row [ 3 ] ) = = 1 )
{
2001-04-24 00:31:34 +03:00
if ( keynr + + )
putc ( ' ) ' , sql_file ) ;
if ( atoi ( row [ 1 ] ) ) /* Test if duplicate key */
/* Duplicate allowed */
2003-09-03 18:48:10 +03:00
fprintf ( sql_file , " , \n KEY %s ( " , quote_name ( row [ 2 ] , name_buff , 0 ) ) ;
2001-04-24 00:31:34 +03:00
else if ( keynr = = primary_key )
fputs ( " , \n PRIMARY KEY ( " , sql_file ) ; /* First UNIQUE is primary */
else
2003-11-04 09:40:36 +02:00
fprintf ( sql_file , " , \n UNIQUE %s ( " , quote_name ( row [ 2 ] , name_buff ,
0 ) ) ;
2000-10-29 13:49:42 +01:00
}
else
2001-04-24 00:31:34 +03:00
putc ( ' , ' , sql_file ) ;
2003-09-03 18:48:10 +03:00
fputs ( quote_name ( row [ 4 ] , name_buff , 0 ) , sql_file ) ;
2000-10-29 13:49:42 +01:00
if ( row [ 7 ] )
2001-04-24 00:31:34 +03:00
fprintf ( sql_file , " (%s) " , row [ 7 ] ) ; /* Sub key */
2000-10-29 13:49:42 +01:00
}
2003-10-30 14:58:30 +04:00
if ( ! opt_xml )
{
if ( keynr )
putc ( ' ) ' , sql_file ) ;
fputs ( " \n ) " , sql_file ) ;
}
2000-10-29 13:49:42 +01:00
/* Get MySQL specific create options */
if ( create_options )
2000-07-31 21:29:14 +02:00
{
2003-12-14 06:39:52 +02:00
char show_name_buff [ FN_REFLEN ] ;
sprintf ( buff , " show table status like %s " ,
quote_for_like ( table , show_name_buff ) ) ;
2000-10-29 13:49:42 +01:00
if ( mysql_query ( sock , buff ) )
{
2001-04-24 00:31:34 +03:00
if ( mysql_errno ( sock ) ! = ER_PARSE_ERROR )
{ /* If old MySQL version */
if ( verbose )
fprintf ( stderr ,
2003-09-03 18:48:10 +03:00
" -- Warning: Couldn't get status information for table %s (%s) \n " ,
result_table , mysql_error ( sock ) ) ;
2001-04-24 00:31:34 +03:00
}
2000-10-29 13:49:42 +01:00
}
else if ( ! ( tableRes = mysql_store_result ( sock ) ) | |
2001-04-24 00:31:34 +03:00
! ( row = mysql_fetch_row ( tableRes ) ) )
2000-10-29 13:49:42 +01:00
{
2001-04-24 00:31:34 +03:00
fprintf ( stderr ,
2003-09-03 18:48:10 +03:00
" Error: Couldn't read status information for table %s (%s) \n " ,
result_table , mysql_error ( sock ) ) ;
2000-10-29 13:49:42 +01:00
}
else
{
2003-10-30 14:58:30 +04:00
if ( opt_xml )
{
print_xml_row ( sql_file , " options " , tableRes , & row ) ;
}
else
{
fputs ( " /*! " , sql_file ) ;
print_value ( sql_file , tableRes , row , " type= " , " Type " , 0 ) ;
print_value ( sql_file , tableRes , row , " " , " Create_options " , 0 ) ;
print_value ( sql_file , tableRes , row , " comment= " , " Comment " , 1 ) ;
fputs ( " */ " , sql_file ) ;
}
2000-10-29 13:49:42 +01:00
}
mysql_free_result ( tableRes ) ; /* Is always safe to free */
2000-07-31 21:29:14 +02:00
}
2003-10-30 14:58:30 +04:00
if ( ! opt_xml )
fputs ( " ; \n " , sql_file ) ;
else
fputs ( " \t </table_structure> \n " , sql_file ) ;
2000-07-31 21:29:14 +02:00
}
}
if ( cFlag )
{
strpos = strmov ( strpos , " ) VALUES " ) ;
if ( ! extended_insert )
strpos = strmov ( strpos , " ( " ) ;
}
2002-01-02 21:29:41 +02:00
if ( sql_file ! = md_result_file )
my_fclose ( sql_file , MYF ( MY_WME ) ) ;
2000-07-31 21:29:14 +02:00
DBUG_RETURN ( numFields ) ;
} /* getTableStructure */
static char * add_load_option ( char * ptr , const char * object ,
const char * statement )
{
if ( object )
{
2001-01-02 14:29:47 +02: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 ;
} /* add_load_option */
/*
* * 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 .
*/
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 ;
} /* field_escape */
/*
* * dumpTable saves database contents as a series of INSERT statements .
*/
static void dumpTable ( uint numFields , char * table )
{
2002-04-27 12:09:59 +03:00
char query [ QUERY_LENGTH ] , * end , buff [ 256 ] , table_buff [ NAME_LEN + 3 ] ;
2003-09-05 06:56:28 +03:00
char * result_table , table_buff2 [ NAME_LEN * 2 + 3 ] , * opt_quoted_table ;
2000-07-31 21:29:14 +02:00
MYSQL_RES * res ;
2003-02-16 22:50:00 +02:00
MYSQL_FIELD * field ;
MYSQL_ROW row ;
2000-07-31 21:29:14 +02:00
ulong rownr , row_break , total_length , init_length ;
2003-12-14 06:39:52 +02:00
const char * table_type ;
2000-07-31 21:29:14 +02:00
2003-09-03 18:48:10 +03:00
result_table = quote_name ( table , table_buff , 1 ) ;
opt_quoted_table = quote_name ( table , table_buff2 , 0 ) ;
2003-12-14 06:39:52 +02:00
/* Check table type */
if ( ( table_type = check_if_ignore_table ( table ) ) )
{
if ( verbose )
fprintf ( stderr ,
" -- Skipping data for table '%s' because it's of type %s \n " ,
table , table_type ) ;
return ;
}
if ( verbose )
fprintf ( stderr , " -- Sending SELECT query... \n " ) ;
2000-07-31 21:29:14 +02:00
if ( path )
{
char filename [ FN_REFLEN ] , tmp_path [ FN_REFLEN ] ;
2001-10-08 04:58:07 +03:00
convert_dirname ( tmp_path , path , NullS ) ;
2000-07-31 21:29:14 +02:00
my_load_path ( tmp_path , tmp_path , NULL ) ;
fn_format ( filename , table , tmp_path , " .txt " , 4 ) ;
my_delete ( filename , MYF ( 0 ) ) ; /* 'INTO OUTFILE' doesn't work, if
filename wasn ' t deleted */
to_unix_path ( filename ) ;
2002-04-27 12:09:59 +03:00
sprintf ( query , " SELECT /*!40001 SQL_NO_CACHE */ * INTO OUTFILE '%s' " ,
filename ) ;
2000-07-31 21:29:14 +02:00
end = strend ( query ) ;
if ( replace )
end = strmov ( end , " REPLACE " ) ;
if ( ignore )
end = strmov ( end , " IGNORE " ) ;
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 " ) ;
* end = ' \0 ' ;
2003-09-03 18:48:10 +03:00
sprintf ( buff , " FROM %s " , result_table ) ;
2000-07-31 21:29:14 +02:00
end = strmov ( end , buff ) ;
if ( where )
end = strxmov ( end , " WHERE " , where , NullS ) ;
if ( mysql_query ( sock , query ) )
{
DBerror ( sock , " when executing 'SELECT INTO OUTFILE' " ) ;
return ;
}
}
else
{
2003-12-11 17:37:45 +04:00
if ( ! opt_xml & & opt_comments )
2003-09-03 18:48:10 +03:00
fprintf ( md_result_file , " \n -- \n -- Dumping data for table %s \n -- \n " ,
result_table ) ;
2002-04-27 12:09:59 +03:00
sprintf ( query , " SELECT /*!40001 SQL_NO_CACHE */ * FROM %s " ,
2003-09-05 06:56:28 +03:00
result_table ) ;
2000-07-31 21:29:14 +02:00
if ( where )
{
2003-12-11 17:37:45 +04:00
if ( ! opt_xml & & opt_comments )
2001-11-05 23:48:03 +02:00
fprintf ( md_result_file , " -- WHERE: %s \n " , where ) ;
2000-07-31 21:29:14 +02:00
strxmov ( strend ( query ) , " WHERE " , where , NullS ) ;
}
2001-11-05 23:48:03 +02:00
if ( ! opt_xml )
2002-04-03 13:36:01 +03:00
fputs ( " \n " , md_result_file ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_query ( sock , query ) )
{
DBerror ( sock , " when retrieving data from server " ) ;
return ;
}
if ( quick )
res = mysql_use_result ( sock ) ;
else
res = mysql_store_result ( sock ) ;
if ( ! res )
{
DBerror ( sock , " when retrieving data from server " ) ;
return ;
}
if ( verbose )
2001-07-04 09:39:58 +03:00
fprintf ( stderr , " -- Retrieving rows... \n " ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_num_fields ( res ) ! = numFields )
{
2003-09-03 18:48:10 +03:00
fprintf ( stderr , " %s: Error in field count for table: %s ! Aborting. \n " ,
my_progname , result_table ) ;
2000-07-31 21:29:14 +02:00
safe_exit ( EX_CONSCHECK ) ;
return ;
}
2002-04-03 13:36:01 +03:00
if ( opt_disable_keys )
2003-09-05 08:16:13 +03:00
fprintf ( md_result_file , " \n /*!40000 ALTER TABLE %s DISABLE KEYS */; \n " ,
2003-09-05 06:56:28 +03:00
opt_quoted_table ) ;
2000-07-31 21:29:14 +02:00
if ( opt_lock )
2003-09-03 18:48:10 +03:00
fprintf ( md_result_file , " LOCK TABLES %s WRITE; \n " , opt_quoted_table ) ;
2000-07-31 21:29:14 +02:00
total_length = net_buffer_length ; /* Force row break */
row_break = 0 ;
rownr = 0 ;
2000-08-22 00:18:32 +03:00
init_length = ( uint ) strlen ( insert_pat ) + 4 ;
2001-11-05 23:48:03 +02:00
if ( opt_xml )
2003-11-03 16:49:39 +04:00
print_xml_tag1 ( md_result_file , " \t " , " table_data name= " , table , " \n " ) ;
2000-07-31 21:29:14 +02:00
2001-10-04 20:43:06 +03:00
if ( opt_autocommit )
fprintf ( md_result_file , " set autocommit=0; \n " ) ;
2000-07-31 21:29:14 +02:00
while ( ( row = mysql_fetch_row ( res ) ) )
{
uint i ;
ulong * lengths = mysql_fetch_lengths ( res ) ;
rownr + + ;
2001-11-05 23:48:03 +02:00
if ( ! extended_insert & & ! opt_xml )
2001-08-22 01:45:07 +03:00
fputs ( insert_pat , md_result_file ) ;
2000-07-31 21:29:14 +02:00
mysql_field_seek ( res , 0 ) ;
2002-04-03 13:36:01 +03:00
if ( opt_xml )
2003-10-30 14:58:30 +04:00
fputs ( " \t <row> \n " , md_result_file ) ;
2002-04-03 13:36:01 +03:00
2000-07-31 21:29:14 +02:00
for ( i = 0 ; i < mysql_num_fields ( res ) ; i + + )
{
if ( ! ( field = mysql_fetch_field ( res ) ) )
{
2003-09-03 18:48:10 +03:00
sprintf ( query , " %s: Not enough fields from table %s! Aborting. \n " ,
my_progname , result_table ) ;
2000-07-31 21:29:14 +02:00
fputs ( query , stderr ) ;
safe_exit ( EX_CONSCHECK ) ;
return ;
}
2003-10-14 12:19:54 +02:00
if ( extended_insert )
2000-07-31 21:29:14 +02:00
{
ulong length = lengths [ i ] ;
if ( i = = 0 )
dynstr_set ( & extended_row , " ( " ) ;
else
dynstr_append ( & extended_row , " , " ) ;
if ( row [ i ] )
{
if ( length )
{
2000-09-02 07:58:42 +03:00
if ( ! IS_NUM_FIELD ( field ) )
2000-07-31 21:29:14 +02:00
{
if ( dynstr_realloc ( & extended_row , length * 2 + 2 ) )
{
fputs ( " Aborting dump (out of memory) " , stderr ) ;
safe_exit ( EX_EOM ) ;
}
dynstr_append ( & extended_row , " \' " ) ;
2000-08-30 21:40:59 +04:00
extended_row . length + =
2000-07-31 21:29:14 +02:00
mysql_real_escape_string ( & mysql_connection ,
& extended_row . str [ extended_row . length ] , row [ i ] , length ) ;
extended_row . str [ extended_row . length ] = ' \0 ' ;
dynstr_append ( & extended_row , " \' " ) ;
}
else
2001-09-03 16:09:25 +03:00
{
2003-12-09 17:33:29 +04:00
/* change any strings ("inf", "-inf", "nan") into NULL */
2001-09-03 16:09:25 +03:00
char * ptr = row [ i ] ;
2003-12-17 17:35:34 +02:00
if ( my_isalpha ( charset_info , * ptr ) | | ( * ptr = = ' - ' & &
my_isalpha ( charset_info , ptr [ 1 ] ) ) )
2003-12-09 17:33:29 +04:00
dynstr_append ( & extended_row , " NULL " ) ;
else
{
if ( field - > type = = FIELD_TYPE_DECIMAL )
{
/* add " signs around */
dynstr_append ( & extended_row , " \" " ) ;
dynstr_append ( & extended_row , ptr ) ;
dynstr_append ( & extended_row , " \" " ) ;
}
else
dynstr_append ( & extended_row , ptr ) ;
}
2001-09-03 16:09:25 +03:00
}
2000-07-31 21:29:14 +02:00
}
else
dynstr_append ( & extended_row , " \' \' " ) ;
}
else if ( dynstr_append ( & extended_row , " NULL " ) )
{
fputs ( " Aborting dump (out of memory) " , stderr ) ;
safe_exit ( EX_EOM ) ;
}
}
else
{
2001-11-05 23:48:03 +02:00
if ( i & & ! opt_xml )
fputc ( ' , ' , md_result_file ) ;
2000-07-31 21:29:14 +02:00
if ( row [ i ] )
{
2000-09-02 07:58:42 +03:00
if ( ! IS_NUM_FIELD ( field ) )
2001-12-20 18:14:55 +00:00
{
2001-11-26 19:14:26 +02:00
if ( opt_xml )
2003-10-30 14:58:30 +04:00
{
2003-11-03 16:49:39 +04:00
print_xml_tag1 ( md_result_file , " \t \t " , " field name= " ,
field - > name , " " ) ;
2003-10-30 14:58:30 +04:00
print_quoted_xml ( md_result_file , row [ i ] , lengths [ i ] ) ;
fputs ( " </field> \n " , md_result_file ) ;
}
2001-11-26 19:14:26 +02:00
else
unescape ( md_result_file , row [ i ] , lengths [ i ] ) ;
2001-11-05 23:48:03 +02:00
}
2000-07-31 21:29:14 +02:00
else
2001-09-03 16:09:25 +03:00
{
2003-12-09 17:33:29 +04:00
/* change any strings ("inf", "-inf", "nan") into NULL */
2001-09-03 16:09:25 +03:00
char * ptr = row [ i ] ;
2001-11-26 19:14:26 +02:00
if ( opt_xml )
2003-10-30 14:58:30 +04:00
{
2003-11-03 16:49:39 +04:00
print_xml_tag1 ( md_result_file , " \t \t " , " field name= " ,
field - > name , " " ) ;
2003-10-30 14:58:30 +04:00
fputs ( ! my_isalpha ( charset_info , * ptr ) ? ptr : " NULL " ,
md_result_file ) ;
fputs ( " </field> \n " , md_result_file ) ;
}
2003-12-17 17:35:34 +02:00
else if ( my_isalpha ( charset_info , * ptr ) | |
( * ptr = = ' - ' & & my_isalpha ( charset_info , ptr [ 1 ] ) ) )
fputs ( " NULL " , md_result_file ) ;
else if ( field - > type = = FIELD_TYPE_DECIMAL )
2003-12-19 16:25:50 +02:00
{
/* add " signs around */
fputs ( " \" " , md_result_file ) ;
fputs ( ptr , md_result_file ) ;
fputs ( " \" " , md_result_file ) ;
}
2001-11-26 19:14:26 +02:00
else
2003-12-19 16:25:50 +02:00
fputs ( ptr , md_result_file ) ;
2001-09-03 16:09:25 +03:00
}
2000-07-31 21:29:14 +02:00
}
}
}
2002-04-03 13:36:01 +03:00
if ( opt_xml )
2003-10-30 14:58:30 +04:00
fputs ( " \t </row> \n " , md_result_file ) ;
2002-04-03 13:36:01 +03:00
2003-10-14 12:19:54 +02:00
if ( extended_insert )
2000-07-31 21:29:14 +02:00
{
ulong row_length ;
dynstr_append ( & extended_row , " ) " ) ;
row_length = 2 + extended_row . length ;
if ( total_length + row_length < net_buffer_length )
{
total_length + = row_length ;
2001-08-22 01:45:07 +03:00
fputc ( ' , ' , md_result_file ) ; /* Always row break */
fputs ( extended_row . str , md_result_file ) ;
2000-07-31 21:29:14 +02:00
}
else
{
2003-10-14 12:19:54 +02:00
if ( row_break )
2001-08-22 01:45:07 +03:00
fputs ( " ; \n " , md_result_file ) ;
2000-07-31 21:29:14 +02:00
row_break = 1 ; /* This is first row */
2001-12-06 14:10:51 +02:00
2003-10-14 12:19:54 +02:00
fputs ( insert_pat , md_result_file ) ;
fputs ( extended_row . str , md_result_file ) ;
2000-07-31 21:29:14 +02:00
total_length = row_length + init_length ;
}
}
2001-11-05 23:48:03 +02:00
else if ( ! opt_xml )
2001-08-22 01:45:07 +03:00
fputs ( " ); \n " , md_result_file ) ;
2000-07-31 21:29:14 +02:00
}
2001-12-06 14:10:51 +02:00
2002-01-23 00:04:43 +02:00
/* XML - close table tag and supress regular output */
2001-11-05 23:48:03 +02:00
if ( opt_xml )
2003-10-30 14:58:30 +04:00
fputs ( " \t </table_data> \n " , md_result_file ) ;
2001-11-05 23:48:03 +02:00
else if ( extended_insert & & row_break )
2001-08-22 01:45:07 +03:00
fputs ( " ; \n " , md_result_file ) ; /* If not empty table */
fflush ( md_result_file ) ;
2000-07-31 21:29:14 +02:00
if ( mysql_errno ( sock ) )
{
2003-09-03 18:48:10 +03:00
sprintf ( query , " %s: Error %d: %s when dumping table %s at row: %ld \n " ,
2000-07-31 21:29:14 +02:00
my_progname ,
mysql_errno ( sock ) ,
mysql_error ( sock ) ,
2003-09-03 18:48:10 +03:00
result_table ,
2000-07-31 21:29:14 +02:00
rownr ) ;
fputs ( query , stderr ) ;
safe_exit ( EX_CONSCHECK ) ;
return ;
}
if ( opt_lock )
2001-08-22 01:45:07 +03:00
fputs ( " UNLOCK TABLES; \n " , md_result_file ) ;
2002-04-03 13:36:01 +03:00
if ( opt_disable_keys )
fprintf ( md_result_file , " /*!40000 ALTER TABLE %s ENABLE KEYS */; \n " ,
2003-09-05 06:56:28 +03:00
opt_quoted_table ) ;
2001-10-04 20:43:06 +03:00
if ( opt_autocommit )
fprintf ( md_result_file , " commit; \n " ) ;
2001-10-10 23:44:07 +03:00
mysql_free_result ( res ) ;
2000-07-31 21:29:14 +02:00
}
} /* dumpTable */
static char * getTableName ( int reset )
{
static MYSQL_RES * res = NULL ;
MYSQL_ROW row ;
if ( ! res )
{
if ( ! ( res = mysql_list_tables ( sock , NullS ) ) )
return ( NULL ) ;
}
if ( ( row = mysql_fetch_row ( res ) ) )
return ( ( char * ) row [ 0 ] ) ;
2000-09-07 04:55:17 +03:00
2000-07-31 21:29:14 +02:00
if ( reset )
mysql_data_seek ( res , 0 ) ; /* We want to read again */
else
{
mysql_free_result ( res ) ;
res = NULL ;
}
return ( NULL ) ;
} /* getTableName */
static int dump_all_databases ( )
{
MYSQL_ROW row ;
MYSQL_RES * tableres ;
int result = 0 ;
if ( mysql_query ( sock , " SHOW DATABASES " ) | |
! ( tableres = mysql_store_result ( sock ) ) )
{
my_printf_error ( 0 , " Error: Couldn't execute 'SHOW DATABASES': %s " ,
MYF ( 0 ) , mysql_error ( sock ) ) ;
return 1 ;
}
while ( ( row = mysql_fetch_row ( tableres ) ) )
{
if ( dump_all_tables_in_db ( row [ 0 ] ) )
result = 1 ;
}
return result ;
}
/* dump_all_databases */
static int dump_databases ( char * * db_names )
{
int result = 0 ;
for ( ; * db_names ; db_names + + )
2001-12-20 18:14:55 +00:00
{
2000-07-31 21:29:14 +02:00
if ( dump_all_tables_in_db ( * db_names ) )
result = 1 ;
}
return result ;
} /* dump_databases */
static int init_dumping ( char * database )
{
if ( mysql_select_db ( sock , database ) )
{
DBerror ( sock , " when selecting the database " ) ;
return 1 ; /* If --force */
}
2002-11-18 20:50:42 +04:00
if ( ! path & & ! opt_xml )
2000-07-31 21:29:14 +02:00
{
if ( opt_databases | | opt_alldbs )
{
2003-11-04 09:40:36 +02:00
/*
length of table name * 2 ( if name contain quotas ) , 2 quotas and 0
*/
2003-10-24 17:26:26 -04:00
char quoted_database_buf [ 64 * 2 + 3 ] ;
char * qdatabase = quote_name ( database , quoted_database_buf , opt_quoted ) ;
2003-12-11 17:37:45 +04:00
if ( opt_comments )
2003-12-17 17:35:34 +02:00
fprintf ( md_result_file , " \n -- \n -- Current Database: %s \n -- \n " , qdatabase ) ;
2000-07-31 21:29:14 +02:00
if ( ! opt_create_db )
2002-11-05 15:10:13 +04:00
{
2003-11-04 09:40:36 +02:00
char qbuf [ 256 ] ;
2002-11-05 15:10:13 +04:00
MYSQL_ROW row ;
MYSQL_RES * dbinfo ;
2003-10-14 12:19:54 +02:00
2003-11-04 09:40:36 +02:00
sprintf ( qbuf , " SHOW CREATE DATABASE WITH IF NOT EXISTS %s " ,
2003-10-24 17:26:26 -04:00
qdatabase ) ;
2003-10-14 12:19:54 +02:00
2002-11-05 15:10:13 +04:00
if ( mysql_query ( sock , qbuf ) | | ! ( dbinfo = mysql_store_result ( sock ) ) )
{
/* Old server version, dump generic CREATE DATABASE */
2003-03-30 19:36:14 +03:00
fprintf ( md_result_file ,
2003-11-04 09:40:36 +02:00
" \n CREATE DATABASE /*!32312 IF NOT EXISTS*/ %s; \n " ,
qdatabase ) ;
2002-11-05 15:10:13 +04:00
}
else
{
row = mysql_fetch_row ( dbinfo ) ;
if ( row [ 1 ] )
{
fprintf ( md_result_file , " \n %s; \n " , row [ 1 ] ) ;
}
}
}
2003-11-04 09:40:36 +02:00
fprintf ( md_result_file , " \n USE %s; \n " , qdatabase ) ;
2000-07-31 21:29:14 +02:00
}
}
2003-11-04 09:40:36 +02:00
if ( extended_insert & & init_dynamic_string ( & extended_row , " " , 1024 , 1024 ) )
exit ( EX_EOM ) ;
2000-07-31 21:29:14 +02:00
return 0 ;
} /* init_dumping */
2003-11-04 09:40:36 +02:00
2000-07-31 21:29:14 +02:00
static int dump_all_tables_in_db ( char * database )
{
char * table ;
uint numrows ;
2003-09-03 18:48:10 +03:00
char table_buff [ NAME_LEN * 2 + 3 ] ;
2000-07-31 21:29:14 +02:00
if ( init_dumping ( database ) )
return 1 ;
2002-11-18 20:50:42 +04:00
if ( opt_xml )
2003-11-03 16:49:39 +04:00
print_xml_tag1 ( md_result_file , " " , " database name= " , database , " \n " ) ;
2000-07-31 21:29:14 +02:00
if ( lock_tables )
{
DYNAMIC_STRING query ;
init_dynamic_string ( & query , " LOCK TABLES " , 256 , 1024 ) ;
for ( numrows = 0 ; ( table = getTableName ( 1 ) ) ; numrows + + )
{
2003-09-03 18:48:10 +03:00
dynstr_append ( & query , quote_name ( table , table_buff , 1 ) ) ;
2000-07-31 21:29:14 +02:00
dynstr_append ( & query , " READ /*!32311 LOCAL */, " ) ;
}
if ( numrows & & mysql_real_query ( sock , query . str , query . length - 1 ) )
2000-09-07 04:55:17 +03:00
DBerror ( sock , " when using LOCK TABLES " ) ;
2000-07-31 21:29:14 +02:00
/* We shall continue here, if --force was given */
dynstr_free ( & query ) ;
}
if ( flush_logs )
{
if ( mysql_refresh ( sock , REFRESH_LOG ) )
2000-09-07 04:55:17 +03:00
DBerror ( sock , " when doing refresh " ) ;
2000-07-31 21:29:14 +02:00
/* We shall continue here, if --force was given */
}
while ( ( table = getTableName ( 0 ) ) )
{
numrows = getTableStructure ( table , database ) ;
if ( ! dFlag & & numrows > 0 )
dumpTable ( numrows , table ) ;
}
2002-11-18 20:50:42 +04:00
if ( opt_xml )
2003-10-30 14:58:30 +04:00
fputs ( " </database> \n " , md_result_file ) ;
2000-07-31 21:29:14 +02:00
if ( lock_tables )
mysql_query ( sock , " UNLOCK_TABLES " ) ;
return 0 ;
} /* dump_all_tables_in_db */
static int dump_selected_tables ( char * db , char * * table_names , int tables )
{
uint numrows ;
2003-09-03 18:48:10 +03:00
char table_buff [ NAME_LEN * + 3 ] ;
2000-07-31 21:29:14 +02:00
if ( init_dumping ( db ) )
return 1 ;
if ( lock_tables )
{
DYNAMIC_STRING query ;
int i ;
init_dynamic_string ( & query , " LOCK TABLES " , 256 , 1024 ) ;
for ( i = 0 ; i < tables ; i + + )
{
2003-09-03 18:48:10 +03:00
dynstr_append ( & query , quote_name ( table_names [ i ] , table_buff , 1 ) ) ;
2000-07-31 21:29:14 +02:00
dynstr_append ( & query , " READ /*!32311 LOCAL */, " ) ;
}
if ( mysql_real_query ( sock , query . str , query . length - 1 ) )
2000-09-07 04:55:17 +03:00
DBerror ( sock , " when doing LOCK TABLES " ) ;
2000-07-31 21:29:14 +02:00
/* We shall countinue here, if --force was given */
dynstr_free ( & query ) ;
}
if ( flush_logs )
{
if ( mysql_refresh ( sock , REFRESH_LOG ) )
2000-09-07 04:55:17 +03:00
DBerror ( sock , " when doing refresh " ) ;
2000-07-31 21:29:14 +02:00
/* We shall countinue here, if --force was given */
}
2002-11-18 20:50:42 +04:00
if ( opt_xml )
2003-11-03 16:49:39 +04:00
print_xml_tag1 ( md_result_file , " " , " database name= " , db , " \n " ) ;
2000-07-31 21:29:14 +02:00
for ( ; tables > 0 ; tables - - , table_names + + )
{
numrows = getTableStructure ( * table_names , db ) ;
if ( ! dFlag & & numrows > 0 )
dumpTable ( numrows , * table_names ) ;
}
2002-11-18 20:50:42 +04:00
if ( opt_xml )
2003-10-30 14:58:30 +04:00
fputs ( " </database> \n " , md_result_file ) ;
2000-07-31 21:29:14 +02:00
if ( lock_tables )
mysql_query ( sock , " UNLOCK_TABLES " ) ;
return 0 ;
} /* dump_selected_tables */
2003-01-16 05:35:59 +02:00
static ulong find_set ( TYPELIB * lib , const char * x , uint length ,
char * * err_pos , uint * err_len )
{
const char * end = x + length ;
ulong found = 0 ;
uint find ;
char buff [ 255 ] ;
2003-03-06 13:51:37 +01:00
* err_pos = 0 ; /* No error yet */
2003-03-16 11:20:45 +04:00
while ( end > x & & my_isspace ( charset_info , end [ - 1 ] ) )
2003-01-16 05:35:59 +02:00
end - - ;
* err_len = 0 ;
if ( x ! = end )
{
const char * start = x ;
for ( ; ; )
{
const char * pos = start ;
uint var_len ;
for ( ; pos ! = end & & * pos ! = ' , ' ; pos + + ) ;
var_len = ( uint ) ( pos - start ) ;
strmake ( buff , start , min ( sizeof ( buff ) , var_len ) ) ;
find = find_type ( buff , lib , var_len ) ;
if ( ! find )
{
* err_pos = ( char * ) start ;
* err_len = var_len ;
}
else
found | = ( ( longlong ) 1 < < ( find - 1 ) ) ;
if ( pos = = end )
break ;
start = pos + 1 ;
}
}
return found ;
}
2000-07-31 21:29:14 +02:00
/* Print a value with a prefix on file */
static void print_value ( FILE * file , MYSQL_RES * result , MYSQL_ROW row ,
const char * prefix , const char * name ,
int string_value )
{
MYSQL_FIELD * field ;
mysql_field_seek ( result , 0 ) ;
for ( ; ( field = mysql_fetch_field ( result ) ) ; row + + )
{
if ( ! strcmp ( field - > name , name ) )
{
if ( row [ 0 ] & & row [ 0 ] [ 0 ] & & strcmp ( row [ 0 ] , " 0 " ) ) /* Skip default */
{
fputc ( ' ' , file ) ;
fputs ( prefix , file ) ;
if ( string_value )
2000-08-22 00:18:32 +03:00
unescape ( file , row [ 0 ] , ( uint ) strlen ( row [ 0 ] ) ) ;
2000-07-31 21:29:14 +02:00
else
fputs ( row [ 0 ] , file ) ;
return ;
}
}
}
return ; /* This shouldn't happen */
} /* print_value */
2003-12-14 06:39:52 +02:00
/*
Check if we the table is one of the table types that should be ignored :
MRG_ISAM , MRG_MYISAM
SYNOPSIS
check_if_ignore_table ( )
table_name Table name to check
GLOBAL VARIABLES
sock MySQL socket
verbose Write warning messages
RETURN
0 Table should be backuped
# Type of table (that should be skipped)
*/
static const char * check_if_ignore_table ( const char * table_name )
{
char buff [ FN_REFLEN + 80 ] , show_name_buff [ FN_REFLEN ] ;
MYSQL_RES * res ;
MYSQL_ROW row ;
const char * result = 0 ;
sprintf ( buff , " show table status like %s " ,
quote_for_like ( table_name , show_name_buff ) ) ;
if ( mysql_query ( sock , buff ) )
{
if ( mysql_errno ( sock ) ! = ER_PARSE_ERROR )
{ /* If old MySQL version */
if ( verbose )
fprintf ( stderr ,
" -- Warning: Couldn't get status information for table %s (%s) \n " ,
table_name , mysql_error ( sock ) ) ;
return 0 ; /* assume table is ok */
}
}
if ( ! ( res = mysql_store_result ( sock ) ) | |
! ( row = mysql_fetch_row ( res ) ) )
{
fprintf ( stderr ,
" Error: Couldn't read status information for table %s (%s) \n " ,
table_name , mysql_error ( sock ) ) ;
if ( res )
mysql_free_result ( res ) ;
return 0 ; /* assume table is ok */
}
if ( strcmp ( row [ 1 ] , ( result = " MRG_MyISAM " ) ) & &
strcmp ( row [ 1 ] , ( result = " MRG_ISAM " ) ) )
result = 0 ;
mysql_free_result ( res ) ;
return result ;
}
2000-07-31 21:29:14 +02:00
int main ( int argc , char * * argv )
{
2001-10-04 20:43:06 +03:00
MYSQL_ROW row ;
MYSQL_RES * master ;
2000-07-31 21:29:14 +02:00
MY_INIT ( argv [ 0 ] ) ;
if ( get_options ( & argc , & argv ) )
{
my_end ( 0 ) ;
exit ( EX_USAGE ) ;
}
2000-11-13 23:55:10 +02:00
if ( dbConnect ( current_host , current_user , opt_password ) )
2000-07-31 21:29:14 +02:00
exit ( EX_MYSQLERR ) ;
if ( ! path )
2002-01-02 21:29:41 +02:00
write_header ( md_result_file , * argv ) ;
if ( opt_first_slave )
{
lock_tables = 0 ; /* No other locks needed */
if ( mysql_query ( sock , " FLUSH TABLES WITH READ LOCK " ) )
{
my_printf_error ( 0 , " Error: Couldn't execute 'FLUSH TABLES WITH READ LOCK': %s " ,
MYF ( 0 ) , mysql_error ( sock ) ) ;
my_end ( 0 ) ;
return ( first_error ) ;
}
}
2002-06-11 13:37:48 +04:00
else if ( opt_single_transaction )
2002-06-14 15:14:30 +04:00
{
/* There is no sense to start transaction if all tables are locked */
if ( mysql_query ( sock , " BEGIN " ) )
2002-06-11 13:37:48 +04:00
{
2002-06-14 15:14:30 +04:00
my_printf_error ( 0 , " Error: Couldn't execute 'BEGIN': %s " ,
2002-06-11 13:37:48 +04:00
MYF ( 0 ) , mysql_error ( sock ) ) ;
2002-06-14 15:14:30 +04:00
my_end ( 0 ) ;
return ( first_error ) ;
}
}
2000-07-31 21:29:14 +02:00
if ( opt_alldbs )
dump_all_databases ( ) ;
else if ( argc > 1 & & ! opt_databases )
2002-06-11 11:20:31 +03:00
{
/* Only one database and selected table(s) */
2000-07-31 21:29:14 +02:00
dump_selected_tables ( * argv , ( argv + 1 ) , ( argc - 1 ) ) ;
2002-06-11 11:20:31 +03:00
}
2000-09-07 04:55:17 +03:00
else
2002-06-11 11:20:31 +03:00
{
/* One or more databases, all tables */
2000-07-31 21:29:14 +02:00
dump_databases ( argv ) ;
2002-06-11 11:20:31 +03:00
}
2000-07-31 21:29:14 +02:00
if ( opt_first_slave )
{
2003-03-19 15:16:51 +01:00
if ( opt_delete_master_logs & & mysql_query ( sock , " FLUSH MASTER " ) )
{
my_printf_error ( 0 , " Error: Couldn't execute 'FLUSH MASTER': %s " ,
MYF ( 0 ) , mysql_error ( sock ) ) ;
}
2001-10-04 20:43:06 +03:00
if ( opt_master_data )
{
if ( mysql_query ( sock , " SHOW MASTER STATUS " ) | |
! ( master = mysql_store_result ( sock ) ) )
my_printf_error ( 0 , " Error: Couldn't execute 'SHOW MASTER STATUS': %s " ,
MYF ( 0 ) , mysql_error ( sock ) ) ;
2001-12-20 18:14:55 +00:00
else
2001-10-04 20:43:06 +03:00
{
row = mysql_fetch_row ( master ) ;
2003-05-08 00:12:46 +03:00
if ( row & & row [ 0 ] & & row [ 1 ] )
2002-06-11 11:20:31 +03:00
{
2003-12-11 17:37:45 +04:00
if ( opt_comments )
fprintf ( md_result_file ,
" \n -- \n -- Position to start replication from \n -- \n \n " ) ;
2001-10-04 20:43:06 +03:00
fprintf ( md_result_file ,
2 minor edits, plus
fix for BUG#1113 "INSERT into non-trans table SELECT ; ROLLBACK" does not send warning"
and
fix for BUG#873 "In transaction, INSERT to non-trans table is written too early to binlog".
Now we don't always write the non-trans update immediately to the binlog;
if there is something in the binlog cache we write it to the binlog cache
(because the non-trans update could depend on a trans table which was modified
earlier in the transaction); then in case of ROLLBACK, we write the binlog
cache to the binlog, wrapped with BEGIN/ROLLBACK.
This guarantees that the slave does the same updates.
For ROLLBACK TO SAVEPOINT: when we execute a SAVEPOINT command we write it
to the binlog cache. At ROLLBACK TO SAVEPOINT, if some non-trans table was updated,
we write ROLLBACK TO SAVEPOINT to the binlog cache; when the transaction
terminates (COMMIT/ROLLBACK), the binlog cache will be flushed to the binlog
(because of the non-trans update) so we'll have SAVEPOINT and ROLLBACK TO
SAVEPOINT in the binlog.
Apart from this rare case of updates of mixed table types in transaction, the
usual way is still clear the binlog cache at ROLLBACK, or chop it at
ROLLBACK TO SAVEPOINT (meaning the SAVEPOINT command is also chopped, which
is fine).
Note that BUG#873 encompasses subbugs 1) and 2) of BUG#333 "3 binlogging bugs when doing INSERT with mixed InnoDB/MyISAM".
2003-08-22 15:39:24 +02:00
" CHANGE MASTER TO MASTER_LOG_FILE='%s', \
MASTER_LOG_POS = % s ; \ n " ,row[0],row[1]);
2001-10-04 20:43:06 +03:00
}
mysql_free_result ( master ) ;
}
}
2000-07-31 21:29:14 +02:00
if ( mysql_query ( sock , " UNLOCK TABLES " ) )
my_printf_error ( 0 , " Error: Couldn't execute 'UNLOCK TABLES': %s " ,
MYF ( 0 ) , mysql_error ( sock ) ) ;
}
2002-06-11 13:37:48 +04:00
else if ( opt_single_transaction ) /* Just to make it beautiful enough */
2002-11-15 00:16:30 +05:00
# ifdef HAVE_SMEM
my_free ( shared_memory_base_name , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
# endif
2002-06-11 13:37:48 +04:00
{
2002-06-14 15:14:30 +04:00
/*
2002-06-19 00:22:30 +03:00
In case we were locking all tables , we did not start transaction
so there is no need to commit it .
2002-06-14 15:14:30 +04:00
*/
2002-06-11 13:37:48 +04:00
/* This should just free locks as we did not change anything */
if ( mysql_query ( sock , " COMMIT " ) )
{
my_printf_error ( 0 , " Error: Couldn't execute 'COMMIT': %s " ,
2003-10-14 12:19:54 +02:00
MYF ( 0 ) , mysql_error ( sock ) ) ;
}
2002-06-11 13:37:48 +04:00
}
2000-07-31 21:29:14 +02:00
dbDisconnect ( current_host ) ;
2002-11-18 20:50:42 +04:00
write_footer ( md_result_file ) ;
2001-08-22 01:45:07 +03:00
if ( md_result_file ! = stdout )
my_fclose ( md_result_file , MYF ( 0 ) ) ;
2000-11-13 23:55:10 +02:00
my_free ( opt_password , MYF ( MY_ALLOW_ZERO_PTR ) ) ;
2003-10-14 12:19:54 +02:00
if ( extended_insert )
2000-07-31 21:29:14 +02:00
dynstr_free ( & extended_row ) ;
my_end ( 0 ) ;
return ( first_error ) ;
} /* main */