Commit 4fed1772 authored by unknown's avatar unknown

Merge neptunus.(none):/home/msvensson/mysql/mysql-5.0

into  neptunus.(none):/home/msvensson/mysql/bug11316/my50-bug11316


mysql-test/mysql-test-run.pl:
  Auto merged
mysql-test/mysql-test-run.sh:
  Auto merged
mysql-test/r/mysqltest.result:
  Auto merged
mysql-test/r/type_newdecimal.result:
  Auto merged
mysql-test/t/create.test:
  Auto merged
mysql-test/t/create_select_tmp.test:
  Auto merged
mysql-test/t/drop.test:
  Auto merged
mysql-test/t/flush.test:
  Auto merged
mysql-test/t/handler.test:
  Auto merged
mysql-test/t/innodb-deadlock.test:
  Auto merged
mysql-test/t/innodb-lock.test:
  Auto merged
mysql-test/t/innodb.test:
  Auto merged
mysql-test/t/kill.test:
  Auto merged
mysql-test/t/ndb_autodiscover2.test:
  Auto merged
mysql-test/t/rpl000001.test:
  Auto merged
mysql-test/t/rpl_EE_error.test:
  Auto merged
mysql-test/t/rpl_change_master.test:
  Auto merged
mysql-test/t/rpl_deadlock.test:
  Auto merged
mysql-test/t/rpl_drop.test:
  Auto merged
mysql-test/t/rpl_drop_temp.test:
  Auto merged
mysql-test/t/rpl_error_ignored_table.test:
  Auto merged
mysql-test/t/rpl_flush_log_loop.test:
  Auto merged
mysql-test/t/rpl_insert_id.test:
  Auto merged
mysql-test/t/rpl_loaddata.test:
  Auto merged
mysql-test/t/rpl_rotate_logs.test:
  Auto merged
mysql-test/t/rpl_sp.test:
  Auto merged
mysql-test/t/rpl_until.test:
  Auto merged
mysql-test/t/strict.test:
  Auto merged
mysql-test/t/type_newdecimal.test:
  Auto merged
client/mysqltest.c:
  Merge with --parsing-disabled
mysql-test/t/mysqltest.test:
  Manual merge
parents f5db9c9a e2965fb2
...@@ -105,12 +105,11 @@ enum {OPT_MANAGER_USER=256,OPT_MANAGER_HOST,OPT_MANAGER_PASSWD, ...@@ -105,12 +105,11 @@ enum {OPT_MANAGER_USER=256,OPT_MANAGER_HOST,OPT_MANAGER_PASSWD,
/* ************************************************************************ */ /* ************************************************************************ */
/* /*
A line that starts with !$ or $S, and the list of error codes to The list of error codes to --error are stored in an internal array of
--error are stored in an internal array of structs. This struct can structs. This struct can hold numeric SQL error codes or SQLSTATE codes
hold numeric SQL error codes or SQLSTATE codes as strings. The as strings. The element next to the last active element in the list is
element next to the last active element in the list is set to type set to type ERR_EMPTY. When an SQL statement return an error we use
ERR_EMPTY. When an SQL statement return an error we use this list to this list to check if this is an expected error.
check if this is an expected error.
*/ */
enum match_err_type enum match_err_type
...@@ -148,13 +147,13 @@ static uint global_expected_errors; ...@@ -148,13 +147,13 @@ static uint global_expected_errors;
static int record = 0, opt_sleep=0; static int record = 0, opt_sleep=0;
static char *db = 0, *pass=0; static char *db = 0, *pass=0;
const char* user = 0, *host = 0, *unix_sock = 0, *opt_basedir="./"; const char *user = 0, *host = 0, *unix_sock = 0, *opt_basedir="./";
static int port = 0; static int port = 0;
static my_bool opt_big_test= 0, opt_compress= 0, silent= 0, verbose = 0; static my_bool opt_big_test= 0, opt_compress= 0, silent= 0, verbose = 0;
static my_bool tty_password= 0, ps_protocol= 0, ps_protocol_enabled= 0; static my_bool tty_password= 0, ps_protocol= 0, ps_protocol_enabled= 0;
static int parsing_disabled= 0; static int parsing_disabled= 0;
static uint start_lineno, *lineno; static uint start_lineno, *lineno;
const char* manager_user="root",*manager_host=0; const char *manager_user="root",*manager_host=0;
char *manager_pass=0; char *manager_pass=0;
int manager_port=MYSQL_MANAGER_PORT; int manager_port=MYSQL_MANAGER_PORT;
int manager_wait_timeout=3; int manager_wait_timeout=3;
...@@ -164,9 +163,16 @@ static char **default_argv; ...@@ -164,9 +163,16 @@ static char **default_argv;
static const char *load_default_groups[]= { "mysqltest","client",0 }; static const char *load_default_groups[]= { "mysqltest","client",0 };
static char line_buffer[MAX_DELIMITER], *line_buffer_pos= line_buffer; static char line_buffer[MAX_DELIMITER], *line_buffer_pos= line_buffer;
static FILE* file_stack[MAX_INCLUDE_DEPTH]; typedef struct
static FILE** cur_file; {
static FILE** file_stack_end; FILE* file;
const char *file_name;
} test_file;
static test_file file_stack[MAX_INCLUDE_DEPTH];
static test_file* cur_file;
static test_file* file_stack_end;
static uint lineno_stack[MAX_INCLUDE_DEPTH]; static uint lineno_stack[MAX_INCLUDE_DEPTH];
static char TMPDIR[FN_REFLEN]; static char TMPDIR[FN_REFLEN];
static char delimiter[MAX_DELIMITER]= DEFAULT_DELIMITER; static char delimiter[MAX_DELIMITER]= DEFAULT_DELIMITER;
...@@ -320,7 +326,7 @@ Q_COMMENT_WITH_COMMAND ...@@ -320,7 +326,7 @@ Q_COMMENT_WITH_COMMAND
/* this should really be called command */ /* this should really be called command */
struct st_query struct st_query
{ {
char *query, *query_buf,*first_argument,*end; char *query, *query_buf,*first_argument,*last_argument,*end;
int first_word_len; int first_word_len;
my_bool abort_on_error, require_file; my_bool abort_on_error, require_file;
match_err expected_errno[MAX_EXPECTED_ERRORS]; match_err expected_errno[MAX_EXPECTED_ERRORS];
...@@ -336,7 +342,7 @@ const char *command_names[]= ...@@ -336,7 +342,7 @@ const char *command_names[]=
"connect", "connect",
/* the difference between sleep and real_sleep is that sleep will use /* the difference between sleep and real_sleep is that sleep will use
the delay from command line (--sleep) if there is one. the delay from command line (--sleep) if there is one.
real_sleep always uses delay from it's argument. real_sleep always uses delay from mysqltest's command line argument.
the logic is that sometimes delays are cpu-dependent (and --sleep the logic is that sometimes delays are cpu-dependent (and --sleep
can be used to set this delay. real_sleep is used for cpu-independent can be used to set this delay. real_sleep is used for cpu-independent
delays delays
...@@ -423,7 +429,7 @@ static void var_free(void* v); ...@@ -423,7 +429,7 @@ static void var_free(void* v);
int dyn_string_cmp(DYNAMIC_STRING* ds, const char *fname); int dyn_string_cmp(DYNAMIC_STRING* ds, const char *fname);
void reject_dump(const char *record_file, char *buf, int size); void reject_dump(const char *record_file, char *buf, int size);
int close_connection(struct st_query* q); int close_connection(struct st_query*);
static void set_charset(struct st_query*); static void set_charset(struct st_query*);
VAR* var_get(const char *var_name, const char** var_name_end, my_bool raw, VAR* var_get(const char *var_name, const char** var_name_end, my_bool raw,
my_bool ignore_not_existing); my_bool ignore_not_existing);
...@@ -449,9 +455,9 @@ static int insert_pointer_name(reg1 POINTER_ARRAY *pa,my_string name); ...@@ -449,9 +455,9 @@ static int insert_pointer_name(reg1 POINTER_ARRAY *pa,my_string name);
void free_pointer_array(POINTER_ARRAY *pa); void free_pointer_array(POINTER_ARRAY *pa);
static int initialize_replace_buffer(void); static int initialize_replace_buffer(void);
static void free_replace_buffer(void); static void free_replace_buffer(void);
static void do_eval(DYNAMIC_STRING* query_eval, const char *query); static void do_eval(DYNAMIC_STRING *query_eval, const char *query);
void str_to_file(const char *fname, char *str, int size); void str_to_file(const char *fname, char *str, int size);
int do_server_op(struct st_query* q,const char *op); int do_server_op(struct st_query *q,const char *op);
struct st_replace *glob_replace; struct st_replace *glob_replace;
static char *out_buff; static char *out_buff;
...@@ -479,9 +485,9 @@ static int normal_handle_error(const char *query, struct st_query *q, ...@@ -479,9 +485,9 @@ static int normal_handle_error(const char *query, struct st_query *q,
MYSQL *mysql, DYNAMIC_STRING *ds); MYSQL *mysql, DYNAMIC_STRING *ds);
static int normal_handle_no_error(struct st_query *q); static int normal_handle_no_error(struct st_query *q);
static void do_eval(DYNAMIC_STRING* query_eval, const char* query) static void do_eval(DYNAMIC_STRING* query_eval, const char *query)
{ {
const char* p; const char *p;
register char c; register char c;
register int escaped = 0; register int escaped = 0;
VAR* v; VAR* v;
...@@ -538,10 +544,13 @@ static void close_cons() ...@@ -538,10 +544,13 @@ static void close_cons()
static void close_files() static void close_files()
{ {
DBUG_ENTER("close_files"); DBUG_ENTER("close_files");
for (; cur_file != file_stack ; cur_file--) for (; cur_file != (file_stack-1) ; cur_file--)
{ {
if (*cur_file != stdin && *cur_file) DBUG_PRINT("info", ("file_name: %s", cur_file->file_name));
my_fclose(*cur_file,MYF(0)); if (cur_file->file && cur_file->file != stdin)
my_fclose(cur_file->file, MYF(0));
my_free((gptr)cur_file->file_name, MYF(MY_ALLOW_ZERO_PTR));
cur_file->file_name= 0;
} }
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
} }
...@@ -584,14 +593,18 @@ static void free_used_memory() ...@@ -584,14 +593,18 @@ static void free_used_memory()
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
} }
static void die(const char* fmt, ...) static void die(const char *fmt, ...)
{ {
va_list args; va_list args;
DBUG_ENTER("die"); DBUG_ENTER("die");
va_start(args, fmt); va_start(args, fmt);
if (fmt) if (fmt)
{ {
fprintf(stderr, "%s: ", my_progname); fprintf(stderr, "mysqltest: ");
if (cur_file && cur_file != file_stack)
fprintf(stderr, "In included file \"%s\": ",
cur_file->file_name);
fprintf(stderr, "At line %u: ", start_lineno);
vfprintf(stderr, fmt, args); vfprintf(stderr, fmt, args);
fprintf(stderr, "\n"); fprintf(stderr, "\n");
fflush(stderr); fflush(stderr);
...@@ -615,7 +628,7 @@ static void abort_not_supported_test() ...@@ -615,7 +628,7 @@ static void abort_not_supported_test()
exit(62); exit(62);
} }
static void verbose_msg(const char* fmt, ...) static void verbose_msg(const char *fmt, ...)
{ {
va_list args; va_list args;
DBUG_ENTER("verbose_msg"); DBUG_ENTER("verbose_msg");
...@@ -624,7 +637,7 @@ static void verbose_msg(const char* fmt, ...) ...@@ -624,7 +637,7 @@ static void verbose_msg(const char* fmt, ...)
va_start(args, fmt); va_start(args, fmt);
fprintf(stderr, "%s: At line %u: ", my_progname, start_lineno); fprintf(stderr, "mysqltest: At line %u: ", start_lineno);
vfprintf(stderr, fmt, args); vfprintf(stderr, fmt, args);
fprintf(stderr, "\n"); fprintf(stderr, "\n");
va_end(args); va_end(args);
...@@ -634,12 +647,12 @@ static void verbose_msg(const char* fmt, ...) ...@@ -634,12 +647,12 @@ static void verbose_msg(const char* fmt, ...)
void init_parser() void init_parser()
{ {
parser.current_line = parser.read_lines = 0; parser.current_line= parser.read_lines= 0;
memset(&var_reg,0, sizeof(var_reg)); memset(&var_reg, 0, sizeof(var_reg));
} }
int dyn_string_cmp(DYNAMIC_STRING* ds, const char* fname) int dyn_string_cmp(DYNAMIC_STRING* ds, const char *fname)
{ {
MY_STAT stat_info; MY_STAT stat_info;
char *tmp, *res_ptr; char *tmp, *res_ptr;
...@@ -706,7 +719,7 @@ int dyn_string_cmp(DYNAMIC_STRING* ds, const char* fname) ...@@ -706,7 +719,7 @@ int dyn_string_cmp(DYNAMIC_STRING* ds, const char* fname)
DBUG_RETURN(res); DBUG_RETURN(res);
} }
static int check_result(DYNAMIC_STRING* ds, const char* fname, static int check_result(DYNAMIC_STRING* ds, const char *fname,
my_bool require_option) my_bool require_option)
{ {
int error = 0; int error = 0;
...@@ -735,7 +748,7 @@ static int check_result(DYNAMIC_STRING* ds, const char* fname, ...@@ -735,7 +748,7 @@ static int check_result(DYNAMIC_STRING* ds, const char* fname,
} }
VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw, VAR* var_get(const char *var_name, const char** var_name_end, my_bool raw,
my_bool ignore_not_existing) my_bool ignore_not_existing)
{ {
int digit; int digit;
...@@ -748,7 +761,7 @@ VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw, ...@@ -748,7 +761,7 @@ VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw,
digit = *++var_name - '0'; digit = *++var_name - '0';
if (digit < 0 || digit >= 10) if (digit < 0 || digit >= 10)
{ {
const char* save_var_name = var_name, *end; const char *save_var_name = var_name, *end;
uint length; uint length;
end = (var_name_end) ? *var_name_end : 0; end = (var_name_end) ? *var_name_end : 0;
while (my_isvar(charset_info,*var_name) && var_name != end) while (my_isvar(charset_info,*var_name) && var_name != end)
...@@ -790,7 +803,7 @@ VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw, ...@@ -790,7 +803,7 @@ VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw,
DBUG_RETURN(0); DBUG_RETURN(0);
} }
static VAR *var_obtain(const char* name, int len) static VAR *var_obtain(const char *name, int len)
{ {
VAR* v; VAR* v;
if ((v = (VAR*)hash_search(&var_hash, name, len))) if ((v = (VAR*)hash_search(&var_hash, name, len)))
...@@ -827,9 +840,11 @@ int var_set(const char *var_name, const char *var_name_end, ...@@ -827,9 +840,11 @@ int var_set(const char *var_name, const char *var_name_end,
} }
int open_file(const char* name) int open_file(const char *name)
{ {
char buff[FN_REFLEN]; char buff[FN_REFLEN];
DBUG_ENTER("open_file");
DBUG_PRINT("enter", ("name: %s", name));
if (!test_if_hard_path(name)) if (!test_if_hard_path(name))
{ {
strxmov(buff, opt_basedir, name, NullS); strxmov(buff, opt_basedir, name, NullS);
...@@ -837,19 +852,51 @@ int open_file(const char* name) ...@@ -837,19 +852,51 @@ int open_file(const char* name)
} }
fn_format(buff,name,"","",4); fn_format(buff,name,"","",4);
if (*cur_file && cur_file == file_stack_end) if (cur_file == file_stack_end)
die("Source directives are nesting too deep"); die("Source directives are nesting too deep");
if (!(*(cur_file+1) = my_fopen(buff, O_RDONLY | FILE_BINARY, MYF(MY_WME))))
die(NullS);
cur_file++; cur_file++;
if (!(cur_file->file = my_fopen(buff, O_RDONLY | FILE_BINARY, MYF(0))))
{
cur_file--;
die("Could not open file %s", buff);
}
cur_file->file_name= my_strdup(buff, MYF(MY_FAE));
*++lineno=1; *++lineno=1;
DBUG_RETURN(0);
}
return 0;
/*
Check for unexpected "junk" after the end of query
This is normally caused by missing delimiters
*/
int check_eol_junk(const char *eol)
{
const char *p= eol;
DBUG_ENTER("check_eol_junk");
DBUG_PRINT("enter", ("eol: %s", eol));
/* Remove all spacing chars except new line */
while (*p && my_isspace(charset_info, *p) && (*p != '\n'))
p++;
/* Check for extra delimiter */
if (*p && !strncmp(p, delimiter, delimiter_length))
die("Extra delimiter \"%s\" found", delimiter);
/* Allow trailing # comment */
if (*p && *p != '#')
{
if (*p == '\n')
die("Missing delimiter");
die("End of line junk detected: \"%s\"", p);
}
DBUG_RETURN(0);
} }
/* ugly long name, but we are following the convention */ /* ugly long name, but we are following the convention */
int do_wait_for_slave_to_stop(struct st_query* q __attribute__((unused))) int do_wait_for_slave_to_stop(struct st_query *q __attribute__((unused)))
{ {
MYSQL* mysql = &cur_con->mysql; MYSQL* mysql = &cur_con->mysql;
for (;;) for (;;)
...@@ -877,7 +924,7 @@ int do_wait_for_slave_to_stop(struct st_query* q __attribute__((unused))) ...@@ -877,7 +924,7 @@ int do_wait_for_slave_to_stop(struct st_query* q __attribute__((unused)))
return 0; return 0;
} }
int do_require_manager(struct st_query* a __attribute__((unused))) int do_require_manager(struct st_query *query __attribute__((unused)) )
{ {
if (!manager) if (!manager)
abort_not_supported_test(); abort_not_supported_test();
...@@ -885,95 +932,121 @@ int do_require_manager(struct st_query* a __attribute__((unused))) ...@@ -885,95 +932,121 @@ int do_require_manager(struct st_query* a __attribute__((unused)))
} }
#ifndef EMBEDDED_LIBRARY #ifndef EMBEDDED_LIBRARY
int do_server_start(struct st_query* q) int do_server_start(struct st_query *q)
{ {
return do_server_op(q,"start"); return do_server_op(q, "start");
} }
int do_server_stop(struct st_query* q) int do_server_stop(struct st_query *q)
{ {
return do_server_op(q,"stop"); return do_server_op(q, "stop");
} }
int do_server_op(struct st_query* q,const char* op) int do_server_op(struct st_query *q, const char *op)
{ {
char* p=q->first_argument; char *p= q->first_argument;
char com_buf[256],*com_p; char com_buf[256], *com_p;
if (!manager) if (!manager)
{ {
die("Manager is not initialized, manager commands are not possible"); die("Manager is not initialized, manager commands are not possible");
} }
com_p=strmov(com_buf,op); com_p= strmov(com_buf,op);
com_p=strmov(com_p,"_exec "); com_p= strmov(com_p,"_exec ");
if (!*p) if (!*p)
die("Missing server name in server_%s\n",op); die("Missing server name in server_%s", op);
while (*p && !my_isspace(charset_info,*p)) while (*p && !my_isspace(charset_info, *p))
*com_p++= *p++; *com_p++= *p++;
*com_p++=' '; *com_p++= ' ';
com_p=int10_to_str(manager_wait_timeout,com_p,10); com_p= int10_to_str(manager_wait_timeout, com_p, 10);
*com_p++ = '\n'; *com_p++= '\n';
*com_p=0; *com_p= 0;
if (mysql_manager_command(manager,com_buf,(int)(com_p-com_buf))) if (mysql_manager_command(manager, com_buf, (int)(com_p-com_buf)))
die("Error in command: %s(%d)",manager->last_error,manager->last_errno); die("Error in command: %s(%d)", manager->last_error, manager->last_errno);
while (!manager->eof) while (!manager->eof)
{ {
if (mysql_manager_fetch_line(manager,com_buf,sizeof(com_buf))) if (mysql_manager_fetch_line(manager, com_buf, sizeof(com_buf)))
die("Error fetching result line: %s(%d)", manager->last_error, die("Error fetching result line: %s(%d)", manager->last_error,
manager->last_errno); manager->last_errno);
} }
q->last_argument= p;
return 0; return 0;
} }
#endif #endif
int do_source(struct st_query* q)
/*
Source and execute the given file
SYNOPSIS
do_source()
query called command
DESCRIPTION
source <file_name>
Open the file <file_name> and execute it
*/
int do_source(struct st_query *query)
{ {
char* p=q->first_argument, *name; char *p= query->first_argument, *name;
if (!*p) if (!*p)
die("Missing file name in source\n"); die("Missing file name in source");
name = p; name= p;
while (*p && !my_isspace(charset_info,*p)) while (*p && !my_isspace(charset_info,*p))
p++; p++;
*p = 0; if (*p)
*p++= 0;
query->last_argument= p;
/*
If this file has already been sourced, dont source it again.
It's already available in the q_lines cache
*/
if (parser.current_line < (parser.read_lines - 1))
return 0;
return open_file(name); return open_file(name);
} }
/* /*
Execute given command. Execute given command.
SYNOPSIS SYNOPSIS
do_exec() do_exec()
q called command query called command
DESCRIPTION DESCRIPTION
If one uses --exec command [args] command in .test file exec <command>
we will execute the command and record its output.
Execute the text between exec and end of line in a subprocess.
The error code returned from the subprocess is checked against the
expected error array, previously set with the --error command.
It can thus be used to execute a command that shall fail.
RETURN VALUES
0 ok
1 error
*/ */
static void do_exec(struct st_query* q) static void do_exec(struct st_query *query)
{ {
int error; int error;
DYNAMIC_STRING *ds= NULL; /* Assign just to avoid warning */ DYNAMIC_STRING *ds= NULL;
DYNAMIC_STRING ds_tmp; DYNAMIC_STRING ds_tmp;
char buf[1024]; char buf[1024];
FILE *res_file; FILE *res_file;
char *cmd= q->first_argument; char *cmd= query->first_argument;
DBUG_ENTER("do_exec"); DBUG_ENTER("do_exec");
while (*cmd && my_isspace(charset_info, *cmd)) while (*cmd && my_isspace(charset_info, *cmd))
cmd++; cmd++;
if (!*cmd) if (!*cmd)
die("Missing argument in exec\n"); die("Missing argument in exec");
query->last_argument= query->end;
DBUG_PRINT("info", ("Executing '%s'", cmd)); DBUG_PRINT("info", ("Executing '%s'", cmd));
if (!(res_file= popen(cmd, "r")) && q->abort_on_error) if (!(res_file= popen(cmd, "r")) && query->abort_on_error)
die("popen() failed\n"); die("popen(\"%s\", \"r\") failed", cmd);
if (disable_result_log) if (disable_result_log)
{ {
...@@ -985,7 +1058,7 @@ static void do_exec(struct st_query* q) ...@@ -985,7 +1058,7 @@ static void do_exec(struct st_query* q)
} }
else else
{ {
if (q->record_file[0]) if (query->record_file[0])
{ {
init_dynamic_string(&ds_tmp, "", 16384, 65536); init_dynamic_string(&ds_tmp, "", 16384, 65536);
ds= &ds_tmp; ds= &ds_tmp;
...@@ -1002,33 +1075,35 @@ static void do_exec(struct st_query* q) ...@@ -1002,33 +1075,35 @@ static void do_exec(struct st_query* q)
uint status= WEXITSTATUS(error), i; uint status= WEXITSTATUS(error), i;
my_bool ok= 0; my_bool ok= 0;
if (q->abort_on_error) if (query->abort_on_error)
die("At line %u: command \"%s\" failed", start_lineno, cmd); die("command \"%s\" failed", cmd);
DBUG_PRINT("info", DBUG_PRINT("info",
("error: %d, status: %d", error, status)); ("error: %d, status: %d", error, status));
for (i=0 ; (uint) i < q->expected_errors ; i++) for (i= 0; i < query->expected_errors; i++)
{ {
DBUG_PRINT("info", DBUG_PRINT("info",
("error: %d, status: %d", error, status)); ("error: %d, status: %d", error, status));
DBUG_PRINT("info", ("expected error: %d", DBUG_PRINT("info", ("expected error: %d",
q->expected_errno[i].code.errnum)); query->expected_errno[i].code.errnum));
if ((q->expected_errno[i].type == ERR_ERRNO) && if ((query->expected_errno[i].type == ERR_ERRNO) &&
(q->expected_errno[i].code.errnum == status)) (query->expected_errno[i].code.errnum == status))
{
ok= 1; ok= 1;
verbose_msg("At line %u: command \"%s\" failed with expected error: %d", verbose_msg("command \"%s\" failed with expected error: %d",
start_lineno, cmd, status); cmd, status);
}
} }
if (!ok) if (!ok)
die("At line: %u: command \"%s\" failed with wrong error: %d", die("command \"%s\" failed with wrong error: %d",
start_lineno, cmd, status); cmd, status);
} }
else if (q->expected_errno[0].type == ERR_ERRNO && else if (query->expected_errno[0].type == ERR_ERRNO &&
q->expected_errno[0].code.errnum != 0) query->expected_errno[0].code.errnum != 0)
{ {
/* Error code we wanted was != 0, i.e. not an expected success */ /* Error code we wanted was != 0, i.e. not an expected success */
die("At line: %u: command \"%s\" succeeded - should have failed with errno %d...", die("command \"%s\" succeeded - should have failed with errno %d...",
start_lineno, cmd, q->expected_errno[0].code.errnum); cmd, query->expected_errno[0].code.errnum);
} }
if (!disable_result_log) if (!disable_result_log)
...@@ -1038,14 +1113,14 @@ static void do_exec(struct st_query* q) ...@@ -1038,14 +1113,14 @@ static void do_exec(struct st_query* q)
if (record) if (record)
{ {
if (!q->record_file[0] && !result_file) if (!query->record_file[0] && !result_file)
die("At line %u: Missing result file", start_lineno); die("Missing result file");
if (!result_file) if (!result_file)
str_to_file(q->record_file, ds->str, ds->length); str_to_file(query->record_file, ds->str, ds->length);
} }
else if (q->record_file[0]) else if (query->record_file[0])
{ {
error= check_result(ds, q->record_file, q->require_file); error= check_result(ds, query->record_file, query->require_file);
} }
if (ds == &ds_tmp) if (ds == &ds_tmp)
dynstr_free(&ds_tmp); dynstr_free(&ds_tmp);
...@@ -1053,7 +1128,7 @@ static void do_exec(struct st_query* q) ...@@ -1053,7 +1128,7 @@ static void do_exec(struct st_query* q)
} }
int var_query_set(VAR* v, const char* p, const char** p_end) int var_query_set(VAR* v, const char *p, const char** p_end)
{ {
char* end = (char*)((p_end && *p_end) ? *p_end : p + strlen(p)); char* end = (char*)((p_end && *p_end) ? *p_end : p + strlen(p));
MYSQL_RES *res; MYSQL_RES *res;
...@@ -1104,19 +1179,27 @@ int var_query_set(VAR* v, const char* p, const char** p_end) ...@@ -1104,19 +1179,27 @@ int var_query_set(VAR* v, const char* p, const char** p_end)
return 0; return 0;
} }
void var_copy(VAR* dest, VAR* src) void var_copy(VAR *dest, VAR *src)
{ {
dest->int_val=src->int_val; dest->int_val= src->int_val;
dest->int_dirty=src->int_dirty; dest->int_dirty= src->int_dirty;
/* Alloc/realloc data for str_val in dest */
if (dest->alloced_len < src->alloced_len && if (dest->alloced_len < src->alloced_len &&
!(dest->str_val=my_realloc(dest->str_val,src->alloced_len+1, !(dest->str_val= dest->str_val
MYF(MY_WME)))) ? my_realloc(dest->str_val, src->alloced_len, MYF(MY_WME))
: my_malloc(src->alloced_len, MYF(MY_WME))))
die("Out of memory"); die("Out of memory");
dest->str_val_len=src->str_val_len; else
memcpy(dest->str_val,src->str_val,src->str_val_len+1); dest->alloced_len= src->alloced_len;
/* Copy str_val data to dest */
dest->str_val_len= src->str_val_len;
if (src->str_val_len)
memcpy(dest->str_val, src->str_val, src->str_val_len);
} }
int eval_expr(VAR* v, const char* p, const char** p_end) int eval_expr(VAR* v, const char *p, const char** p_end)
{ {
VAR* vp; VAR* vp;
if (*p == '$') if (*p == '$')
...@@ -1157,29 +1240,58 @@ int eval_expr(VAR* v, const char* p, const char** p_end) ...@@ -1157,29 +1240,58 @@ int eval_expr(VAR* v, const char* p, const char** p_end)
return 1; return 1;
} }
int do_inc(struct st_query* q)
enum enum_operator
{ {
char* p=q->first_argument; DO_DEC,
VAR* v; DO_INC
v = var_get(p, 0, 1, 0); };
v->int_val++;
v->int_dirty = 1; /*
return 0; Decrease or increase the value of a variable
}
SYNOPSIS
do_modify_var()
query called command
name human readable name of operator
operator operation to perform on the var
DESCRIPTION
dec $var_name
inc $var_name
int do_dec(struct st_query* q) */
int do_modify_var(struct st_query *query, const char *name,
enum enum_operator operator)
{ {
char* p=q->first_argument; const char *p= query->first_argument;
VAR* v; VAR* v;
v = var_get(p, 0, 1, 0); if (!*p)
die("Missing arguments to %s", name);
if (*p != '$')
die("First argument to %s must be a variable (start with $)", name);
v= var_get(p, &p, 1, 0);
switch (operator){
case DO_DEC:
v->int_val--; v->int_val--;
v->int_dirty = 1; break;
case DO_INC:
v->int_val++;
break;
default:
die("Invalid operator to do_operator");
break;
}
v->int_dirty= 1;
query->last_argument= (char*)++p;
return 0; return 0;
} }
int do_system(struct st_query* q)
int do_system(struct st_query *q)
{ {
char* p=q->first_argument; char *p=q->first_argument;
VAR v; VAR v;
var_init(&v, 0, 0, 0, 0); var_init(&v, 0, 0, 0, 0);
eval_expr(&v, p, 0); /* NULL terminated */ eval_expr(&v, p, 0); /* NULL terminated */
...@@ -1191,49 +1303,85 @@ int do_system(struct st_query* q) ...@@ -1191,49 +1303,85 @@ int do_system(struct st_query* q)
memcpy(expr_buf, v.str_val, v.str_val_len); memcpy(expr_buf, v.str_val, v.str_val_len);
expr_buf[v.str_val_len] = 0; expr_buf[v.str_val_len] = 0;
DBUG_PRINT("info", ("running system command '%s'", expr_buf)); DBUG_PRINT("info", ("running system command '%s'", expr_buf));
if (system(expr_buf) && q->abort_on_error) if (system(expr_buf))
{
if (q->abort_on_error)
die("system command '%s' failed", expr_buf); die("system command '%s' failed", expr_buf);
/* If ! abort_on_error, display message and continue */
verbose_msg("system command '%s' failed", expr_buf);
} }
}
else
die("Missing arguments to system, nothing to do!");
var_free(&v); var_free(&v);
q->last_argument= q->end;
return 0; return 0;
} }
int do_echo(struct st_query* q)
/*
Print the content between echo and <delimiter> to result file.
If content is a variable, the variable value will be retrieved
SYNOPSIS
do_echo()
q called command
DESCRIPTION
Usage 1:
echo text
Print the text after echo until end of command to result file
Usage 2:
echo $<var_name>
Print the content of the variable <var_name> to result file
*/
int do_echo(struct st_query *q)
{ {
char* p=q->first_argument; char *p= q->first_argument;
DYNAMIC_STRING *ds;
DYNAMIC_STRING ds_tmp;
VAR v; VAR v;
var_init(&v,0,0,0,0); var_init(&v,0,0,0,0);
eval_expr(&v, p, 0); /* NULL terminated */
if (v.str_val_len) if (q->record_file[0])
{ {
fflush(stdout); init_dynamic_string(&ds_tmp, "", 256, 512);
write(1, v.str_val, v.str_val_len); ds= &ds_tmp;
} }
write(1, "\n", 1); else
ds= &ds_res;
eval_expr(&v, p, 0); /* NULL terminated */
if (v.str_val_len)
dynstr_append_mem(ds, v.str_val, v.str_val_len);
dynstr_append_mem(ds, "\n", 1);
var_free(&v); var_free(&v);
if (ds == &ds_tmp)
dynstr_free(&ds_tmp);
q->last_argument= q->end;
return 0; return 0;
} }
int do_sync_with_master2(const char* p) int do_sync_with_master2(long offset)
{ {
MYSQL_RES* res; MYSQL_RES* res;
MYSQL_ROW row; MYSQL_ROW row;
MYSQL* mysql = &cur_con->mysql; MYSQL* mysql= &cur_con->mysql;
char query_buf[FN_REFLEN+128]; char query_buf[FN_REFLEN+128];
int offset= 0, tries= 0; int tries= 0;
int rpl_parse; int rpl_parse;
if (!master_pos.file[0]) if (!master_pos.file[0])
{ {
die("Line %u: Calling 'sync_with_master' without calling 'save_master_pos'", start_lineno); die("Line %u: Calling 'sync_with_master' without calling 'save_master_pos'", start_lineno);
} }
rpl_parse = mysql_rpl_parse_enabled(mysql); rpl_parse= mysql_rpl_parse_enabled(mysql);
mysql_disable_rpl_parse(mysql); mysql_disable_rpl_parse(mysql);
if (*p)
offset = atoi(p);
sprintf(query_buf, "select master_pos_wait('%s', %ld)", master_pos.file, sprintf(query_buf, "select master_pos_wait('%s', %ld)", master_pos.file,
master_pos.pos + offset); master_pos.pos + offset);
...@@ -1243,10 +1391,10 @@ int do_sync_with_master2(const char* p) ...@@ -1243,10 +1391,10 @@ int do_sync_with_master2(const char* p)
die("line %u: failed in %s: %d: %s", start_lineno, query_buf, die("line %u: failed in %s: %d: %s", start_lineno, query_buf,
mysql_errno(mysql), mysql_error(mysql)); mysql_errno(mysql), mysql_error(mysql));
if (!(last_result = res = mysql_store_result(mysql))) if (!(last_result= res= mysql_store_result(mysql)))
die("line %u: mysql_store_result() returned NULL for '%s'", start_lineno, die("line %u: mysql_store_result() returned NULL for '%s'", start_lineno,
query_buf); query_buf);
if (!(row = mysql_fetch_row(res))) if (!(row= mysql_fetch_row(res)))
die("line %u: empty result in %s", start_lineno, query_buf); die("line %u: empty result in %s", start_lineno, query_buf);
if (!row[0]) if (!row[0])
{ {
...@@ -1271,9 +1419,21 @@ int do_sync_with_master2(const char* p) ...@@ -1271,9 +1419,21 @@ int do_sync_with_master2(const char* p)
return 0; return 0;
} }
int do_sync_with_master(struct st_query* q) int do_sync_with_master(struct st_query *query)
{ {
return do_sync_with_master2(q->first_argument); long offset= 0;
char *p= query->first_argument;
const char *offset_start= p;
if (*offset_start)
{
for (; my_isdigit(charset_info, *p); p++)
offset = offset * 10 + *p - '0';
if(*p && !my_isspace(charset_info, *p))
die("Invalid integer argument \"%s\"", offset_start);
query->last_argument= p;
}
return do_sync_with_master2(offset);
} }
int do_save_master_pos() int do_save_master_pos()
...@@ -1288,7 +1448,7 @@ int do_save_master_pos() ...@@ -1288,7 +1448,7 @@ int do_save_master_pos()
mysql_disable_rpl_parse(mysql); mysql_disable_rpl_parse(mysql);
if (mysql_query(mysql, query= "show master status")) if (mysql_query(mysql, query= "show master status"))
die("At line %u: failed in show master status: %d: %s", start_lineno, die("failed in show master status: %d: %s",
mysql_errno(mysql), mysql_error(mysql)); mysql_errno(mysql), mysql_error(mysql));
if (!(last_result =res = mysql_store_result(mysql))) if (!(last_result =res = mysql_store_result(mysql)))
...@@ -1307,21 +1467,51 @@ int do_save_master_pos() ...@@ -1307,21 +1467,51 @@ int do_save_master_pos()
} }
int do_let(struct st_query* q) /*
Assign the variable <var_name> with <var_val>
SYNOPSIS
do_let()
query called command
DESCRIPTION
let $<var_name>=<var_val><delimiter>
<var_name> - is the string string found between the $ and =
<var_val> - is the content between the = and <delimiter>, it may span
multiple line and contain any characters except <delimiter>
<delimiter> - is a string containing of one or more chars, default is ;
RETURN VALUES
Program will die if error detected
*/
int do_let(struct st_query *query)
{ {
char* p=q->first_argument; char *p= query->first_argument;
char *var_name, *var_name_end, *var_val_start; char *var_name, *var_name_end, *var_val_start;
/* Find <var_name> */
if (!*p) if (!*p)
die("Missing variable name in let\n"); die("Missing arguments to let");
var_name = p; var_name= p;
while (*p && (*p != '=' || my_isspace(charset_info,*p))) while (*p && (*p != '=') && !my_isspace(charset_info,*p))
p++;
var_name_end= p;
if (var_name+1 == var_name_end)
die("Missing variable name in let");
while (my_isspace(charset_info,*p))
p++; p++;
var_name_end = p; if (*p++ != '=')
if (*p == '=') p++; die("Missing assignment operator in let");
/* Find start of <var_val> */
while (*p && my_isspace(charset_info,*p)) while (*p && my_isspace(charset_info,*p))
p++; p++;
var_val_start = p; var_val_start= p;
return var_set(var_name, var_name_end, var_val_start, q->end); query->last_argument= query->end;
/* Assign var_val to var_name */
return var_set(var_name, var_name_end, var_val_start, query->end);
} }
...@@ -1340,7 +1530,7 @@ int var_set_errno(int sql_errno) ...@@ -1340,7 +1530,7 @@ int var_set_errno(int sql_errno)
} }
int do_rpl_probe(struct st_query* q __attribute__((unused))) int do_rpl_probe(struct st_query *query __attribute__((unused)))
{ {
DBUG_ENTER("do_rpl_probe"); DBUG_ENTER("do_rpl_probe");
if (mysql_rpl_probe(&cur_con->mysql)) if (mysql_rpl_probe(&cur_con->mysql))
...@@ -1349,71 +1539,104 @@ int do_rpl_probe(struct st_query* q __attribute__((unused))) ...@@ -1349,71 +1539,104 @@ int do_rpl_probe(struct st_query* q __attribute__((unused)))
} }
int do_enable_rpl_parse(struct st_query* q __attribute__((unused))) int do_enable_rpl_parse(struct st_query *query __attribute__((unused)))
{ {
mysql_enable_rpl_parse(&cur_con->mysql); mysql_enable_rpl_parse(&cur_con->mysql);
return 0; return 0;
} }
int do_disable_rpl_parse(struct st_query* q __attribute__((unused))) int do_disable_rpl_parse(struct st_query *query __attribute__((unused)))
{ {
mysql_disable_rpl_parse(&cur_con->mysql); mysql_disable_rpl_parse(&cur_con->mysql);
return 0; return 0;
} }
int do_sleep(struct st_query* q, my_bool real_sleep) /*
Sleep the number of specifed seconds
SYNOPSIS
do_sleep()
q called command
real_sleep use the value from opt_sleep as number of seconds to sleep
DESCRIPTION
sleep <seconds>
real_sleep
*/
int do_sleep(struct st_query *query, my_bool real_sleep)
{ {
char *p=q->first_argument; int error= 0;
while (*p && my_isspace(charset_info,*p)) char *p= query->first_argument;
char *sleep_start, *sleep_end= query->end;
double sleep_val;
while (my_isspace(charset_info, *p))
p++; p++;
if (!*p) if (!*p)
die("Missing argument in sleep\n"); die("Missing argument to sleep");
sleep_start= p;
/* Check that arg starts with a digit, not handled by my_strtod */
if (!my_isdigit(charset_info, *sleep_start))
die("Invalid argument to sleep \"%s\"", query->first_argument);
sleep_val= my_strtod(sleep_start, &sleep_end, &error);
if (error)
die("Invalid argument to sleep \"%s\"", query->first_argument);
/* Fixed sleep time selected by --sleep option */
if (opt_sleep && !real_sleep) if (opt_sleep && !real_sleep)
my_sleep(opt_sleep * 1000000L); sleep_val= opt_sleep;
else
my_sleep((ulong) (atof(p) * 1000000L)); my_sleep((ulong) (sleep_val * 1000000L));
query->last_argument= sleep_end;
return 0; return 0;
} }
static void get_file_name(char *filename, struct st_query* q) static void get_file_name(char *filename, struct st_query *q)
{ {
char* p=q->first_argument; char *p= q->first_argument, *name;
strnmov(filename, p, FN_REFLEN); if (!*p)
/* Remove end space */ die("Missing file name argument");
while (p > filename && my_isspace(charset_info,p[-1])) name= p;
p--; while (*p && !my_isspace(charset_info,*p))
p[0]=0; p++;
if (*p)
*p++= 0;
q->last_argument= p;
strmake(filename, name, FN_REFLEN);
} }
static void set_charset(struct st_query* q) static void set_charset(struct st_query *q)
{ {
char* charset_name= q->first_argument; char *charset_name= q->first_argument;
char* tmp; char *p;
if (!charset_name || !*charset_name) if (!charset_name || !*charset_name)
die("Missing charset name in 'character_set'\n"); die("Missing charset name in 'character_set'");
/* Remove end space */ /* Remove end space */
tmp= charset_name; p= charset_name;
while (*tmp && !my_isspace(charset_info,*tmp)) while (*p && !my_isspace(charset_info,*p))
tmp++; p++;
*tmp= 0; if(*p)
*p++= 0;
q->last_argument= p;
charset_info= get_charset_by_csname(charset_name,MY_CS_PRIMARY,MYF(MY_WME)); charset_info= get_charset_by_csname(charset_name,MY_CS_PRIMARY,MYF(MY_WME));
if (!charset_info) if (!charset_info)
abort_not_supported_test(); abort_not_supported_test();
} }
static uint get_errcodes(match_err *to,struct st_query* q) static uint get_errcodes(match_err *to,struct st_query *q)
{ {
char* p= q->first_argument; char *p= q->first_argument;
uint count= 0; uint count= 0;
DBUG_ENTER("get_errcodes"); DBUG_ENTER("get_errcodes");
if (!*p) if (!*p)
die("Missing argument in %s\n", q->query); die("Missing argument in %s", q->query);
do do
{ {
...@@ -1447,20 +1670,20 @@ static uint get_errcodes(match_err *to,struct st_query* q) ...@@ -1447,20 +1670,20 @@ static uint get_errcodes(match_err *to,struct st_query* q)
} }
} }
if (!e->name) if (!e->name)
die("Unknown SQL error '%s'\n", start); die("Unknown SQL error '%s'", start);
} }
else else
{ {
long val; long val;
if (!(p= str2int(p,10,(long) INT_MIN, (long) INT_MAX, &val))) if (!(p= str2int(p,10,(long) INT_MIN, (long) INT_MAX, &val)))
die("Invalid argument in %s\n", q->query); die("Invalid argument in %s", q->query);
to[count].code.errnum= (uint) val; to[count].code.errnum= (uint) val;
to[count].type= ERR_ERRNO; to[count].type= ERR_ERRNO;
} }
count++; count++;
} while (*(p++) == ','); } while (*(p++) == ',');
q->last_argument= (p - 1);
to[count].type= ERR_EMPTY; /* End of data */ to[count].type= ERR_EMPTY; /* End of data */
DBUG_RETURN(count); DBUG_RETURN(count);
} }
...@@ -1474,7 +1697,7 @@ static uint get_errcodes(match_err *to,struct st_query* q) ...@@ -1474,7 +1697,7 @@ static uint get_errcodes(match_err *to,struct st_query* q)
static char *get_string(char **to_ptr, char **from_ptr, static char *get_string(char **to_ptr, char **from_ptr,
struct st_query* q) struct st_query *q)
{ {
reg1 char c,sep; reg1 char c,sep;
char *to= *to_ptr, *from= *from_ptr, *start=to; char *to= *to_ptr, *from= *from_ptr, *start=to;
...@@ -1522,7 +1745,7 @@ static char *get_string(char **to_ptr, char **from_ptr, ...@@ -1522,7 +1745,7 @@ static char *get_string(char **to_ptr, char **from_ptr,
*to++=c; *to++=c;
} }
if (*from != ' ' && *from) if (*from != ' ' && *from)
die("Wrong string argument in %s\n", q->query); die("Wrong string argument in %s", q->query);
while (my_isspace(charset_info,*from)) /* Point to next string */ while (my_isspace(charset_info,*from)) /* Point to next string */
from++; from++;
...@@ -1557,7 +1780,7 @@ static char *get_string(char **to_ptr, char **from_ptr, ...@@ -1557,7 +1780,7 @@ static char *get_string(char **to_ptr, char **from_ptr,
static void get_replace(struct st_query *q) static void get_replace(struct st_query *q)
{ {
uint i; uint i;
char *from=q->first_argument; char *from= q->first_argument;
char *buff,*start; char *buff,*start;
char word_end_chars[256],*pos; char word_end_chars[256],*pos;
POINTER_ARRAY to_array,from_array; POINTER_ARRAY to_array,from_array;
...@@ -1568,14 +1791,14 @@ static void get_replace(struct st_query *q) ...@@ -1568,14 +1791,14 @@ static void get_replace(struct st_query *q)
bzero((char*) &to_array,sizeof(to_array)); bzero((char*) &to_array,sizeof(to_array));
bzero((char*) &from_array,sizeof(from_array)); bzero((char*) &from_array,sizeof(from_array));
if (!*from) if (!*from)
die("Missing argument in %s\n", q->query); die("Missing argument in %s", q->query);
start=buff=my_malloc(strlen(from)+1,MYF(MY_WME | MY_FAE)); start=buff=my_malloc(strlen(from)+1,MYF(MY_WME | MY_FAE));
while (*from) while (*from)
{ {
char *to=buff; char *to=buff;
to=get_string(&buff, &from, q); to=get_string(&buff, &from, q);
if (!*from) if (!*from)
die("Wrong number of arguments to replace in %s\n", q->query); die("Wrong number of arguments to replace_result in '%s'", q->query);
insert_pointer_name(&from_array,to); insert_pointer_name(&from_array,to);
to=get_string(&buff, &from, q); to=get_string(&buff, &from, q);
insert_pointer_name(&to_array,to); insert_pointer_name(&to_array,to);
...@@ -1589,10 +1812,11 @@ static void get_replace(struct st_query *q) ...@@ -1589,10 +1812,11 @@ static void get_replace(struct st_query *q)
(uint) from_array.typelib.count, (uint) from_array.typelib.count,
word_end_chars)) || word_end_chars)) ||
initialize_replace_buffer()) initialize_replace_buffer())
die("Can't initialize replace from %s\n", q->query); die("Can't initialize replace from '%s'", q->query);
free_pointer_array(&from_array); free_pointer_array(&from_array);
free_pointer_array(&to_array); free_pointer_array(&to_array);
my_free(start, MYF(0)); my_free(start, MYF(0));
q->last_argument= q->end;
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
} }
...@@ -1608,25 +1832,18 @@ void free_replace() ...@@ -1608,25 +1832,18 @@ void free_replace()
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
} }
int select_connection(char *p)
int select_connection_name(const char *name)
{ {
char* name;
struct connection *con; struct connection *con;
DBUG_ENTER("select_connection"); DBUG_ENTER("select_connection2");
DBUG_PRINT("enter",("name: '%s'",p)); DBUG_PRINT("enter",("name: '%s'", name));
if (!*p) for (con= cons; con < next_con; con++)
die("Missing connection name in connect\n");
name = p;
while (*p && !my_isspace(charset_info,*p))
p++;
*p = 0;
for (con = cons; con < next_con; con++)
{ {
if (!strcmp(con->name, name)) if (!strcmp(con->name, name))
{ {
cur_con = con; cur_con= con;
DBUG_RETURN(0); DBUG_RETURN(0);
} }
} }
...@@ -1634,21 +1851,42 @@ int select_connection(char *p) ...@@ -1634,21 +1851,42 @@ int select_connection(char *p)
DBUG_RETURN(1); /* Never reached */ DBUG_RETURN(1); /* Never reached */
} }
int close_connection(struct st_query* q)
int select_connection(struct st_query *query)
{
char *name;
char *p= query->first_argument;
DBUG_ENTER("select_connection");
if (!*p)
die("Missing connection name in connect");
name= p;
while (*p && !my_isspace(charset_info,*p))
p++;
if (*p)
*p++= 0;
query->last_argument= p;
return select_connection_name(name);
}
int close_connection(struct st_query *q)
{ {
char* p=q->first_argument, *name; char *p= q->first_argument, *name;
struct connection *con; struct connection *con;
DBUG_ENTER("close_connection"); DBUG_ENTER("close_connection");
DBUG_PRINT("enter",("name: '%s'",p)); DBUG_PRINT("enter",("name: '%s'",p));
if (!*p) if (!*p)
die("Missing connection name in connect\n"); die("Missing connection name in connect");
name = p; name= p;
while (*p && !my_isspace(charset_info,*p)) while (*p && !my_isspace(charset_info,*p))
p++; p++;
*p = 0;
for (con = cons; con < next_con; con++) if (*p)
*p++= 0;
q->last_argument= p;
for (con= cons; con < next_con; con++)
{ {
if (!strcmp(con->name, name)) if (!strcmp(con->name, name))
{ {
...@@ -1678,21 +1916,21 @@ int close_connection(struct st_query* q) ...@@ -1678,21 +1916,21 @@ int close_connection(struct st_query* q)
) are delimiters/terminators ) are delimiters/terminators
*/ */
char* safe_get_param(char* str, char** arg, const char* msg) char* safe_get_param(char *str, char** arg, const char *msg)
{ {
DBUG_ENTER("safe_get_param"); DBUG_ENTER("safe_get_param");
while (*str && my_isspace(charset_info,*str)) while (*str && my_isspace(charset_info,*str))
str++; str++;
*arg = str; *arg= str;
for (; *str && *str != ',' && *str != ')' ; str++) for (; *str && *str != ',' && *str != ')' ; str++)
{ {
if (my_isspace(charset_info,*str)) if (my_isspace(charset_info,*str))
*str = 0; *str= 0;
} }
if (!*str) if (!*str)
die(msg); die(msg);
*str++ = 0; *str++= 0;
DBUG_RETURN(str); DBUG_RETURN(str);
} }
...@@ -1732,9 +1970,9 @@ void init_manager() ...@@ -1732,9 +1970,9 @@ void init_manager()
0 - success, non-0 - failure 0 - success, non-0 - failure
*/ */
int safe_connect(MYSQL* con, const char* host, const char* user, int safe_connect(MYSQL* con, const char *host, const char *user,
const char* pass, const char *pass,
const char* db, int port, const char* sock) const char *db, int port, const char *sock)
{ {
int con_error = 1; int con_error = 1;
int i; int i;
...@@ -1856,7 +2094,7 @@ int connect_n_handle_errors(struct st_query *q, MYSQL* con, const char* host, ...@@ -1856,7 +2094,7 @@ int connect_n_handle_errors(struct st_query *q, MYSQL* con, const char* host,
if (record) if (record)
{ {
if (!q->record_file[0] && !result_file) if (!q->record_file[0] && !result_file)
die("At line %u: Missing result file", start_lineno); die("Missing result file");
if (!result_file) if (!result_file)
str_to_file(q->record_file, ds->str, ds->length); str_to_file(q->record_file, ds->str, ds->length);
} }
...@@ -1871,14 +2109,14 @@ int connect_n_handle_errors(struct st_query *q, MYSQL* con, const char* host, ...@@ -1871,14 +2109,14 @@ int connect_n_handle_errors(struct st_query *q, MYSQL* con, const char* host,
} }
int do_connect(struct st_query* q) int do_connect(struct st_query *q)
{ {
char* con_name, *con_user,*con_pass, *con_host, *con_port_str, char *con_name, *con_user,*con_pass, *con_host, *con_port_str,
*con_db, *con_sock; *con_db, *con_sock;
char* p=q->first_argument; char *p= q->first_argument;
char buff[FN_REFLEN]; char buff[FN_REFLEN];
int con_port; int con_port;
int free_con_sock = 0; int free_con_sock= 0;
int error= 0; int error= 0;
int create_conn= 1; int create_conn= 1;
...@@ -1888,40 +2126,41 @@ int do_connect(struct st_query* q) ...@@ -1888,40 +2126,41 @@ int do_connect(struct st_query* q)
if (*p != '(') if (*p != '(')
die("Syntax error in connect - expected '(' found '%c'", *p); die("Syntax error in connect - expected '(' found '%c'", *p);
p++; p++;
p = safe_get_param(p, &con_name, "missing connection name"); p= safe_get_param(p, &con_name, "missing connection name");
p = safe_get_param(p, &con_host, "missing connection host"); p= safe_get_param(p, &con_host, "missing connection host");
p = safe_get_param(p, &con_user, "missing connection user"); p= safe_get_param(p, &con_user, "missing connection user");
p = safe_get_param(p, &con_pass, "missing connection password"); p= safe_get_param(p, &con_pass, "missing connection password");
p = safe_get_param(p, &con_db, "missing connection db"); p= safe_get_param(p, &con_db, "missing connection db");
if (!*p || *p == ';') /* Default port and sock */ if (!*p || *p == ';') /* Default port and sock */
{ {
con_port=port; con_port= port;
con_sock=(char*) unix_sock; con_sock= (char*) unix_sock;
} }
else else
{ {
VAR* var_port, *var_sock; VAR* var_port, *var_sock;
p = safe_get_param(p, &con_port_str, "missing connection port"); p= safe_get_param(p, &con_port_str, "missing connection port");
if (*con_port_str == '$') if (*con_port_str == '$')
{ {
if (!(var_port = var_get(con_port_str, 0, 0, 0))) if (!(var_port= var_get(con_port_str, 0, 0, 0)))
die("Unknown variable '%s'", con_port_str+1); die("Unknown variable '%s'", con_port_str+1);
con_port = var_port->int_val; con_port= var_port->int_val;
} }
else else
con_port=atoi(con_port_str); con_port= atoi(con_port_str);
p = safe_get_param(p, &con_sock, "missing connection socket"); p= safe_get_param(p, &con_sock, "missing connection socket");
if (*con_sock == '$') if (*con_sock == '$')
{ {
if (!(var_sock = var_get(con_sock, 0, 0, 0))) if (!(var_sock= var_get(con_sock, 0, 0, 0)))
die("Unknown variable '%s'", con_sock+1); die("Unknown variable '%s'", con_sock+1);
if (!(con_sock = (char*)my_malloc(var_sock->str_val_len+1, MYF(0)))) if (!(con_sock= (char*)my_malloc(var_sock->str_val_len+1, MYF(0))))
die("Out of memory"); die("Out of memory");
free_con_sock = 1; free_con_sock= 1;
memcpy(con_sock, var_sock->str_val, var_sock->str_val_len); memcpy(con_sock, var_sock->str_val, var_sock->str_val_len);
con_sock[var_sock->str_val_len] = 0; con_sock[var_sock->str_val_len]= 0;
} }
} }
q->last_argument= p;
if (next_con == cons_end) if (next_con == cons_end)
die("Connection limit exhausted - increase MAX_CONS in mysqltest.c"); die("Connection limit exhausted - increase MAX_CONS in mysqltest.c");
...@@ -1941,11 +2180,10 @@ int do_connect(struct st_query* q) ...@@ -1941,11 +2180,10 @@ int do_connect(struct st_query* q)
if (con_sock && !free_con_sock && *con_sock && *con_sock != FN_LIBCHAR) if (con_sock && !free_con_sock && *con_sock && *con_sock != FN_LIBCHAR)
con_sock=fn_format(buff, con_sock, TMPDIR, "",0); con_sock=fn_format(buff, con_sock, TMPDIR, "",0);
if (!con_db[0]) if (!con_db[0])
con_db=db; con_db= db;
/* Special database to allow one to connect without a database name */ /* Special database to allow one to connect without a database name */
if (con_db && !strcmp(con_db,"*NO-ONE*")) if (con_db && !strcmp(con_db,"*NO-ONE*"))
con_db=0; con_db= 0;
if (q->abort_on_error) if (q->abort_on_error)
{ {
if ((safe_connect(&next_con->mysql, con_host, con_user, con_pass, if ((safe_connect(&next_con->mysql, con_host, con_user, con_pass,
...@@ -1970,14 +2208,15 @@ int do_connect(struct st_query* q) ...@@ -1970,14 +2208,15 @@ int do_connect(struct st_query* q)
} }
int do_done(struct st_query* q) int do_done(struct st_query *q)
{ {
/* Dummy statement to eliminate compiler warning */
q->type = Q_END_BLOCK;
/* Check if empty block stack */ /* Check if empty block stack */
if (cur_block == block_stack) if (cur_block == block_stack)
{
if (*q->query != '}')
die("Stray 'end' command - end of block before beginning");
die("Stray '}' - end of block before beginning"); die("Stray '}' - end of block before beginning");
}
/* Test if inner block has been executed */ /* Test if inner block has been executed */
if (cur_block->ok && cur_block->cmd == cmd_while) if (cur_block->ok && cur_block->cmd == cmd_while)
...@@ -1998,9 +2237,10 @@ int do_done(struct st_query* q) ...@@ -1998,9 +2237,10 @@ int do_done(struct st_query* q)
int do_block(enum block_cmd cmd, struct st_query* q) int do_block(enum block_cmd cmd, struct st_query* q)
{ {
char* p=q->first_argument; char *p= q->first_argument;
const char* expr_start, *expr_end; const char *expr_start, *expr_end;
VAR v; VAR v;
const char *cmd_name= (cmd == cmd_while ? "while" : "if");
/* Check stack overflow */ /* Check stack overflow */
if (cur_block == block_stack_end) if (cur_block == block_stack_end)
...@@ -2020,12 +2260,21 @@ int do_block(enum block_cmd cmd, struct st_query* q) ...@@ -2020,12 +2260,21 @@ int do_block(enum block_cmd cmd, struct st_query* q)
} }
/* Parse and evaluate test expression */ /* Parse and evaluate test expression */
expr_start = strchr(p, '('); expr_start= strchr(p, '(');
if (!expr_start) if (!expr_start)
die("missing '(' in while"); die("missing '(' in %s", cmd_name);
expr_end = strrchr(expr_start, ')'); expr_end= strrchr(expr_start, ')');
if (!expr_end) if (!expr_end)
die("missing ')' in while"); die("missing ')' in %s", cmd_name);
p= (char*)expr_end+1;
while (*p && my_isspace(charset_info, *p))
p++;
if (*p == '{')
die("Missing newline between %s and '{'", cmd_name);
if (*p)
die("Missing '{' after %s. Found \"%s\"", cmd_name, p);
var_init(&v,0,0,0,0); var_init(&v,0,0,0,0);
eval_expr(&v, ++expr_start, &expr_end); eval_expr(&v, ++expr_start, &expr_end);
...@@ -2071,7 +2320,7 @@ my_bool end_of_query(int c) ...@@ -2071,7 +2320,7 @@ my_bool end_of_query(int c)
return 0; return 0;
for (i= 1; i < delimiter_length && for (i= 1; i < delimiter_length &&
(c= my_getc(*cur_file)) == *(delimiter + i); (c= my_getc(cur_file->file)) == *(delimiter + i);
i++) i++)
tmp[i]= c; tmp[i]= c;
...@@ -2086,11 +2335,35 @@ my_bool end_of_query(int c) ...@@ -2086,11 +2335,35 @@ my_bool end_of_query(int c)
} }
int read_line(char* buf, int size) /*
Read one "line" from the file
SYNOPSIS
read_line
buf buffer for the read line
size size of the buffer i.e max size to read
DESCRIPTION
This function actually reads several lines an adds them to the
buffer buf. It will continue to read until it finds what it believes
is a complete query.
Normally that means it will read lines until it reaches the
"delimiter" that marks end of query. Default delimiter is ';'
The function should be smart enough not to detect delimiter's
found inside strings sorrounded with '"' and '\'' escaped strings.
If the first line in a query starts with '#' or '-' this line is treated
as a comment. A comment is always terminated when end of line '\n' is
reached.
*/
int read_line(char *buf, int size)
{ {
int c; int c;
char quote; char quote;
char* p= buf, *buf_end= buf + size - 1; char *p= buf, *buf_end= buf + size - 1;
int no_save= 0; int no_save= 0;
enum {R_NORMAL, R_Q, R_Q_IN_Q, R_SLASH_IN_Q, enum {R_NORMAL, R_Q, R_Q_IN_Q, R_SLASH_IN_Q,
R_COMMENT, R_LINE_START} state= R_LINE_START; R_COMMENT, R_LINE_START} state= R_LINE_START;
...@@ -2100,19 +2373,30 @@ int read_line(char* buf, int size) ...@@ -2100,19 +2373,30 @@ int read_line(char* buf, int size)
for (; p < buf_end ;) for (; p < buf_end ;)
{ {
no_save= 0; no_save= 0;
c= my_getc(*cur_file); c= my_getc(cur_file->file);
if (feof(*cur_file)) if (feof(cur_file->file))
{ {
found_eof: found_eof:
if ((*cur_file) != stdin) if (cur_file->file != stdin)
my_fclose(*cur_file, MYF(0)); my_fclose(cur_file->file, MYF(0));
cur_file--; my_free((gptr)cur_file->file_name, MYF(MY_ALLOW_ZERO_PTR));
cur_file->file_name= 0;
lineno--; lineno--;
start_lineno= *lineno;
if (cur_file == file_stack) if (cur_file == file_stack)
{ {
/* We're back at the first file, check if
all { have matching }
*/
if (cur_block != block_stack)
{
start_lineno= *(lineno+1);
die("Missing end of block");
}
DBUG_PRINT("info", ("end of file")); DBUG_PRINT("info", ("end of file"));
DBUG_RETURN(1); DBUG_RETURN(1);
} }
cur_file--;
continue; continue;
} }
...@@ -2146,7 +2430,8 @@ int read_line(char* buf, int size) ...@@ -2146,7 +2430,8 @@ int read_line(char* buf, int size)
} }
break; break;
case R_LINE_START: case R_LINE_START:
if (c == '#' || c == '-' || parsing_disabled) /* Only accept start of comment if this is the first line in query */
if ((*lineno == start_lineno) && (c == '#' || c == '-' || parsing_diabled))
{ {
state = R_COMMENT; state = R_COMMENT;
} }
...@@ -2216,9 +2501,9 @@ int read_line(char* buf, int size) ...@@ -2216,9 +2501,9 @@ int read_line(char* buf, int size)
for (i= 1; i < charlen; i++) for (i= 1; i < charlen; i++)
{ {
if (feof(*cur_file)) if (feof(cur_file->file))
goto found_eof; /* FIXME: could we just break here?! */ goto found_eof; /* FIXME: could we just break here?! */
c= my_getc(*cur_file); c= my_getc(cur_file->file);
*p++ = c; *p++ = c;
} }
if (! my_ismbchar(charset_info, mb_start, p)) if (! my_ismbchar(charset_info, mb_start, p))
...@@ -2235,15 +2520,34 @@ int read_line(char* buf, int size) ...@@ -2235,15 +2520,34 @@ int read_line(char* buf, int size)
} }
} }
*p= 0; /* Always end with \0 */ *p= 0; /* Always end with \0 */
DBUG_RETURN(feof(*cur_file)); DBUG_RETURN(feof(cur_file->file));
} }
/*
Create a query from a set of lines
SYNOPSIS
read_query()
q_ptr pointer where to return the new query
DESCRIPTION
Converts lines returned by read_line into a query, this involves
parsing the first word in the read line to find the query type.
A -- comment may contain a valid query as the first word after the
comment start. Thus it's always checked to see if that is the case.
The advantage with this approach is to be able to execute commands
terminated by new line '\n' regardless how many "delimiter" it contain.
If query starts with @<file_name> this will specify a file to ....
*/
static char read_query_buf[MAX_QUERY]; static char read_query_buf[MAX_QUERY];
int read_query(struct st_query** q_ptr) int read_query(struct st_query** q_ptr)
{ {
char *p = read_query_buf, * p1 ; char *p= read_query_buf, *p1;
struct st_query* q; struct st_query* q;
DBUG_ENTER("read_query"); DBUG_ENTER("read_query");
...@@ -2260,7 +2564,7 @@ int read_query(struct st_query** q_ptr) ...@@ -2260,7 +2564,7 @@ int read_query(struct st_query** q_ptr)
q->require_file= 0; q->require_file= 0;
q->first_word_len= 0; q->first_word_len= 0;
q->type = Q_UNKNOWN; q->type= Q_UNKNOWN;
q->query_buf= q->query= 0; q->query_buf= q->query= 0;
if (read_line(read_query_buf, sizeof(read_query_buf))) if (read_line(read_query_buf, sizeof(read_query_buf)))
{ {
...@@ -2269,7 +2573,7 @@ int read_query(struct st_query** q_ptr) ...@@ -2269,7 +2573,7 @@ int read_query(struct st_query** q_ptr)
DBUG_PRINT("info", ("query: %s", read_query_buf)); DBUG_PRINT("info", ("query: %s", read_query_buf));
if (*p == '#') if (*p == '#')
{ {
q->type = Q_COMMENT; q->type= Q_COMMENT;
/* This goto is to avoid losing the "expected error" info. */ /* This goto is to avoid losing the "expected error" info. */
goto end; goto end;
} }
...@@ -2279,8 +2583,6 @@ int read_query(struct st_query** q_ptr) ...@@ -2279,8 +2583,6 @@ int read_query(struct st_query** q_ptr)
sizeof(global_expected_errno)); sizeof(global_expected_errno));
q->expected_errors= global_expected_errors; q->expected_errors= global_expected_errors;
q->abort_on_error= (global_expected_errors == 0 && abort_on_error); q->abort_on_error= (global_expected_errors == 0 && abort_on_error);
bzero((gptr) global_expected_errno, sizeof(global_expected_errno));
global_expected_errors=0;
} }
if (p[0] == '-' && p[1] == '-') if (p[0] == '-' && p[1] == '-')
...@@ -2290,34 +2592,6 @@ int read_query(struct st_query** q_ptr) ...@@ -2290,34 +2592,6 @@ int read_query(struct st_query** q_ptr)
} }
else if (!parsing_disabled) else if (!parsing_disabled)
{ {
if (*p == '!')
{
q->abort_on_error= 0;
p++;
if (*p == '$')
{
int expected_errno= 0;
p++;
for (; my_isdigit(charset_info, *p); p++)
expected_errno = expected_errno * 10 + *p - '0';
q->expected_errno[0].code.errnum = expected_errno;
q->expected_errno[0].type= ERR_ERRNO;
q->expected_errno[1].type= ERR_EMPTY;
q->expected_errors=1;
}
else if (*p == 'S') /* SQLSTATE */
{
int i;
p++;
for (i = 0; my_isalnum(charset_info, *p) && i < SQLSTATE_LENGTH; p++, i++)
q->expected_errno[0].code.sqlstate[i]= *p;
q->expected_errno[0].code.sqlstate[i]= '\0';
q->expected_errno[0].type= ERR_SQLSTATE;
q->expected_errno[1].type= ERR_EMPTY;
q->expected_errors=1;
}
}
while (*p && my_isspace(charset_info, *p)) while (*p && my_isspace(charset_info, *p))
p++ ; p++ ;
if (*p == '@') if (*p == '@')
...@@ -2478,7 +2752,9 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)), ...@@ -2478,7 +2752,9 @@ get_one_option(int optid, const struct my_option *opt __attribute__((unused)),
argument= buff; argument= buff;
} }
fn_format(buff, argument, "", "", 4); fn_format(buff, argument, "", "", 4);
if (!(*++cur_file = my_fopen(buff, O_RDONLY | FILE_BINARY, MYF(MY_WME)))) DBUG_ASSERT(cur_file->file == 0);
if (!(cur_file->file=
my_fopen(buff, O_RDONLY | FILE_BINARY, MYF(MY_WME))))
die("Could not open %s: errno = %d", argument, errno); die("Could not open %s: errno = %d", argument, errno);
break; break;
} }
...@@ -2564,7 +2840,7 @@ int parse_args(int argc, char **argv) ...@@ -2564,7 +2840,7 @@ int parse_args(int argc, char **argv)
return 0; return 0;
} }
char* safe_str_append(char* buf, const char* str, int size) char* safe_str_append(char *buf, const char *str, int size)
{ {
int i,c ; int i,c ;
for (i = 0; (c = *str++) && i < size - 1; i++) for (i = 0; (c = *str++) && i < size - 1; i++)
...@@ -2573,7 +2849,7 @@ char* safe_str_append(char* buf, const char* str, int size) ...@@ -2573,7 +2849,7 @@ char* safe_str_append(char* buf, const char* str, int size)
return buf; return buf;
} }
void str_to_file(const char* fname, char* str, int size) void str_to_file(const char *fname, char *str, int size)
{ {
int fd; int fd;
char buff[FN_REFLEN]; char buff[FN_REFLEN];
...@@ -2592,7 +2868,7 @@ void str_to_file(const char* fname, char* str, int size) ...@@ -2592,7 +2868,7 @@ void str_to_file(const char* fname, char* str, int size)
my_close(fd, MYF(0)); my_close(fd, MYF(0));
} }
void reject_dump(const char* record_file, char* buf, int size) void reject_dump(const char *record_file, char *buf, int size)
{ {
char reject_file[FN_REFLEN]; char reject_file[FN_REFLEN];
str_to_file(fn_format(reject_file, record_file,"",".reject",2), buf, size); str_to_file(fn_format(reject_file, record_file,"",".reject",2), buf, size);
...@@ -2608,7 +2884,7 @@ static void replace_dynstr_append_mem(DYNAMIC_STRING *ds, const char *val, ...@@ -2608,7 +2884,7 @@ static void replace_dynstr_append_mem(DYNAMIC_STRING *ds, const char *val,
{ {
len=(int) replace_strings(glob_replace, &out_buff, &out_length, val); len=(int) replace_strings(glob_replace, &out_buff, &out_length, val);
if (len == -1) if (len == -1)
die("Out of memory in replace\n"); die("Out of memory in replace");
val=out_buff; val=out_buff;
} }
dynstr_append_mem(ds, val, len); dynstr_append_mem(ds, val, len);
...@@ -2747,8 +3023,8 @@ static int run_query_normal(MYSQL* mysql, struct st_query* q, int flags) ...@@ -2747,8 +3023,8 @@ static int run_query_normal(MYSQL* mysql, struct st_query* q, int flags)
{ {
got_error_on_send= mysql_send_query(mysql, query, query_len); got_error_on_send= mysql_send_query(mysql, query, query_len);
if (got_error_on_send && q->expected_errno[0].type == ERR_EMPTY) if (got_error_on_send && q->expected_errno[0].type == ERR_EMPTY)
die("At line %u: unable to send query '%s' (mysql_errno=%d , errno=%d)", die("unable to send query '%s' (mysql_errno=%d , errno=%d)",
start_lineno, query, mysql_errno(mysql), errno); query, mysql_errno(mysql), errno);
} }
do do
...@@ -2826,7 +3102,7 @@ static int run_query_normal(MYSQL* mysql, struct st_query* q, int flags) ...@@ -2826,7 +3102,7 @@ static int run_query_normal(MYSQL* mysql, struct st_query* q, int flags)
warn_res= mysql_store_result(mysql); warn_res= mysql_store_result(mysql);
} }
if (!warn_res) if (!warn_res)
verbose_msg("Warning count is %u but didn't get any warnings\n", verbose_msg("Warning count is %u but didn't get any warnings",
count); count);
else else
{ {
...@@ -2852,7 +3128,7 @@ static int run_query_normal(MYSQL* mysql, struct st_query* q, int flags) ...@@ -2852,7 +3128,7 @@ static int run_query_normal(MYSQL* mysql, struct st_query* q, int flags)
if (record) if (record)
{ {
if (!q->record_file[0] && !result_file) if (!q->record_file[0] && !result_file)
die("At line %u: Missing result file", start_lineno); die("Missing result file");
if (!result_file) if (!result_file)
str_to_file(q->record_file, ds->str, ds->length); str_to_file(q->record_file, ds->str, ds->length);
} }
...@@ -2917,7 +3193,7 @@ static int normal_handle_error(const char *query, struct st_query *q, ...@@ -2917,7 +3193,7 @@ static int normal_handle_error(const char *query, struct st_query *q,
abort_not_supported_test(); abort_not_supported_test();
if (q->abort_on_error) if (q->abort_on_error)
die("At line %u: query '%s' failed: %d: %s", start_lineno, query, die("query '%s' failed: %d: %s", query,
mysql_errno(mysql), mysql_error(mysql)); mysql_errno(mysql), mysql_error(mysql));
else else
{ {
...@@ -3044,7 +3320,7 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3044,7 +3320,7 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
may be a new connection. may be a new connection.
*/ */
if (!(stmt= mysql_stmt_init(mysql))) if (!(stmt= mysql_stmt_init(mysql)))
die("At line %u: unable init stmt structure"); die("unable init stmt structure");
if (q->type != Q_EVAL) if (q->type != Q_EVAL)
{ {
...@@ -3088,9 +3364,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3088,9 +3364,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
{ {
if (q->abort_on_error) if (q->abort_on_error)
{ {
die("At line %u: unable to prepare statement '%s': " die("unable to prepare statement '%s': "
"%s (mysql_stmt_errno=%d returned=%d)", "%s (mysql_stmt_errno=%d returned=%d)",
start_lineno, query, query,
mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err); mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err);
} }
else else
...@@ -3123,9 +3399,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3123,9 +3399,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
if (q->abort_on_error) if (q->abort_on_error)
{ {
/* We got an error, unexpected */ /* We got an error, unexpected */
die("At line %u: unable to execute statement '%s': " die("unable to execute statement '%s': "
"%s (mysql_stmt_errno=%d returned=%d)", "%s (mysql_stmt_errno=%d returned=%d)",
start_lineno, query, mysql_stmt_error(stmt), query, mysql_stmt_error(stmt),
mysql_stmt_errno(stmt), got_error_on_execute); mysql_stmt_errno(stmt), got_error_on_execute);
} }
else else
...@@ -3145,9 +3421,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3145,9 +3421,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
my_bool one= 1; my_bool one= 1;
if (mysql_stmt_attr_set(stmt, STMT_ATTR_UPDATE_MAX_LENGTH, if (mysql_stmt_attr_set(stmt, STMT_ATTR_UPDATE_MAX_LENGTH,
(void*) &one) != 0) (void*) &one) != 0)
die("At line %u: unable to set stmt attribute " die("unable to set stmt attribute "
"'STMT_ATTR_UPDATE_MAX_LENGTH': %s (returned=%d)", "'STMT_ATTR_UPDATE_MAX_LENGTH': %s (returned=%d)",
start_lineno, query, err); query, err);
} }
/* /*
...@@ -3159,9 +3435,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3159,9 +3435,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
if (q->abort_on_error) if (q->abort_on_error)
{ {
/* We got an error, unexpected */ /* We got an error, unexpected */
die("At line %u: unable to execute statement '%s': " die("unable to execute statement '%s': "
"%s (mysql_stmt_errno=%d returned=%d)", "%s (mysql_stmt_errno=%d returned=%d)",
start_lineno, query, mysql_stmt_error(stmt), query, mysql_stmt_error(stmt),
mysql_stmt_errno(stmt), got_error_on_execute); mysql_stmt_errno(stmt), got_error_on_execute);
} }
else else
...@@ -3251,18 +3527,18 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3251,18 +3527,18 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
/* Fill in the data into the structures created above */ /* Fill in the data into the structures created above */
if ((err= mysql_stmt_bind_result(stmt, bind)) != 0) if ((err= mysql_stmt_bind_result(stmt, bind)) != 0)
die("At line %u: unable to bind result to statement '%s': " die("unable to bind result to statement '%s': "
"%s (mysql_stmt_errno=%d returned=%d)", "%s (mysql_stmt_errno=%d returned=%d)",
start_lineno, query, query,
mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err); mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err);
/* Read result from each row */ /* Read result from each row */
for (row_idx= 0; row_idx < num_rows; row_idx++) for (row_idx= 0; row_idx < num_rows; row_idx++)
{ {
if ((err= mysql_stmt_fetch(stmt)) != 0) if ((err= mysql_stmt_fetch(stmt)) != 0)
die("At line %u: unable to fetch all rows from statement '%s': " die("unable to fetch all rows from statement '%s': "
"%s (mysql_stmt_errno=%d returned=%d)", "%s (mysql_stmt_errno=%d returned=%d)",
start_lineno, query, query,
mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err); mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err);
/* Read result from each column */ /* Read result from each column */
...@@ -3300,9 +3576,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3300,9 +3576,9 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
} }
if ((err= mysql_stmt_fetch(stmt)) != MYSQL_NO_DATA) if ((err= mysql_stmt_fetch(stmt)) != MYSQL_NO_DATA)
die("At line %u: fetch didn't end with MYSQL_NO_DATA from statement " die("fetch didn't end with MYSQL_NO_DATA from statement "
"'%s': %s (mysql_stmt_errno=%d returned=%d)", "'%s': %s (mysql_stmt_errno=%d returned=%d)",
start_lineno, query, query,
mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err); mysql_stmt_error(stmt), mysql_stmt_errno(stmt), err);
free_replace_column(); free_replace_column();
...@@ -3339,7 +3615,7 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags) ...@@ -3339,7 +3615,7 @@ static int run_query_stmt(MYSQL *mysql, struct st_query *q, int flags)
if (record) if (record)
{ {
if (!q->record_file[0] && !result_file) if (!q->record_file[0] && !result_file)
die("At line %u: Missing result file", start_lineno); die("Missing result file");
if (!result_file) if (!result_file)
str_to_file(q->record_file, ds->str, ds->length); str_to_file(q->record_file, ds->str, ds->length);
} }
...@@ -3444,7 +3720,7 @@ static void run_query_stmt_handle_warnings(MYSQL *mysql, DYNAMIC_STRING *ds) ...@@ -3444,7 +3720,7 @@ static void run_query_stmt_handle_warnings(MYSQL *mysql, DYNAMIC_STRING *ds)
{ {
MYSQL_RES *warn_res= mysql_store_result(mysql); MYSQL_RES *warn_res= mysql_store_result(mysql);
if (!warn_res) if (!warn_res)
verbose_msg("Warning count is %u but didn't get any warnings\n", verbose_msg("Warning count is %u but didn't get any warnings",
count); count);
else else
{ {
...@@ -3467,7 +3743,7 @@ static int run_query_stmt_handle_error(char *query, struct st_query *q, ...@@ -3467,7 +3743,7 @@ static int run_query_stmt_handle_error(char *query, struct st_query *q,
} }
if (q->abort_on_error) if (q->abort_on_error)
die("At line %u: query '%s' failed: %d: %s", start_lineno, query, die("query '%s' failed: %d: %s", query,
mysql_stmt_errno(stmt), mysql_stmt_error(stmt)); mysql_stmt_errno(stmt), mysql_stmt_error(stmt));
else else
{ {
...@@ -3726,14 +4002,14 @@ int main(int argc, char **argv) ...@@ -3726,14 +4002,14 @@ int main(int argc, char **argv)
memset(file_stack, 0, sizeof(file_stack)); memset(file_stack, 0, sizeof(file_stack));
memset(&master_pos, 0, sizeof(master_pos)); memset(&master_pos, 0, sizeof(master_pos));
file_stack_end = file_stack + MAX_INCLUDE_DEPTH; file_stack_end= file_stack + MAX_INCLUDE_DEPTH - 1;
cur_file = file_stack; cur_file= file_stack;
lineno = lineno_stack; lineno = lineno_stack;
my_init_dynamic_array(&q_lines, sizeof(struct st_query*), INIT_Q_LINES, my_init_dynamic_array(&q_lines, sizeof(struct st_query*), INIT_Q_LINES,
INIT_Q_LINES); INIT_Q_LINES);
memset(block_stack, 0, sizeof(block_stack)); memset(block_stack, 0, sizeof(block_stack));
block_stack_end= block_stack + BLOCK_STACK_DEPTH; block_stack_end= block_stack + BLOCK_STACK_DEPTH - 1;
cur_block= block_stack; cur_block= block_stack;
cur_block->ok= TRUE; /* Outer block should always be executed */ cur_block->ok= TRUE; /* Outer block should always be executed */
cur_block->cmd= cmd_none; cur_block->cmd= cmd_none;
...@@ -3747,7 +4023,11 @@ int main(int argc, char **argv) ...@@ -3747,7 +4023,11 @@ int main(int argc, char **argv)
(char**) embedded_server_groups)) (char**) embedded_server_groups))
die("Can't initialize MySQL server"); die("Can't initialize MySQL server");
if (cur_file == file_stack) if (cur_file == file_stack)
*++cur_file = stdin; {
DBUG_ASSERT(cur_file->file == 0);
cur_file->file= stdin;
cur_file->file_name= my_strdup("<stdin>", MYF(MY_WME));
}
*lineno=1; *lineno=1;
#ifndef EMBEDDED_LIBRARY #ifndef EMBEDDED_LIBRARY
if (manager_host) if (manager_host)
...@@ -3793,12 +4073,13 @@ int main(int argc, char **argv) ...@@ -3793,12 +4073,13 @@ int main(int argc, char **argv)
get_query_type(q); get_query_type(q);
if (cur_block->ok) if (cur_block->ok)
{ {
q->last_argument= q->first_argument;
processed = 1; processed = 1;
switch (q->type) { switch (q->type) {
case Q_CONNECT: case Q_CONNECT:
error|= do_connect(q); error|= do_connect(q);
break; break;
case Q_CONNECTION: select_connection(q->first_argument); break; case Q_CONNECTION: select_connection(q); break;
case Q_DISCONNECT: case Q_DISCONNECT:
case Q_DIRTY_CLOSE: case Q_DIRTY_CLOSE:
close_connection(q); break; close_connection(q); break;
...@@ -3828,19 +4109,24 @@ int main(int argc, char **argv) ...@@ -3828,19 +4109,24 @@ int main(int argc, char **argv)
case Q_SERVER_START: do_server_start(q); break; case Q_SERVER_START: do_server_start(q); break;
case Q_SERVER_STOP: do_server_stop(q); break; case Q_SERVER_STOP: do_server_stop(q); break;
#endif #endif
case Q_INC: do_inc(q); break; case Q_INC: do_modify_var(q, "inc", DO_INC); break;
case Q_DEC: do_dec(q); break; case Q_DEC: do_modify_var(q, "dec", DO_DEC); break;
case Q_ECHO: do_echo(q); break; case Q_ECHO: do_echo(q); break;
case Q_SYSTEM: do_system(q); break; case Q_SYSTEM: do_system(q); break;
case Q_DELIMITER: case Q_DELIMITER:
strmake(delimiter, q->first_argument, sizeof(delimiter) - 1); strmake(delimiter, q->first_argument, sizeof(delimiter) - 1);
delimiter_length= strlen(delimiter); delimiter_length= strlen(delimiter);
q->last_argument= q->first_argument+delimiter_length;
break;
case Q_DISPLAY_VERTICAL_RESULTS:
display_result_vertically= TRUE;
break; break;
case Q_DISPLAY_VERTICAL_RESULTS: display_result_vertically= TRUE; break;
case Q_DISPLAY_HORIZONTAL_RESULTS: case Q_DISPLAY_HORIZONTAL_RESULTS:
display_result_vertically= FALSE; break; display_result_vertically= FALSE;
break;
case Q_LET: do_let(q); break; case Q_LET: do_let(q); break;
case Q_EVAL_RESULT: eval_result = 1; break; case Q_EVAL_RESULT:
eval_result = 1; break;
case Q_EVAL: case Q_EVAL:
if (q->query == q->query_buf) if (q->query == q->query_buf)
{ {
...@@ -3873,6 +4159,7 @@ int main(int argc, char **argv) ...@@ -3873,6 +4159,7 @@ int main(int argc, char **argv)
} }
error|= run_query(&cur_con->mysql, q, QUERY_REAP|QUERY_SEND); error|= run_query(&cur_con->mysql, q, QUERY_REAP|QUERY_SEND);
display_result_vertically= old_display_result_vertically; display_result_vertically= old_display_result_vertically;
q->last_argument= q->end;
break; break;
} }
case Q_QUERY: case Q_QUERY:
...@@ -3897,6 +4184,7 @@ int main(int argc, char **argv) ...@@ -3897,6 +4184,7 @@ int main(int argc, char **argv)
save_file[0]=0; save_file[0]=0;
} }
error |= run_query(&cur_con->mysql, q, flags); error |= run_query(&cur_con->mysql, q, flags);
q->last_argument= q->end;
break; break;
} }
case Q_SEND: case Q_SEND:
...@@ -3916,6 +4204,7 @@ int main(int argc, char **argv) ...@@ -3916,6 +4204,7 @@ int main(int argc, char **argv)
is given on this connection. is given on this connection.
*/ */
error |= run_query(&cur_con->mysql, q, QUERY_SEND); error |= run_query(&cur_con->mysql, q, QUERY_SEND);
q->last_argument= q->end;
break; break;
case Q_RESULT: case Q_RESULT:
get_file_name(save_file,q); get_file_name(save_file,q);
...@@ -3940,17 +4229,15 @@ int main(int argc, char **argv) ...@@ -3940,17 +4229,15 @@ int main(int argc, char **argv)
{ {
do_save_master_pos(); do_save_master_pos();
if (*q->first_argument) if (*q->first_argument)
select_connection(q->first_argument); select_connection(q);
else else
{ select_connection_name("slave");
char buf[] = "slave"; do_sync_with_master2(0);
select_connection(buf);
}
do_sync_with_master2("");
break; break;
} }
case Q_COMMENT: /* Ignore row */ case Q_COMMENT: /* Ignore row */
case Q_COMMENT_WITH_COMMAND: case Q_COMMENT_WITH_COMMAND:
q->last_argument= q->end;
break; break;
case Q_PING: case Q_PING:
(void) mysql_ping(&cur_con->mysql); (void) mysql_ping(&cur_con->mysql);
...@@ -4011,11 +4298,23 @@ int main(int argc, char **argv) ...@@ -4011,11 +4298,23 @@ int main(int argc, char **argv)
default: current_line_inc = 1; break; default: current_line_inc = 1; break;
} }
} }
else
check_eol_junk(q->last_argument);
if (q->type != Q_ERROR)
{
/*
As soon as any non "error" command has been executed,
the array with expected errors should be cleared
*/
global_expected_errors= 0;
bzero((gptr) global_expected_errno, sizeof(global_expected_errno));
}
parser.current_line += current_line_inc; parser.current_line += current_line_inc;
} }
if (result_file && ds_res.length) if (result_file && ds_res.length && !error)
{ {
if (!record) if (!record)
error |= check_result(&ds_res, result_file, q->require_file); error |= check_result(&ds_res, result_file, q->require_file);
...@@ -4874,7 +5173,7 @@ static void get_replace_column(struct st_query *q) ...@@ -4874,7 +5173,7 @@ static void get_replace_column(struct st_query *q)
free_replace_column(); free_replace_column();
if (!*from) if (!*from)
die("Missing argument in %s\n", q->query); die("Missing argument in %s", q->query);
/* Allocate a buffer for results */ /* Allocate a buffer for results */
start=buff=my_malloc(strlen(from)+1,MYF(MY_WME | MY_FAE)); start=buff=my_malloc(strlen(from)+1,MYF(MY_WME | MY_FAE));
...@@ -4885,15 +5184,16 @@ static void get_replace_column(struct st_query *q) ...@@ -4885,15 +5184,16 @@ static void get_replace_column(struct st_query *q)
to= get_string(&buff, &from, q); to= get_string(&buff, &from, q);
if (!(column_number= atoi(to)) || column_number > MAX_COLUMNS) if (!(column_number= atoi(to)) || column_number > MAX_COLUMNS)
die("Wrong column number to replace_columns in %s\n", q->query); die("Wrong column number to replace_column in '%s'", q->query);
if (!*from) if (!*from)
die("Wrong number of arguments to replace in %s\n", q->query); die("Wrong number of arguments to replace_column in '%s'", q->query);
to= get_string(&buff, &from, q); to= get_string(&buff, &from, q);
my_free(replace_column[column_number-1], MY_ALLOW_ZERO_PTR); my_free(replace_column[column_number-1], MY_ALLOW_ZERO_PTR);
replace_column[column_number-1]= my_strdup(to, MYF(MY_WME | MY_FAE)); replace_column[column_number-1]= my_strdup(to, MYF(MY_WME | MY_FAE));
set_if_bigger(max_replace_column, column_number); set_if_bigger(max_replace_column, column_number);
} }
my_free(start, MYF(0)); my_free(start, MYF(0));
q->last_argument= q->end;
} }
#if defined(__NETWARE__) || defined(__WIN__) #if defined(__NETWARE__) || defined(__WIN__)
...@@ -4944,7 +5244,7 @@ static char *subst_env_var(const char *str) ...@@ -4944,7 +5244,7 @@ static char *subst_env_var(const char *str)
if (!(subst= getenv(env_var))) if (!(subst= getenv(env_var)))
{ {
my_free(result, MYF(0)); my_free(result, MYF(0));
die("MYSQLTEST.NLM: Environment variable %s is not defined\n", die("MYSQLTEST.NLM: Environment variable %s is not defined",
env_var); env_var);
} }
......
let $1 = 10;
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
while ($1)
{
echo $1;
dec $1;
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
...@@ -2484,6 +2484,11 @@ sub run_mysqltest ($) { ...@@ -2484,6 +2484,11 @@ sub run_mysqltest ($) {
mysqld_arguments($args,'master',0,$tinfo->{'master_opt'},[]); mysqld_arguments($args,'master',0,$tinfo->{'master_opt'},[]);
} }
# ----------------------------------------------------------------------
# export MYSQL_TEST variable containing <path>/mysqltest <args>
# ----------------------------------------------------------------------
$ENV{'MYSQL_TEST'}= "$exe_mysqltest " . join(" ", @$args);
return mtr_run_test($exe,$args,$tinfo->{'path'},"",$path_timefile,""); return mtr_run_test($exe,$args,$tinfo->{'path'},"",$path_timefile,"");
} }
......
...@@ -733,6 +733,7 @@ if [ x$USE_TIMER = x1 ] ; then ...@@ -733,6 +733,7 @@ if [ x$USE_TIMER = x1 ] ; then
fi fi
MYSQL_TEST_BIN=$MYSQL_TEST MYSQL_TEST_BIN=$MYSQL_TEST
MYSQL_TEST="$MYSQL_TEST $MYSQL_TEST_ARGS" MYSQL_TEST="$MYSQL_TEST $MYSQL_TEST_ARGS"
export MYSQL_TEST
GDB_CLIENT_INIT=$MYSQL_TMP_DIR/gdbinit.client GDB_CLIENT_INIT=$MYSQL_TMP_DIR/gdbinit.client
GDB_MASTER_INIT=$MYSQL_TMP_DIR/gdbinit.master GDB_MASTER_INIT=$MYSQL_TMP_DIR/gdbinit.master
GDB_SLAVE_INIT=$MYSQL_TMP_DIR/gdbinit.slave GDB_SLAVE_INIT=$MYSQL_TMP_DIR/gdbinit.slave
......
...@@ -7,18 +7,16 @@ otto ...@@ -7,18 +7,16 @@ otto
select otto from (select 1 as otto) as t1; select otto from (select 1 as otto) as t1;
otto otto
1 1
mysqltest: At line 1: query 'select friedrich from (select 1 as otto) as t1' failed: 1054: Unknown column 'friedrich' in 'field list'
select friedrich from (select 1 as otto) as t1; select friedrich from (select 1 as otto) as t1;
ERROR 42S22: Unknown column 'friedrich' in 'field list' ERROR 42S22: Unknown column 'friedrich' in 'field list'
select otto from (select 1 as otto) as t1; select otto from (select 1 as otto) as t1;
otto otto
1 1
select otto from (select 1 as otto) as t1; mysqltest: At line 1: query 'select otto from (select 1 as otto) as t1' succeeded - should have failed with sqlstate 42S22...
otto
1
select friedrich from (select 1 as otto) as t1;
ERROR 42S22: Unknown column 'friedrich' in 'field list'
select friedrich from (select 1 as otto) as t1; select friedrich from (select 1 as otto) as t1;
ERROR 42S22: Unknown column 'friedrich' in 'field list' ERROR 42S22: Unknown column 'friedrich' in 'field list'
mysqltest: At line 1: query 'select friedrich from (select 1 as otto) as t1' failed with wrong sqlstate 42S22 instead of 00000...
select otto from (select 1 as otto) as t1; select otto from (select 1 as otto) as t1;
otto otto
1 1
...@@ -135,6 +133,8 @@ ERROR 42S02: Table 'test.t1' doesn't exist ...@@ -135,6 +133,8 @@ ERROR 42S02: Table 'test.t1' doesn't exist
select 1146 as "after_!errno_masked_error" ; select 1146 as "after_!errno_masked_error" ;
after_!errno_masked_error after_!errno_masked_error
1146 1146
mysqltest: At line 1: query 'select 3 from t1' failed with wrong errno 1146 instead of 1000...
mysqltest: At line 1: query 'select 3 from t1' failed with wrong errno 1146 instead of 1000...
garbage ; garbage ;
ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'garbage' at line 1 ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'garbage' at line 1
select 1064 as "after_--enable_abort_on_error" ; select 1064 as "after_--enable_abort_on_error" ;
...@@ -142,6 +142,219 @@ after_--enable_abort_on_error ...@@ -142,6 +142,219 @@ after_--enable_abort_on_error
1064 1064
select 3 from t1 ; select 3 from t1 ;
ERROR 42S02: Table 'test.t1' doesn't exist ERROR 42S02: Table 'test.t1' doesn't exist
mysqltest: At line 1: query 'select 3 from t1' failed with wrong errno 1146 instead of 1064...
mysqltest: At line 1: query 'select 3 from t1' failed: 1146: Table 'test.t1' doesn't exist
hello
hello
;;;;;;;;
# MySQL: -- The
mysqltest: At line 1: End of line junk detected: "6"
mysqltest: At line 1: End of line junk detected: "6"
mysqltest: At line 1: Missing delimiter
mysqltest: At line 1: Extra delimiter ";" found
MySQL
"MySQL"
MySQL: The world''s most popular open source database
"MySQL: The world's most popular open source database"
MySQL: The world''s
most popular open
source database
# MySQL: The world''s
# most popular open
# source database
- MySQL: The world''s
- most popular open
- source database
- MySQL: The world''s
-- most popular open
-- source database
# MySQL: The
--world''s
# most popular
-- open
- source database
"MySQL: The world's most popular; open source database"
"MySQL: The world's most popular ; open source database"
"MySQL: The world's most popular ;open source database"
echo message echo message
mysqltest: At line 1: Empty variable
sh: -c: line 0: syntax error near unexpected token `;'
sh: -c: line 0: `;'
mysqltest: At line 1: command ";" failed
mysqltest: At line 1: Missing argument in exec
MySQL
"MySQL"
MySQL: The
world''s most
popular open
source database
# MySQL: The
# world''s most
# popular open
# source database
-- MySQL: The
-- world''s most
-- popular open
-- source database
# MySQL: The
- world''s most
-- popular open
# source database
'$message'
"$message"
hej
hej
hej
1
a long variable content
a long variable content
a long $where variable content
mysqltest: At line 1: Missing arguments to let
mysqltest: At line 1: Missing variable name in let
mysqltest: At line 1: Variable name in hi=hi does not start with '$'
mysqltest: At line 1: Missing assignment operator in let
mysqltest: At line 1: Missing assignment operator in let
mysqltest: At line 1: Missing arguments to let
mysqltest: At line 1: Missing variable name in let
mysqltest: At line 1: Variable name in =hi does not start with '$'
mysqltest: At line 1: Missing assignment operator in let
mysqltest: At line 1: Missing file name in source
mysqltest: At line 1: Could not open file ./non_existingFile
mysqltest: In included file "./var/tmp/recursive.sql": At line 1: Source directives are nesting too deep
mysqltest: In included file "./var/tmp/error.sql": At line 1: query 'garbage ' failed: 1064: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'garbage' at line 1
2 = outer loop variable after while
here is the sourced script
2 = outer loop variable before dec
1 = outer loop variable after dec
1 = outer loop variable after while
here is the sourced script
1 = outer loop variable before dec
0 = outer loop variable after dec
2 = outer loop variable after while
here is the sourced script
2 = outer loop variable before dec
1 = outer loop variable after dec
1 = outer loop variable after while
here is the sourced script
1 = outer loop variable before dec
0 = outer loop variable after dec
In loop
here is the sourced script
In loop
here is the sourced script
In loop
here is the sourced script
In loop
here is the sourced script
In loop
here is the sourced script
In loop
here is the sourced script
In loop
here is the sourced script
In loop
here is the sourced script
In loop
here is the sourced script
mysqltest: At line 1: Missing argument to sleep
mysqltest: At line 1: Invalid argument to sleep "abc"
1
2
101
hej
1
mysqltest: At line 1: Missing arguments to inc
mysqltest: At line 1: First argument to inc must be a variable (start with $)
mysqltest: At line 1: End of line junk detected: "1000"
4
4
-1
-2
99
hej
-1
mysqltest: At line 1: Missing arguments to dec
mysqltest: At line 1: First argument to dec must be a variable (start with $)
mysqltest: At line 1: End of line junk detected: "1000"
mysqltest: At line 1: Missing arguments to system, nothing to do!
mysqltest: At line 1: Missing arguments to system, nothing to do!
sh: NonExistsinfComamdn: command not found
mysqltest: At line 1: system command 'NonExistsinfComamdn' failed
test
test2
test3
test4
1
mysqltest: In included file "./include/mysqltest_while.inc": At line 64: Nesting too deeply
mysqltest: At line 1: missing '(' in while
mysqltest: At line 1: missing ')' in while
mysqltest: At line 1: Missing '{' after while. Found "dec $i"
mysqltest: At line 1: Stray '}' - end of block before beginning
mysqltest: At line 1: Stray 'end' command - end of block before beginning
mysqltest: At line 1: query '' failed: 1065: Query was empty
mysqltest: At line 1: Missing '{' after while. Found "echo hej"
mysqltest: At line 3: Missing end of block
mysqltest: At line 1: Missing newline between while and '{'
mysqltest: At line 1: missing '(' in if
mysqltest: At line 1: Stray 'end' command - end of block before beginning
select "b" bs col1, "c" bs col2;
col1 col2
b c
seledt "b" bs dol1, "d" bs dol2;
dol1 dol2
b d
mysqltest: At line 1: Wrong number of arguments to replace_result in 'replace_result a'
mysqltest: At line 1: Wrong number of arguments to replace_result in 'replace_result a;'
mysqltest: At line 1: Wrong number of arguments to replace_result in 'replace_result a'
mysqltest: At line 1: Wrong number of arguments to replace_result in 'replace_result a '
mysqltest: At line 1: Wrong number of arguments to replace_result in 'replace_result a b c'
mysqltest: At line 1: Wrong number of arguments to replace_result in 'replace_result a b c '
select "a" as col1, "c" as col2;
col1 col2
b c
select "a" as col1, "c" as col2;
col1 col2
b d
mysqltest: At line 1: Wrong column number to replace_column in 'replace_column a'
mysqltest: At line 1: Wrong number of arguments to replace_column in 'replace_column 1'
mysqltest: At line 1: Wrong column number to replace_column in 'replace_column a b'
mysqltest: At line 1: Wrong column number to replace_column in 'replace_column a 1'
mysqltest: At line 1: Wrong column number to replace_column in 'replace_column 1 b c '
mysqltest: At line 1: Invalid integer argument "10!"
mysqltest: At line 1: End of line junk detected: "!"
mysqltest: At line 1: Invalid integer argument "a"
failing_statement;
ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'failing_statement' at line 1
failing_statement;
ERROR 42000: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'failing_statement' at line 1
SELECT 1 as a;
a
1
select 1 as `a'b`, 2 as `a"b`; select 1 as `a'b`, 2 as `a"b`;
a'b a"b a'b a"b
1 2 1 2
......
...@@ -4,10 +4,10 @@ reset master; ...@@ -4,10 +4,10 @@ reset master;
reset slave; reset slave;
drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9; drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9;
start slave; start slave;
stop slave;
change master to master_host='127.0.0.1',master_user='root', change master to master_host='127.0.0.1',master_user='root',
master_password='',master_port=MASTER_PORT; master_password='',master_port=MASTER_PORT;
start slave; start slave;
stop slave;
change master to master_host='127.0.0.1',master_user='root', change master to master_host='127.0.0.1',master_user='root',
master_password='',master_port=SLAVE_PORT; master_password='',master_port=SLAVE_PORT;
start slave; start slave;
......
...@@ -684,9 +684,7 @@ set v1 = 1; set v2 = 2; set v3 = 1000000000000; set v4 = 2000000000000; set v5 = ...@@ -684,9 +684,7 @@ set v1 = 1; set v2 = 2; set v3 = 1000000000000; set v4 = 2000000000000; set v5 =
while v5 < 100000 do while v5 < 100000 do
set v1 = v1 + 0.000000000001; set v2 = v2 - 0.000000000001; set v3 = v3 + 1; set v4 = v4 - 1; set v5 = v5 + 1; set v1 = v1 + 0.000000000001; set v2 = v2 - 0.000000000001; set v3 = v3 + 1; set v4 = v4 - 1; set v5 = v5 + 1;
end while; select v1, v2, v3 * 0.000000000001, v4 * 0.000000000001; end;// end while; select v1, v2, v3 * 0.000000000001, v4 * 0.000000000001; end;//
#
call p1()// call p1()//
#
v1 v2 v3 * 0.000000000001 v4 * 0.000000000001 v1 v2 v3 * 0.000000000001 v4 * 0.000000000001
1.000000100000 1.999999900000 1.000000100000 1.999999900000 1.000000100000 1.999999900000 1.000000100000 1.999999900000
drop procedure p1; drop procedure p1;
......
...@@ -249,7 +249,7 @@ drop table t1; ...@@ -249,7 +249,7 @@ drop table t1;
create table `t1 `(a int); create table `t1 `(a int);
--error 1102 --error 1102
create database `db1 `; create database `db1 `;
--error 1166; --error 1166
create table t1(`a ` int); create table t1(`a ` int);
# #
......
...@@ -11,21 +11,21 @@ drop table if exists t1, t2; ...@@ -11,21 +11,21 @@ drop table if exists t1, t2;
--enable_warnings --enable_warnings
CREATE TABLE t1 ( a int ); CREATE TABLE t1 ( a int );
INSERT INTO t1 VALUES (1),(2),(1); INSERT INTO t1 VALUES (1),(2),(1);
--error 1062; --error 1062
CREATE TABLE t2 ( PRIMARY KEY (a) ) ENGINE=INNODB SELECT a FROM t1; CREATE TABLE t2 ( PRIMARY KEY (a) ) ENGINE=INNODB SELECT a FROM t1;
--error 1146; --error 1146
select * from t2; select * from t2;
--error 1062; --error 1062
CREATE TEMPORARY TABLE t2 ( PRIMARY KEY (a) ) ENGINE=INNODB SELECT a FROM t1; CREATE TEMPORARY TABLE t2 ( PRIMARY KEY (a) ) ENGINE=INNODB SELECT a FROM t1;
--error 1146; --error 1146
select * from t2; select * from t2;
--error 1062; --error 1062
CREATE TABLE t2 ( PRIMARY KEY (a) ) ENGINE=MYISAM SELECT a FROM t1; CREATE TABLE t2 ( PRIMARY KEY (a) ) ENGINE=MYISAM SELECT a FROM t1;
--error 1146; --error 1146
select * from t2; select * from t2;
--error 1062; --error 1062
CREATE TEMPORARY TABLE t2 ( PRIMARY KEY (a) ) ENGINE=MYISAM SELECT a FROM t1; CREATE TEMPORARY TABLE t2 ( PRIMARY KEY (a) ) ENGINE=MYISAM SELECT a FROM t1;
--error 1146; --error 1146
select * from t2; select * from t2;
# End of 4.1 tests # End of 4.1 tests
...@@ -6,13 +6,13 @@ drop database if exists mysqltest; ...@@ -6,13 +6,13 @@ drop database if exists mysqltest;
drop database if exists client_test_db; drop database if exists client_test_db;
--enable_warnings --enable_warnings
--error 1051; --error 1051
drop table t1; drop table t1;
create table t1(n int); create table t1(n int);
insert into t1 values(1); insert into t1 values(1);
create temporary table t1( n int); create temporary table t1( n int);
insert into t1 values(2); insert into t1 values(2);
--error 1050; --error 1050
create table t1(n int); create table t1(n int);
drop table t1; drop table t1;
select * from t1; select * from t1;
...@@ -56,13 +56,13 @@ drop database mysqltest; ...@@ -56,13 +56,13 @@ drop database mysqltest;
# test drop/create database and FLUSH TABLES WITH READ LOCK # test drop/create database and FLUSH TABLES WITH READ LOCK
flush tables with read lock; flush tables with read lock;
--error 1209,1223; --error 1209,1223
create database mysqltest; create database mysqltest;
unlock tables; unlock tables;
create database mysqltest; create database mysqltest;
show databases; show databases;
flush tables with read lock; flush tables with read lock;
--error 1208,1223; --error 1208,1223
drop database mysqltest; drop database mysqltest;
unlock tables; unlock tables;
drop database mysqltest; drop database mysqltest;
...@@ -73,7 +73,7 @@ drop database mysqltest; ...@@ -73,7 +73,7 @@ drop database mysqltest;
# test create table and FLUSH TABLES WITH READ LOCK # test create table and FLUSH TABLES WITH READ LOCK
drop table t1; drop table t1;
flush tables with read lock; flush tables with read lock;
--error 1223; --error 1223
create table t1(n int); create table t1(n int);
unlock tables; unlock tables;
create table t1(n int); create table t1(n int);
......
...@@ -34,7 +34,7 @@ send flush tables with read lock; ...@@ -34,7 +34,7 @@ send flush tables with read lock;
connection con2; connection con2;
select ((@id := kill_id) - kill_id) from t1; select ((@id := kill_id) - kill_id) from t1;
--sleep 2; # leave time for FLUSH to block --sleep 2 # leave time for FLUSH to block
kill connection @id; kill connection @id;
connection con1; connection con1;
......
...@@ -300,7 +300,7 @@ handler t5 open as h5; ...@@ -300,7 +300,7 @@ handler t5 open as h5;
handler h5 read first limit 9; handler h5 read first limit 9;
# close first # close first
alter table t1 engine=MyISAM; alter table t1 engine=MyISAM;
--error 1109; --error 1109
handler h1 read first limit 9; handler h1 read first limit 9;
handler h2 read first limit 9; handler h2 read first limit 9;
handler h3 read first limit 9; handler h3 read first limit 9;
...@@ -308,22 +308,22 @@ handler h4 read first limit 9; ...@@ -308,22 +308,22 @@ handler h4 read first limit 9;
handler h5 read first limit 9; handler h5 read first limit 9;
# close last # close last
alter table t5 engine=MyISAM; alter table t5 engine=MyISAM;
--error 1109; --error 1109
handler h1 read first limit 9; handler h1 read first limit 9;
handler h2 read first limit 9; handler h2 read first limit 9;
handler h3 read first limit 9; handler h3 read first limit 9;
handler h4 read first limit 9; handler h4 read first limit 9;
--error 1109; --error 1109
handler h5 read first limit 9; handler h5 read first limit 9;
# close middle # close middle
alter table t3 engine=MyISAM; alter table t3 engine=MyISAM;
--error 1109; --error 1109
handler h1 read first limit 9; handler h1 read first limit 9;
handler h2 read first limit 9; handler h2 read first limit 9;
--error 1109; --error 1109
handler h3 read first limit 9; handler h3 read first limit 9;
handler h4 read first limit 9; handler h4 read first limit 9;
--error 1109; --error 1109
handler h5 read first limit 9; handler h5 read first limit 9;
handler h2 close; handler h2 close;
handler h4 close; handler h4 close;
...@@ -335,11 +335,11 @@ handler h1_1 read first limit 9; ...@@ -335,11 +335,11 @@ handler h1_1 read first limit 9;
handler h1_2 read first limit 9; handler h1_2 read first limit 9;
handler h1_3 read first limit 9; handler h1_3 read first limit 9;
alter table t1 engine=MyISAM; alter table t1 engine=MyISAM;
--error 1109; --error 1109
handler h1_1 read first limit 9; handler h1_1 read first limit 9;
--error 1109; --error 1109
handler h1_2 read first limit 9; handler h1_2 read first limit 9;
--error 1109; --error 1109
handler h1_3 read first limit 9; handler h1_3 read first limit 9;
drop table t1; drop table t1;
drop table t2; drop table t2;
......
...@@ -25,7 +25,7 @@ set autocommit=0; ...@@ -25,7 +25,7 @@ set autocommit=0;
# The following query should hang because con1 is locking the page # The following query should hang because con1 is locking the page
--send --send
update t1 set x=2 where id = 0; update t1 set x=2 where id = 0;
--sleep 2; --sleep 2
connection con1; connection con1;
update t1 set x=1 where id = 0; update t1 set x=1 where id = 0;
...@@ -63,7 +63,7 @@ set autocommit=0; ...@@ -63,7 +63,7 @@ set autocommit=0;
# The following query should hang because con1 is locking the page # The following query should hang because con1 is locking the page
--send --send
update t1 set x=2 where id = 0; update t1 set x=2 where id = 0;
--sleep 2; --sleep 2
connection con1; connection con1;
update t1 set x=1 where id = 0; update t1 set x=1 where id = 0;
...@@ -97,7 +97,7 @@ update t2 set a=2 where b = 0; ...@@ -97,7 +97,7 @@ update t2 set a=2 where b = 0;
select * from t2; select * from t2;
--send --send
update t1 set x=2 where id = 0; update t1 set x=2 where id = 0;
--sleep 2; --sleep 2
connection con1; connection con1;
update t1 set x=1 where id = 0; update t1 set x=1 where id = 0;
......
...@@ -39,7 +39,7 @@ set autocommit=0; ...@@ -39,7 +39,7 @@ set autocommit=0;
# The following statement should hang because con1 is locking the page # The following statement should hang because con1 is locking the page
--send --send
lock table t1 write; lock table t1 write;
--sleep 2; --sleep 2
connection con1; connection con1;
update t1 set x=1 where id = 0; update t1 set x=1 where id = 0;
......
...@@ -1212,7 +1212,7 @@ drop table t1; ...@@ -1212,7 +1212,7 @@ drop table t1;
# #
CREATE TABLE t1 ( a char(10) ) ENGINE=InnoDB; CREATE TABLE t1 ( a char(10) ) ENGINE=InnoDB;
--error 1214; --error 1214
SELECT a FROM t1 WHERE MATCH (a) AGAINST ('test' IN BOOLEAN MODE); SELECT a FROM t1 WHERE MATCH (a) AGAINST ('test' IN BOOLEAN MODE);
DROP TABLE t1; DROP TABLE t1;
......
...@@ -49,7 +49,7 @@ select get_lock("a", 10); ...@@ -49,7 +49,7 @@ select get_lock("a", 10);
connection con2; connection con2;
let $ID= `select connection_id()`; let $ID= `select connection_id()`;
send select get_lock("a", 10); send select get_lock("a", 10);
--real_sleep 2; real_sleep 2;
connection con1; connection con1;
disable_query_log; disable_query_log;
eval kill query $ID; eval kill query $ID;
......
...@@ -3,6 +3,24 @@ ...@@ -3,6 +3,24 @@
# #
# Test of mysqltest itself # Test of mysqltest itself
# #
# There are three rules that determines what belong to each command
# 1. A normal command is delimited by the <delimiter> which by default is
# set to ';'
#
# ex: | select *
# | from t1;
# |
# Command: "select * from t1"
#
# 2. Special case is a line that starts with "--", this is a comment
# ended when the new line character is reached. But the first word
# in the comment may contain a valid command, which then will be
# executed. This can be useful when sending commands that
# contains <delimiter>
#
# 3. Special case is also a line that starts with '#' which is treated
# as a comment and will be ended by new line character
#
# ============================================================================ # ============================================================================
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------
...@@ -37,7 +55,9 @@ select otto from (select 1 as otto) as t1; ...@@ -37,7 +55,9 @@ select otto from (select 1 as otto) as t1;
# expectation <> response # expectation <> response
#--error 0 #--error 0
#select friedrich from (select 1 as otto) as t1; #select friedrich from (select 1 as otto) as t1
--error 1
--exec echo "select friedrich from (select 1 as otto) as t1;" | $MYSQL_TEST 2>&1
# expectation = response # expectation = response
--error 1054 --error 1054
...@@ -55,8 +75,9 @@ select friedrich from (select 1 as otto) as t1; ...@@ -55,8 +75,9 @@ select friedrich from (select 1 as otto) as t1;
# Positive case(statement) # Positive case(statement)
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------
# This syntax not allowed anymore, use --error S00000, see below
# expectation = response # expectation = response
!S00000 select otto from (select 1 as otto) as t1; #!S00000 select otto from (select 1 as otto) as t1;
--error S00000 --error S00000
select otto from (select 1 as otto) as t1; select otto from (select 1 as otto) as t1;
...@@ -65,14 +86,18 @@ select otto from (select 1 as otto) as t1; ...@@ -65,14 +86,18 @@ select otto from (select 1 as otto) as t1;
#!S42S22 select otto from (select 1 as otto) as t1; #!S42S22 select otto from (select 1 as otto) as t1;
#--error S42S22 #--error S42S22
#select otto from (select 1 as otto) as t1; #select otto from (select 1 as otto) as t1;
--error 1
--exec echo "error S42S22; select otto from (select 1 as otto) as t1;" | $MYSQL_TEST 2>&1
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------
# Negative case(statement) # Negative case(statement)
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------
# This syntax not allowed anymore, use --error S42S22, see below
# expectation = response # expectation = response
!S42S22 select friedrich from (select 1 as otto) as t1; #!S42S22 select friedrich from (select 1 as otto) as t1;
--error S42S22 --error S42S22
select friedrich from (select 1 as otto) as t1; select friedrich from (select 1 as otto) as t1;
...@@ -80,7 +105,8 @@ select friedrich from (select 1 as otto) as t1; ...@@ -80,7 +105,8 @@ select friedrich from (select 1 as otto) as t1;
#!S00000 select friedrich from (select 1 as otto) as t1; #!S00000 select friedrich from (select 1 as otto) as t1;
#--error S00000 #--error S00000
#select friedrich from (select 1 as otto) as t1; #select friedrich from (select 1 as otto) as t1;
--error 1
--exec echo "error S00000; select friedrich from (select 1 as otto) as t1;" | $MYSQL_TEST 2>&1
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------
# test cases for $mysql_errno # test cases for $mysql_errno
...@@ -262,6 +288,8 @@ eval select $mysql_errno as "after_!errno_masked_error" ; ...@@ -262,6 +288,8 @@ eval select $mysql_errno as "after_!errno_masked_error" ;
# select 3 from t1 ; # select 3 from t1 ;
# --error 1000 # --error 1000
# select 3 from t1 ; # select 3 from t1 ;
--error 1
--exec echo "disable_abort_on_error; error 1000; select 3 from t1; error 1000; select 3 from t1;" | $MYSQL_TEST 2>&1
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------
# Switch the abort on error on and check the effect on $mysql_errno # Switch the abort on error on and check the effect on $mysql_errno
...@@ -288,6 +316,501 @@ select 3 from t1 ; ...@@ -288,6 +316,501 @@ select 3 from t1 ;
#select 3 from t1 ; #select 3 from t1 ;
# End of 4.1 tests # End of 4.1 tests
--error 1
--exec echo "disable_abort_on_error; enable_abort_on_error; error 1064; select 3 from t1; select 3 from t1;" | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test comments
# ----------------------------------------------------------------------------
# This is a comment
# This is a ; comment
# This is a -- comment
-- This is also a comment
-- # This is also a comment
-- This is also a ; comment
# ----------------------------------------------------------------------------
# Test comments with embedded command
# ----------------------------------------------------------------------------
--echo hello
-- echo hello
-- echo ;;;;;;;;
--echo # MySQL: -- The
# ----------------------------------------------------------------------------
# Test detect end of line "junk"
# Most likely causes by a missing delimiter
# ----------------------------------------------------------------------------
# Too many parameters to function
--error 1
--exec echo "sleep 5 6;" | $MYSQL_TEST 2>&1
# Too many parameters to function
--error 1
--exec echo "--sleep 5 6" | $MYSQL_TEST 2>&1
#
# Missing delimiter
# The comment will be "sucked into" the sleep command since
# delimiter is missing until after "show status"
--error 1
--exec echo -e "sleep 4\n # A comment\nshow status;" | $MYSQL_TEST 2>&1
#
# Extra delimiter
#
--error 1
--exec echo "--sleep 4;" | $MYSQL_TEST 2>&1
# Allow trailing # comment
--sleep 1 # Wait for insert delayed to be executed.
--sleep 1 # Wait for insert delayed to be executed.
# ----------------------------------------------------------------------------
# Test echo command
# ----------------------------------------------------------------------------
echo MySQL;
echo "MySQL";
echo MySQL: The world''s most popular open source database;
echo "MySQL: The world's most popular open source database";
echo MySQL: The world''s
most popular open
source database;
echo # MySQL: The world''s
# most popular open
# source database;
echo - MySQL: The world''s
- most popular open
- source database;
echo - MySQL: The world''s
-- most popular open
-- source database;
echo # MySQL: The
--world''s
# most popular
-- open
- source database;
echo "MySQL: The world's most popular; open source database";
echo "MySQL: The world's most popular ; open source database";
echo "MySQL: The world's most popular ;open source database";
echo echo message echo message;
echo ;
# Illegal use of echo
--error 1
--exec echo "echo $;" | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test exec command
# ----------------------------------------------------------------------------
# Illegal use of exec
--error 1
--exec echo "--exec ;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "--exec " | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test let command
# ----------------------------------------------------------------------------
let $message=MySQL;
echo $message;
let $message="MySQL";
echo $message;
let $message= MySQL: The
world''s most
popular open
source database;
echo $message;
let $message= # MySQL: The
# world''s most
# popular open
# source database;
echo $message;
let $message= -- MySQL: The
-- world''s most
-- popular open
-- source database;
echo $message;
let $message= # MySQL: The
- world''s most
-- popular open
# source database;
echo $message;
echo '$message';
echo "$message";
let $1=hej;
echo $1;
let $1 =hej ;
echo $1;
let $1 = hej;
echo $1;
let $1=1;
let $2=$1;
echo $2;
let $5=$6;
echo $5;
echo $6;
let $where=a long variable content;
echo $where;
let $where2= $where;
echo $where2;
let $where3=a long $where variable content;
echo $where3;
let $novar1= $novar2;
echo $novar1;
# Test illegal uses of let
--error 1
--exec echo "let ;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let $=hi;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let hi=hi;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let $1 hi;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let $m hi;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let $hi;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let $ hi;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let =hi;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let hi;" | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test source command
# ----------------------------------------------------------------------------
# Test illegal uses of source
--error 1
--exec echo "source ;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "source non_existingFile;" | $MYSQL_TEST 2>&1
# Too many source
--exec echo "source var/tmp/recursive.sql;" > var/tmp/recursive.sql
--error 1
--exec echo "source var/tmp/recursive.sql;" | $MYSQL_TEST 2>&1
# Source a file with error
--exec echo "garbage ;" > var/tmp/error.sql
--error 1
--exec echo "source var/tmp/error.sql;" | $MYSQL_TEST 2>&1
# Test execution of source in a while loop
--exec echo "echo here is the sourced script;" > var/tmp/sourced.sql
--disable_query_log
let $outer= 2; # Number of outer loops
while ($outer)
{
eval SELECT '$outer = outer loop variable after while' AS "";
--source var/tmp/sourced.sql
eval SELECT '$outer = outer loop variable before dec' AS "";
dec $outer;
eval SELECT '$outer = outer loop variable after dec' AS "";
}
let $outer= 2; # Number of outer loops
while ($outer)
{
eval SELECT '$outer = outer loop variable after while' AS "";
echo here is the sourced script;
eval SELECT '$outer = outer loop variable before dec' AS "";
dec $outer;
eval SELECT '$outer = outer loop variable after dec' AS "";
}
# Test execution of source in a while loop
--exec echo "--source var/tmp/sourced.sql" > var/tmp/sourced1.sql
--disable_abort_on_error
# Sourcing of a file within while loop, sourced file will
# source other file
let $num= 9;
while ($num)
{
SELECT 'In loop' AS "";
--source var/tmp/sourced1.sql
dec $num;
}
--enable_abort_on_error;
--enable_query_log
# ----------------------------------------------------------------------------
# Test sleep command
# ----------------------------------------------------------------------------
sleep 0.5;
sleep 1;
real_sleep 1;
# Missing parameter
--error 1
--exec echo "sleep ;" | $MYSQL_TEST 2>&1
# Illegal parameter
--error 1
--exec echo "sleep abc;" | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test inc
# ----------------------------------------------------------------------------
inc $i;
echo $i;
inc $i;
echo $i;
let $i=100;
inc $i;
echo $i;
let $i=hej;
echo $i;
inc $i;
echo $i;
--error 1
--exec echo "inc;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "inc i;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let \$i=100; inc \$i 1000; echo \$i;" | $MYSQL_TEST 2>&1
inc $i; inc $i; inc $i; --echo $i
echo $i;
# ----------------------------------------------------------------------------
# Test dec
# ----------------------------------------------------------------------------
dec $d;
echo $d;
dec $d;
echo $d;
let $d=100;
dec $d;
echo $d;
let $d=hej;
echo $d;
dec $d;
echo $d;
--error 1
--exec echo "dec;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "dec i;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let \$i=100; dec \$i 1000; echo \$i;" | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test system
# ----------------------------------------------------------------------------
system ls > /dev/null;
system echo "hej" > /dev/null;
--system ls > /dev/null
--system echo "hej" > /dev/null;
--error 1
--exec echo "system;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "system $NONEXISTSINFVAREABLI;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "system NonExistsinfComamdn;" | $MYSQL_TEST 2>&1
--disable_abort_on_error
system NonExistsinfComamdn;
--enable_abort_on_error
# ----------------------------------------------------------------------------
# Test delimiter
# ----------------------------------------------------------------------------
delimiter stop;
echo teststop
delimiter ;stop
echo test2;
--delimiter stop
echo test3stop
--delimiter ;
echo test4;
# ----------------------------------------------------------------------------
# Test while, { and }
# ----------------------------------------------------------------------------
let $i=1;
while ($i)
{
echo $i;
dec $i;
}
# One liner
#let $i=1;while ($i){echo $i;dec $i;}
# Exceed max nesting level
--error 1
--exec echo "source include/mysqltest_while.inc;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "while \$i;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "while (\$i;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "let \$i=1; while (\$i) dec \$i;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "};" | $MYSQL_TEST 2>&1
--error 1
--exec echo "end;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "{;" | $MYSQL_TEST 2>&1
--error 1
--exec echo -e "while (0)\necho hej;" | $MYSQL_TEST 2>&1
--error 1
--exec echo -e "while (0)\n{echo hej;" | $MYSQL_TEST 2>&1
--error 1
--exec echo -e "while (0){\n echo hej;" | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test error messages returned from comments starting with a command
# ----------------------------------------------------------------------------
--error 1
--exec echo "--if the other server is down" | $MYSQL_TEST 2>&1
--error 1
--exec echo "-- end when ..." | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test replace
# ----------------------------------------------------------------------------
--replace_result a b
select "a" as col1, "c" as col2;
--replace_result a b c d
select "a" as col1, "c" as col2;
--error 1
--exec echo "--replace_result a" | $MYSQL_TEST 2>&1
--error 1
--exec echo "--replace_result a;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "replace_result a;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "replace_result a ;" | $MYSQL_TEST 2>&1
--exec echo "replace_result a b;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "--replace_result a b c" | $MYSQL_TEST 2>&1
--error 1
--exec echo "replace_result a b c ;" | $MYSQL_TEST 2>&1
--replace_column 1 b
select "a" as col1, "c" as col2;
--replace_column 1 b 2 d
select "a" as col1, "c" as col2;
--error 1
--exec echo "--replace_column a" | $MYSQL_TEST 2>&1
--error 1
--exec echo "--replace_column 1" | $MYSQL_TEST 2>&1
--error 1
--exec echo "--replace_column a b" | $MYSQL_TEST 2>&1
--error 1
--exec echo "--replace_column a 1" | $MYSQL_TEST 2>&1
--error 1
--exec echo "--replace_column 1 b c " | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# Test sync_with_master
# ----------------------------------------------------------------------------
--error 1
--exec echo "save_master_pos; sync_with_master 10!;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "save_master_pos; sync_with_master 10 !;" | $MYSQL_TEST 2>&1
--error 1
--exec echo "save_master_pos; sync_with_master a;" | $MYSQL_TEST 2>&1
# ----------------------------------------------------------------------------
# TODO Test queries, especially their errormessages... so it's easy to debug
# new scripts and diagnose errors
# ----------------------------------------------------------------------------
# ----------------------------------------------------------------------------
# Test bug#12386
# ----------------------------------------------------------------------------
let $num= 2;
while ($num)
{
--error 1064
failing_statement;
dec $num;
}
SELECT 1 as a;
# #
# Bug #10251: Identifiers containing quotes not handled correctly # Bug #10251: Identifiers containing quotes not handled correctly
......
...@@ -6,7 +6,7 @@ ...@@ -6,7 +6,7 @@
# The previous step has simply removed the frm file # The previous step has simply removed the frm file
# from disk, but left the table in NDB # from disk, but left the table in NDB
# #
--sleep 3; --sleep 3
select * from t9 order by a; select * from t9 order by a;
# handler_discover should be 1 # handler_discover should be 1
......
...@@ -92,7 +92,7 @@ kill @id; ...@@ -92,7 +92,7 @@ kill @id;
# We don't drop t3 as this is a temporary table # We don't drop t3 as this is a temporary table
drop table t2; drop table t2;
connection master; connection master;
--error 1053; --error 1053
reap; reap;
connection slave; connection slave;
# The SQL slave thread should now have stopped because the query was killed on # The SQL slave thread should now have stopped because the query was killed on
......
...@@ -22,7 +22,7 @@ set sql_log_bin=0; ...@@ -22,7 +22,7 @@ set sql_log_bin=0;
insert into t1 values(2); insert into t1 values(2);
set sql_log_bin=1; set sql_log_bin=1;
save_master_pos; save_master_pos;
--error 1062; --error 1062
insert into t1 values(1),(2); insert into t1 values(1),(2);
drop table t1; drop table t1;
save_master_pos; save_master_pos;
......
...@@ -16,7 +16,7 @@ insert into t1 values(1); ...@@ -16,7 +16,7 @@ insert into t1 values(1);
insert into t1 values(2); insert into t1 values(2);
save_master_pos; save_master_pos;
connection slave; connection slave;
--real_sleep 3; # wait for I/O thread to have read updates --real_sleep 3 # wait for I/O thread to have read updates
stop slave; stop slave;
--replace_result $MASTER_MYPORT MASTER_MYPORT --replace_result $MASTER_MYPORT MASTER_MYPORT
--replace_column 1 # 8 # 9 # 23 # 33 # --replace_column 1 # 8 # 9 # 23 # 33 #
......
...@@ -58,7 +58,7 @@ while ($1) ...@@ -58,7 +58,7 @@ while ($1)
enable_query_log; enable_query_log;
select * from t1 for update; select * from t1 for update;
start slave; start slave;
--sleep 3; # hope that slave is blocked now --sleep 3 # hope that slave is blocked now
insert into t2 values(22); # provoke deadlock, slave should be victim insert into t2 values(22); # provoke deadlock, slave should be victim
commit; commit;
sync_with_master; sync_with_master;
...@@ -76,7 +76,7 @@ change master to master_log_pos=532; # the BEGIN log event ...@@ -76,7 +76,7 @@ change master to master_log_pos=532; # the BEGIN log event
begin; begin;
select * from t2 for update; # hold lock select * from t2 for update; # hold lock
start slave; start slave;
--sleep 10; # slave should have blocked, and be retrying --sleep 10 # slave should have blocked, and be retrying
commit; commit;
sync_with_master; sync_with_master;
select * from t1; # check that slave succeeded finally select * from t1; # check that slave succeeded finally
...@@ -97,7 +97,7 @@ change master to master_log_pos=532; ...@@ -97,7 +97,7 @@ change master to master_log_pos=532;
begin; begin;
select * from t2 for update; select * from t2 for update;
start slave; start slave;
--sleep 10; --sleep 10
commit; commit;
sync_with_master; sync_with_master;
select * from t1; select * from t1;
......
...@@ -5,7 +5,7 @@ source include/master-slave.inc; ...@@ -5,7 +5,7 @@ source include/master-slave.inc;
drop table if exists t1, t2; drop table if exists t1, t2;
--enable_warnings --enable_warnings
create table t1 (a int); create table t1 (a int);
--error 1051; --error 1051
drop table t1, t2; drop table t1, t2;
save_master_pos; save_master_pos;
connection slave; connection slave;
......
...@@ -9,7 +9,7 @@ sync_slave_with_master; ...@@ -9,7 +9,7 @@ sync_slave_with_master;
connection master; connection master;
disconnect master; disconnect master;
connection slave; connection slave;
--real_sleep 3; # time for DROP to be written --real_sleep 3 # time for DROP to be written
show status like 'Slave_open_temp_tables'; show status like 'Slave_open_temp_tables';
connection default; connection default;
drop database mysqltest; drop database mysqltest;
......
...@@ -6,7 +6,7 @@ source include/master-slave.inc; ...@@ -6,7 +6,7 @@ source include/master-slave.inc;
connection master; connection master;
create table t1 (a int primary key); create table t1 (a int primary key);
# generate an error that goes to the binlog # generate an error that goes to the binlog
--error 1062; --error 1062
insert into t1 values (1),(1); insert into t1 values (1),(1);
save_master_pos; save_master_pos;
connection slave; connection slave;
...@@ -45,7 +45,7 @@ select (@id := id) - id from t3; ...@@ -45,7 +45,7 @@ select (@id := id) - id from t3;
kill @id; kill @id;
drop table t2,t3; drop table t2,t3;
connection master; connection master;
--error 0,1053; --error 0,1053
reap; reap;
connection master1; connection master1;
--replace_column 2 # 5 # --replace_column 2 # 5 #
......
# Testing if "flush logs" command bouncing resulting in logs created in a loop # Testing if "flush logs" command bouncing resulting in logs created in a loop
# in case of bi-directional replication # in case of bi-directional replication
source include/master-slave.inc source include/master-slave.inc;
connection slave; connection slave;
stop slave;
--replace_result $MASTER_MYPORT MASTER_PORT --replace_result $MASTER_MYPORT MASTER_PORT
eval change master to master_host='127.0.0.1',master_user='root', eval change master to master_host='127.0.0.1',master_user='root',
master_password='',master_port=$MASTER_MYPORT; master_password='',master_port=$MASTER_MYPORT;
start slave; start slave;
connection master; connection master;
stop slave;
--replace_result $SLAVE_MYPORT SLAVE_PORT --replace_result $SLAVE_MYPORT SLAVE_PORT
eval change master to master_host='127.0.0.1',master_user='root', eval change master to master_host='127.0.0.1',master_user='root',
master_password='',master_port=$SLAVE_MYPORT; master_password='',master_port=$SLAVE_MYPORT;
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
# We also check how the foreign_key_check variable is replicated # We also check how the foreign_key_check variable is replicated
source include/master-slave.inc; source include/master-slave.inc;
source include/have_innodb.inc source include/have_innodb.inc;
connection master; connection master;
create table t1(a int auto_increment, key(a)); create table t1(a int auto_increment, key(a));
create table t2(b int auto_increment, c int, key(b)); create table t2(b int auto_increment, c int, key(b));
......
...@@ -124,7 +124,7 @@ connection master; ...@@ -124,7 +124,7 @@ connection master;
reset master; reset master;
create table t2 (day date,id int(9),category enum('a','b','c'),name varchar(60), create table t2 (day date,id int(9),category enum('a','b','c'),name varchar(60),
unique(day)) engine=MyISAM; # no transactions unique(day)) engine=MyISAM; # no transactions
--error 1062; --error 1062
load data infile '../../std_data/rpl_loaddata2.dat' into table t2 fields load data infile '../../std_data/rpl_loaddata2.dat' into table t2 fields
terminated by ',' optionally enclosed by '%' escaped by '@' lines terminated by terminated by ',' optionally enclosed by '%' escaped by '@' lines terminated by
'\n##\n' starting by '>' ignore 1 lines; '\n##\n' starting by '>' ignore 1 lines;
...@@ -140,7 +140,7 @@ select * from t2; ...@@ -140,7 +140,7 @@ select * from t2;
alter table t2 drop key day; alter table t2 drop key day;
connection master; connection master;
delete from t2; delete from t2;
--error 1062; --error 1062
load data infile '../../std_data/rpl_loaddata2.dat' into table t2 fields load data infile '../../std_data/rpl_loaddata2.dat' into table t2 fields
terminated by ',' optionally enclosed by '%' escaped by '@' lines terminated by terminated by ',' optionally enclosed by '%' escaped by '@' lines terminated by
'\n##\n' starting by '>' ignore 1 lines; '\n##\n' starting by '>' ignore 1 lines;
......
...@@ -103,7 +103,7 @@ show master logs; ...@@ -103,7 +103,7 @@ show master logs;
purge binary logs to 'master-bin.000002'; purge binary logs to 'master-bin.000002';
show binary logs; show binary logs;
# sleeping 10 seconds or more would make the slave believe connection is down # sleeping 10 seconds or more would make the slave believe connection is down
--real_sleep 1; --real_sleep 1
purge master logs before now(); purge master logs before now();
show binary logs; show binary logs;
insert into t2 values (65); insert into t2 values (65);
......
...@@ -27,7 +27,7 @@ drop function if exists fn1; ...@@ -27,7 +27,7 @@ drop function if exists fn1;
--enable_warnings --enable_warnings
delimiter |; delimiter |;
--error 1418; # not deterministic --error 1418 # not deterministic
create procedure foo() create procedure foo()
begin begin
declare b int; declare b int;
...@@ -85,7 +85,7 @@ call foo2(); ...@@ -85,7 +85,7 @@ call foo2();
--replace_column 2 # 5 # --replace_column 2 # 5 #
show binlog events from 518; show binlog events from 518;
--error 1418; --error 1418
alter procedure foo2 contains sql; alter procedure foo2 contains sql;
# SP with definer's right # SP with definer's right
...@@ -106,7 +106,7 @@ grant SELECT, INSERT on mysqltest1.t2 to "zedjzlcsjhd"@127.0.0.1; ...@@ -106,7 +106,7 @@ grant SELECT, INSERT on mysqltest1.t2 to "zedjzlcsjhd"@127.0.0.1;
connect (con1,127.0.0.1,zedjzlcsjhd,,mysqltest1,$MASTER_MYPORT,); connect (con1,127.0.0.1,zedjzlcsjhd,,mysqltest1,$MASTER_MYPORT,);
connection con1; connection con1;
--error 1419; # only full-global-privs user can create a routine --error 1419 # only full-global-privs user can create a routine
create procedure foo4() create procedure foo4()
deterministic deterministic
insert into t1 values (10); insert into t1 values (10);
...@@ -127,7 +127,7 @@ delimiter ;| ...@@ -127,7 +127,7 @@ delimiter ;|
# I add ,0 so that it does not print the error in the test output, # I add ,0 so that it does not print the error in the test output,
# because this error is hostname-dependent # because this error is hostname-dependent
--error 1142,0; --error 1142,0
call foo4(); # invoker has no INSERT grant on table => failure call foo4(); # invoker has no INSERT grant on table => failure
show warnings; show warnings;
...@@ -136,7 +136,7 @@ call foo3(); # success (definer == root) ...@@ -136,7 +136,7 @@ call foo3(); # success (definer == root)
show warnings; show warnings;
--replace_result localhost.localdomain localhost 127.0.0.1 localhost --replace_result localhost.localdomain localhost 127.0.0.1 localhost
--error 1142,0; --error 1142,0
call foo4(); # definer's rights => failure call foo4(); # definer's rights => failure
show warnings; show warnings;
...@@ -226,7 +226,7 @@ select * from mysql.proc where db='mysqltest1'; ...@@ -226,7 +226,7 @@ select * from mysql.proc where db='mysqltest1';
# And now triggers # And now triggers
connection con1; connection con1;
--error 1227; --error 1227
create trigger trg before insert on t1 for each row set new.a= 10; create trigger trg before insert on t1 for each row set new.a= 10;
connection master; connection master;
......
...@@ -315,7 +315,7 @@ INSERT INTO t1 (col2) VALUES(CAST('0000-00-00' AS DATETIME)); ...@@ -315,7 +315,7 @@ INSERT INTO t1 (col2) VALUES(CAST('0000-00-00' AS DATETIME));
## Test INSERT with CAST AS DATETIME into TIMESTAMP ## Test INSERT with CAST AS DATETIME into TIMESTAMP
# All test cases expected to fail should return # All test cases expected to fail should return
# SQLSTATE 22007 <invalid datetime value> # SQLSTATE 22007 <invalid datetime value>
!$1292 --error 1292
INSERT INTO t1 (col3) VALUES(CAST('0000-10-31 15:30' AS DATETIME)); INSERT INTO t1 (col3) VALUES(CAST('0000-10-31 15:30' AS DATETIME));
-- should return OK -- should return OK
-- We accept this to be a failure -- We accept this to be a failure
...@@ -406,7 +406,7 @@ INSERT INTO t1 (col2) VALUES(CONVERT('0000-00-00',DATETIME)); ...@@ -406,7 +406,7 @@ INSERT INTO t1 (col2) VALUES(CONVERT('0000-00-00',DATETIME));
## Test INSERT with CONVERT to DATETIME into DATETIME ## Test INSERT with CONVERT to DATETIME into DATETIME
# All test cases expected to fail should return # All test cases expected to fail should return
# SQLSTATE 22007 <invalid datetime value> # SQLSTATE 22007 <invalid datetime value>
!$1292 --error 1292
INSERT INTO t1 (col3) VALUES(CONVERT('0000-10-31 15:30',DATETIME)); INSERT INTO t1 (col3) VALUES(CONVERT('0000-10-31 15:30',DATETIME));
-- should return OK -- should return OK
-- We accept this to be a failure -- We accept this to be a failure
......
...@@ -494,7 +494,7 @@ select 0.8 = 0.7 + 0.1; ...@@ -494,7 +494,7 @@ select 0.8 = 0.7 + 0.1;
# #
#drop procedure p1; #drop procedure p1;
# #
delimiter // delimiter //;
# #
create procedure p1 () begin create procedure p1 () begin
declare v1, v2, v3, v4 decimal(16,12); declare v5 int; declare v1, v2, v3, v4 decimal(16,12); declare v5 int;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment