home *** CD-ROM | disk | FTP | other *** search
/ PC World 2005 April / PCWorld_2005-04_cd.bin / akce / web / phptriad / phptriad2-2-1.exe / mysql / bench / crash-me < prev    next >
Text File  |  2001-12-29  |  108KB  |  3,543 lines

  1. #!/usr/bin/perl
  2. # Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
  3. #
  4. # This library is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU Library General Public
  6. # License as published by the Free Software Foundation; either
  7. # version 2 of the License, or (at your option) any later version.
  8. #
  9. # This library is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  12. # Library General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU Library General Public
  15. # License along with this library; if not, write to the Free
  16. # Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
  17. # MA 02111-1307, USA
  18.  
  19. # Written by Monty for the TCX/Monty Program/Detron benchmark suite.
  20. # Empress and PostgreSQL patches by Luuk de Boer
  21. # Extensions for ANSI SQL and Mimer by Bengt Gunne
  22. # Some additions and corrections by Matthias Urlich
  23. #
  24. # This programs tries to find all limits for a sql server
  25. # It gets the name from what it does to most servers :)
  26. #
  27. # Be sure to use --help before running this!
  28. #
  29. # If you want to add support for another server, add a new package for the
  30. # server in server-cfg.  You only have to support the 'new' and 'version'
  31. # functions. new doesn't need to have any limits if one doesn't want to
  32. # use the benchmarks.
  33. #
  34.  
  35. # TODO:
  36. # CMT includes types and functions which are synonyms for other types
  37. # and functions, including those in SQL9x. It should label those synonyms
  38. # as such, and clarify ones such as "mediumint" with comments such as
  39. # "3-byte int" or "same as xxx".
  40.  
  41. $version="1.57";
  42.  
  43. use DBI;
  44. use Getopt::Long;
  45. chomp($pwd = `pwd`); $pwd = "." if ($pwd eq '');
  46. require "$pwd/server-cfg" || die "Can't read Configuration file: $!\n";
  47.  
  48. $opt_server="mysql"; $opt_host="localhost"; $opt_database="test";
  49. $opt_dir="limits";
  50. $opt_user=$opt_password="";
  51. $opt_debug=$opt_help=$opt_Information=$opt_restart=$opt_force=$opt_quick=0;
  52. $opt_log_all_queries=$opt_fix_limit_file=$opt_batch_mode=0;
  53. $opt_db_start_cmd="";           # the db server start command
  54. $opt_check_server=0;        # Check if server is alive before each query
  55. $opt_sleep=10;                  # time to sleep while starting the db server
  56. $limit_changed=0;               # For configure file
  57. $reconnect_count=0;
  58. $opt_comment=$opt_config_file=$opt_log_queries_to_file="";
  59. $limits{'crash_me_safe'}='yes';
  60. $prompts{'crash_me_safe'}='crash me safe';
  61. $limits{'operating_system'}= machine();
  62. $prompts{'operating_system'}='crash-me tested on';
  63. $retry_limit=3;
  64.  
  65. GetOptions("Information","help","server=s","debug","user=s","password=s","database=s","restart","force","quick","log-all-queries","comment=s","host=s","fix-limit-file","dir=s","db-start-cmd=s","sleep=s","batch-mode","config-file=s","log-queries-to-file=s","check-server") || usage();
  66. usage() if ($opt_help || $opt_Information);
  67.  
  68. $opt_config_file="$pwd/$opt_dir/$opt_server.cfg" if (length($opt_config_file) == 0);
  69.  
  70. if ($opt_fix_limit_file)
  71. {
  72.   print "Fixing limit file for $opt_server\n";
  73.   read_config_data();
  74.   $limit_changed=1;
  75.   save_all_config_data();
  76.   exit 0;
  77. }
  78.  
  79. $server=get_server($opt_server,$opt_host,$opt_database);
  80. $opt_server=$server->{'cmp_name'};
  81.  
  82. $|=1;                           # For debugging
  83.  
  84. print "Running $0 $version on '",($server_version=$server->version()),"'\n\n";
  85. print "I hope you didn't have anything important running on this server....\n";
  86. read_config_data();
  87. if ($limit_changed)             # Must have been restarted
  88. {
  89.   save_config_data('crash_me_safe','no',"crash me safe");
  90. }
  91.  
  92. if (!$opt_force && !$opt_batch_mode)
  93. {
  94.   server_info();
  95. }
  96. else
  97. {
  98.   print "Using --force.  I assume you know what you are doing...\n";
  99. }
  100. print "\n";
  101.  
  102. save_config_data('crash_me_version',$version,"crash me version");
  103. if ($server_version)
  104. {
  105.   save_config_data('server_version',$server_version,"server version");
  106. }
  107. if (length($opt_comment))
  108. {
  109.   save_config_data('user_comment',$opt_comment,"comment");
  110. }
  111.  
  112. $opt_log=0;
  113. if (length($opt_log_queries_to_file))
  114. {
  115.   open(LOG,">$opt_log_queries_to_file") || die "Can't open file $opt_log_queries_to_file\n";
  116.   $opt_log=1;
  117. }
  118.  
  119. #
  120. # Set up some limits that's regared as unlimited
  121. # We don't want to take up all resources from the server...
  122. #
  123.  
  124. $max_connections="+1000";       # Number of simultaneous connections
  125. $max_buffer_size="+16000000";   # size of communication buffer.
  126. $max_string_size="+8000000";    # Enough for this test
  127. $max_name_length="+512";        # Actually 256, but ...
  128. $max_keys="+64";                # Probably too big.
  129. $max_join_tables="+64";         # Probably too big.
  130. $max_columns="+8192";           # Probably too big.
  131. $max_row_length=$max_string_size;
  132. $max_key_length="+8192";        # Big enough
  133. $max_order_by="+64";        # Big enough
  134. $max_expressions="+10000";
  135. $max_big_expressions="+100";
  136. $max_stacked_expressions="+2000";
  137. $query_size=$max_buffer_size;
  138. $longreadlen=16000000;        # For retrieval buffer
  139.  
  140.  
  141. #
  142. # First do some checks that needed for the rest of the benchmark
  143. #
  144. use sigtrap;               # Must be removed with perl5.005_2 on Win98
  145. $SIG{PIPE} = 'IGNORE';
  146. $SIG{SEGV} = sub {warn('SEGFAULT')};
  147. $dbh=safe_connect();
  148.  
  149. #
  150. # Test if the database require RESTRICT/CASCADE after DROP TABLE
  151. #
  152.  
  153. # Really remove the crash_me table
  154. $prompt="drop table require cascade/restrict";
  155. $drop_attr="";
  156. $dbh->do("drop table crash_me");
  157. $dbh->do("drop table crash_me cascade");
  158. if (!safe_query(["create table crash_me (a integer not null)",
  159.          "drop table crash_me"]))
  160. {
  161.   $dbh->do("drop table crash_me cascade");  
  162.   if (safe_query(["create table crash_me (a integer not null)",
  163.           "drop table crash_me cascade"]))
  164.   {
  165.     save_config_data('drop_requires_cascade',"yes","$prompt");
  166.     $drop_attr="cascade";
  167.   }
  168.   else
  169.   {
  170.     die "Can't create and drop table 'crash_me'\n";
  171.   }
  172. }
  173. else
  174. {
  175.   save_config_data('drop_requires_cascade',"no","$prompt");
  176.   $drop_attr="";
  177. }
  178.  
  179. # Remove tables from old runs
  180. $dbh->do("drop table crash_me $drop_attr");
  181. $dbh->do("drop table crash_me2 $drop_attr");
  182. $dbh->do("drop table crash_me3 $drop_attr");
  183. $dbh->do("drop table crash_q $drop_attr");
  184. $dbh->do("drop table crash_q1 $drop_attr");
  185.  
  186. $prompt="Tables without primary key";
  187. if (!safe_query(["create table crash_me (a integer not null,b char(10) not null)",
  188.          "insert into crash_me (a,b) values (1,'a')"]))
  189. {
  190.   if (!safe_query(["create table crash_me (a integer not null,b char(10) not null, primary key (a))",
  191.            "insert into crash_me (a,b) values (1,'a')"]))
  192.   {
  193.     die "Can't create table 'crash_me' with one record: $DBI::errstr\n";
  194.   }
  195.   save_config_data('no_primary_key',"no",$prompt);
  196. }
  197. else
  198. {
  199.   save_config_data('no_primary_key',"yes",$prompt);
  200. }
  201.  
  202. #
  203. #  Define strings for character NULL and numeric NULL used in expressions
  204. #
  205. $char_null=$server->{'char_null'};
  206. $numeric_null=$server->{'numeric_null'};
  207. if ($char_null eq '')
  208. {
  209.   $char_null="NULL";
  210. }
  211. if ($numeric_null eq '')
  212. {
  213.   $numeric_null="NULL";
  214. }
  215.  
  216. print "$prompt: $limits{'no_primary_key'}\n";
  217.  
  218. report("SELECT without FROM",'select_without_from',"select 1");
  219. if ($limits{'select_without_from'} ne "yes")
  220. {
  221.   $end_query=" from crash_me";
  222.   $check_connect="select a from crash_me";
  223. }
  224. else
  225. {
  226.   $end_query="";
  227.   $check_connect="select 1";
  228. }
  229.  
  230. assert($check_connect);
  231. assert("select a from crash_me where b<'b'");
  232.  
  233. report("Select constants",'select_constants',"select 1 $end_query");
  234. report("Select table_name.*",'table_wildcard',
  235.        "select crash_me.* from crash_me");
  236. report("Allows \' and \" as string markers",'quote_with_"',
  237.        'select a from crash_me where b<"c"');
  238. check_and_report("Double '' as ' in strings",'double_quotes',[],
  239.          "select 'Walker''s' $end_query",[],"Walker's",1);
  240. check_and_report("Multiple line strings","multi_strings",[],
  241.          "select a from crash_me where b < 'a'\n'b'",[],"1",0);
  242. check_and_report("\" as identifier quote (ANSI SQL)",'quote_ident_with_"',[],
  243.          'select "A" from crash_me',[],"1",0);
  244. check_and_report("\` as identifier quote",'quote_ident_with_`',[],
  245.          'select `A` from crash_me',[],"1",0);
  246. check_and_report("[] as identifier quote",'quote_ident_with_[',[],
  247.          'select [A] from crash_me',[],"1",0);
  248.  
  249. report("Column alias","column_alias","select a as ab from crash_me");
  250. report("Table alias","table_alias","select b.a from crash_me as b");
  251. report("Functions",'functions',"select 1+1 $end_query");
  252. report("Group functions",'group_functions',"select count(*) from crash_me");
  253. report("Group functions with distinct",'group_distinct_functions',
  254.        "select count(distinct a) from crash_me");
  255. report("Group by",'group_by',"select a from crash_me group by a");
  256. report("Group by position",'group_by_position',
  257.        "select a from crash_me group by 1");
  258. report("Group by alias",'group_by_alias',
  259.        "select a as ab from crash_me group by ab");
  260. report("Group on unused column",'group_on_unused',
  261.        "select count(*) from crash_me group by a");
  262.  
  263. report("Order by",'order_by',"select a from crash_me order by a");
  264. report("Order by position",'order_by_position',
  265.        "select a from crash_me order by 1");
  266. report("Order by function","order_by_function",
  267.        "select a from crash_me order by a+1");
  268. report("Order by on unused column",'order_on_unused',
  269.        "select b from crash_me order by a");
  270. check_and_report("Order by DESC is remembered",'order_by_remember_desc',
  271.          ["create table crash_q (s int,s1 int)",
  272.           "insert into crash_q values(1,1)",
  273.           "insert into crash_q values(3,1)",
  274.           "insert into crash_q values(2,1)"],
  275.          "select s,s1 from crash_q order by s1 DESC,s",
  276.          ["drop table crash_q $drop_attr"],[3,2,1],7,undef(),3);
  277. report("Compute",'compute',
  278.        "select a from crash_me order by a compute sum(a) by a");
  279. report("INSERT with Value lists",'insert_multi_value',
  280.        "create table crash_q (s char(10))",
  281.        "insert into crash_q values ('a'),('b')",
  282.        "drop table crash_q $drop_attr");
  283. report("INSERT with set syntax",'insert_with_set',
  284.        "create table crash_q (a integer)",
  285.        "insert into crash_q SET a=1",
  286.        "drop table crash_q $drop_attr");
  287. report("allows end ';'","end_colon", "select * from crash_me;");
  288. try_and_report("LIMIT number of rows","select_limit",
  289.            ["with LIMIT",
  290.         "select * from crash_me limit 1"],
  291.            ["with TOP",
  292.         "select TOP 1 * from crash_me"]);
  293. report("SELECT with LIMIT #,#","select_limit2", "select * from crash_me limit 1,1");
  294.  
  295. # The following alter table commands MUST be kept together!
  296. if ($dbh->do("create table crash_q (a integer, b integer,c1 CHAR(10))"))
  297. {
  298.   report("Alter table add column",'alter_add_col',
  299.      "alter table crash_q add d integer");
  300.   report_one("Alter table add many columns",'alter_add_multi_col',
  301.          [["alter table crash_q add (f integer,g integer)","yes"],
  302.           ["alter table crash_q add f integer, add g integer","with add"],
  303.           ["alter table crash_q add f integer,g integer","without add"]] );
  304.   report("Alter table change column",'alter_change_col',
  305.      "alter table crash_q change a e char(50)");
  306.  
  307.   # informix can only change data type with modify
  308.   report_one("Alter table modify column",'alter_modify_col',
  309.          [["alter table crash_q modify c1 CHAR(20)","yes"],
  310.           ["alter table crash_q alter c1 CHAR(20)","with alter"]]);
  311.   report("Alter table alter column default",'alter_alter_col',
  312.      "alter table crash_q alter b set default 10");
  313.   report_one("Alter table drop column",'alter_drop_col',
  314.          [["alter table crash_q drop column b","yes"],
  315.           ["alter table crash_q drop column b restrict","with restrict/cascade"]]);
  316.   report("Alter table rename table",'alter_rename_table',
  317.      "alter table crash_q rename to crash_q1");
  318. }
  319. # Make sure both tables will be dropped, even if rename fails.
  320. $dbh->do("drop table crash_q1 $drop_attr");
  321. $dbh->do("drop table crash_q $drop_attr");
  322.  
  323. report("rename table","rename_table",
  324.        "create table crash_q (a integer, b integer,c1 CHAR(10))",
  325.        "rename table crash_q to crash_q1",
  326.        "drop table crash_q1 $drop_attr");
  327. # Make sure both tables will be dropped, even if rename fails.
  328. $dbh->do("drop table crash_q1 $drop_attr");
  329. $dbh->do("drop table crash_q $drop_attr");
  330.  
  331. report("truncate","truncate_table",
  332.        "create table crash_q (a integer, b integer,c1 CHAR(10))",
  333.        "truncate table crash_q",
  334.        "drop table crash_q $drop_attr");
  335.  
  336. if ($dbh->do("create table crash_q (a integer, b integer,c1 CHAR(10))") &&
  337.     $dbh->do("create table crash_q1 (a integer, b integer,c1 CHAR(10) not null)"))
  338. {
  339.   report("Alter table add constraint",'alter_add_constraint',
  340.      "alter table crash_q add constraint c2 check(a > b)");
  341.   report_one("Alter table drop constraint",'alter_drop_constraint',
  342.          [["alter table crash_q drop constraint c2","yes"],
  343.           ["alter table crash_q drop constraint c2 restrict","with restrict/cascade"]]);
  344.   report("Alter table add unique",'alter_add_unique',
  345.      "alter table crash_q add constraint u1 unique(c1)");
  346.   try_and_report("Alter table drop unique",'alter_drop_unique',
  347.          ["with constraint",
  348.           "alter table crash_q drop constraint u1"],
  349.          ["with constraint and restrict/cascade",
  350.           "alter table crash_q drop constraint u1 restrict"],
  351.          ["with drop key",
  352.           "alter table crash_q drop key c1"]);
  353.   try_and_report("Alter table add primary key",'alter_add_primary_key',
  354.          ["with constraint",
  355.           "alter table crash_q1 add constraint p1 primary key(c1)"],
  356.          ["with add primary key",
  357.           "alter table crash_q1 add primary key(c1)"]);
  358.   report("Alter table add foreign key",'alter_add_foreign_key',
  359.      "alter table crash_q add constraint f1 foreign key(c1) references crash_q1(c1)");
  360.   try_and_report("Alter table drop foreign key",'alter_drop_foreign_key',
  361.          ["with drop constraint",
  362.           "alter table crash_q drop constraint f1"],
  363.          ["with drop constraint and restrict/cascade",
  364.           "alter table crash_q drop constraint f1 restrict"],
  365.          ["with drop foreign key",
  366.           "alter table crash_q drop foreign key f1"]);
  367.   try_and_report("Alter table drop primary key",'alter_drop_primary_key',
  368.          ["drop constraint",
  369.           "alter table crash_q1 drop constraint p1 restrict"],
  370.          ["drop primary key",
  371.           "alter table crash_q1 drop primary key"]);
  372. }
  373. $dbh->do("drop table crash_q $drop_attr");
  374. $dbh->do("drop table crash_q1 $drop_attr");
  375.  
  376. check_and_report("Case insensitive compare","case_insensitive_strings",
  377.          [],"select b from crash_me where b = 'A'",[],'a',1);
  378. check_and_report("Ignore end space in compare","ignore_end_space",
  379.          [],"select b from crash_me where b = 'a '",[],'a',1);
  380. check_and_report("Group on column with null values",'group_by_null',
  381.          ["create table crash_q (s char(10))",
  382.           "insert into crash_q values(null)",
  383.           "insert into crash_q values(null)"],
  384.          "select count(*),s from crash_q group by s",
  385.          ["drop table crash_q $drop_attr"],2,0);
  386.  
  387. $prompt="Having";
  388. if (!defined($limits{'having'}))
  389. {                               # Complicated because of postgreSQL
  390.   if (!safe_query_result("select a from crash_me group by a having a > 0",1,0))
  391.   {
  392.     if (!safe_query_result("select a from crash_me group by a having a < 0",
  393.                1,0))
  394.     { save_config_data("having","error",$prompt); }
  395.     else
  396.     { save_config_data("having","yes",$prompt); }
  397.   }
  398.   else
  399.   { save_config_data("having","no",$prompt); }
  400. }
  401. print "$prompt: $limits{'having'}\n";
  402.  
  403. if ($limits{'having'} eq 'yes')
  404. {
  405.   report("Having with group function","having_with_group",
  406.      "select a from crash_me group by a having count(*) = 1");
  407. }
  408.  
  409. if ($limits{'column_alias'} eq 'yes')
  410. {
  411.   report("Order by alias",'order_by_alias',
  412.      "select a as ab from crash_me order by ab");
  413.   if ($limits{'having'} eq 'yes')
  414.   {
  415.     report("Having on alias","having_with_alias",
  416.        "select a as ab from crash_me group by a having ab > 0");
  417.   }
  418. }
  419. report("binary numbers (0b1001)","binary_numbers","select 0b1001 $end_query");
  420. report("hex numbers (0x41)","hex_numbers","select 0x41 $end_query");
  421. report("binary strings (b'0110')","binary_strings","select b'0110' $end_query");
  422. report("hex strings (x'1ace')","hex_strings","select x'1ace' $end_query");
  423.  
  424. report_result("Value of logical operation (1=1)","logical_value",
  425.           "select (1=1) $end_query");
  426.  
  427. $logical_value= $limits{'logical_value'};
  428.  
  429. $false=0;
  430. $result="no";
  431. if ($res=safe_query("select (1=1)=true $end_query")) {
  432.   $false="false";
  433.   $result="yes";
  434. }
  435. save_config_data('has_true_false',$result,"TRUE and FALSE");
  436.  
  437. #
  438. # Check how many connections the server can handle:
  439. # We can't test unlimited connections, because this may take down the
  440. # server...
  441. #
  442.  
  443. $prompt="Simultaneous connections (installation default)";
  444. print "$prompt: ";
  445. if (defined($limits{'connections'}))
  446. {
  447.   print "$limits{'connections'}\n";
  448. }
  449. else
  450. {
  451.   @connect=($dbh);
  452.  
  453.   for ($i=1; $i < $max_connections ; $i++)
  454.   {
  455.     if (!($dbh=DBI->connect($server->{'data_source'},$opt_user,$opt_password,
  456.               { PrintError => 0})))
  457.     {
  458.       print "Last connect error: $DBI::errstr\n" if ($opt_debug);
  459.       last;
  460.     }
  461.     $dbh->{LongReadLen}= $longreadlen; # Set retrieval buffer
  462.     print "." if ($opt_debug);
  463.     push(@connect,$dbh);
  464.   }
  465.   print "$i\n";
  466.   save_config_data('connections',$i,$prompt);
  467.   foreach $dbh (@connect)
  468.   {
  469.     print "#" if ($opt_debug);
  470.     $dbh->disconnect || warn $dbh->errstr;           # close connection
  471.   }
  472.  
  473.   $#connect=-1;                 # Free connections
  474.  
  475.   if ($i == 0)
  476.   {
  477.     print "Can't connect to server: $DBI::errstr.  Please start it and try again\n";
  478.     exit 1;
  479.   }
  480.   $dbh=safe_connect();
  481. }
  482.  
  483.  
  484. #
  485. # Check size of communication buffer, strings...
  486. #
  487.  
  488. $prompt="query size";
  489. print "$prompt: ";
  490. if (!defined($limits{'query_size'}))
  491. {
  492.   $query="select ";
  493.   $first=64;
  494.   $end=$max_buffer_size;
  495.   $select= $limits{'select_without_from'} eq 'yes' ? 1 : 'a';
  496.  
  497.   assert($query . "$select$end_query");
  498.  
  499.   $first=$limits{'restart'}{'low'} if ($limits{'restart'}{'low'});
  500.  
  501.   if ($limits{'restart'}{'tohigh'})
  502.   {
  503.     $end = $limits{'restart'}{'tohigh'} - 1;
  504.     print "\nRestarting this with low limit: $first and high limit: $end\n";
  505.     delete $limits{'restart'};
  506.     $first=$first+int(($end-$first+4)/5);           # Prefere lower on errors
  507.   }
  508.   for ($i=$first ; $i < $end ; $i*=2)
  509.   {
  510.     last if (!safe_query($query . (" " x ($i - length($query)-length($end_query) -1)) . "$select$end_query"));
  511.     $first=$i;
  512.     save_config_data("restart",$i,"") if ($opt_restart);
  513.   }
  514.   $end=$i;
  515.  
  516.   if ($i < $max_buffer_size)
  517.   {
  518.     while ($first != $end)
  519.     {
  520.       $i=int(($first+$end+1)/2);
  521.       if (safe_query($query .
  522.              (" " x ($i - length($query)-length($end_query) -1)) .
  523.              "$select$end_query"))
  524.       {
  525.     $first=$i;
  526.       }
  527.       else
  528.       {
  529.     $end=$i-1;
  530.       }
  531.     }
  532.   }
  533.   save_config_data('query_size',$end,$prompt);
  534. }
  535. $query_size=$limits{'query_size'};
  536.  
  537. print "$limits{'query_size'}\n";
  538. #
  539. # Test database types
  540. #
  541.  
  542. @sql_types=("character(1)","char(1)","char varying(1)", "character varying(1)",
  543.         "boolean",
  544.         "varchar(1)",
  545.         "integer","int","smallint",
  546.         "numeric(9,2)","decimal(6,2)","dec(6,2)",
  547.         "bit", "bit(2)","bit varying(2)","float","float(8)","real",
  548.         "double precision", "date","time","timestamp",
  549.         "interval year", "interval year to month",
  550.             "interval month",
  551.             "interval day", "interval day to hour", "interval day to minute",
  552.             "interval day to second",
  553.             "interval hour", "interval hour to minute", "interval hour to second",
  554.             "interval minute", "interval minute to second",
  555.             "interval second",
  556.         "national character varying(20)",
  557.         "national character(20)","nchar(1)",
  558.         "national char varying(20)","nchar varying(20)",
  559.         "national character varying(20)",
  560.         "timestamp with time zone");
  561. @odbc_types=("binary(1)","varbinary(1)","tinyint","bigint",
  562.          "datetime");
  563. @extra_types=("blob","byte","long varbinary","image","text","text(10)",
  564.           "mediumtext",
  565.           "long varchar(1)", "varchar2(257)",
  566.           "mediumint","middleint","int unsigned",
  567.           "int1","int2","int3","int4","int8","uint",
  568.           "money","smallmoney","float4","float8","smallfloat",
  569.           "float(6,2)","double",
  570.           "enum('red')","set('red')", "int(5) zerofill", "serial",
  571.           "char(10) binary","int not null auto_increment,unique(q)",
  572.           "abstime","year","datetime","smalldatetime","timespan","reltime",
  573.           # Sybase types
  574.           "int not null identity,unique(q)",
  575.           # postgres types
  576.           "box","bool","circle","polygon","point","line","lseg","path",
  577.           "interval", "serial", "inet", "cidr", "macaddr",
  578.  
  579.           # oracle types
  580.           "varchar2(16)","nvarchar2(16)","number(9,2)","number(9)",
  581.           "number", "long","raw(16)","long raw","rowid","mlslabel","clob",
  582.           "nclob","bfile"
  583.           );
  584.  
  585. @types=(["sql",\@sql_types],
  586.     ["odbc",\@odbc_types],
  587.     ["extra",\@extra_types]);
  588.  
  589. foreach $types (@types)
  590. {
  591.   print "\nSupported $types->[0] types\n";
  592.   $tmp=@$types->[1];
  593.   foreach $use_type (@$tmp)
  594.   {
  595.     $type=$use_type;
  596.     $type =~ s/\(.*\)/(1 arg)/;
  597.     if (index($use_type,",")>= 0)
  598.     {
  599.       $type =~ s/\(1 arg\)/(2 arg)/;
  600.     }
  601.     if (($tmp2=index($type,",unique")) >= 0)
  602.     {
  603.       $type=substr($type,0,$tmp2);
  604.     }
  605.     $tmp2=$type;
  606.     $tmp2 =~ s/ /_/g;
  607.     $tmp2 =~ s/_not_null//g;
  608.     report("Type $type","type_$types->[0]_$tmp2",
  609.        "create table crash_q (q $use_type)",
  610.        "drop table crash_q $drop_attr");
  611.   }
  612. }
  613.  
  614. #
  615. # Test some type limits
  616. #
  617.  
  618. check_and_report("Remembers end space in char()","remember_end_space",
  619.          ["create table crash_q (a char(10))",
  620.           "insert into crash_q values('hello ')"],
  621.          "select a from crash_q where a = 'hello '",
  622.          ["drop table crash_q $drop_attr"],
  623.          'hello ',6);
  624.  
  625. check_and_report("Remembers end space in varchar()",
  626.          "remember_end_space_varchar",
  627.          ["create table crash_q (a varchar(10))",
  628.           "insert into crash_q values('hello ')"],
  629.          "select a from crash_q where a = 'hello '",
  630.          ["drop table crash_q $drop_attr"],
  631.          'hello ',6);
  632.  
  633. check_and_report("Supports 0000-00-00 dates","date_zero",
  634.          ["create table crash_me2 (a date not null)",
  635.           "insert into crash_me2 values ('0000-00-00')"],
  636.          "select a from crash_me2",
  637.          ["drop table crash_me2 $drop_attr"],
  638.          "0000-00-00",1);
  639.  
  640. check_and_report("Supports 0001-01-01 dates","date_one",
  641.          ["create table crash_me2 (a date not null)",
  642.           "insert into crash_me2 values (DATE '0001-01-01')"],
  643.          "select a from crash_me2",
  644.          ["drop table crash_me2 $drop_attr"],
  645.          "0001-01-01",1);
  646.  
  647. check_and_report("Supports 9999-12-31 dates","date_last",
  648.          ["create table crash_me2 (a date not null)",
  649.           "insert into crash_me2 values (DATE '9999-12-31')"],
  650.          "select a from crash_me2",
  651.          ["drop table crash_me2 $drop_attr"],
  652.          "9999-12-31",1);
  653.  
  654. check_and_report("Supports 'infinity dates","date_infinity",
  655.          ["create table crash_me2 (a date not null)",
  656.           "insert into crash_me2 values ('infinity')"],
  657.          "select a from crash_me2",
  658.          ["drop table crash_me2 $drop_attr"],
  659.          "infinity",1);
  660.  
  661. if (!defined($limits{'date_with_YY'}))
  662. {
  663.     check_and_report("Supports YY-MM-DD dates","date_with_YY",
  664.              ["create table crash_me2 (a date not null)",
  665.               "insert into crash_me2 values ('98-03-03')"],
  666.              "select a from crash_me2",
  667.              ["drop table crash_me2 $drop_attr"],
  668.              "1998-03-03",5);
  669.     if ($limits{'date_with_YY'} eq "yes")
  670.     {
  671.     undef($limits{'date_with_YY'});
  672.     check_and_report("Supports YY-MM-DD 2000 compilant dates",
  673.              "date_with_YY",
  674.              ["create table crash_me2 (a date not null)",
  675.               "insert into crash_me2 values ('10-03-03')"],
  676.              "select a from crash_me2",
  677.              ["drop table crash_me2 $drop_attr"],
  678.              "2010-03-03",5);
  679.     }
  680. }
  681.  
  682. if (($limits{'type_extra_float(2_arg)'} eq "yes" ||
  683.     $limits{'type_sql_decimal(2_arg)'} eq "yes") &&
  684.     (!defined($limits{'storage_of_float'})))
  685. {
  686.   my $type=$limits{'type_extra_float(2_arg)'} eq "yes" ? "float(4,1)" :
  687.     "decimal(4,1)";
  688.   my $result="undefined";
  689.   if (execute_and_check(["create table crash_q (q1 $type)",
  690.              "insert into crash_q values(1.14)"],
  691.             "select q1 from crash_q",
  692.             ["drop table crash_q $drop_attr"],1.1,0) &&
  693.       execute_and_check(["create table crash_q (q1 $type)",
  694.              "insert into crash_q values(1.16)"],
  695.             "select q1 from crash_q",
  696.             ["drop table crash_q $drop_attr"],1.1,0))
  697.   {
  698.     $result="truncate";
  699.   }
  700.   elsif (execute_and_check(["create table crash_q (q1 $type)",
  701.                 "insert into crash_q values(1.14)"],
  702.                "select q1 from crash_q",
  703.                ["drop table crash_q $drop_attr"],1.1,0) &&
  704.      execute_and_check(["create table crash_q (q1 $type)",
  705.                 "insert into crash_q values(1.16)"],
  706.                "select q1 from crash_q",
  707.                ["drop table crash_q $drop_attr"],1.2,0))
  708.   {
  709.     $result="round";
  710.   }
  711.   elsif (execute_and_check(["create table crash_q (q1 $type)",
  712.                 "insert into crash_q values(1.14)"],
  713.                "select q1 from crash_q",
  714.                ["drop table crash_q $drop_attr"],1.14,0) &&
  715.      execute_and_check(["create table crash_q (q1 $type)",
  716.                 "insert into crash_q values(1.16)"],
  717.                "select q1 from crash_q",
  718.                ["drop table crash_q $drop_attr"],1.16,0))
  719.   {
  720.     $result="exact";
  721.   }
  722.   $prompt="Storage of float values";
  723.   print "$prompt: $result\n";
  724.   save_config_data("storage_of_float", $result, $prompt);
  725. }
  726.  
  727. try_and_report("Type for row id", "rowid",
  728.            ["rowid",
  729.         "create table crash_q (a rowid)","drop table crash_q $drop_attr"],
  730.            ["auto_increment",
  731.         "create table crash_q (a int not null auto_increment, primary key(a))","drop table crash_q $drop_attr"],
  732.            ["oid",
  733.         "create table crash_q (a oid, primary key(a))","drop table crash_q $drop_attr"],
  734.            ["serial",
  735.         "create table crash_q (a serial, primary key(a))","drop table crash_q $drop_attr"]);
  736.  
  737. try_and_report("Automatic row id", "automatic_rowid",
  738.            ["_rowid",
  739.         "create table crash_q (a int not null, primary key(a))",
  740.         "insert into crash_q values (1)",
  741.         "select _rowid from crash_q",
  742.         "drop table crash_q $drop_attr"]);
  743.  
  744. #
  745. # Test functions
  746. #
  747.  
  748. @sql_functions=
  749.   (["+, -, * and /","+","5*3-4/2+1",14,0],
  750.    ["ANSI SQL SUBSTRING","substring","substring('abcd' from 2 for 2)","bc",1],
  751.    ["BIT_LENGTH","bit_length","bit_length('abc')",24,0],
  752.    ["searched CASE","searched_case","case when 1 > 2 then 'false' when 2 > 1 then 'true' end", "true",1],
  753.    ["simple CASE","simple_case","case 2 when 1 then 'false' when 2 then 'true' end", "true",1],
  754.    ["CAST","cast","CAST(1 as CHAR)","1",1],
  755.    ["CHARACTER_LENGTH","character_length","character_length('abcd')","4",0],
  756.    ["CHAR_LENGTH","char_length","char_length(b)","10",0],
  757.    ["CHAR_LENGTH(constant)","char_length(constant)","char_length('abcd')","4",0],
  758.    ["COALESCE","coalesce","coalesce($char_null,'bcd','qwe')","bcd",1],
  759.    ["CURRENT_DATE","current_date","current_date",0,2],
  760.    ["CURRENT_TIME","current_time","current_time",0,2],
  761.    ["CURRENT_TIMESTAMP","current_timestamp","current_timestamp",0,2],
  762.    ["CURRENT_USER","current_user","current_user",0,2],
  763.    ["EXTRACT","extract_sql","extract(minute from timestamp '2000-02-23 18:43:12.987')",43,0],
  764.    ["LOCALTIME","localtime","localtime",0,2],
  765.    ["LOCALTIMESTAMP","localtimestamp","localtimestamp",0,2],
  766.    ["LOWER","lower","LOWER('ABC')","abc",1],
  767.    ["NULLIF with strings","nullif_string","NULLIF(NULLIF('first','second'),'first')",undef(),4],
  768.    ["NULLIF with numbers","nullif_num","NULLIF(NULLIF(1,2),1)",undef(),4],
  769.    ["OCTET_LENGTH","octet_length","octet_length('abc')",3,0],
  770.    ["POSITION","position","position('ll' in 'hello')",3,0],
  771.    ["SESSION_USER","session_user","session_user",0,2],
  772.    ["SYSTEM_USER","system_user","system_user",0,2],
  773.    ["TRIM","trim","trim(trailing from trim(LEADING FROM ' abc '))","abc",3],
  774.    ["UPPER","upper","UPPER('abc')","ABC",1],
  775.    ["USER","user","user"],
  776.    ["concatenation with ||","concat_as_||","'abc' || 'def'","abcdef",1],
  777.    );
  778.  
  779. @odbc_functions=
  780.   (["ASCII", "ascii", "ASCII('A')","65",0],
  781.    ["CHAR", "char", "CHAR(65)"  ,"A",1],
  782.    ["CONCAT(2 arg)","concat", "concat('a','b')","ab",1],
  783.    ["DIFFERENCE()","difference","difference('abc','abe')",0,2],
  784.    ["INSERT","insert","insert('abcd',2,2,'ef')","aefd",1],
  785.    ["LEFT","left","left('abcd',2)","ab",1],
  786.    ["LTRIM","ltrim","ltrim('   abcd')","abcd",1],
  787.    ["REAL LENGTH","length","length('abcd ')","5",0],
  788.    ["ODBC LENGTH","length_without_space","length('abcd ')","4",0],
  789.    ["LOCATE(2 arg)","locate_2","locate('bcd','abcd')","2",0],
  790.    ["LOCATE(3 arg)","locate_3","locate('bcd','abcd',3)","0",0],
  791.    ["LCASE","lcase","lcase('ABC')","abc",1],
  792.    ["REPEAT","repeat","repeat('ab',3)","ababab",1],
  793.    ["REPLACE","replace","replace('abbaab','ab','ba')","bababa",1],
  794.    ["RIGHT","right","right('abcd',2)","cd",1],
  795.    ["RTRIM","rtrim","rtrim(' abcd  ')"," abcd",1],
  796.    ["SPACE","space","space(5)","     ",3],
  797.    ["SOUNDEX","soundex","soundex('hello')",0,2],
  798.    ["ODBC SUBSTRING","substring","substring('abcd',3,2)","cd",1],
  799.    ["UCASE","ucase","ucase('abc')","ABC",1],
  800.  
  801.    ["ABS","abs","abs(-5)",5,0],
  802.    ["ACOS","acos","acos(0)","1.570796",0],
  803.    ["ASIN","asin","asin(1)","1.570796",0],
  804.    ["ATAN","atan","atan(1)","0.785398",0],
  805.    ["ATAN2","atan2","atan2(1,0)","1.570796",0],
  806.    ["CEILING","ceiling","ceiling(-4.5)",-4,0],
  807.    ["COS","cos","cos(0)","1.00000",0],
  808.    ["COT","cot","cot(1)","0.64209262",0],
  809.    ["DEGREES","degrees","degrees(6.283185)","360",0],
  810.    ["EXP","exp","exp(1.0)","2.718282",0],
  811.    ["FLOOR","floor","floor(2.5)","2",0],
  812.    ["LOG","log","log(2)","0.693147",0],
  813.    ["LOG10","log10","log10(10)","1",0],
  814.    ["MOD","mod","mod(11,7)","4",0],
  815.    ["PI","pi","pi()","3.141593",0],
  816.    ["POWER","power","power(2,4)","16",0],
  817.    ["RAND","rand","rand(1)",0,2],       # Any value is acceptable
  818.    ["RADIANS","radians","radians(360)","6.283185",0],
  819.    ["ROUND(2 arg)","round","round(5.63,2)","5.6",0],
  820.    ["SIGN","sign","sign(-5)",-1,0],
  821.    ["SIN","sin","sin(1)","0.841471",0],
  822.    ["SQRT","sqrt","sqrt(4)",2,0],
  823.    ["TAN","tan","tan(1)","1.557408",0],
  824.    ["TRUNCATE","truncate","truncate(18.18,-1)",10,0],
  825.    ["NOW","now","now()",0,2],           # Any value is acceptable
  826.    ["CURDATE","curdate","curdate()",0,2],
  827.    ["DAYNAME","dayname","dayname(DATE '1997-02-01')","",2],
  828.    ["MONTH","month","month(DATE '1997-02-01')","",2],
  829.    ["MONTHNAME","monthname","monthname(DATE '1997-02-01')","",2],
  830.    ["DAYOFMONTH","dayofmonth","dayofmonth(DATE '1997-02-01')",1,0],
  831.    ["DAYOFWEEK","dayofweek","dayofweek(DATE '1997-02-01')",7,0],
  832.    ["DAYOFYEAR","dayofyear","dayofyear(DATE '1997-02-01')",32,0],
  833.    ["QUARTER","quarter","quarter(DATE '1997-02-01')",1,0],
  834.    ["WEEK","week","week(DATE '1997-02-01')",5,0],
  835.    ["YEAR","year","year(DATE '1997-02-01')",1997,0],
  836.    ["CURTIME","curtime","curtime()",0,2],
  837.    ["HOUR","hour","hour('12:13:14')",12,0],
  838.    ["ANSI HOUR","hour_time","hour(TIME '12:13:14')",12,0],
  839.    ["MINUTE","minute","minute('12:13:14')",13,0],
  840.    ["SECOND","second","second('12:13:14')",14,0],
  841.    ["TIMESTAMPADD","timestampadd",
  842.     "timestampadd(SQL_TSI_SECOND,1,'1997-01-01 00:00:00')",
  843.     "1997-01-01 00:00:01",1],
  844.    ["TIMESTAMPDIFF","timestampdiff",
  845.     "timestampdiff(SQL_TSI_SECOND,'1997-01-01 00:00:02', '1997-01-01 00:00:01')","1",0],
  846.    ["USER()","user()","user()",0,2],
  847.    ["DATABASE","database","database()",0,2],
  848.    ["IFNULL","ifnull","ifnull(2,3)",2,0],
  849.    ["ODBC syntax LEFT & RIGHT", "fn_left",
  850.     "{ fn LEFT( { fn RIGHT('abcd',2) },1) }","c",1],
  851.    );
  852.  
  853. @extra_functions=
  854.   (
  855.    ["& (bitwise and)",'&',"5 & 3",1,0],
  856.    ["| (bitwise or)",'|',"1 | 2",3,0],
  857.    ["<< and >> (bitwise shifts)",'binary_shifts',"(1 << 4) >> 2",4,0],
  858.    ["<> in SELECT","<>","1<>1","0",0],
  859.    ["=","=","(1=1)",1,$logical_value],
  860.    ["~* (case insensitive compare)","~*","'hi' ~* 'HI'",1,$logical_value],
  861.    ["ADD_MONTHS","add_months","add_months('1997-01-01',1)","1997-02-01",0], # oracle the date plus n months
  862.    ["AND and OR in SELECT","and_or","1=1 AND 2=2",$logical_value,0],
  863.    ["AND as '&&'",'&&',"1=1 && 2=2",$logical_value,0],
  864.    ["ASCII_CHAR", "ascii_char", "ASCII_CHAR(65)","A",1],
  865.    ["ASCII_CODE", "ascii_code", "ASCII_CODE('A')","65",0],
  866.    ["ATN2","atn2","atn2(1,0)","1.570796",0],
  867.    ["BETWEEN in SELECT","between","5 between 4 and 6",$logical_value,0],
  868.    ["BIT_COUNT","bit_count","bit_count(5)",2,0],
  869.    ["CEIL","ceil","ceil(-4.5)",-4,0], # oracle
  870.    ["CHARINDEX","charindex","charindex('a','crash')",3,0],
  871.    ["CHR", "chr", "CHR(65)"  ,"A",1], # oracle
  872.    ["CONCAT(list)","concat_list", "concat('a','b','c','d')","abcd",1],
  873.    ["CONVERT","convert","convert(CHAR,5)","5",1],
  874.    ["COSH","cosh","cosh(0)","1",0], # oracle hyperbolic cosine of n.
  875.    ["DATEADD","dateadd","dateadd(day,3,'Nov 30 1997')",0,2],
  876.    ["DATEDIFF","datediff","datediff(month,'Oct 21 1997','Nov 30 1997')",0,2],
  877.    ["DATENAME","datename","datename(month,'Nov 30 1997')",0,2],
  878.    ["DATEPART","datepart","datepart(month,'July 20 1997')",0,2],
  879.    ["DATE_FORMAT","date_format", "date_format('1997-01-02 03:04:05','M W D Y y m d h i s w')", 0,2],
  880.    ["ELT","elt","elt(2,'ONE','TWO','THREE')","TWO",1],
  881.    ["ENCRYPT","encrypt","encrypt('hello')",0,2],
  882.    ["FIELD","field","field('IBM','NCA','ICL','SUN','IBM','DIGITAL')",4,0],
  883.    ["FORMAT","format","format(1234.5555,2)","1,234.56",1],
  884.    ["FROM_DAYS","from_days","from_days(729024)","1996-01-01",1],
  885.    ["FROM_UNIXTIME","from_unixtime","from_unixtime(0)",0,2],
  886.    ["GETDATE","getdate","getdate()",0,2],
  887.    ["GREATEST","greatest","greatest('HARRY','HARRIOT','HAROLD')","HARRY",1], # oracle
  888.    ["IF","if", "if(5,6,7)",6,0],
  889.    ["IN on numbers in SELECT","in_num","2 in (3,2,5,9,5,1)",$logical_value,0],
  890.    ["IN on strings in SELECT","in_str","'monty' in ('david','monty','allan')", $logical_value,0],
  891.    ["INITCAP","initcap","initcap('the soap')","The Soap",1], # oracle Returns char, with the first letter of each word in uppercase
  892.    ["INSTR (Oracle syntax)", "instr_oracle", "INSTR('CORPORATE FLOOR','OR',3,2)"  ,"14",0], # oracle instring
  893.    ["INSTRB", "instrb", "INSTRB('CORPORATE FLOOR','OR',5,2)"  ,"27",0], # oracle instring in bytes
  894.    ["INTERVAL","interval","interval(55,10,20,30,40,50,60,70,80,90,100)",5,0],
  895.    ["LAST_DAY","last_day","last_day('1997-04-01')","1997-04-30",0], # oracle last day of month of date
  896.    ["LAST_INSERT_ID","last_insert_id","last_insert_id()",0,2],
  897.    ["LEAST","least","least('HARRY','HARRIOT','HAROLD')","HAROLD",1], # oracle
  898.    ["LENGTHB","lengthb","lengthb('CANDIDE')","14",0], # oracle length in bytes
  899.    ["LIKE ESCAPE in SELECT","like_escape","'%' like 'a%' escape 'a'",$logical_value,0],
  900.    ["LIKE in SELECT","like","'a' like 'a%'",$logical_value,0],
  901.    ["LN","ln","ln(95)","4.55387689",0], # oracle natural logarithm of n
  902.    ["LOCATE as INSTR","instr","instr('hello','ll')",3,0],
  903.    ["LOG(m,n)","log(m_n)","log(10,100)","2",0], # oracle logarithm, base m, of n
  904.    ["LOGN","logn","logn(2)","0.693147",0], # informix
  905.    ["LPAD","lpad","lpad('hi',4,'??')",'??hi',3],
  906.    ["MDY","mdy","mdy(7,1,1998)","1998-07-01",0], # informix
  907.    ["MOD as %","%","10%7","3",0],
  908.    ["MONTHS_BETWEEN","months_between","months_between('1997-02-02','1997-01-01')","1.03225806",0], # oracle number of months between 2 dates
  909.    ["NOT BETWEEN in SELECT","not_between","5 not between 4 and 6",0,0],
  910.    ["NOT LIKE in SELECT","not_like","'a' not like 'a%'",0,0],
  911.    ["NOT as '!' in SELECT","!","! 1",0,0],
  912.    ["NOT in SELECT","not","not $false",$logical_value,0],
  913.    ["ODBC CONVERT","odbc_convert","convert(5,SQL_CHAR)","5",1],
  914.    ["OR as '||'",'||',"1=0 || 1=1",$logical_value,0],
  915.    ["PASSWORD","password","password('hello')",0,2],
  916.    ["PASTE", "paste", "paste('ABCDEFG',3,2,'1234')","AB1234EFG",1],
  917.    ["PATINDEX","patindex","patindex('%a%','crash')",3,0],
  918.    ["PERIOD_ADD","period_add","period_add(9602,-12)",199502,0],
  919.    ["PERIOD_DIFF","period_diff","period_diff(199505,199404)",13,0],
  920.    ["POW","pow","pow(3,2)",9,0],
  921.    ["RANGE","range","range(a)","0.0",0], # informix range(a) = max(a) - min(a)
  922.    ["REGEXP in SELECT","regexp","'a' regexp '^(a|b)*\$'",$logical_value,0],
  923.    ["REPLICATE","replicate","replicate('a',5)","aaaaa",1],
  924.    ["REVERSE","reverse","reverse('abcd')","dcba",1],
  925.    ["ROOT","root","root(4)",2,0], # informix
  926.    ["ROUND(1 arg)","round1","round(5.63)","6",0],
  927.    ["RPAD","rpad","rpad('hi',4,'??')",'hi??',3],
  928.    ["SEC_TO_TIME","sec_to_time","sec_to_time(5001)","01:23:21",1],
  929.    ["SINH","sinh","sinh(1)","1.17520119",0], # oracle hyperbolic sine of n
  930.    ["STR","str","str(123.45,5,1)",123.5,3],
  931.    ["STRCMP","strcmp","strcmp('abc','adc')",-1,0],
  932.    ["STUFF","stuff","stuff('abc',2,3,'xyz')",'axyz',3],
  933.    ["SUBSTRB", "substrb", "SUBSTRB('ABCDEFG',5,4.2)"  ,"CD",1], # oracle substring with bytes
  934.    ["SUBSTRING as MID","mid","mid('hello',3,2)","ll",1],
  935.    ["SUBSTRING_INDEX","substring_index","substring_index('www.tcx.se','.',-2)", "tcx.se",1],
  936.    ["SYSDATE","sysdate","sysdate()",0,2],
  937.    ["TAIL","tail","tail('ABCDEFG',3)","EFG",0],
  938.    ["TANH","tanh","tanh(1)","0.462117157",0], # oracle hyperbolic tangent of n
  939.    ["TIME_TO_SEC","time_to_sec","time_to_sec('01:23:21')","5001",0],
  940.    ["TO_DAYS","to_days","to_days(DATE '1996-01-01')",729024,0],
  941.    ["TRANSLATE","translate","translate('abc','bc','de')",'ade',3],
  942.    ["TRIM; Many char extension","trim_many_char","trim(':!' FROM ':abc!')","abc",3],
  943.    ["TRIM; Substring extension","trim_substring","trim('cb' FROM 'abccb')","abc",3],
  944.    ["TRUNC","trunc","trunc(18.18,-1)",10,0], # oracle
  945.    ["UID","uid","uid",0,2], # oracle uid from user
  946.    ["UNIX_TIMESTAMP","unix_timestamp","unix_timestamp()",0,2],
  947.    ["USERENV","userenv","userenv",0,2], # oracle user enviroment
  948.    ["VERSION","version","version()",0,2],
  949.    ["WEEKDAY","weekday","weekday(DATE '1997-11-29')",5,0],
  950.    ["automatic num->string convert","auto_num2string","concat('a',2)","a2",1],
  951.    ["automatic string->num convert","auto_string2num","'1'+2",3,0],
  952.    ["concatenation with +","concat_as_+","'abc' + 'def'","abcdef",1],
  953.    );
  954.  
  955. @sql_group_functions=
  956.   (
  957.    ["AVG","avg","avg(a)",1,0],
  958.    ["COUNT (*)","count_*","count(*)",1,0],
  959.    ["COUNT column name","count_column","count(a)",1,0],
  960.    ["COUNT(DISTINCT expr)","count_distinct","count(distinct a)",1,0],
  961.    ["MAX on numbers","max","max(a)",1,0],
  962.    ["MAX on strings","max_str","max(b)","a",1],
  963.    ["MIN on numbers","min","min(a)",1,0],
  964.    ["MIN on strings","min_str","min(b)","a",1],
  965.    ["SUM","sum","sum(a)",1,0],
  966.    ["ANY","any","any(a)",$logical_value,0],
  967.    ["EVERY","every","every(a)",$logical_value,0],
  968.    ["SOME","some","some(a)",$logical_value,0],
  969.    );
  970.  
  971. @extra_group_functions=
  972.   (
  973.    ["BIT_AND",'bit_and',"bit_and(a)",1,0],
  974.    ["BIT_OR", 'bit_or', "bit_or(a)",1,0],
  975.    ["COUNT(DISTINCT expr,expr,...)","count_distinct_list","count(distinct a,b)",1,0],
  976.    ["STD","std","std(a)",0,0],
  977.    ["STDDEV","stddev","stddev(a)",0,0],
  978.    ["VARIANCE","variance","variance(a)",0,0],
  979.    );
  980.  
  981. @where_functions=
  982. (
  983.  ["= ALL","eq_all","b =all (select b from crash_me)",1,0],
  984.  ["= ANY","eq_any","b =any (select b from crash_me)",1,0],
  985.  ["= SOME","eq_some","b =some (select b from crash_me)",1,0],
  986.  ["BETWEEN","between","5 between 4 and 6",1,0],
  987.  ["EXISTS","exists","exists (select * from crash_me)",1,0],
  988.  ["IN on numbers","in_num","2 in (3,2,5,9,5,1)",1,0],
  989.  ["LIKE ESCAPE","like_escape","b like '%' escape 'a'",1,0],
  990.  ["LIKE","like","b like 'a%'",1,0],
  991.  ["MATCH UNIQUE","match_unique","1 match unique (select a from crash_me)",1,0],
  992.  ["MATCH","match","1 match (select a from crash_me)",1,0],
  993.  ["MATCHES","matches","b matcjhes 'a*'",1,0],
  994.  ["NOT BETWEEN","not_between","7 not between 4 and 6",1,0],
  995.  ["NOT EXISTS","not_exists","not exists (select * from crash_me where a = 2)",1,0],
  996.  ["NOT LIKE","not_like","b not like 'b%'",1,0],
  997.  ["NOT UNIQUE","not_unique","not unique (select * from crash_me where a = 2)",1,0],
  998.  ["UNIQUE","unique","unique (select * from crash_me)",1,0],
  999.  );
  1000.  
  1001. @types=(["sql",\@sql_functions,0],
  1002.     ["odbc",\@odbc_functions,0],
  1003.     ["extra",\@extra_functions,0],
  1004.     ["where",\@where_functions,0]);
  1005.  
  1006. @group_types=(["sql",\@sql_group_functions,0],
  1007.           ["extra",\@extra_group_functions,0]);
  1008.  
  1009.  
  1010. foreach $types (@types)
  1011. {
  1012.   print "\nSupported $types->[0] functions\n";
  1013.   $tmp=@$types->[1];
  1014.   foreach $type (@$tmp)
  1015.   {
  1016.     if (defined($limits{"func_$types->[0]_$type->[1]"}))
  1017.     {
  1018.       next;
  1019.     }
  1020.     if ($types->[0] eq "where")
  1021.     {
  1022.       check_and_report("Function $type->[0]","func_$types->[0]_$type->[1]",
  1023.                [],"select a from crash_me where $type->[2]",[],
  1024.                $type->[3],$type->[4]);
  1025.     }
  1026.     elsif ($limits{'functions'} eq 'yes')
  1027.     {
  1028.       if (($type->[2] =~ /char_length\(b\)/) && (!$end_query))
  1029.       {
  1030.     my $tmp= $type->[2];
  1031.     $tmp .= " from crash_me ";
  1032.     undef($limits{"func_$types->[0]_$type->[1]"});
  1033.     check_and_report("Function $type->[0]",
  1034.              "func_$types->[0]_$type->[1]",
  1035.              [],"select $tmp ",[],
  1036.              $type->[3],$type->[4]);
  1037.       }
  1038.       else
  1039.       {
  1040.     undef($limits{"func_$types->[0]_$type->[1]"});
  1041.     $result = check_and_report("Function $type->[0]",
  1042.                 "func_$types->[0]_$type->[1]",
  1043.                 [],"select $type->[2] $end_query",[],
  1044.                 $type->[3],$type->[4]);
  1045.     if (!$result)
  1046.     {
  1047.       # check without type specifyer
  1048.       if ($type->[2] =~ /DATE /)
  1049.       {
  1050.         my $tmp= $type->[2];
  1051.         $tmp =~ s/DATE //;
  1052.         undef($limits{"func_$types->[0]_$type->[1]"});
  1053.         $result = check_and_report("Function $type->[0]",
  1054.                   "func_$types->[0]_$type->[1]",
  1055.                   [],"select $tmp $end_query",[],
  1056.                   $type->[3],$type->[4]);
  1057.       }
  1058.       if (!$result)
  1059.       {
  1060.         if ($types->[0] eq "odbc" && ! ($type->[2] =~ /\{fn/))
  1061.         {
  1062.          my $tmp= $type->[2];
  1063.          # Check by converting to ODBC format
  1064.          undef($limits{"func_$types->[0]_$type->[1]"});
  1065.          $tmp= "{fn $tmp }";
  1066.          $tmp =~ s/('1997-\d\d-\d\d \d\d:\d\d:\d\d')/{ts $1}/g;
  1067.          $tmp =~ s/(DATE '1997-\d\d-\d\d')/{d $1}/g;
  1068.          $tmp =~ s/(TIME '12:13:14')/{t $1}/g;
  1069.          $tmp =~ s/DATE //;
  1070.          $tmp =~ s/TIME //;
  1071.          check_and_report("Function $type->[0]",
  1072.                   "func_$types->[0]_$type->[1]",
  1073.                   [],"select $tmp $end_query",[],
  1074.                   $type->[3],$type->[4]);
  1075.         }
  1076.       }
  1077.         }
  1078.       }
  1079.     }
  1080.   }
  1081. }
  1082.  
  1083. if ($limits{'functions'} eq 'yes')
  1084. {
  1085.   foreach $types (@group_types)
  1086.   {
  1087.     print "\nSupported $types->[0] group functions\n";
  1088.     $tmp=@$types->[1];
  1089.     foreach $type (@$tmp)
  1090.     {
  1091.       check_and_report("Group function $type->[0]",
  1092.                "group_func_$types->[0]_$type->[1]",
  1093.                [],"select $type->[2],a from crash_me group by a",[],
  1094.                $type->[3],$type->[4]);
  1095.     }
  1096.   }
  1097.   print "\n";
  1098.   report("mixing of integer and float in expression","float_int_expr",
  1099.      "select 1+1.0 $end_query");
  1100.   if ($limits{'func_odbc_exp'} eq 'yes')
  1101.   {
  1102.     report("No need to cast from integer to float",
  1103.        "dont_require_cast_to_float", "select exp(1) $end_query");
  1104.   }
  1105.   check_and_report("Is 1+NULL = NULL","null_num_expr",
  1106.            [],"select 1+$numeric_null $end_query",[],undef(),4);
  1107.   $tmp=sql_concat("'a'",$char_null);
  1108.   if (defined($tmp))
  1109.   {
  1110.     check_and_report("Is $tmp = NULL", "null_concat_expr", [],
  1111.              "select $tmp $end_query",[], undef(),4);
  1112.   }
  1113.   $prompt="Need to cast NULL for arithmetic";
  1114.   save_config_data("Need_cast_for_null",
  1115.            ($numeric_null eq "NULL") ? "no" : "yes",
  1116.            $prompt);
  1117. }
  1118. else
  1119. {
  1120.   print "\n";
  1121. }
  1122.  
  1123.  
  1124. report("LIKE on numbers","like_with_number",
  1125.        "create table crash_q (a int,b int)",
  1126.        "insert into crash_q values(10,10)",
  1127.        "select * from crash_q where a like '10'",
  1128.        "drop table crash_q $drop_attr");
  1129.  
  1130. report("column LIKE column","like_with_column",
  1131.        "create table crash_q (a char(10),b char(10))",
  1132.        "insert into crash_q values('abc','abc')",
  1133.        "select * from crash_q where a like b",
  1134.        "drop table crash_q $drop_attr");
  1135.  
  1136. report("update of column= -column","NEG",
  1137.        "create table crash_q (a integer)",
  1138.        "insert into crash_q values(10)",
  1139.        "update crash_q set a=-a",
  1140.        "drop table crash_q $drop_attr");
  1141.  
  1142. if ($limits{'func_odbc_left'} eq 'yes' ||
  1143.     $limits{'func_odbc_substring'} eq 'yes')
  1144. {
  1145.   my $type= ($limits{'func_odbc_left'} eq 'yes' ?
  1146.          "left(a,4)" : "substring(a for 4)");
  1147.  
  1148.     check_and_report("String functions on date columns","date_as_string",
  1149.              ["create table crash_me2 (a date not null)",
  1150.               "insert into crash_me2 values ('1998-03-03')"],
  1151.              "select $type from crash_me2",
  1152.              ["drop table crash_me2 $drop_attr"],
  1153.              "1998",1);
  1154. }
  1155.  
  1156.  
  1157. $tmp=sql_concat("b","b");
  1158. if (defined($tmp))
  1159. {
  1160.   check_and_report("char are space filled","char_is_space_filled",
  1161.            [],"select $tmp from crash_me where b = 'a         '",[],
  1162.            'a         a         ',6);
  1163. }
  1164.  
  1165. if (!defined($limits{'multi_table_update'}))
  1166. {
  1167.   if (check_and_report("Update with many tables","multi_table_update",
  1168.            ["create table crash_q (a integer,b char(10))",
  1169.             "insert into crash_q values(1,'c')",
  1170.             "update crash_q left join crash_me on crash_q.a=crash_me.a set crash_q.b=crash_me.b"],
  1171.            "select b from crash_q",
  1172.            ["drop table crash_q $drop_attr"],
  1173.            "a",1,undef(),2))
  1174.   {
  1175.     check_and_report("Update with many tables","multi_table_update",
  1176.              ["create table crash_q (a integer,b char(10))",
  1177.               "insert into crash_q values(1,'c')",
  1178.               "update crash_q,crash_me set crash_q.b=crash_me.b where crash_q.a=crash_me.a"],
  1179.              "select b from crash_q",
  1180.              ["drop table crash_q $drop_attr"],
  1181.              "a",1,
  1182.             1);
  1183.   }
  1184. }
  1185.  
  1186. report("DELETE FROM table1,table2...","multi_table_delete",
  1187.        "create table crash_q (a integer,b char(10))",
  1188.        "insert into crash_q values(1,'c')",
  1189.        "delete crash_q.* from crash_q,crash_me where crash_q.a=crash_me.a",
  1190.        "drop table crash_q $drop_attr");
  1191.  
  1192. check_and_report("Update with sub select","select_table_update",
  1193.          ["create table crash_q (a integer,b char(10))",
  1194.           "insert into crash_q values(1,'c')",
  1195.           "update crash_q set b= (select b from crash_me where crash_q.a = crash_me.a)"],
  1196.          "select b from crash_q",
  1197.          ["drop table crash_q $drop_attr"],
  1198.          "a",1);
  1199.  
  1200. check_and_report("Calculate 1--1","minus_neg",[],
  1201.          "select a--1 from crash_me",[],0,2);
  1202.  
  1203. report("ANSI SQL simple joins","simple_joins",
  1204.        "select crash_me.a from crash_me, crash_me t0");
  1205.  
  1206. #
  1207. # Check max string size, and expression limits
  1208. #
  1209. $found=undef;
  1210. foreach $type (('mediumtext','text','text()','blob','long'))
  1211. {
  1212.   if ($limits{"type_extra_$type"} eq 'yes')
  1213.   {
  1214.     $found=$type;
  1215.     last;
  1216.   }
  1217. }
  1218. if (defined($found))
  1219. {
  1220.   $found =~ s/\(\)/\(%d\)/;
  1221.   find_limit("max text or blob size","max_text_size",
  1222.          new query_many(["create table crash_q (q $found)",
  1223.                  "insert into crash_q values ('%s')"],
  1224.                 "select * from crash_q","%s",
  1225.                 ["drop table crash_q $drop_attr"],
  1226.                 min($max_string_size,$limits{'query_size'}-30)));
  1227.  
  1228. }
  1229.  
  1230. # It doesn't make lots of sense to check for string lengths much bigger than
  1231. # what can be stored...
  1232.  
  1233. find_limit(($prompt="constant string size in where"),"where_string_size",
  1234.        new query_repeat([],"select a from crash_me where b >='",
  1235.                 "","","1","","'"));
  1236. if ($limits{'where_string_size'} == 10)
  1237. {
  1238.   save_config_data('where_string_size','nonstandard',$prompt);
  1239. }
  1240.  
  1241. if ($limits{'select_constants'} eq 'yes')
  1242. {
  1243.   find_limit("constant string size in SELECT","select_string_size",
  1244.          new query_repeat([],"select '","","","a","","'$end_query"));
  1245. }
  1246.  
  1247. goto no_functions if ($limits{'functions'} ne "yes");
  1248.  
  1249. if ($limits{'func_odbc_repeat'} eq 'yes')
  1250. {
  1251.   find_limit("return string size from function","repeat_string_size",
  1252.          new query_many([],
  1253.                 "select repeat('a',%d) $end_query","%s",
  1254.                 [],
  1255.                 $max_string_size,0));
  1256. }
  1257.  
  1258. $tmp=find_limit("simple expressions","max_expressions",
  1259.         new query_repeat([],"select 1","","","+1","",$end_query,
  1260.                  undef(),$max_expressions));
  1261.  
  1262. if ($tmp > 10)
  1263. {
  1264.   $tmp= "(1" . ( '+1' x ($tmp-10) ) . ")";
  1265.   find_limit("big expressions", "max_big_expressions",
  1266.          new query_repeat([],"select 0","","","+$tmp","",$end_query,
  1267.                   undef(),$max_big_expressions));
  1268. }
  1269.  
  1270. find_limit("stacked expressions", "max_stack_expression",
  1271.        new query_repeat([],"select 1","","","+(1",")",$end_query,
  1272.                 undef(),$max_stacked_expressions));
  1273.  
  1274. no_functions:
  1275.  
  1276. if (!defined($limits{'max_conditions'}))
  1277. {
  1278.   find_limit("OR and AND in WHERE","max_conditions",
  1279.          new query_repeat([],
  1280.                   "select a from crash_me where a=1 and b='a'","",
  1281.                   "", " or a=%d and b='%d'","","","",
  1282.                   [],($query_size-42)/29,undef,2));
  1283.   $limits{'max_conditions'}*=2;
  1284. }
  1285. # The 42 is the length of the constant part.
  1286. # The 29 is the length of the variable part, plus two seven-digit numbers.
  1287.  
  1288. find_limit("tables in join", "join_tables",
  1289.        new query_repeat([],
  1290.                 "select crash_me.a",",t%d.a","from crash_me",
  1291.                 ",crash_me t%d","","",[],$max_join_tables,undef,
  1292.                 1));
  1293.  
  1294. # Different CREATE TABLE options
  1295.  
  1296. report("primary key in create table",'primary_key_in_create',
  1297.        "create table crash_q (q integer not null,primary key (q))",
  1298.        "drop table crash_q $drop_attr");
  1299.  
  1300. report("unique in create table",'unique_in_create',
  1301.        "create table crash_q (q integer not null,unique (q))",
  1302.        "drop table crash_q $drop_attr");
  1303.  
  1304. if ($limits{'unique_in_create'} eq 'yes')
  1305. {
  1306.   report("unique null in create",'unique_null_in_create',
  1307.      "create table crash_q (q integer,unique (q))",
  1308.      "insert into crash_q (q) values (NULL)",
  1309.      "insert into crash_q (q) values (NULL)",
  1310.      "insert into crash_q (q) values (1)",
  1311.      "drop table crash_q $drop_attr");
  1312. }
  1313.  
  1314. report("default value for column",'create_default',
  1315.        "create table crash_q (q integer default 10 not null)",
  1316.        "drop table crash_q $drop_attr");
  1317.  
  1318. report("default value function for column",'create_default_func',
  1319.        "create table crash_q (q integer not null,q1 integer default (1+1))",
  1320.        "drop table crash_q $drop_attr");
  1321.  
  1322. report("temporary tables",'temporary_table',
  1323.        "create temporary table crash_q (q integer not null)",
  1324.        "drop table crash_q $drop_attr");
  1325.  
  1326. report_one("create table from select",'create_table_select',
  1327.        [["create table crash_q SELECT * from crash_me","yes"],
  1328.         ["create table crash_q AS SELECT * from crash_me","with AS"]]);
  1329. $dbh->do("drop table crash_q $drop_attr");
  1330.  
  1331. report("index in create table",'index_in_create',
  1332.        "create table crash_q (q integer not null,index (q))",
  1333.        "drop table crash_q $drop_attr");
  1334.  
  1335. # The following must be executed as we need the value of end_drop_keyword
  1336. # later
  1337. if (!(defined($limits{'create_index'}) && defined($limits{'drop_index'})))
  1338. {
  1339.   if ($res=safe_query("create index crash_q on crash_me (a)"))
  1340.   {
  1341.     $res="yes";
  1342.     $drop_res="yes";
  1343.     $end_drop_keyword="";
  1344.     if (!safe_query("drop index crash_q"))
  1345.     {
  1346.       # Can't drop the standard way; Check if mSQL
  1347.       if (safe_query("drop index crash_q from crash_me"))
  1348.       {
  1349.         $drop_res="with 'FROM'";    # Drop is not ANSI SQL
  1350.         $end_drop_keyword="drop index %i from %t";
  1351.       }
  1352.       # else check if Access or MySQL
  1353.       elsif (safe_query("drop index crash_q on crash_me"))
  1354.       {
  1355.         $drop_res="with 'ON'";    # Drop is not ANSI SQL
  1356.         $end_drop_keyword="drop index %i on %t";
  1357.       }
  1358.       # else check if MS-SQL
  1359.       elsif (safe_query("drop index crash_me.crash_q"))
  1360.       {
  1361.         $drop_res="with 'table.index'"; # Drop is not ANSI SQL
  1362.         $end_drop_keyword="drop index %t.%i";
  1363.       }
  1364.     }
  1365.     else
  1366.     {
  1367.       # Old MySQL 3.21 supports only the create index syntax
  1368.       # This means that the second create doesn't give an error.
  1369.       $res=safe_query(["create index crash_q on crash_me (a)",
  1370.                    "create index crash_q on crash_me (a)",
  1371.                    "drop index crash_q"]);
  1372.       $res= $res ? 'ignored' : 'yes';
  1373.     }
  1374.   }
  1375.   else
  1376.   {
  1377.     $drop_res=$res='no';
  1378.   }
  1379.   save_config_data('create_index',$res,"create index");
  1380.   save_config_data('drop_index',$drop_res,"drop index");
  1381.  
  1382.   print "create index: $limits{'create_index'}\n";
  1383.   print "drop index: $limits{'drop_index'}\n";
  1384. }
  1385.  
  1386. # check if we can have 'NULL' as a key
  1387. check_and_report("null in index","null_in_index",
  1388.          [create_table("crash_q",["a char(10)"],["(a)"]),
  1389.           "insert into crash_q values (NULL)"],
  1390.          "select * from crash_q",
  1391.          ["drop table crash_q $drop_attr"],
  1392.          undef(),4);
  1393.  
  1394. if ($limits{'unique_in_create'} eq 'yes')
  1395. {
  1396.   report("null in unique index",'null_in_unique',
  1397.           create_table("crash_q",["q integer"],["unique(q)"]),
  1398.      "insert into crash_q (q) values(NULL)",
  1399.      "insert into crash_q (q) values(NULL)",
  1400.      "drop table crash_q $drop_attr");
  1401.   report("null combination in unique index",'nulls_in_unique',
  1402.           create_table("crash_q",["q integer,q1 integer"],["unique(q,q1)"]),
  1403.      "insert into crash_q (q,q1) values(1,NULL)",
  1404.      "insert into crash_q (q,q1) values(1,NULL)",
  1405.      "drop table crash_q $drop_attr");
  1406. }
  1407.  
  1408. if ($limits{'null_in_unique'} eq 'yes')
  1409. {
  1410.   report("null in unique index",'multi_null_in_unique',
  1411.           create_table("crash_q",["q integer, x integer"],["unique(q)"]),
  1412.      "insert into crash_q(x) values(1)",
  1413.      "insert into crash_q(x) values(2)",
  1414.      "drop table crash_q $drop_attr");
  1415. }
  1416.  
  1417. if ($limits{'create_index'} ne 'no')
  1418. {
  1419.   $end_drop=$end_drop_keyword;
  1420.   $end_drop =~ s/%i/crash_q/;
  1421.   $end_drop =~ s/%t/crash_me/;
  1422.   report("index on column part (extension)","index_parts",,
  1423.      "create index crash_q on crash_me (b(5))",
  1424.      $end_drop);
  1425.   $end_drop=$end_drop_keyword;
  1426.   $end_drop =~ s/%i/crash_me/;
  1427.   $end_drop =~ s/%t/crash_me/;
  1428.   report("different namespace for index",
  1429.      "index_namespace",
  1430.      "create index crash_me on crash_me (b)",
  1431.      $end_drop);
  1432. }
  1433.  
  1434. if (!report("case independent table names","table_name_case",
  1435.         "create table crash_q (q integer)",
  1436.         "drop table CRASH_Q $drop_attr"))
  1437. {
  1438.   safe_query("drop table crash_q $drop_attr");
  1439. }
  1440.  
  1441. if (!report("drop table if exists","drop_if_exists",
  1442.         "create table crash_q (q integer)",
  1443.         "drop table if exists crash_q $drop_attr"))
  1444. {
  1445.   safe_query("drop table crash_q $drop_attr");
  1446. }
  1447.  
  1448. report("create table if not exists","create_if_not_exists",
  1449.        "create table crash_q (q integer)",
  1450.        "create table if not exists crash_q (q integer)");
  1451. safe_query("drop table crash_q $drop_attr");
  1452.  
  1453. #
  1454. # test of different join types
  1455. #
  1456.  
  1457. assert("create table crash_me2 (a integer not null,b char(10) not null, c1 integer)");
  1458. assert("insert into crash_me2 (a,b,c1) values (1,'b',1)");
  1459. assert("create table crash_me3 (a integer not null,b char(10) not null)");
  1460. assert("insert into crash_me3 (a,b) values (1,'b')");
  1461.  
  1462. report("inner join","inner_join",
  1463.        "select crash_me.a from crash_me inner join crash_me2 ON crash_me.a=crash_me2.a");
  1464. report("left outer join","left_outer_join",
  1465.        "select crash_me.a from crash_me left join crash_me2 ON crash_me.a=crash_me2.a");
  1466. report("natural left outer join","natural_left_outer_join",
  1467.        "select c1 from crash_me natural left join crash_me2");
  1468. report("left outer join using","left_outer_join_using",
  1469.        "select c1 from crash_me left join crash_me2 using (a)");
  1470. report("left outer join odbc style","odbc_left_outer_join",
  1471.        "select crash_me.a from { oj crash_me left outer join crash_me2 ON crash_me.a=crash_me2.a }");
  1472. report("right outer join","right_outer_join",
  1473.        "select crash_me.a from crash_me right join crash_me2 ON crash_me.a=crash_me2.a");
  1474. report("full outer join","full_outer_join",
  1475.        "select crash_me.a from crash_me full join crash_me2 ON crash_me.a=crash_me2.a");
  1476. report("cross join (same as from a,b)","cross_join",
  1477.        "select crash_me.a from crash_me cross join crash_me3");
  1478. report("natural join","natural_join",
  1479.        "select * from crash_me natural join crash_me3");
  1480. report("union","union",
  1481.        "select * from crash_me union select a,b from crash_me3");
  1482. report("union all","union_all",
  1483.        "select * from crash_me union all select a,b from crash_me3");
  1484. report("intersect","intersect",
  1485.        "select * from crash_me intersect select * from crash_me3");
  1486. report("intersect all","intersect_all",
  1487.        "select * from crash_me intersect all select * from crash_me3");
  1488. report("except","except",
  1489.        "select * from crash_me except select * from crash_me3");
  1490. report("except all","except_all",
  1491.        "select * from crash_me except all select * from crash_me3");
  1492. report("except","except",
  1493.        "select * from crash_me except select * from crash_me3");
  1494. report("except all","except_all",
  1495.        "select * from crash_me except all select * from crash_me3");
  1496. report("minus","minus",
  1497.        "select * from crash_me minus select * from crash_me3"); # oracle ...
  1498.  
  1499. report("natural join (incompatible lists)","natural_join_incompat",
  1500.        "select c1 from crash_me natural join crash_me2");
  1501. report("union (incompatible lists)","union_incompat",
  1502.        "select * from crash_me union select a,b from crash_me2");
  1503. report("union all (incompatible lists)","union_all_incompat",
  1504.        "select * from crash_me union all select a,b from crash_me2");
  1505. report("intersect (incompatible lists)","intersect_incompat",
  1506.        "select * from crash_me intersect select * from crash_me2");
  1507. report("intersect all (incompatible lists)","intersect_all_incompat",
  1508.        "select * from crash_me intersect all select * from crash_me2");
  1509. report("except (incompatible lists)","except_incompat",
  1510.        "select * from crash_me except select * from crash_me2");
  1511. report("except all (incompatible lists)","except_all_incompat",
  1512.        "select * from crash_me except all select * from crash_me2");
  1513. report("except (incompatible lists)","except_incompat",
  1514.        "select * from crash_me except select * from crash_me2");
  1515. report("except all (incompatible lists)","except_all_incompat",
  1516.        "select * from crash_me except all select * from crash_me2");
  1517. report("minus (incompatible lists)","minus_incompat",
  1518.        "select * from crash_me minus select * from crash_me2"); # oracle ...
  1519.  
  1520. assert("drop table crash_me2 $drop_attr");
  1521. assert("drop table crash_me3 $drop_attr");
  1522.  
  1523. # somethings to be added here ....
  1524. # FOR UNION - INTERSECT - EXCEPT -> CORRESPONDING [ BY ]
  1525. # after subqueries:
  1526. # >ALL | ANY | SOME - EXISTS - UNIQUE
  1527.  
  1528. if (report("subqueries","subqueries",
  1529.        "select a from crash_me where crash_me.a in (select max(a) from crash_me)"))
  1530. {
  1531.     $tmp=new query_repeat([],"select a from crash_me","","",
  1532.               " where a in (select a from crash_me",")",
  1533.               "",[],$max_join_tables);
  1534.     find_limit("recursive subqueries", "recursive_subqueries",$tmp);
  1535. }
  1536.  
  1537. report("insert INTO ... SELECT ...","insert_select",
  1538.        "create table crash_q (a int)",
  1539.        "insert into crash_q (a) SELECT crash_me.a from crash_me",
  1540.        "drop table crash_q $drop_attr");
  1541.  
  1542. if (!defined($limits{"transactions"}))
  1543. {
  1544.   my ($limit,$type);
  1545.   $limit="transactions";
  1546.   print "$limit: ";
  1547.   foreach $type (('', 'type=bdb', 'type=innodb', 'type=gemini'))
  1548.   {
  1549.     undef($limits{$limit});
  1550.     last if (!report_trans($limit,
  1551.                [create_table("crash_q",["a integer not null"],[],
  1552.                      $type),
  1553.                 "insert into crash_q values (1)"],
  1554.                "select * from crash_q",
  1555.                "drop table crash_q $drop_attr"
  1556.               ));
  1557.   }
  1558.   print "$limits{$limit}\n";
  1559. }
  1560.  
  1561. report("atomic updates","atomic_updates",
  1562.        create_table("crash_q",["a integer not null"],["primary key (a)"]),
  1563.        "insert into crash_q values (2)",
  1564.        "insert into crash_q values (3)",
  1565.        "insert into crash_q values (1)",
  1566.        "update crash_q set a=a+1",
  1567.        "drop table crash_q $drop_attr");
  1568.  
  1569. if ($limits{'atomic_updates'} eq 'yes')
  1570. {
  1571.   report_fail("atomic_updates_with_rollback","atomic_updates_with_rollback",
  1572.           create_table("crash_q",["a integer not null"],
  1573.                ["primary key (a)"]),
  1574.           "insert into crash_q values (2)",
  1575.           "insert into crash_q values (3)",
  1576.           "insert into crash_q values (1)",
  1577.           "update crash_q set a=a+1 where a < 3",
  1578.           "drop table crash_q $drop_attr");
  1579. }
  1580.  
  1581. # To add with the views:
  1582. # DROP VIEW - CREAT VIEW *** [ WITH [ CASCADE | LOCAL ] CHECK OPTION ]
  1583. report("views","views",
  1584.        "create view crash_q as select a from crash_me",
  1585.        "drop view crash_q $drop_attr");
  1586.  
  1587. report("foreign key syntax","foreign_key_syntax",
  1588.        create_table("crash_q",["a integer not null"],["primary key (a)"]),
  1589.        create_table("crash_q2",["a integer not null",
  1590.                 "foreign key (a) references crash_q (a)"],
  1591.             []),
  1592.        "insert into crash_q values (1)",
  1593.        "insert into crash_q2 values (1)",
  1594.        "drop table crash_q2 $drop_attr",
  1595.        "drop table crash_q $drop_attr");
  1596.  
  1597. if ($limits{'foreign_key_syntax'} eq 'yes')
  1598. {
  1599.   report_fail("foreign keys","foreign_key",
  1600.           create_table("crash_q",["a integer not null"],
  1601.                ["primary key (a)"]),
  1602.           create_table("crash_q2",["a integer not null",
  1603.                        "foreign key (a) references crash_q (a)"],
  1604.                []),
  1605.           "insert into crash_q values (1)",
  1606.           "insert into crash_q2 values (2)",
  1607.           "drop table crash_q2 $drop_attr",
  1608.           "drop table crash_q $drop_attr");
  1609. }
  1610.  
  1611. report("Create SCHEMA","create_schema",
  1612.        "create schema crash_schema create table crash_q (a int) create table crash_q2(b int)",
  1613.        "drop schema crash_schema cascade");
  1614.  
  1615. if ($limits{'foreign_key'} eq 'yes')
  1616. {
  1617.   if ($limits{'create_schema'} eq 'yes')
  1618.   {
  1619.     report("Circular foreign keys","foreign_key_circular",
  1620.            "create schema crash_schema create table crash_q (a int primary key, b int, foreign key (b) references crash_q2(a)) create table crash_q2(a int, b int, primary key(a), foreign key (b) references crash_q(a))",
  1621.            "drop schema crash_schema cascade");
  1622.   }
  1623. }
  1624.  
  1625. report("Column constraints","constraint_check",
  1626.        "create table crash_q (a int check (a>0))",
  1627.        "drop table crash_q $drop_attr");
  1628.  
  1629. report("Table constraints","constraint_check_table",
  1630.        "create table crash_q (a int ,b int, check (a>b))",
  1631.        "drop table crash_q $drop_attr");
  1632.  
  1633. report("Named constraints","constraint_check",
  1634.        "create table crash_q (a int ,b int, constraint abc check (a>b))",
  1635.        "drop table crash_q $drop_attr");
  1636.  
  1637. report("NULL constraint (SyBase style)","constraint_null",
  1638.        "create table crash_q (a int null)",
  1639.        "drop table crash_q $drop_attr");
  1640.  
  1641. report("Triggers (ANSI SQL)","psm_trigger",
  1642.        "create table crash_q (a int ,b int)",
  1643.        "create trigger crash_trigger after insert on crash_q referencing new table as new_a when (localtime > time '18:00:00') begin atomic end",
  1644.        "insert into crash_q values(1,2)",
  1645.        "drop trigger crash_trigger",
  1646.        "drop table crash_q $drop_attr");
  1647.  
  1648. report("PSM procedures (ANSI SQL)","psm_procedures",
  1649.        "create table crash_q (a int,b int)",
  1650.        "create procedure crash_proc(in a1 int, in b1 int) language sql modifies sql data begin declare c1 int; set c1 = a1 + b1; insert into crash_q(a,b) values (a1,c1); end",
  1651.        "call crash_proc(1,10)",
  1652.        "drop procedure crash_proc",
  1653.        "drop table crash_q $drop_attr");
  1654.  
  1655. report("PSM modules (ANSI SQL)","psm_modules",
  1656.        "create table crash_q (a int,b int)",
  1657.        "create module crash_m declare procedure crash_proc(in a1 int, in b1 int) language sql modifies sql data begin declare c1 int; set c1 = a1 + b1; insert into crash_q(a,b) values (a1,c1); end; declare procedure crash_proc2(INOUT a int, in b int) contains sql set a = b + 10; end module",
  1658.        "call crash_proc(1,10)",
  1659.        "drop module crash_m cascade",
  1660.        "drop table crash_q cascade $drop_attr");
  1661.  
  1662. report("PSM functions (ANSI SQL)","psm_functions",
  1663.        "create table crash_q (a int)",
  1664.        "create function crash_func(in a1 int, in b1 int) returns int language sql deterministic contains sql begin return a1 * b1; end",
  1665.        "insert into crash_q values(crash_func(2,4))",
  1666.        "select a,crash_func(a,2) from crash_q",
  1667.        "drop function crash_func cascade",
  1668.        "drop table crash_q $drop_attr");
  1669.  
  1670. report("Domains (ANSI SQL)","domains",
  1671.        "create domain crash_d as varchar(10) default 'Empty' check (value <> 'abcd')",
  1672.        "create table crash_q(a crash_d, b int)",
  1673.        "insert into crash_q(a,b) values('xyz',10)",
  1674.        "insert into crash_q(b) values(10)",
  1675.        "drop table crash_q $drop_attr",
  1676.        "drop domain crash_d");
  1677.  
  1678.  
  1679. if (!defined($limits{'lock_tables'}))
  1680. {
  1681.   report("lock table","lock_tables",
  1682.      "lock table crash_me READ",
  1683.      "unlock tables");
  1684.   if ($limits{'lock_tables'} eq 'no')
  1685.   {
  1686.     delete $limits{'lock_tables'};
  1687.     report("lock table","lock_tables",
  1688.        "lock table crash_me IN SHARE MODE");
  1689.   }
  1690. }
  1691.  
  1692. if (!report("many tables to drop table","multi_drop",
  1693.        "create table crash_q (a int)",
  1694.        "create table crash_q2 (a int)",
  1695.        "drop table crash_q,crash_q2 $drop_attr"))
  1696. {
  1697.   $dbh->do("drop table crash_q $drop_attr");
  1698.   $dbh->do("drop table crash_q2 $drop_attr");
  1699. }
  1700.  
  1701. if (!report("drop table with cascade/restrict","drop_restrict",
  1702.        "create table crash_q (a int)",
  1703.        "drop table crash_q restrict"))
  1704. {
  1705.   $dbh->do("drop table crash_q $drop_attr");
  1706. }
  1707.  
  1708.  
  1709. report("-- as comment (ANSI)","comment_--",
  1710.        "select * from crash_me -- Testing of comments");
  1711. report("// as comment","comment_//",
  1712.        "select * from crash_me // Testing of comments");
  1713. report("# as comment","comment_#",
  1714.        "select * from crash_me # Testing of comments");
  1715. report("/* */ as comment","comment_/**/",
  1716.        "select * from crash_me /* Testing of comments */");
  1717.  
  1718. #
  1719. # Check things that fails one some servers
  1720. #
  1721.  
  1722. # Empress can't insert empty strings in a char() field
  1723. report("insert empty string","insert_empty_string",
  1724.        create_table("crash_q",["a char(10) not null,b char(10)"],[]),
  1725.        "insert into crash_q values ('','')",
  1726.        "drop table crash_q $drop_attr");
  1727.  
  1728. report("Having with alias","having_with_alias",
  1729.        create_table("crash_q",["a integer"],[]),
  1730.        "insert into crash_q values (10)",
  1731.        "select sum(a) as b from crash_q group by a having b > 0",
  1732.        "drop table crash_q $drop_attr");
  1733.  
  1734. #
  1735. # test name limits
  1736. #
  1737.  
  1738. find_limit("table name length","max_table_name",
  1739.        new query_many(["create table crash_q%s (q integer)",
  1740.                "insert into crash_q%s values(1)"],
  1741.                "select * from crash_q%s",1,
  1742.                ["drop table crash_q%s $drop_attr"],
  1743.                $max_name_length,7,1));
  1744.  
  1745. find_limit("column name length","max_column_name",
  1746.        new query_many(["create table crash_q (q%s integer)",
  1747.               "insert into crash_q (q%s) values(1)"],
  1748.               "select q%s from crash_q",1,
  1749.               ["drop table crash_q $drop_attr"],
  1750.                $max_name_length,1));
  1751.  
  1752. if ($limits{'column_alias'} eq 'yes')
  1753. {
  1754.   find_limit("select alias name length","max_select_alias_name",
  1755.        new query_many(undef,
  1756.               "select b as %s from crash_me",undef,
  1757.               undef, $max_name_length));
  1758. }
  1759.  
  1760. find_limit("table alias name length","max_table_alias_name",
  1761.        new query_many(undef,
  1762.               "select %s.b from crash_me %s",
  1763.               undef,
  1764.               undef, $max_name_length));
  1765.  
  1766. $end_drop_keyword = "drop index %i" if (!$end_drop_keyword);
  1767. $end_drop=$end_drop_keyword;
  1768. $end_drop =~ s/%i/crash_q%s/;
  1769. $end_drop =~ s/%t/crash_me/;
  1770.  
  1771. if ($limits{'create_index'} ne 'no')
  1772. {
  1773.   find_limit("index name length","max_index_name",
  1774.          new query_many(["create index crash_q%s on crash_me (a)"],
  1775.                 undef,undef,
  1776.                 [$end_drop],
  1777.                 $max_name_length,7));
  1778. }
  1779.  
  1780. find_limit("max char() size","max_char_size",
  1781.        new query_many(["create table crash_q (q char(%d))",
  1782.                "insert into crash_q values ('%s')"],
  1783.               "select * from crash_q","%s",
  1784.               ["drop table crash_q $drop_attr"],
  1785.               min($max_string_size,$limits{'query_size'})));
  1786.  
  1787. if ($limits{'type_sql_varchar(1_arg)'} eq 'yes')
  1788. {
  1789.   find_limit("max varchar() size","max_varchar_size",
  1790.          new query_many(["create table crash_q (q varchar(%d))",
  1791.                  "insert into crash_q values ('%s')"],
  1792.                 "select * from crash_q","%s",
  1793.                 ["drop table crash_q $drop_attr"],
  1794.                 min($max_string_size,$limits{'query_size'})));
  1795. }
  1796.  
  1797. $found=undef;
  1798. foreach $type (('mediumtext','text','text()','blob','long'))
  1799. {
  1800.   if ($limits{"type_extra_$type"} eq 'yes')
  1801.   {
  1802.     $found=$type;
  1803.     last;
  1804.   }
  1805. }
  1806. if (defined($found))
  1807. {
  1808.   $found =~ s/\(\)/\(%d\)/;
  1809.   find_limit("max text or blob size","max_text_size",
  1810.          new query_many(["create table crash_q (q $found)",
  1811.                  "insert into crash_q values ('%s')"],
  1812.                 "select * from crash_q","%s",
  1813.                 ["drop table crash_q $drop_attr"],
  1814.                 min($max_string_size,$limits{'query_size'}-30)));
  1815.  
  1816. }
  1817.  
  1818. $tmp=new query_repeat([],"create table crash_q (a integer","","",
  1819.               ",a%d integer","",")",["drop table crash_q $drop_attr"],
  1820.               $max_columns);
  1821. $tmp->{'offset'}=1;
  1822. find_limit("Columns in table","max_columns",$tmp);
  1823.  
  1824. # Make a field definition to be used when testing keys
  1825.  
  1826. $key_definitions="q0 integer not null";
  1827. $key_fields="q0";
  1828. for ($i=1; $i < min($limits{'max_columns'},$max_keys) ; $i++)
  1829. {
  1830.   $key_definitions.=",q$i integer not null";
  1831.   $key_fields.=",q$i";
  1832. }
  1833. $key_values="1," x $i;
  1834. chop($key_values);
  1835.  
  1836. if ($limits{'unique_in_create'} eq 'yes')
  1837. {
  1838.   find_limit("unique indexes","max_unique_index",
  1839.          new query_table("create table crash_q (q integer",
  1840.                  ",q%d integer not null,unique (q%d)",")",
  1841.                  ["insert into crash_q (q,%f) values (1,%v)"],
  1842.                  "select q from crash_q",1,
  1843.                  "drop table crash_q $drop_attr",
  1844.                  $max_keys,0));
  1845.  
  1846.   find_limit("index parts","max_index_parts",
  1847.          new query_table("create table crash_q ($key_definitions,unique (q0",
  1848.                  ",q%d","))",
  1849.                  ["insert into crash_q ($key_fields) values ($key_values)"],
  1850.                  "select q0 from crash_q",1,
  1851.                  "drop table crash_q $drop_attr",
  1852.                  $max_keys,1));
  1853.  
  1854.   find_limit("max index part length","max_index_part_length",
  1855.          new query_many(["create table crash_q (q char(%d) not null,unique(q))",
  1856.                  "insert into crash_q (q) values ('%s')"],
  1857.                 "select q from crash_q","%s",
  1858.                 ["drop table crash_q $drop_attr"],
  1859.                 $limits{'max_char_size'},0));
  1860.  
  1861.   if ($limits{'type_sql_varchar(1_arg)'} eq 'yes')
  1862.   {
  1863.     find_limit("index varchar part length","max_index_varchar_part_length",
  1864.          new query_many(["create table crash_q (q varchar(%d) not null,unique(q))",
  1865.                  "insert into crash_q (q) values ('%s')"],
  1866.                 "select q from crash_q","%s",
  1867.                 ["drop table crash_q $drop_attr"],
  1868.                 $limits{'max_varchar_size'},0));
  1869.   }
  1870. }
  1871.  
  1872.  
  1873. if ($limits{'create_index'} ne 'no')
  1874. {
  1875.   if ($limits{'create_index'} eq 'ignored' ||
  1876.       $limits{'unique_in_create'} eq 'yes')
  1877.   {                                     # This should be true
  1878.     save_config_data('max_index',$limits{'max_unique_index'},"max index");
  1879.     print "indexes: $limits{'max_index'}\n";
  1880.   }
  1881.   else
  1882.   {
  1883.     if (!defined($limits{'max_index'}))
  1884.     {
  1885.       assert("create table crash_q ($key_definitions)");
  1886.       for ($i=1; $i <= min($limits{'max_columns'},$max_keys) ; $i++)
  1887.       {
  1888.     last if (!safe_query("create index crash_q$i on crash_q (q$i)"));
  1889.       }
  1890.       save_config_data('max_index',$i == $max_keys ? $max_keys : $i,
  1891.                "max index");
  1892.       while ( --$i > 0)
  1893.       {
  1894.     $end_drop=$end_drop_keyword;
  1895.     $end_drop =~ s/%i/crash_q$i/;
  1896.     $end_drop =~ s/%t/crash_q/;
  1897.     assert($end_drop);
  1898.       }
  1899.       assert("drop table crash_q $drop_attr");
  1900.     }
  1901.     print "indexs: $limits{'max_index'}\n";
  1902.     if (!defined($limits{'max_unique_index'}))
  1903.     {
  1904.       assert("create table crash_q ($key_definitions)");
  1905.       for ($i=0; $i < min($limits{'max_columns'},$max_keys) ; $i++)
  1906.       {
  1907.     last if (!safe_query("create unique index crash_q$i on crash_q (q$i)"));
  1908.       }
  1909.       save_config_data('max_unique_index',$i == $max_keys ? $max_keys : $i,
  1910.                "max unique index");
  1911.       while ( --$i >= 0)
  1912.       {
  1913.     $end_drop=$end_drop_keyword;
  1914.     $end_drop =~ s/%i/crash_q$i/;
  1915.     $end_drop =~ s/%t/crash_q/;
  1916.     assert($end_drop);
  1917.       }
  1918.       assert("drop table crash_q $drop_attr");
  1919.     }
  1920.     print "unique indexes: $limits{'max_unique_index'}\n";
  1921.     if (!defined($limits{'max_index_parts'}))
  1922.     {
  1923.       assert("create table crash_q ($key_definitions)");
  1924.       $end_drop=$end_drop_keyword;
  1925.       $end_drop =~ s/%i/crash_q1%d/;
  1926.       $end_drop =~ s/%t/crash_q/;
  1927.       find_limit("index parts","max_index_parts",
  1928.          new query_table("create index crash_q1%d on crash_q (q0",
  1929.                  ",q%d",")",
  1930.                  [],
  1931.                  undef,undef,
  1932.                  $end_drop,
  1933.                  $max_keys,1));
  1934.       assert("drop table crash_q $drop_attr");
  1935.     }
  1936.     else
  1937.     {
  1938.       print "index parts: $limits{'max_index_parts'}\n";
  1939.     }
  1940.     $end_drop=$end_drop_keyword;
  1941.     $end_drop =~ s/%i/crash_q2%d/;
  1942.     $end_drop =~ s/%t/crash_me/;
  1943.  
  1944.     find_limit("index part length","max_index_part_length",
  1945.            new query_many(["create table crash_q (q char(%d))",
  1946.                    "create index crash_q2%d on crash_q (q)",
  1947.                    "insert into crash_q values('%s')"],
  1948.                   "select q from crash_q",
  1949.                   "%s",
  1950.                   [ $end_drop,
  1951.                    "drop table crash_q $drop_attr"],
  1952.                   min($limits{'max_char_size'},"+8192")));
  1953.   }
  1954. }
  1955.  
  1956. find_limit("index length","max_index_length",
  1957.        new query_index_length("create table crash_q ",
  1958.                   "drop table crash_q $drop_attr",
  1959.                   $max_key_length));
  1960.  
  1961. find_limit("max table row length (without blobs)","max_row_length",
  1962.        new query_row_length("crash_q ",
  1963.                 "not null",
  1964.                 "drop table crash_q $drop_attr",
  1965.                 min($max_row_length,
  1966.                     $limits{'max_columns'}*
  1967.                     min($limits{'max_char_size'},255))));
  1968.  
  1969. find_limit("table row length with nulls (without blobs)",
  1970.        "max_row_length_with_null",
  1971.        new query_row_length("crash_q ",
  1972.                 "",
  1973.                 "drop table crash_q $drop_attr",
  1974.                 $limits{'max_row_length'}*2));
  1975.  
  1976. find_limit("number of columns in order by","columns_in_order_by",
  1977.        new query_many(["create table crash_q (%F)",
  1978.                "insert into crash_q values(%v)",
  1979.                "insert into crash_q values(%v)"],
  1980.               "select * from crash_q order by %f",
  1981.               undef(),
  1982.               ["drop table crash_q $drop_attr"],
  1983.               $max_order_by));
  1984.  
  1985. find_limit("number of columns in group by","columns_in_group_by",
  1986.        new query_many(["create table crash_q (%F)",
  1987.                "insert into crash_q values(%v)",
  1988.                "insert into crash_q values(%v)"],
  1989.               "select %f from crash_q group by %f",
  1990.               undef(),
  1991.               ["drop table crash_q $drop_attr"],
  1992.               $max_order_by));
  1993.  
  1994. #
  1995. # End of test
  1996. #
  1997.  
  1998. $dbh->do("drop table crash_me $drop_attr");        # Remove temporary table
  1999.  
  2000. print "crash-me safe: $limits{'crash_me_safe'}\n";
  2001. print "reconnected $reconnect_count times\n";
  2002.  
  2003. $dbh->disconnect || warn $dbh->errstr;
  2004. save_all_config_data();
  2005. exit 0;
  2006.  
  2007. sub usage
  2008. {
  2009.     print <<EOF;
  2010. $0  Ver $version
  2011.  
  2012. This program tries to find all limits and capabilities for a SQL
  2013. server.  As it will use the server in some 'unexpected' ways, one
  2014. shouldn\'t have anything important running on it at the same time this
  2015. program runs!  There is a slight chance that something unexpected may
  2016. happen....
  2017.  
  2018. As all used queries are legal according to some SQL standard. any
  2019. reasonable SQL server should be able to run this test without any
  2020. problems.
  2021.  
  2022. All questions is cached in $opt_dir/'server_name'.cfg that future runs will use
  2023. limits found in previous runs. Remove this file if you want to find the
  2024. current limits for your version of the database server.
  2025.  
  2026. This program uses some table names while testing things. If you have any
  2027. tables with the name of 'crash_me' or 'crash_qxxxx' where 'x' is a number,
  2028. they will be deleted by this test!
  2029.  
  2030. $0 takes the following options:
  2031.  
  2032. --help or --Information
  2033.   Shows this help
  2034.  
  2035. --batch-mode
  2036.   Don\'t ask any questions, quit on errors.
  2037.  
  2038. --comment='some comment'
  2039.   Add this comment to the crash-me limit file
  2040.  
  2041. --check-server
  2042.   Do a new connection to the server every time crash-me checks if the server
  2043.   is alive.  This can help in cases where the server starts returning wrong
  2044.   data because of an earlier select.
  2045.  
  2046. --database='database' (Default $opt_database)
  2047.   Create test tables in this database.
  2048.  
  2049. --dir='limits'
  2050.   Save crash-me output in this directory
  2051.  
  2052. --debug
  2053.   Lots of printing to help debugging if something goes wrong.
  2054.  
  2055. --fix-limit-file
  2056.   Reformat the crash-me limit file.  crash-me is not run!
  2057.  
  2058. --force
  2059.   Start test at once, without a warning screen and without questions.
  2060.   This is a option for the very brave.
  2061.   Use this in your cron scripts to test your database every night.
  2062.  
  2063. --log-all-queries
  2064.   Prints all queries that are executed. Mostly used for debugging crash-me.
  2065.  
  2066. --log-queries-to-file='filename'
  2067.   Log full queries to file.
  2068.  
  2069. --host='hostname' (Default $opt_host)
  2070.   Run tests on this host.
  2071.  
  2072. --password='password'
  2073.   Password for the current user.
  2074.  
  2075. --restart
  2076.   Save states during each limit tests. This will make it possible to continue
  2077.   by restarting with the same options if there is some bug in the DBI or
  2078.   DBD driver that caused $0 to die!
  2079.  
  2080. --server='server name'  (Default $opt_server)
  2081.   Run the test on the given server.
  2082.   Known servers names are: Access, Adabas, AdabasD, Empress, Oracle, Informix, DB2, Mimer, mSQL, MS-SQL, MySQL, Pg, Solid or Sybase.
  2083.   For others $0 can\'t report the server version.
  2084.  
  2085. --user='user_name'
  2086.   User name to log into the SQL server.
  2087.  
  2088. --start-cmd='command to restart server'
  2089.   Automaticly restarts server with this command if the database server dies.
  2090.  
  2091. --sleep='time in seconds' (Default $opt_sleep)
  2092.   Wait this long before restarting server.
  2093.  
  2094. EOF
  2095.   exit(0);
  2096. }
  2097.  
  2098.  
  2099. sub server_info
  2100. {
  2101.   my ($ok,$tmp);
  2102.   $ok=0;
  2103.   print "\nNOTE: You should be familiar with '$0 --help' before continuing!\n\n";
  2104.   if (lc($opt_server) eq "mysql")
  2105.   {
  2106.     $ok=1;
  2107.     print <<EOF;
  2108. This test should not crash MySQL if it was distributed together with the
  2109. running MySQL version.
  2110. If this is the case you can probably continue without having to worry about
  2111. destroying something.
  2112. EOF
  2113.   }
  2114.   elsif (lc($opt_server) eq "msql")
  2115.   {
  2116.     print <<EOF;
  2117. This test will take down mSQL repeatedly while finding limits.
  2118. To make this test easier, start mSQL in another terminal with something like:
  2119.  
  2120. while (true); do /usr/local/mSQL/bin/msql2d ; done
  2121.  
  2122. You should be sure that no one is doing anything important with mSQL and that
  2123. you have privileges to restart it!
  2124. It may take awhile to determinate the number of joinable tables, so prepare to
  2125. wait!
  2126. EOF
  2127.   }
  2128.   elsif (lc($opt_server) eq "solid")
  2129.   {
  2130.     print <<EOF;
  2131. This test will take down Solid server repeatedly while finding limits.
  2132. You should be sure that no one is doing anything important with Solid
  2133. and that you have privileges to restart it!
  2134.  
  2135. If you are running Solid without logging and/or backup YOU WILL LOSE!
  2136. Solid does not write data from the cache often enough. So if you continue
  2137. you may lose tables and data that you entered hours ago!
  2138.  
  2139. Solid will also take a lot of memory running this test. You will nead
  2140. at least 234M free!
  2141.  
  2142. When doing the connect test Solid server or the perl api will hang when
  2143. freeing connections. Kill this program and restart it to continue with the
  2144. test. You don\'t have to use --restart for this case.
  2145. EOF
  2146.     if (!$opt_restart)
  2147.     {
  2148.       print "\nWhen DBI/Solid dies you should run this program repeatedly\n";
  2149.       print "with --restart until all tests have completed\n";
  2150.     }
  2151.   }
  2152.   elsif (lc($opt_server) eq "pg")
  2153.   {
  2154.     print <<EOF;
  2155. This test will crash postgreSQL when calculating the number of joinable tables!
  2156. You should be sure that no one is doing anything important with postgreSQL
  2157. and that you have privileges to restart it!
  2158. EOF
  2159.   }
  2160.   else
  2161.   {
  2162.     print <<EOF;
  2163. This test may crash $opt_server repeatedly while finding limits!
  2164. You should be sure that no one is doing anything important with $opt_server
  2165. and that you have privileges to restart it!
  2166. EOF
  2167.   }
  2168.   print <<EOF;
  2169.  
  2170. Some of the tests you are about to execute may require a lot of
  2171. memory.  Your tests WILL adversely affect system performance. It's
  2172. not uncommon that either this crash-me test program, or the actual
  2173. database back-end, will DIE with an out-of-memory error. So might
  2174. any other program on your system if it requests more memory at the
  2175. wrong time.
  2176.  
  2177. Note also that while crash-me tries to find limits for the database server
  2178. it will make a lot of queries that can't be categorized as 'normal'.  It's
  2179. not unlikely that crash-me finds some limit bug in your server so if you
  2180. run this test you have to be prepared that your server may die during it!
  2181.  
  2182. We, the creators of this utility, are not responsible in any way if your
  2183. database server unexpectedly crashes while this program tries to find the
  2184. limitations of your server. By accepting the following question with 'yes',
  2185. you agree to the above!
  2186.  
  2187. You have been warned!
  2188.  
  2189. EOF
  2190.  
  2191.   #
  2192.   # No default reply here so no one can blame us for starting the test
  2193.   # automaticly.
  2194.   #
  2195.   for (;;)
  2196.   {
  2197.     print "Start test (yes/no) ? ";
  2198.     $tmp=<STDIN>; chomp($tmp); $tmp=lc($tmp);
  2199.     last if ($tmp =~ /^yes$/i);
  2200.     exit 1 if ($tmp =~ /^n/i);
  2201.     print "\n";
  2202.   }
  2203. }
  2204.  
  2205. sub machine
  2206. {
  2207.   $name= `uname -s -r -m`;
  2208.   if ($?)
  2209.   {
  2210.     $name= `uname -s -m`;
  2211.   }
  2212.   if ($?)
  2213.   {
  2214.     $name= `uname -s`;
  2215.   }
  2216.   if ($?)
  2217.   {
  2218.     $name= `uname`;
  2219.   }
  2220.   if ($?)
  2221.   {
  2222.     $name="unknown";
  2223.   }
  2224.   chomp($name); $name =~ s/[\n\r]//g;
  2225.   return $name;
  2226. }
  2227.  
  2228.  
  2229. #
  2230. # Help functions that we need
  2231. #
  2232.  
  2233. sub safe_connect
  2234. {
  2235.   my ($object)=@_;
  2236.   my ($dbh,$tmp);
  2237.  
  2238.   for (;;)
  2239.   {
  2240.     if (($dbh=DBI->connect($server->{'data_source'},$opt_user,$opt_password,
  2241.                { PrintError => 0, AutoCommit => 1})))
  2242.     {
  2243.       $dbh->{LongReadLen}= 16000000; # Set max retrieval buffer
  2244.       return $dbh;
  2245.     }
  2246.     print "Error: $DBI::errstr;  $server->{'data_source'}  - '$opt_user' - '$opt_password'\n";
  2247.     print "I got the above error when connecting to $opt_server\n";
  2248.     if (defined($object) && defined($object->{'limit'}))
  2249.     {
  2250.       print "This check was done with limit: $object->{'limit'}.\nNext check will be done with a smaller limit!\n";
  2251.       $object=undef();
  2252.     }
  2253.     save_config_data('crash_me_safe','no',"crash me safe");
  2254.     if ($opt_db_start_cmd)
  2255.     {
  2256.       print "Restarting the db server with:\n'$opt_db_start_cmd'\n";
  2257.       system("$opt_db_start_cmd");
  2258.       print "Waiting $opt_sleep seconds so the server can initialize\n";
  2259.       sleep $opt_sleep;
  2260.     }
  2261.     else
  2262.     {
  2263.       exit(1) if ($opt_batch_mode);
  2264.       print "Can you check/restart it so I can continue testing?\n";
  2265.       for (;;)
  2266.       {
  2267.     print "Continue test (yes/no) ? [yes] ";
  2268.     $tmp=<STDIN>; chomp($tmp); $tmp=lc($tmp);
  2269.     $tmp = "yes" if ($tmp eq "");
  2270.     last if (index("yes",$tmp) >= 0);
  2271.     exit 1 if (index("no",$tmp) >= 0);
  2272.     print "\n";
  2273.       }
  2274.     }
  2275.   }
  2276. }
  2277.  
  2278. #
  2279. # Check if the server is upp and running. If not, ask the user to restart it
  2280. #
  2281.  
  2282. sub check_connect
  2283. {
  2284.   my ($object)=@_;
  2285.   my ($sth);
  2286.   print "Checking connection\n" if ($opt_log_all_queries);
  2287.   # The following line will not work properly with interbase
  2288.   if ($opt_check_server && defined($check_connect) && $dbh->{AutoCommit} != 0)
  2289.   {
  2290.     
  2291.     $dbh->disconnect;
  2292.     $dbh=safe_connect($object);
  2293.     return;
  2294.   }
  2295.   return if (defined($check_connect) && defined($dbh->do($check_connect)));
  2296.   $dbh->disconnect || warn $dbh->errstr;
  2297.   print "\nreconnecting\n" if ($opt_debug);
  2298.   $reconnect_count++;
  2299.   undef($dbh);
  2300.   $dbh=safe_connect($object);
  2301. }
  2302.  
  2303. #
  2304. # print query if debugging
  2305. #
  2306. sub print_query
  2307. {
  2308.   my ($query)=@_;
  2309.   $last_error=$DBI::errstr;
  2310.   if ($opt_debug)
  2311.   {
  2312.     if (length($query) > 130)
  2313.     {
  2314.       $query=substr($query,0,120) . "...(" . (length($query)-120) . ")";
  2315.     }
  2316.     printf "\nGot error from query: '%s'\n%s\n",$query,$DBI::errstr;
  2317.   }
  2318. }
  2319.  
  2320. #
  2321. # Do one or many queries. Return 1 if all was ok
  2322. # Note that all rows are executed (to ensure that we execute drop table commands)
  2323. #
  2324.  
  2325. sub safe_query
  2326. {
  2327.   my($queries)=@_;
  2328.   my($query,$ok,$retry_ok,$retry,@tmp,$sth);
  2329.   $ok=1;
  2330.   if (ref($queries) ne "ARRAY")
  2331.   {
  2332.     push(@tmp,$queries);
  2333.     $queries= \@tmp;
  2334.   }
  2335.   foreach $query (@$queries)
  2336.   {
  2337.     printf "query1: %-80.80s ...(%d - %d)\n",$query,length($query),$retry_limit  if ($opt_log_all_queries);
  2338.     print LOG "$query;\n" if ($opt_log);
  2339.     if (length($query) > $query_size)
  2340.     {
  2341.       $ok=0;
  2342.       next;
  2343.     }
  2344.  
  2345.     $retry_ok=0;
  2346.     for ($retry=0; $retry < $retry_limit ; $retry++)
  2347.     {
  2348.       if (! ($sth=$dbh->prepare($query)))
  2349.       {
  2350.     print_query($query);
  2351.     $retry=100 if (!$server->abort_if_fatal_error());
  2352.     # Force a reconnect because of Access drop table bug!
  2353.     if ($retry == $retry_limit-2)
  2354.     {
  2355.       print "Forcing disconnect to retry query\n" if ($opt_debug);
  2356.       $dbh->disconnect || warn $dbh->errstr;
  2357.     }
  2358.     check_connect();        # Check that server is still up
  2359.       }
  2360.       else
  2361.       {
  2362.         if (!$sth->execute())
  2363.         {
  2364.        print_query($query);
  2365.       $retry=100 if (!$server->abort_if_fatal_error());
  2366.       # Force a reconnect because of Access drop table bug!
  2367.       if ($retry == $retry_limit-2)
  2368.       {
  2369.         print "Forcing disconnect to retry query\n" if ($opt_debug);
  2370.         $dbh->disconnect || warn $dbh->errstr;
  2371.       }
  2372.       check_connect();        # Check that server is still up
  2373.         }
  2374.         else
  2375.         {
  2376.       $retry = $retry_limit;
  2377.       $retry_ok = 1;
  2378.         }
  2379.         $sth->finish;
  2380.       }
  2381.     }
  2382.     $ok=0 if (!$retry_ok);
  2383.     if ($query =~ /create/i && $server->reconnect_on_errors())
  2384.     {
  2385.       print "Forcing disconnect to retry query\n" if ($opt_debug);
  2386.       $dbh->disconnect || warn $dbh->errstr;
  2387.       $dbh=safe_connect();
  2388.     }
  2389.   }
  2390.   return $ok;
  2391. }
  2392.  
  2393.  
  2394. #
  2395. # Do a query on a query package object.
  2396. #
  2397.  
  2398. sub limit_query
  2399. {
  2400.   my($object,$limit)=@_;
  2401.   my ($query,$result,$retry,$sth);
  2402.  
  2403.   $query=$object->query($limit);
  2404.   $result=safe_query($query);
  2405.   if (!$result)
  2406.   {
  2407.     $object->cleanup();
  2408.     return 0;
  2409.   }
  2410.   if (defined($query=$object->check_query()))
  2411.   {
  2412.     for ($retry=0 ; $retry < $retry_limit ; $retry++)
  2413.     {
  2414.       printf "query2: %-80.80s\n",$query if ($opt_log_all_queries);
  2415.       print LOG "$query;\n" if ($opt_log);
  2416.       if (($sth= $dbh->prepare($query)))
  2417.       {
  2418.     if ($sth->execute)
  2419.     {
  2420.       $result= $object->check($sth);
  2421.       $sth->finish;
  2422.       $object->cleanup();
  2423.       return $result;
  2424.     }
  2425.     print_query($query);
  2426.     $sth->finish;
  2427.       }
  2428.       else
  2429.       {
  2430.     print_query($query);
  2431.       }
  2432.       $retry=100 if (!$server->abort_if_fatal_error()); # No need to continue
  2433.       if ($retry == $retry_limit-2)
  2434.       {
  2435.     print "Forcing discoennect to retry query\n" if ($opt_debug);
  2436.     $dbh->disconnect || warn $dbh->errstr;
  2437.       }
  2438.       check_connect($object);   # Check that server is still up
  2439.     }
  2440.     $result=0;                  # Query failed
  2441.   }
  2442.   $object->cleanup();
  2443.   return $result;               # Server couldn't handle the query
  2444. }
  2445.  
  2446.  
  2447. sub report
  2448. {
  2449.   my ($prompt,$limit,@queries)=@_;
  2450.   print "$prompt: ";
  2451.   if (!defined($limits{$limit}))
  2452.   {
  2453.     save_config_data($limit,safe_query(\@queries) ? "yes" : "no",$prompt);
  2454.   }
  2455.   print "$limits{$limit}\n";
  2456.   return $limits{$limit} ne "no";
  2457. }
  2458.  
  2459. sub report_fail
  2460. {
  2461.   my ($prompt,$limit,@queries)=@_;
  2462.   print "$prompt: ";
  2463.   if (!defined($limits{$limit}))
  2464.   {
  2465.     save_config_data($limit,safe_query(\@queries) ? "no" : "yes",$prompt);
  2466.   }
  2467.   print "$limits{$limit}\n";
  2468.   return $limits{$limit} ne "no";
  2469. }
  2470.  
  2471.  
  2472. # Return true if one of the queries is ok
  2473.  
  2474. sub report_one
  2475. {
  2476.   my ($prompt,$limit,$queries)=@_;
  2477.   my ($query,$res,$result);
  2478.   print "$prompt: ";
  2479.   if (!defined($limits{$limit}))
  2480.   {
  2481.     $result="no";
  2482.     foreach $query (@$queries)
  2483.     {
  2484.       if (safe_query($query->[0]))
  2485.       {
  2486.     $result= $query->[1];
  2487.     last;
  2488.       }
  2489.     }
  2490.     save_config_data($limit,$result,$prompt);
  2491.   }
  2492.   print "$limits{$limit}\n";
  2493.   return $limits{$limit} ne "no";
  2494. }
  2495.  
  2496.  
  2497. # Execute query and save result as limit value.
  2498.  
  2499. sub report_result
  2500. {
  2501.   my ($prompt,$limit,$query)=@_;
  2502.   my($error);
  2503.   print "$prompt: ";
  2504.   if (!defined($limits{$limit}))
  2505.   {
  2506.     $error=safe_query_result($query,"1",2);
  2507.     save_config_data($limit,$error ? "not supported" : $last_result,$prompt);
  2508.   }
  2509.   print "$limits{$limit}\n";
  2510.   return $limits{$limit} ne "no";
  2511. }
  2512.  
  2513. sub report_trans
  2514. {
  2515.   my ($limit,$queries,$check,$clear)=@_;
  2516.   if (!defined($limits{$limit}))
  2517.   {
  2518.     eval {undef($dbh->{AutoCommit})};
  2519.     if (!$@)
  2520.     {
  2521.       if (safe_query(\@$queries))
  2522.       {
  2523.       $rc = $dbh->rollback;
  2524.       if ($rc) {
  2525.         $dbh->{AutoCommit} = 1;
  2526.         if (safe_query_result($check,"","")) {
  2527.           save_config_data($limit,"yes",$prompt);
  2528.         }
  2529.         safe_query($clear);
  2530.       } else {
  2531.         $dbh->{AutoCommit} = 1;
  2532.         save_config_data($limit,"error",$prompt);
  2533.       }
  2534.       } else {
  2535.         save_config_data($limit,"error",$prompt);
  2536.       }
  2537.       $dbh->{AutoCommit} = 1;
  2538.     }
  2539.     else
  2540.     {
  2541.       save_config_data($limit,"no",$prompt);
  2542.     }
  2543.     safe_query($clear);
  2544.   }
  2545.   return $limits{$limit} ne "yes";
  2546. }
  2547.  
  2548.  
  2549. sub check_and_report
  2550. {
  2551.   my ($prompt,$limit,$pre,$query,$post,$answer,$string_type,$skip_prompt,
  2552.       $function)=@_;
  2553.   my ($tmp);
  2554.   $function=0 if (!defined($function));
  2555.  
  2556.   print "$prompt: " if (!defined($skip_prompt));
  2557.   if (!defined($limits{$limit}))
  2558.   {
  2559.     $tmp=1-safe_query(\@$pre);
  2560.     $tmp=safe_query_result($query,$answer,$string_type) if (!$tmp);
  2561.     safe_query(\@$post);
  2562.     if ($function == 3)        # Report error as 'no'.
  2563.     {
  2564.       $function=0;
  2565.       $tmp= -$tmp;
  2566.     }
  2567.     if ($function == 0 ||
  2568.     $tmp != 0 && $function == 1 ||
  2569.     $tmp == 0 && $function== 2)
  2570.     {
  2571.       save_config_data($limit, $tmp == 0 ? "yes" : $tmp == 1 ? "no" : "error",
  2572.                $prompt);
  2573.       print "$limits{$limit}\n";
  2574.       return $function == 0 ? $limits{$limit} eq "yes" : 0;
  2575.     }
  2576.     return 1;            # more things to check
  2577.   }
  2578.   print "$limits{$limit}\n";
  2579.   return 0 if ($function);
  2580.   return $limits{$limit} eq "yes";
  2581. }
  2582.  
  2583.  
  2584. sub try_and_report
  2585. {
  2586.   my ($prompt,$limit,@tests)=@_;
  2587.   my ($tmp,$test,$type);
  2588.  
  2589.   print "$prompt: ";
  2590.   if (!defined($limits{$limit}))
  2591.   {
  2592.     $type="no";            # Not supported
  2593.     foreach $test (@tests)
  2594.     {
  2595.       my $tmp_type= shift(@$test);
  2596.       if (safe_query(\@$test))
  2597.       {
  2598.     $type=$tmp_type;
  2599.     goto outer;
  2600.       }
  2601.     }
  2602.   outer:
  2603.     save_config_data($limit, $type, $prompt);
  2604.   }
  2605.   print "$limits{$limit}\n";
  2606.   return $limits{$limit} ne "no";
  2607. }
  2608.  
  2609. #
  2610. # Just execute the query and check values;  Returns 1 if ok
  2611. #
  2612.  
  2613. sub execute_and_check
  2614. {
  2615.   my ($pre,$query,$post,$answer,$string_type)=@_;
  2616.   my ($tmp);
  2617.  
  2618.   $tmp=safe_query(\@$pre);
  2619.   $tmp=safe_query_result($query,$answer,$string_type) == 0 if ($tmp);
  2620.   safe_query(\@$post);
  2621.   return $tmp;
  2622. }
  2623.  
  2624.  
  2625. # returns 0 if ok, 1 if error, -1 if wrong answer
  2626. # Sets $last_result to value of query
  2627.  
  2628. sub safe_query_result
  2629. {
  2630.   my ($query,$answer,$result_type)=@_;
  2631.   my ($sth,$row,$result,$retry);
  2632.   undef($last_result);
  2633.  
  2634.   printf "\nquery3: %-80.80s\n",$query  if ($opt_log_all_queries);
  2635.   print LOG "$query;\n" if ($opt_log);
  2636.   for ($retry=0; $retry < $retry_limit ; $retry++)
  2637.   {
  2638.     if (!($sth=$dbh->prepare($query)))
  2639.     {
  2640.       print_query($query);
  2641.       if ($server->abort_if_fatal_error())
  2642.       {
  2643.     check_connect();    # Check that server is still up
  2644.     next;            # Retry again
  2645.       }
  2646.       check_connect();        # Check that server is still up
  2647.       return 1;
  2648.     }
  2649.     if (!$sth->execute)
  2650.     {
  2651.       print_query($query);
  2652.       if ($server->abort_if_fatal_error())
  2653.       {
  2654.     check_connect();    # Check that server is still up
  2655.     next;            # Retry again
  2656.       }
  2657.       check_connect();        # Check that server is still up
  2658.       return 1;
  2659.     }
  2660.     else
  2661.     {
  2662.       last;
  2663.     }
  2664.   }
  2665.   if (!($row=$sth->fetchrow_arrayref))
  2666.   {
  2667.     print "\nquery: $query didn't return any result\n" if ($opt_debug);
  2668.     $sth->finish;
  2669.     return ($result_type == 8) ? 0 : 1;
  2670.   }
  2671.   if($result_type == 8) {
  2672.     $sth->finish;
  2673.     return 1;
  2674.   }
  2675.   $result=0;                      # Ok
  2676.   $last_result= $row->[0];    # Save for report_result;
  2677.   if ($result_type == 0)    # Compare numbers
  2678.   {
  2679.     $row->[0] =~ s/,/,/;    # Fix if ',' is used instead of '.'
  2680.     if ($row->[0] != $answer && (abs($row->[0]- $answer)/
  2681.                  (abs($row->[0]) + abs($answer))) > 0.01)
  2682.     {
  2683.       $result=-1;
  2684.     }
  2685.   }
  2686.   elsif ($result_type == 1)    # Compare where end space may differ
  2687.   {
  2688.     $row->[0] =~ s/\s+$//;
  2689.     $result=-1 if ($row->[0] ne $answer);
  2690.   }
  2691.   elsif ($result_type == 3)    # This should be a exact match
  2692.   {
  2693.     $result= -1 if ($row->[0] ne $answer);
  2694.   }
  2695.   elsif ($result_type == 4)    # If results should be NULL
  2696.   {
  2697.     $result= -1 if (defined($row->[0]));
  2698.   }
  2699.   elsif ($result_type == 5)    # Result should have given prefix
  2700.   {
  2701.     $result= -1 if (length($row->[0]) < length($answer) &&
  2702.             substring($row->[0],1,length($answer)) ne $answer);
  2703.   }
  2704.   elsif ($result_type == 6)    # Exact match but ignore errors
  2705.   {
  2706.     $result= 1 if ($row->[0] ne $answer);
  2707.   }
  2708.   elsif ($result_type == 7)    # Compare against array of numbers
  2709.   {
  2710.     if ($row->[0] != $answer->[0])
  2711.     {
  2712.       $result= -1;
  2713.     }
  2714.     else
  2715.     {
  2716.       my ($value);
  2717.       shift @$answer;
  2718.       while (($row=$sth->fetchrow_arrayref))
  2719.       {
  2720.     $value=shift(@$answer);
  2721.     if (!defined($value))
  2722.     {
  2723.       print "\nquery: $query returned to many results\n"
  2724.         if ($opt_debug);
  2725.       $result= 1;
  2726.       last;
  2727.     }
  2728.     if ($row->[0] != $value)
  2729.     {
  2730.       $result= -1;
  2731.       last;
  2732.     }
  2733.       }
  2734.       if ($#$answer != -1)
  2735.       {
  2736.     print "\nquery: $query returned too few results\n"
  2737.       if ($opt_debug);
  2738.     $result= 1;
  2739.       }
  2740.     }
  2741.   }
  2742.   $sth->finish;
  2743.   print "\nquery: '$query' returned '$row->[0]' instead of '$answer'\n"
  2744.     if ($opt_debug && $result && $result_type != 7);
  2745.   return $result;
  2746. }
  2747.  
  2748. #
  2749. # Find limit using binary search.  This is a weighed binary search that
  2750. # will prefere lower limits to get the server to crash as few times as possible
  2751. #
  2752.  
  2753. sub find_limit()
  2754. {
  2755.   my ($prompt,$limit,$query)=@_;
  2756.   my ($first,$end,$i,$tmp);
  2757.   print "$prompt: ";
  2758.   if (defined($end=$limits{$limit}))
  2759.   {
  2760.     print "$end (cache)\n";
  2761.     return $end;
  2762.   }
  2763.   if (defined($query->{'init'}) && !defined($end=$limits{'restart'}{'tohigh'}))
  2764.   {
  2765.     if (!safe_query($query->{'init'}))
  2766.     {
  2767.       $query->cleanup();
  2768.       return "error";
  2769.     }
  2770.   }
  2771.  
  2772.   if (!limit_query($query,1))           # This must work
  2773.   {
  2774.     print "\nMaybe fatal error: Can't check '$prompt' for limit=1\nerror: $last_error\n";
  2775.     return "error";
  2776.   }
  2777.  
  2778.   $first=0;
  2779.   $first=$limits{'restart'}{'low'} if ($limits{'restart'}{'low'});
  2780.  
  2781.   if (defined($end=$limits{'restart'}{'tohigh'}))
  2782.   {
  2783.     $end--;
  2784.     print "\nRestarting this with low limit: $first and high limit: $end\n";
  2785.     delete $limits{'restart'};
  2786.     $i=$first+int(($end-$first+4)/5);           # Prefere lower on errors
  2787.   }
  2788.   else
  2789.   {
  2790.     $end= $query->max_limit();
  2791.     $i=int(($end+$first)/2);
  2792.   }
  2793.  
  2794.   unless(limit_query($query,0+$end)) {
  2795.     while ($first < $end)
  2796.     {
  2797.       print "." if ($opt_debug);
  2798.       save_config_data("restart",$i,"") if ($opt_restart);
  2799.       if (limit_query($query,$i))
  2800.       {
  2801.         $first=$i;
  2802.         $i=$first+int(($end-$first+1)/2); # to be a bit faster to go up
  2803.       }
  2804.       else
  2805.       {
  2806.         $end=$i-1;
  2807.         $i=$first+int(($end-$first+4)/5); # Prefere lower on errors
  2808.       }
  2809.     }
  2810.   }
  2811.   $end+=$query->{'offset'} if ($end && defined($query->{'offset'}));
  2812.   if ($end >= $query->{'max_limit'} &&
  2813.       substr($query->{'max_limit'},0,1) eq '+')
  2814.   {
  2815.     $end= $query->{'max_limit'};
  2816.   }
  2817.   print "$end\n";
  2818.   save_config_data($limit,$end,$prompt);
  2819.   delete $limits{'restart'};
  2820.   return $end;
  2821. }
  2822.  
  2823. #
  2824. # Check that the query works!
  2825. #
  2826.  
  2827. sub assert
  2828. {
  2829.   my($query)=@_;
  2830.  
  2831.   if (!safe_query($query))
  2832.   {
  2833.     $query=join("; ",@$query) if (ref($query) eq "ARRAY");
  2834.     print "\nFatal error:\nquery: '$query'\nerror: $DBI::errstr\n";
  2835.     exit 1;
  2836.   }
  2837. }
  2838.  
  2839.  
  2840. sub read_config_data
  2841. {
  2842.   my ($key,$limit,$prompt);
  2843.   if (-e $opt_config_file)
  2844.   {
  2845.     open(CONFIG_FILE,"+<$opt_config_file") ||
  2846.       die "Can't open configure file $opt_config_file\n";
  2847.     print "Reading old values from cache: $opt_config_file\n";
  2848.   }
  2849.   else
  2850.   {
  2851.     open(CONFIG_FILE,"+>>$opt_config_file") ||
  2852.       die "Can't create configure file $opt_config_file: $!\n";
  2853.   }
  2854.   select CONFIG_FILE;
  2855.   $|=1;
  2856.   select STDOUT;
  2857.   while (<CONFIG_FILE>)
  2858.   {
  2859.     chomp;
  2860.     if (/^(\S+)=([^\#]*[^\#\s])\s*(\# .*)*$/)
  2861.     {
  2862.       $key=$1; $limit=$2 ; $prompt=$3;
  2863.       if (!$opt_quick || $limit =~ /\d/ || $key =~ /crash_me/)
  2864.       {
  2865.     if ($key !~ /restart/i)
  2866.     {
  2867.       $limits{$key}=$limit;
  2868.       $prompts{$key}=length($prompt) ? substr($prompt,2) : "";
  2869.       delete $limits{'restart'};
  2870.     }
  2871.     else
  2872.     {
  2873.       $limit_changed=1;
  2874.       if ($limit > $limits{'restart'}{'tohigh'})
  2875.       {
  2876.         $limits{'restart'}{'low'} = $limits{'restart'}{'tohigh'};
  2877.       }
  2878.       $limits{'restart'}{'tohigh'} = $limit;
  2879.     }
  2880.       }
  2881.     }
  2882.     elsif (!/^\s*$/ && !/^\#/)
  2883.     {
  2884.       die "Wrong config row: $_\n";
  2885.     }
  2886.   }
  2887. }
  2888.  
  2889.  
  2890. sub save_config_data
  2891. {
  2892.   my ($key,$limit,$prompt)=@_;
  2893.   $prompts{$key}=$prompt;
  2894.   return if (defined($limits{$key}) && $limits{$key} eq $limit);
  2895.   if (!defined($limit) || $limit eq "")
  2896.   {
  2897.     die "Undefined limit for $key\n";
  2898.   }
  2899.   print CONFIG_FILE "$key=$limit\t# $prompt\n";
  2900.   $limits{$key}=$limit;
  2901.   $limit_changed=1;
  2902.   if (($opt_restart && $limits{'operating_system'} =~ /windows/i) ||
  2903.                ($limits{'operating_system'} =~ /NT/))
  2904.   {
  2905.     # If perl crashes in windows, everything is lost (Wonder why? :)
  2906.     close CONFIG_FILE;
  2907.     open(CONFIG_FILE,"+>>$opt_config_file") ||
  2908.       die "Can't reopen configure file $opt_config_file: $!\n";
  2909.   }
  2910. }
  2911.  
  2912.  
  2913. sub save_all_config_data
  2914. {
  2915.   my ($key,$tmp);
  2916.   close CONFIG_FILE;
  2917.   return if (!$limit_changed);
  2918.   open(CONFIG_FILE,">$opt_config_file") ||
  2919.     die "Can't create configure file $opt_config_file: $!\n";
  2920.   select CONFIG_FILE;
  2921.   $|=1;
  2922.   select STDOUT;
  2923.   delete $limits{'restart'};
  2924.  
  2925.   print CONFIG_FILE "#This file is automaticly generated by crash-me $version\n\n";
  2926.   foreach $key (sort keys %limits)
  2927.   {
  2928.     $tmp="$key=$limits{$key}";
  2929.     print CONFIG_FILE $tmp . ("\t" x (int((32-min(length($tmp),32)+7)/8)+1)) .
  2930.       "# $prompts{$key}\n";
  2931.   }
  2932.   close CONFIG_FILE;
  2933. }
  2934.  
  2935.  
  2936. sub check_repeat
  2937. {
  2938.   my ($sth,$limit)=@_;
  2939.   my ($row);
  2940.  
  2941.   return 0 if (!($row=$sth->fetchrow_arrayref));
  2942.   return (defined($row->[0]) && ('a' x $limit) eq $row->[0]) ? 1 : 0;
  2943. }
  2944.  
  2945.  
  2946. sub min
  2947. {
  2948.   my($min)=$_[0];
  2949.   my($i);
  2950.   for ($i=1 ; $i <= $#_; $i++)
  2951.   {
  2952.     $min=$_[$i] if ($min > $_[$i]);
  2953.   }
  2954.   return $min;
  2955. }
  2956.  
  2957. sub sql_concat
  2958. {
  2959.   my ($a,$b)= @_;
  2960.   return "$a || $b" if ($limits{'func_sql_concat_as_||'} eq 'yes');
  2961.   return "concat($a,$b)" if ($limits{'func_odbc_concat'} eq 'yes');
  2962.   return "$a + $b" if ($limits{'func_extra_concat_as_+'} eq 'yes');
  2963.   return undef;
  2964. }
  2965.  
  2966. #
  2967. # Returns a list of statements to create a table in a portable manner
  2968. # but still utilizing features in the databases.
  2969. #
  2970.  
  2971. sub create_table
  2972. {
  2973.   my($table_name,$fields,$index,$extra) = @_;
  2974.   my($query,$nr,$parts,@queries,@index);
  2975.  
  2976.   $extra="" if (!defined($extra));
  2977.  
  2978.   $query="create table $table_name (";
  2979.   $nr=0;
  2980.   foreach $field (@$fields)
  2981.   {
  2982.     $query.= $field . ',';
  2983.   }
  2984.   foreach $index (@$index)
  2985.   {
  2986.     $index =~ /\(([^\(]*)\)$/i;
  2987.     $parts=$1;
  2988.     if ($index =~ /^primary key/)
  2989.     {
  2990.       if ($limits{'primary_key_in_create'} eq 'yes')
  2991.       {
  2992.     $query.= $index . ',';
  2993.       }
  2994.       else
  2995.       {
  2996.     push(@queries,
  2997.          "create unique index ${table_name}_prim on $table_name ($parts)");
  2998.       }
  2999.     }
  3000.     elsif ($index =~ /^unique/)
  3001.     {
  3002.       if ($limits{'unique_in_create'} eq 'yes')
  3003.       {
  3004.     $query.= "unique ($parts),";
  3005.       }
  3006.       else
  3007.       {
  3008.     $nr++;
  3009.     push(@queries,
  3010.          "create unique index ${table_name}_$nr on $table_name ($parts)");
  3011.  
  3012.       }
  3013.     }
  3014.     else
  3015.     {
  3016.       if ($limits{'index_in_create'} eq 'yes')
  3017.       {
  3018.     $query.= "index ($parts),";
  3019.       }
  3020.       else
  3021.       {
  3022.     $nr++;
  3023.     push(@queries,
  3024.          "create index ${table_name}_$nr on $table_name ($1)");
  3025.       }
  3026.     }
  3027.   }
  3028.   chop($query);
  3029.   $query.= ") $extra";
  3030.   unshift(@queries,$query);
  3031.   return @queries;
  3032. }
  3033.  
  3034.  
  3035. #
  3036. # This is used by some query packages to change:
  3037. # %d -> limit
  3038. # %s -> 'a' x limit
  3039. # %v -> "1,1,1,1,1" where there are 'limit' number of ones
  3040. # %f -> q1,q2,q3....
  3041. # %F -> q1 integer,q2 integer,q3 integer....
  3042.  
  3043. sub fix_query
  3044. {
  3045.   my ($query,$limit)=@_;
  3046.   my ($repeat,$i);
  3047.  
  3048.   return $query if !(defined($query));
  3049.   $query =~ s/%d/$limit/g;
  3050.   if ($query =~ /%s/)
  3051.   {
  3052.     $repeat= 'a' x $limit;
  3053.     $query =~ s/%s/$repeat/g;
  3054.   }
  3055.   if ($query =~ /%v/)
  3056.   {
  3057.     $repeat= '1,' x $limit;
  3058.     chop($repeat);
  3059.     $query =~ s/%v/$repeat/g;
  3060.   }
  3061.   if ($query =~ /%f/)
  3062.   {
  3063.     $repeat="";
  3064.     for ($i=1 ; $i <= $limit ; $i++)
  3065.     {
  3066.       $repeat.="q$i,";
  3067.     }
  3068.     chop($repeat);
  3069.     $query =~ s/%f/$repeat/g;
  3070.   }
  3071.   if ($query =~ /%F/)
  3072.   {
  3073.     $repeat="";
  3074.     for ($i=1 ; $i <= $limit ; $i++)
  3075.     {
  3076.       $repeat.="q$i integer,";
  3077.     }
  3078.     chop($repeat);
  3079.     $query =~ s/%F/$repeat/g;
  3080.   }
  3081.   return $query;
  3082. }
  3083.  
  3084.  
  3085. #
  3086. # Different query packages
  3087. #
  3088.  
  3089. package query_repeat;
  3090.  
  3091. sub new
  3092. {
  3093.   my ($type,$init,$query,$add1,$add_mid,$add,$add_end,$end_query,$cleanup,
  3094.       $max_limit, $check, $offset)=@_;
  3095.   my $self={};
  3096.   if (defined($init) && $#$init != -1)
  3097.   {
  3098.     $self->{'init'}=$init;
  3099.   }
  3100.   $self->{'query'}=$query;
  3101.   $self->{'add1'}=$add1;
  3102.   $self->{'add_mid'}=$add_mid;
  3103.   $self->{'add'}=$add;
  3104.   $self->{'add_end'}=$add_end;
  3105.   $self->{'end_query'}=$end_query;
  3106.   $self->{'cleanup'}=$cleanup;
  3107.   $self->{'max_limit'}=(defined($max_limit) ? $max_limit : $main::query_size);
  3108.   $self->{'check'}=$check;
  3109.   $self->{'offset'}=$offset;
  3110.   $self->{'printf'}= ($add =~ /%d/);
  3111.   bless $self;
  3112. }
  3113.  
  3114. sub query
  3115. {
  3116.   my ($self,$limit)=@_;
  3117.   if (!$self->{'printf'})
  3118.   {
  3119.     return $self->{'query'} . ($self->{'add'} x $limit) .
  3120.       ($self->{'add_end'} x $limit) . $self->{'end_query'};
  3121.   }
  3122.   my ($tmp,$tmp2,$tmp3,$i);
  3123.   $tmp=$self->{'query'};
  3124.   if ($self->{'add1'})
  3125.   {
  3126.     for ($i=0; $i < $limit ; $i++)
  3127.     {
  3128.       $tmp3 = $self->{'add1'};
  3129.       $tmp3 =~ s/%d/$i/g;
  3130.       $tmp  .= $tmp3;
  3131.     }
  3132.   }
  3133.   $tmp .= " ".$self->{'add_mid'};
  3134.   if ($self->{'add'})
  3135.   {
  3136.     for ($i=0; $i < $limit ; $i++)
  3137.     {
  3138.       $tmp2 = $self->{'add'};
  3139.       $tmp2 =~ s/%d/$i/g;
  3140.       $tmp  .= $tmp2;
  3141.     }
  3142.   }
  3143.   return ($tmp .
  3144.       ($self->{'add_end'} x $limit) . $self->{'end_query'});
  3145. }
  3146.  
  3147. sub max_limit
  3148. {
  3149.   my ($self)=@_;
  3150.   my $tmp;
  3151.   $tmp=int(($main::limits{"query_size"}-length($self->{'query'})
  3152.         -length($self->{'add_mid'})-length($self->{'end_query'}))/
  3153.        (length($self->{'add1'})+
  3154.        length($self->{'add'})+length($self->{'add_end'})));
  3155.   return main::min($self->{'max_limit'},$tmp);
  3156. }
  3157.  
  3158.  
  3159. sub cleanup
  3160. {
  3161.   my ($self)=@_;
  3162.   my($tmp,$statement);
  3163.   $tmp=$self->{'cleanup'};
  3164.   foreach $statement (@$tmp)
  3165.   {
  3166.     main::safe_query($statement) if (defined($statement) && length($statement));
  3167.   }
  3168. }
  3169.  
  3170. sub check
  3171. {
  3172.   my ($self,$sth)=@_;
  3173.   my $check=$self->{'check'};
  3174.   return &$check($sth,$self->{'limit'}) if (defined($check));
  3175.   return 1;
  3176. }
  3177.  
  3178. sub check_query
  3179. {
  3180.   return undef;
  3181. }
  3182.  
  3183.  
  3184. package query_num;
  3185.  
  3186. sub new
  3187. {
  3188.   my ($type,$query,$end_query,$cleanup,$max_limit,$check)=@_;
  3189.   my $self={};
  3190.   $self->{'query'}=$query;
  3191.   $self->{'end_query'}=$end_query;
  3192.   $self->{'cleanup'}=$cleanup;
  3193.   $self->{'max_limit'}=$max_limit;
  3194.   $self->{'check'}=$check;
  3195.   bless $self;
  3196. }
  3197.  
  3198.  
  3199. sub query
  3200. {
  3201.   my ($self,$i)=@_;
  3202.   $self->{'limit'}=$i;
  3203.   return "$self->{'query'}$i$self->{'end_query'}";
  3204. }
  3205.  
  3206. sub max_limit
  3207. {
  3208.   my ($self)=@_;
  3209.   return $self->{'max_limit'};
  3210. }
  3211.  
  3212. sub cleanup
  3213. {
  3214.   my ($self)=@_;
  3215.   my($statement);
  3216.   foreach $statement ($self->{'$cleanup'})
  3217.   {
  3218.     main::safe_query($statement) if (defined($statement) && length($statement));
  3219.   }
  3220. }
  3221.  
  3222.  
  3223. sub check
  3224. {
  3225.   my ($self,$sth)=@_;
  3226.   my $check=$self->{'check'};
  3227.   return &$check($sth,$self->{'limit'}) if (defined($check));
  3228.   return 1;
  3229. }
  3230.  
  3231. sub check_query
  3232. {
  3233.   return undef;
  3234. }
  3235.  
  3236. #
  3237. # This package is used when testing CREATE TABLE!
  3238. #
  3239.  
  3240. package query_table;
  3241.  
  3242. sub new
  3243. {
  3244.   my ($type,$query, $add, $end_query, $extra_init, $safe_query, $check,
  3245.       $cleanup, $max_limit, $offset)=@_;
  3246.   my $self={};
  3247.   $self->{'query'}=$query;
  3248.   $self->{'add'}=$add;
  3249.   $self->{'end_query'}=$end_query;
  3250.   $self->{'extra_init'}=$extra_init;
  3251.   $self->{'safe_query'}=$safe_query;
  3252.   $self->{'check'}=$check;
  3253.   $self->{'cleanup'}=$cleanup;
  3254.   $self->{'max_limit'}=$max_limit;
  3255.   $self->{'offset'}=$offset;
  3256.   bless $self;
  3257. }
  3258.  
  3259.  
  3260. sub query
  3261. {
  3262.   my ($self,$limit)=@_;
  3263.   $self->{'limit'}=$limit;
  3264.   $self->cleanup();     # Drop table before create
  3265.  
  3266.   my ($tmp,$tmp2,$i,$query,@res);
  3267.   $tmp =$self->{'query'};
  3268.   $tmp =~ s/%d/$limit/g;
  3269.   for ($i=1; $i <= $limit ; $i++)
  3270.   {
  3271.     $tmp2 = $self->{'add'};
  3272.     $tmp2 =~ s/%d/$i/g;
  3273.     $tmp  .= $tmp2;
  3274.   }
  3275.   push(@res,$tmp . $self->{'end_query'});
  3276.   $tmp=$self->{'extra_init'};
  3277.   foreach $query (@$tmp)
  3278.   {
  3279.     push(@res,main::fix_query($query,$limit));
  3280.   }
  3281.   return \@res;
  3282. }
  3283.  
  3284.  
  3285. sub max_limit
  3286. {
  3287.   my ($self)=@_;
  3288.   return $self->{'max_limit'};
  3289. }
  3290.  
  3291.  
  3292. sub check_query
  3293. {
  3294.   my ($self)=@_;
  3295.   return main::fix_query($self->{'safe_query'},$self->{'limit'});
  3296. }
  3297.  
  3298. sub check
  3299. {
  3300.   my ($self,$sth)=@_;
  3301.   my $check=$self->{'check'};
  3302.   return 0 if (!($row=$sth->fetchrow_arrayref));
  3303.   if (defined($check))
  3304.   {
  3305.     return (defined($row->[0]) &&
  3306.         $row->[0] eq main::fix_query($check,$self->{'limit'})) ? 1 : 0;
  3307.   }
  3308.   return 1;
  3309. }
  3310.  
  3311.  
  3312. # Remove table before and after create table query
  3313.  
  3314. sub cleanup()
  3315. {
  3316.   my ($self)=@_;
  3317.   main::safe_query(main::fix_query($self->{'cleanup'},$self->{'limit'}));
  3318. }
  3319.  
  3320. #
  3321. # Package to do many queries with %d, and %s substitution
  3322. #
  3323.  
  3324. package query_many;
  3325.  
  3326. sub new
  3327. {
  3328.   my ($type,$query,$safe_query,$check_result,$cleanup,$max_limit,$offset,
  3329.       $safe_cleanup)=@_;
  3330.   my $self={};
  3331.   $self->{'query'}=$query;
  3332.   $self->{'safe_query'}=$safe_query;
  3333.   $self->{'check'}=$check_result;
  3334.   $self->{'cleanup'}=$cleanup;
  3335.   $self->{'max_limit'}=$max_limit;
  3336.   $self->{'offset'}=$offset;
  3337.   $self->{'safe_cleanup'}=$safe_cleanup;
  3338.   bless $self;
  3339. }
  3340.  
  3341.  
  3342. sub query
  3343. {
  3344.   my ($self,$limit)=@_;
  3345.   my ($queries,$query,@res);
  3346.   $self->{'limit'}=$limit;
  3347.   $self->cleanup() if (defined($self->{'safe_cleanup'}));
  3348.   $queries=$self->{'query'};
  3349.   foreach $query (@$queries)
  3350.   {
  3351.     push(@res,main::fix_query($query,$limit));
  3352.   }
  3353.   return \@res;
  3354. }
  3355.  
  3356. sub check_query
  3357. {
  3358.   my ($self)=@_;
  3359.   return main::fix_query($self->{'safe_query'},$self->{'limit'});
  3360. }
  3361.  
  3362. sub cleanup
  3363. {
  3364.   my ($self)=@_;
  3365.   my($tmp,$statement);
  3366.   return if (!defined($self->{'cleanup'}));
  3367.   $tmp=$self->{'cleanup'};
  3368.   foreach $statement (@$tmp)
  3369.   {
  3370.     if (defined($statement) && length($statement))
  3371.     {
  3372.       main::safe_query(main::fix_query($statement,$self->{'limit'}));
  3373.     }
  3374.   }
  3375. }
  3376.  
  3377.  
  3378. sub check
  3379. {
  3380.   my ($self,$sth)=@_;
  3381.   my ($check,$row);
  3382.   return 0 if (!($row=$sth->fetchrow_arrayref));
  3383.   $check=$self->{'check'};
  3384.   if (defined($check))
  3385.   {
  3386.     return (defined($row->[0]) &&
  3387.         $row->[0] eq main::fix_query($check,$self->{'limit'})) ? 1 : 0;
  3388.   }
  3389.   return 1;
  3390. }
  3391.  
  3392. sub max_limit
  3393. {
  3394.   my ($self)=@_;
  3395.   return $self->{'max_limit'};
  3396. }
  3397.  
  3398. #
  3399. # Used to find max supported row length
  3400. #
  3401.  
  3402. package query_row_length;
  3403.  
  3404. sub new
  3405. {
  3406.   my ($type,$create,$null,$drop,$max_limit)=@_;
  3407.   my $self={};
  3408.   $self->{'table_name'}=$create;
  3409.   $self->{'null'}=$null;
  3410.   $self->{'cleanup'}=$drop;
  3411.   $self->{'max_limit'}=$max_limit;
  3412.   bless $self;
  3413. }
  3414.  
  3415.  
  3416. sub query
  3417. {
  3418.   my ($self,$limit)=@_;
  3419.   my ($res,$values,$size,$length,$i);
  3420.   $self->{'limit'}=$limit;
  3421.  
  3422.   $res="";
  3423.   $size=main::min($main::limits{'max_char_size'},255);
  3424.   $size = 255 if (!$size); # Safety
  3425.   for ($length=$i=0; $length + $size <= $limit ; $length+=$size, $i++)
  3426.   {
  3427.     $res.= "q$i char($size) $self->{'null'},";
  3428.     $values.="'" . ('a' x $size) . "',";
  3429.   }
  3430.   if ($length < $limit)
  3431.   {
  3432.     $size=$limit-$length;
  3433.     $res.= "q$i char($size) $self->{'null'},";
  3434.     $values.="'" . ('a' x $size) . "',";
  3435.   }
  3436.   chop($res);
  3437.   chop($values);
  3438.   return ["create table " . $self->{'table_name'} . " ($res)",
  3439.       "insert into " . $self->{'table_name'} . " values ($values)"];
  3440. }
  3441.  
  3442. sub max_limit
  3443. {
  3444.   my ($self)=@_;
  3445.   return $self->{'max_limit'};
  3446. }
  3447.  
  3448. sub cleanup
  3449. {
  3450.   my ($self)=@_;
  3451.   main::safe_query($self->{'cleanup'});
  3452. }
  3453.  
  3454.  
  3455. sub check
  3456. {
  3457.   return 1;
  3458. }
  3459.  
  3460. sub check_query
  3461. {
  3462.   return undef;
  3463. }
  3464.  
  3465. #
  3466. # Used to find max supported index length
  3467. #
  3468.  
  3469. package query_index_length;
  3470.  
  3471. sub new
  3472. {
  3473.   my ($type,$create,$drop,$max_limit)=@_;
  3474.   my $self={};
  3475.   $self->{'create'}=$create;
  3476.   $self->{'cleanup'}=$drop;
  3477.   $self->{'max_limit'}=$max_limit;
  3478.   bless $self;
  3479. }
  3480.  
  3481.  
  3482. sub query
  3483. {
  3484.   my ($self,$limit)=@_;
  3485.   my ($res,$size,$length,$i,$parts,$values);
  3486.   $self->{'limit'}=$limit;
  3487.  
  3488.   $res=$parts=$values="";
  3489.   $size=main::min($main::limits{'max_index_part_length'},$main::limits{'max_char_size'});
  3490.   $size=1 if ($size == 0);    # Avoid infinite loop errors
  3491.   for ($length=$i=0; $length + $size <= $limit ; $length+=$size, $i++)
  3492.   {
  3493.     $res.= "q$i char($size) not null,";
  3494.     $parts.= "q$i,";
  3495.     $values.= "'" . ('a' x $size) . "',";
  3496.   }
  3497.   if ($length < $limit)
  3498.   {
  3499.     $size=$limit-$length;
  3500.     $res.= "q$i char($size) not null,";
  3501.     $parts.="q$i,";
  3502.     $values.= "'" . ('a' x $size) . "',";
  3503.   }
  3504.   chop($parts);
  3505.   chop($res);
  3506.   chop($values);
  3507.   if ($main::limits{'unique_in_create'} eq 'yes')
  3508.   {
  3509.     return [$self->{'create'} . "($res,unique ($parts))",
  3510.         "insert into crash_q values($values)"];
  3511.   }
  3512.   return [$self->{'create'} . "($res)",
  3513.       "create index crash_q_index on crash_q ($parts)",
  3514.       "insert into crash_q values($values)"];
  3515. }
  3516.  
  3517. sub max_limit
  3518. {
  3519.   my ($self)=@_;
  3520.   return $self->{'max_limit'};
  3521. }
  3522.  
  3523. sub cleanup
  3524. {
  3525.   my ($self)=@_;
  3526.   main::safe_query($self->{'cleanup'});
  3527. }
  3528.  
  3529.  
  3530. sub check
  3531. {
  3532.   return 1;
  3533. }
  3534.  
  3535. sub check_query
  3536. {
  3537.   return undef;
  3538. }
  3539.  
  3540.  
  3541. ### TODO:
  3542. # OID test instead of / in addition to _rowid
  3543.