64
60
/* Added this for string translation. */
65
61
#include "drizzled/gettext.h"
66
62
#include "drizzled/hash.h"
67
#include "drizzled/drizzle_time.h"
68
#include "drizzled/charset.h"
70
64
#ifndef DRIZZLE_RETURN_SERVER_GONE
71
65
#define DRIZZLE_RETURN_HANDSHAKE_FAILED DRIZZLE_RETURN_ERROR_CODE
74
68
using namespace std;
75
using namespace drizzled;
78
unsigned char *get_var_key(const unsigned char* var, size_t *len, bool);
80
int get_one_option(int optid, const struct option *, char *argument);
72
unsigned char *get_var_key(const unsigned char* var, size_t *len, bool);
73
bool get_one_option(int optid, const struct my_option *, char *argument);
82
76
#define MAX_VAR_NAME_LENGTH 256
83
77
#define MAX_COLUMNS 256
78
#define MAX_EMBEDDED_SERVER_ARGS 64
84
79
#define MAX_DELIMITER_LENGTH 16
85
80
/* Flags controlling send and reap */
86
81
#define QUERY_SEND_FLAG 1
103
98
const char *opt_testdir= NULL;
104
99
static uint32_t opt_port= 0;
105
100
static int opt_max_connect_retries;
106
static bool silent= false, verbose= false;
101
static bool opt_compress= false, silent= false, verbose= false;
102
static bool debug_info_flag= false, debug_check_flag= false;
107
103
static bool tty_password= false;
108
104
static bool opt_mark_progress= false;
109
105
static bool parsing_disabled= false;
115
111
static bool abort_on_error= true;
116
112
static bool server_initialized= false;
117
113
static bool is_windows= false;
118
static bool opt_mysql= false;
119
114
static char **default_argv;
120
115
static const char *load_default_groups[]= { "drizzletest", "client", 0 };
121
116
static char line_buffer[MAX_DELIMITER_LENGTH], *line_buffer_pos= line_buffer;
123
118
static uint32_t start_lineno= 0; /* Start line of current command */
119
static uint32_t my_end_arg= 0;
125
121
/* Number of lines of the result to include in failure report */
126
122
static uint32_t opt_tail_lines= 0;
1143
1144
char buff[512];
1145
if ((fd= internal::my_open(filename, O_RDONLY, MYF(0))) < 0)
1146
if ((fd= my_open(filename, O_RDONLY, MYF(0))) < 0)
1146
1147
die("Failed to open file '%s'", filename);
1147
while((len= internal::my_read(fd, (unsigned char*)&buff,
1148
while((len= my_read(fd, (unsigned char*)&buff,
1148
1149
sizeof(buff), MYF(0))) > 0)
1150
1151
char *p= buff, *start= buff;
1345
static int compare_files2(int fd, const char* filename2)
1346
static int compare_files2(File fd, const char* filename2)
1347
1348
int error= RESULT_OK;
1349
1350
uint32_t len, len2;
1350
1351
char buff[512], buff2[512];
1351
1352
const char *fname= filename2;
1352
1353
string tmpfile;
1354
if ((fd2= internal::my_open(fname, O_RDONLY, MYF(0))) < 0)
1355
if ((fd2= my_open(fname, O_RDONLY, MYF(0))) < 0)
1356
internal::my_close(fd, MYF(0));
1357
my_close(fd, MYF(0));
1357
1358
if (opt_testdir != NULL)
1359
1360
tmpfile= opt_testdir;
1362
1363
tmpfile.append(filename2);
1363
1364
fname= tmpfile.c_str();
1365
if ((fd2= internal::my_open(fname, O_RDONLY, MYF(0))) < 0)
1366
if ((fd2= my_open(fname, O_RDONLY, MYF(0))) < 0)
1367
internal::my_close(fd, MYF(0));
1368
my_close(fd, MYF(0));
1369
1370
die("Failed to open second file: '%s'", fname);
1372
while((len= internal::my_read(fd, (unsigned char*)&buff,
1373
while((len= my_read(fd, (unsigned char*)&buff,
1373
1374
sizeof(buff), MYF(0))) > 0)
1375
if ((len2= internal::my_read(fd2, (unsigned char*)&buff2,
1376
if ((len2= my_read(fd2, (unsigned char*)&buff2,
1376
1377
sizeof(buff2), MYF(0))) < len)
1378
1379
/* File 2 was smaller */
1395
if (!error && internal::my_read(fd2, (unsigned char*)&buff2,
1396
if (!error && my_read(fd2, (unsigned char*)&buff2,
1396
1397
sizeof(buff2), MYF(0)) > 0)
1398
1399
/* File 1 was smaller */
1399
1400
error= RESULT_LENGTH_MISMATCH;
1402
internal::my_close(fd2, MYF(0));
1403
my_close(fd2, MYF(0));
1421
1422
static int compare_files(const char* filename1, const char* filename2)
1426
if ((fd= internal::my_open(filename1, O_RDONLY, MYF(0))) < 0)
1427
if ((fd= my_open(filename1, O_RDONLY, MYF(0))) < 0)
1427
1428
die("Failed to open first file: '%s'", filename1);
1429
1430
error= compare_files2(fd, filename2);
1431
internal::my_close(fd, MYF(0));
1432
my_close(fd, MYF(0));
1449
1450
static int string_cmp(string* ds, const char *fname)
1453
1454
char temp_file_path[FN_REFLEN];
1455
if ((fd= internal::create_temp_file(temp_file_path, NULL,
1456
if ((fd= create_temp_file(temp_file_path, NULL,
1456
1457
"tmp", MYF(MY_WME))) < 0)
1457
1458
die("Failed to create temporary file for ds");
1459
1460
/* Write ds to temporary file and set file pos to beginning*/
1460
if (internal::my_write(fd, (unsigned char *) ds->c_str(), ds->length(),
1461
if (my_write(fd, (unsigned char *) ds->c_str(), ds->length(),
1461
1462
MYF(MY_FNABP | MY_WME)) ||
1462
1463
lseek(fd, 0, SEEK_SET) == MY_FILEPOS_ERROR)
1464
internal::my_close(fd, MYF(0));
1465
my_close(fd, MYF(0));
1465
1466
/* Remove the temporary file */
1466
internal::my_delete(temp_file_path, MYF(0));
1467
my_delete(temp_file_path, MYF(0));
1467
1468
die("Failed to write file '%s'", temp_file_path);
1470
1471
error= compare_files2(fd, fname);
1472
internal::my_close(fd, MYF(0));
1473
my_close(fd, MYF(0));
1473
1474
/* Remove the temporary file */
1474
internal::my_delete(temp_file_path, MYF(0));
1475
my_delete(temp_file_path, MYF(0));
1514
1515
char reject_file[FN_REFLEN];
1515
1516
size_t reject_length;
1516
internal::dirname_part(reject_file, result_file_name, &reject_length);
1517
dirname_part(reject_file, result_file_name, &reject_length);
1518
1519
if (access(reject_file, W_OK) == 0)
1520
1521
/* Result file directory is writable, save reject file there */
1521
internal::fn_format(reject_file, result_file_name, NULL,
1522
fn_format(reject_file, result_file_name, NULL,
1522
1523
".reject", MY_REPLACE_EXT);
1526
1527
/* Put reject file in opt_logdir */
1527
internal::fn_format(reject_file, result_file_name, opt_logdir,
1528
fn_format(reject_file, result_file_name, opt_logdir,
1528
1529
".reject", MY_REPLACE_DIR | MY_REPLACE_EXT);
1530
1531
str_to_file(reject_file, ds->c_str(), ds->length());
1565
1566
if (string_cmp(ds, fname.c_str()))
1567
1568
char reason[FN_REFLEN];
1568
internal::fn_format(reason, fname.c_str(), "", "", MY_REPLACE_EXT | MY_REPLACE_DIR);
1569
fn_format(reason, fname.c_str(), "", "", MY_REPLACE_EXT | MY_REPLACE_DIR);
1569
1570
abort_not_supported_test("Test requires: '%s'", reason);
2146
2147
char buff[FN_REFLEN];
2148
if (!internal::test_if_hard_path(name))
2149
if (!test_if_hard_path(name))
2150
snprintf(buff, sizeof(buff), "%s%s",opt_basedir,name);
2151
sprintf(buff,"%s%s",opt_basedir,name);
2153
internal::fn_format(buff, name, "", "", MY_UNPACK_FILENAME);
2154
fn_format(buff, name, "", "", MY_UNPACK_FILENAME);
2155
2156
if (cur_file == file_stack_end)
2156
2157
die("Source directives are nesting too deep");
2503
2504
sizeof(copy_file_args)/sizeof(struct command_arg),
2506
error= (internal::my_copy(ds_from_file.c_str(), ds_to_file.c_str(),
2507
error= (my_copy(ds_from_file.c_str(), ds_to_file.c_str(),
2507
2508
MYF(MY_DONT_OVERWRITE_FILE)) != 0);
2508
2509
handle_command_error(command, error);
2992
2993
read_until_delimiter(&ds_script, &ds_delimiter);
2994
2995
/* Create temporary file name */
2995
if ((fd= internal::create_temp_file(temp_file_path, getenv("MYSQLTEST_VARDIR"),
2996
if ((fd= create_temp_file(temp_file_path, getenv("MYSQLTEST_VARDIR"),
2996
2997
"tmp", MYF(MY_WME))) < 0)
2997
2998
die("Failed to create temporary file for perl command");
2998
internal::my_close(fd, MYF(0));
2999
my_close(fd, MYF(0));
3000
3001
str_to_file(temp_file_path, ds_script.c_str(), ds_script.length());
3121
3122
if (!master_pos.file[0])
3122
3123
die("Calling 'sync_with_master' without calling 'save_master_pos'");
3124
snprintf(query_buf, sizeof(query_buf), "select master_pos_wait('%s', %ld)", master_pos.file,
3125
sprintf(query_buf, "select master_pos_wait('%s', %ld)", master_pos.file,
3125
3126
master_pos.pos + offset);
3127
3128
wait_for_position:
3316
3317
die("Missing argument to %.*s", command->first_word_len, command->query);
3317
3318
sleep_start= p;
3318
/* Check that arg starts with a digit, not handled by internal::my_strtod */
3319
/* Check that arg starts with a digit, not handled by my_strtod */
3319
3320
if (!my_isdigit(charset_info, *sleep_start))
3320
3321
die("Invalid argument to %.*s \"%s\"", command->first_word_len,
3321
3322
command->query,command->first_argument);
4156
static int my_strnncoll_simple(const CHARSET_INFO * const cs, const unsigned char *s, size_t slen,
4157
const unsigned char *t, size_t tlen,
4160
size_t len = ( slen > tlen ) ? tlen : slen;
4161
unsigned char *map= cs->sort_order;
4162
if (t_is_prefix && slen > tlen)
4166
if (map[*s++] != map[*t++])
4167
return ((int) map[s[-1]] - (int) map[t[-1]]);
4170
We can't use (slen - tlen) here as the result may be outside of the
4171
precision of a signed int
4173
return slen > tlen ? 1 : slen < tlen ? -1 : 0 ;
4176
4154
static int read_line(char *buf, int size)
4178
4156
char c, last_quote= 0;
4587
4565
{"character-sets-dir", OPT_CHARSETS_DIR,
4588
4566
"Directory where character sets are.", (char**) &opt_charsets_dir,
4589
4567
(char**) &opt_charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4568
{"compress", 'C', "Use the compressed server/client protocol.",
4569
(char**) &opt_compress, (char**) &opt_compress, 0, GET_BOOL, NO_ARG, 0, 0, 0,
4590
4571
{"database", 'D', "Database to use.", (char**) &opt_db, (char**) &opt_db, 0,
4591
4572
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4573
{"debug-check", OPT_DEBUG_CHECK, "Check memory and open file usage at exit.",
4574
(char**) &debug_check_flag, (char**) &debug_check_flag, 0,
4575
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4576
{"debug-info", OPT_DEBUG_INFO, "Print some debug info at exit.",
4577
(char**) &debug_info_flag, (char**) &debug_info_flag,
4578
0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4592
4579
{"host", 'h', "Connect to host.", (char**) &opt_host, (char**) &opt_host, 0,
4593
4580
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4594
4581
{"include", 'i', "Include SQL before each test case.", (char**) &opt_include,
4606
4593
"Max number of connection attempts when connecting to server",
4607
4594
(char**) &opt_max_connect_retries, (char**) &opt_max_connect_retries, 0,
4608
4595
GET_INT, REQUIRED_ARG, 500, 1, 10000, 0, 0, 0},
4609
{"mysql", 'm', N_("Use MySQL Protocol."),
4610
(char**) &opt_mysql, (char**) &opt_mysql, 0, GET_BOOL, NO_ARG, 1, 0, 0,
4612
4596
{"password", 'P', "Password to use when connecting to server.",
4613
4597
0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0},
4614
4598
{"port", 'p', "Port number to use for connection or 0 for default to, in "
4622
4606
{"result-file", 'R', "Read/Store result from/in this file.",
4623
4607
(char**) &result_file_name, (char**) &result_file_name, 0,
4624
4608
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4609
{"server-arg", 'A', "Send option value to embedded server as a parameter.",
4610
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4611
{"server-file", 'F', "Read embedded server arguments from file.",
4612
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4625
4613
{"silent", 's', "Suppress all normal output. Synonym for --quiet.",
4626
4614
(char**) &silent, (char**) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4627
4615
{"sleep", 'T', "Sleep always this many seconds on sleep commands.",
4650
4638
static void print_version(void)
4652
printf("%s Ver %s Distrib %s, for %s-%s (%s)\n",internal::my_progname,MTEST_VERSION,
4640
printf("%s Ver %s Distrib %s, for %s-%s (%s)\n",my_progname,MTEST_VERSION,
4653
4641
drizzle_version(),HOST_VENDOR,HOST_OS,HOST_CPU);
4658
4646
print_version();
4659
4647
printf("MySQL AB, by Sasha, Matt, Monty & Jani\n");
4660
printf("Drizzle version modified by Brian, Jay, Monty Taylor, PatG and Stewart\n");
4661
4648
printf("This software comes with ABSOLUTELY NO WARRANTY\n\n");
4662
4649
printf("Runs a test against the DRIZZLE server and compares output with a results file.\n\n");
4663
printf("Usage: %s [OPTIONS] [database] < test_file\n", internal::my_progname);
4650
printf("Usage: %s [OPTIONS] [database] < test_file\n", my_progname);
4664
4651
my_print_help(my_long_options);
4665
4652
printf(" --no-defaults Don't read default options from any options file.\n");
4666
4653
my_print_variables(my_long_options);
4669
int get_one_option(int optid, const struct option *, char *argument)
4657
Read arguments for embedded server and put them into
4658
embedded_server_args[]
4661
static void read_embedded_server_arguments(const char *name)
4663
char argument[1024],buff[FN_REFLEN], *str=0;
4666
if (!test_if_hard_path(name))
4668
sprintf(buff,"%s%s",opt_basedir,name);
4671
fn_format(buff, name, "", "", MY_UNPACK_FILENAME);
4673
if (!embedded_server_arg_count)
4675
embedded_server_arg_count=1;
4676
embedded_server_args[0]= (char*) ""; /* Progname */
4678
if (!(file= fopen(buff, "r")))
4679
die("Failed to open file '%s'", buff);
4681
while (embedded_server_arg_count < MAX_EMBEDDED_SERVER_ARGS &&
4682
(str=fgets(argument,sizeof(argument), file)))
4684
*(strchr(str, '\0')-1)=0; /* Remove end newline */
4685
if (!(embedded_server_args[embedded_server_arg_count]=
4686
(char*) strdup(str)))
4689
die("Out of memory");
4692
embedded_server_arg_count++;
4696
die("Too many arguments in option file: %s",name);
4702
bool get_one_option(int optid, const struct my_option *, char *argument)
4671
4704
char *endchar= NULL;
4672
4705
uint64_t temp_drizzle_port= 0;
4680
4713
char buff[FN_REFLEN];
4681
if (!internal::test_if_hard_path(argument))
4714
if (!test_if_hard_path(argument))
4683
snprintf(buff, sizeof(buff), "%s%s",opt_basedir,argument);
4716
sprintf(buff,"%s%s",opt_basedir,argument);
4684
4717
argument= buff;
4686
internal::fn_format(buff, argument, "", "", MY_UNPACK_FILENAME);
4719
fn_format(buff, argument, "", "", MY_UNPACK_FILENAME);
4687
4720
assert(cur_file == file_stack && cur_file->file == 0);
4688
4721
if (!(cur_file->file= fopen(buff, "r")))
4690
fprintf(stderr, _("Could not open '%s' for reading: errno = %d"), buff, errno);
4691
return EXIT_ARGUMENT_INVALID;
4722
die("Could not open '%s' for reading: errno = %d", buff, errno);
4693
4723
if (!(cur_file->file_name= strdup(buff)))
4695
fprintf(stderr, _("Out of memory"));
4696
return EXIT_OUT_OF_MEMORY;
4724
die("Out of memory");
4698
4725
cur_file->lineno= 1;
4703
4730
static char buff[FN_REFLEN];
4704
if (!internal::test_if_hard_path(argument))
4731
if (!test_if_hard_path(argument))
4706
snprintf(buff, sizeof(buff), "%s%s",opt_basedir,argument);
4733
sprintf(buff,"%s%s",opt_basedir,argument);
4707
4734
argument= buff;
4709
internal::fn_format(buff, argument, "", "", MY_UNPACK_FILENAME);
4736
fn_format(buff, argument, "", "", MY_UNPACK_FILENAME);
4710
4737
timer_file= buff;
4711
4738
unlink(timer_file); /* Ignore error, may not exist */
4757
4781
strncpy(TMPDIR, argument, sizeof(TMPDIR));
4784
if (!embedded_server_arg_count)
4786
embedded_server_arg_count=1;
4787
embedded_server_args[0]= (char*) "";
4789
if (embedded_server_arg_count == MAX_EMBEDDED_SERVER_ARGS-1 ||
4790
!(embedded_server_args[embedded_server_arg_count++]=
4793
die("Can't use server argument");
4797
read_embedded_server_arguments(argument);
4760
4800
print_version();
4770
4810
static int parse_args(int argc, char **argv)
4772
internal::load_defaults("drizzle",load_default_groups,&argc,&argv);
4812
load_defaults("drizzle",load_default_groups,&argc,&argv);
4773
4813
default_argv= argv;
4775
4815
if ((handle_options(&argc, &argv, my_long_options, get_one_option)))
4805
4849
char buff[FN_REFLEN];
4806
4850
int flags= O_WRONLY | O_CREAT;
4807
if (!internal::test_if_hard_path(fname))
4851
if (!test_if_hard_path(fname))
4809
snprintf(buff, sizeof(buff), "%s%s",opt_basedir,fname);
4853
sprintf(buff,"%s%s",opt_basedir,fname);
4812
internal::fn_format(buff, fname, "", "", MY_UNPACK_FILENAME);
4856
fn_format(buff, fname, "", "", MY_UNPACK_FILENAME);
4815
4859
flags|= O_TRUNC;
4816
if ((fd= internal::my_open(buff, flags,
4860
if ((fd= my_open(buff, flags,
4817
4861
MYF(MY_WME | MY_FFNF))) < 0)
4818
4862
die("Could not open '%s' for writing: errno = %d", buff, errno);
4819
4863
if (append && lseek(fd, 0, SEEK_END) == MY_FILEPOS_ERROR)
4820
4864
die("Could not find end of file '%s': errno = %d", buff, errno);
4821
if (internal::my_write(fd, (unsigned char*)str, size, MYF(MY_WME|MY_FNABP)))
4865
if (my_write(fd, (unsigned char*)str, size, MYF(MY_WME|MY_FNABP)))
4822
4866
die("write failed");
4823
internal::my_close(fd, MYF(0));
4867
my_close(fd, MYF(0));
4842
4886
void dump_result_to_log_file(const char *buf, int size)
4844
4888
char log_file[FN_REFLEN];
4845
str_to_file(internal::fn_format(log_file, result_file_name, opt_logdir, ".log",
4889
str_to_file(fn_format(log_file, result_file_name, opt_logdir, ".log",
4846
4890
*opt_logdir ? MY_REPLACE_DIR | MY_REPLACE_EXT :
4847
4891
MY_REPLACE_EXT),
4865
4909
char warn_file[FN_REFLEN];
4867
str_to_file(internal::fn_format(warn_file, result_file_name, opt_logdir, ".warnings",
4911
str_to_file(fn_format(warn_file, result_file_name, opt_logdir, ".warnings",
4868
4912
*opt_logdir ? MY_REPLACE_DIR | MY_REPLACE_EXT :
4869
4913
MY_REPLACE_EXT),
4870
4914
ds_warning_messages.c_str(), ds_warning_messages.length());
6049
6091
uint64_t timer_now(void)
6051
#if defined(HAVE_GETHRTIME)
6052
return gethrtime()/1000/1000;
6057
The following loop is here because gettimeofday may fail on some systems
6059
while (gettimeofday(&t, NULL) != 0)
6061
newtime= (uint64_t)t.tv_sec * 1000000 + t.tv_usec;
6062
return newtime/1000;
6063
#endif /* defined(HAVE_GETHRTIME) */
6093
return my_micro_time() / 1000;