# -*- cperl -*- # Copyright (C) 2005-2006 MySQL AB # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; version 2 of the License. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA # This is a library file used by the Perl version of mysql-test-run, # and is part of the translation of the Bourne shell script with the # same name. package mtr_cases; use strict; use base qw(Exporter); our @EXPORT= qw(collect_option collect_test_cases); use mtr_report; # Options used for the collect phase our $start_from; our $print_testcases; our $skip_rpl; our $do_test; our $skip_test; our $opt_skip_combination; our $binlog_format; our $enable_disabled; our $default_storage_engine; our $opt_with_ndbcluster_only; our $defaults_file; our $defaults_extra_file; our $reorder; sub collect_option { my ($opt, $value)= @_; # Convert - to _ in option name $opt =~ s/-/_/; no strict 'refs'; ${$opt}= $value; } use File::Basename; use IO::File(); use My::Config; use My::Platform; require "mtr_match.pl"; require "mtr_misc.pl"; # Precompiled regex's for tests to do or skip my $do_test_reg; my $skip_test_reg; sub init_pattern { my ($from, $what)= @_; return undef unless defined $from; if ( $from =~ /^[a-z0-9\.]*$/ ) { # Does not contain any regex (except . that we allow as # separator betwen suite and testname), make the pattern match # beginning of string $from= "^$from"; mtr_verbose("$what='$from'"); } # Check that pattern is a valid regex eval { "" =~/$from/; 1 } or mtr_error("Invalid regex '$from' passed to $what\nPerl says: $@"); return $from; } ############################################################################## # # Collect information about test cases to be run # ############################################################################## sub collect_test_cases ($$) { my $suites= shift; # Semicolon separated list of test suites my $opt_cases= shift; my $cases= []; # Array of hash(one hash for each testcase) $do_test_reg= init_pattern($do_test, "--do-test"); $skip_test_reg= init_pattern($skip_test, "--skip-test"); foreach my $suite (split(",", $suites)) { push(@$cases, collect_one_suite($suite, $opt_cases)); } if ( @$opt_cases ) { # A list of tests was specified on the command line # Check that the tests specified was found # in at least one suite foreach my $test_name_spec ( @$opt_cases ) { my $found= 0; my ($sname, $tname, $extension)= split_testname($test_name_spec); foreach my $test ( @$cases ) { # test->{name} is always in suite.name format if ( $test->{name} =~ /.*\.$tname/ ) { $found= 1; } } if ( not $found ) { mtr_error("Could not find $tname in any suite"); } } } if ( $reorder ) { # Reorder the test cases in an order that will make them faster to run my %sort_criteria; # Make a mapping of test name to a string that represents how that test # should be sorted among the other tests. Put the most important criterion # first, then a sub-criterion, then sub-sub-criterion, etc. foreach my $tinfo (@$cases) { my @criteria = (); # Look for tests that must be run in a defined order - that is # defined by test having the same name except for the ending digit # Put variables into hash my $test_name= $tinfo->{'name'}; my $depend_on_test_name; if ( $test_name =~ /^([\D]+)([0-9]{1})$/ ) { my $base_name= $1; my $idx= $2; mtr_verbose("$test_name => $base_name idx=$idx"); if ( $idx > 1 ) { $idx-= 1; $base_name= "$base_name$idx"; mtr_verbose("New basename $base_name"); } foreach my $tinfo2 (@$cases) { if ( $tinfo2->{'name'} eq $base_name ) { mtr_verbose("found dependent test $tinfo2->{'name'}"); $depend_on_test_name=$base_name; } } } if ( defined $depend_on_test_name ) { mtr_verbose("Giving $test_name same critera as $depend_on_test_name"); $sort_criteria{$test_name} = $sort_criteria{$depend_on_test_name}; } else { # # Append the criteria for sorting, in order of importance. # push(@criteria, "ndb=" . ($tinfo->{'ndb_test'} ? "1" : "0")); # Group test with equal options together. # Ending with "~" makes empty sort later than filled my $opts= $tinfo->{'master_opt'} ? $tinfo->{'master_opt'} : []; push(@criteria, join("!", sort @{$opts}) . "~"); $sort_criteria{$test_name} = join(" ", @criteria); } } @$cases = sort { $sort_criteria{$a->{'name'}} . $a->{'name'} cmp $sort_criteria{$b->{'name'}} . $b->{'name'}; } @$cases; # For debugging the sort-order # foreach my $tinfo (@$cases) # { # print("$sort_criteria{$tinfo->{'name'}} -> \t$tinfo->{'name'}\n"); # } } if (defined $print_testcases){ print_testcases(@$cases); exit(1); } return $cases; } # Returns (suitename, testname, extension) sub split_testname { my ($test_name)= @_; # Get rid of directory part and split name on .'s my @parts= split(/\./, basename($test_name)); if (@parts == 1){ # Only testname given, ex: alias return (undef , $parts[0], undef); } elsif (@parts == 2) { # Either testname.test or suite.testname given # Ex. main.alias or alias.test if ($parts[1] eq "test") { return (undef , $parts[0], $parts[1]); } else { return ($parts[0], $parts[1], undef); } } elsif (@parts == 3) { # Fully specified suitename.testname.test # ex main.alias.test return ( $parts[0], $parts[1], $parts[2]); } mtr_error("Illegal format of test name: $test_name"); } sub collect_one_suite($) { my $suite= shift; # Test suite name my $opt_cases= shift; my @cases; # Array of hash mtr_verbose("Collecting: $suite"); my $suitedir= "$::glob_mysql_test_dir"; # Default if ( $suite ne "main" ) { $suitedir= mtr_path_exists("$suitedir/suite/$suite", "$suitedir/$suite"); mtr_verbose("suitedir: $suitedir"); } my $testdir= "$suitedir/t"; my $resdir= "$suitedir/r"; # Check if t/ exists if (-d $testdir){ # t/ exists if ( -d $resdir ) { # r/exists } else { # No r/, use t/ as result dir $resdir= $testdir; } } else { # No t/ dir => there can' be any r/ dir mtr_error("Can't have r/ dir without t/") if -d $resdir; # No t/ or r/ => use suitedir $resdir= $testdir= $suitedir; } mtr_verbose("testdir: $testdir"); mtr_verbose("resdir: $resdir"); # ---------------------------------------------------------------------- # Build a hash of disabled testcases for this suite # ---------------------------------------------------------------------- my %disabled; if ( open(DISABLED, "$testdir/disabled.def" ) ) { while ( <DISABLED> ) { chomp; if ( /^\s*(\S+)\s*:\s*(.*?)\s*$/ ) { $disabled{$1}= $2; } } close DISABLED; } # Read suite.opt file my $suite_opt_file= "$testdir/suite.opt"; my $suite_opts= []; if ( -f $suite_opt_file ) { $suite_opts= opts_from_file($suite_opt_file); } if ( @$opt_cases ) { # Collect in specified order foreach my $test_name_spec ( @$opt_cases ) { my ($sname, $tname, $extension)= split_testname($test_name_spec); # The test name parts have now been defined #print " suite_name: $sname\n"; #print " tname: $tname\n"; #print " extension: $extension\n"; # Check cirrect suite if suitename is defined next if (defined $sname and $suite ne $sname); if ( defined $extension ) { my $full_name= "$testdir/$tname.$extension"; # Extension was specified, check if the test exists if ( ! -f $full_name) { # This is only an error if suite was specified, otherwise it # could exist in another suite mtr_error("Test '$full_name' was not found in suite '$sname'") if $sname; next; } } else { # No extension was specified, use default $extension= "test"; my $full_name= "$testdir/$tname.$extension"; # Test not found here, could exist in other suite next if ( ! -f $full_name ); } push(@cases, collect_one_test_case($suitedir, $testdir, $resdir, $suite, $tname, "$tname.$extension", \%disabled, $suite_opts)); } } else { opendir(TESTDIR, $testdir) or mtr_error("Can't open dir \"$testdir\": $!"); foreach my $elem ( sort readdir(TESTDIR) ) { my $tname= mtr_match_extension($elem, 'test'); next unless defined $tname; # Skip tests that does not match the --do-test= filter next if ($do_test_reg and not $tname =~ /$do_test_reg/o); push(@cases, collect_one_test_case($suitedir, $testdir, $resdir, $suite, $tname, $elem, \%disabled, $suite_opts)); } closedir TESTDIR; } # Return empty list if no testcases found return if (@cases == 0); # ---------------------------------------------------------------------- # Read combinations for this suite and build testcases x combinations # if any combinations exists # ---------------------------------------------------------------------- if ( ! $opt_skip_combination ) { my @combinations; my $combination_file= "$suitedir/combinations"; #print "combination_file: $combination_file\n"; if (@::opt_combinations) { # take the combination from command-line mtr_verbose("Take the combination from command line"); foreach my $combination (@::opt_combinations) { my $comb= {}; $comb->{name}= $combination; push(@{$comb->{comb_opt}}, $combination); push(@combinations, $comb); } } elsif (-f $combination_file ) { # Read combinations file in my.cnf format mtr_verbose("Read combinations file"); my $config= My::Config->new($combination_file); foreach my $group ($config->groups()) { my $comb= {}; $comb->{name}= $group->name(); foreach my $option ( $group->options() ) { push(@{$comb->{comb_opt}}, $option->name()."=".$option->value()); } push(@combinations, $comb); } } if (@combinations) { print " - adding combinations\n"; #print_testcases(@cases); my @new_cases; foreach my $comb (@combinations) { foreach my $test (@cases) { next if ( $test->{'skip'} ); # Skip this combination if the values it provides # already are set in master_opt or slave_opt if (My::Options::is_set($test->{master_opt}, $comb->{comb_opt}) && My::Options::is_set($test->{slave_opt}, $comb->{comb_opt}) ){ next; } # Copy test options my $new_test= {}; while (my ($key, $value) = each(%$test)) { if (ref $value eq "ARRAY") { push(@{$new_test->{$key}}, @$value); } else { $new_test->{$key}= $value; } } # Append the combination options to master_opt and slave_opt push(@{$new_test->{master_opt}}, @{$comb->{comb_opt}}); push(@{$new_test->{slave_opt}}, @{$comb->{comb_opt}}); # Add combination name short name $new_test->{combination}= $comb->{name}; # Add the new test to new test cases list push(@new_cases, $new_test); } } # Add the plain test if it was not already added # as part of a combination my %added; foreach my $new_test (@new_cases){ $added{$new_test->{name}}= 1; } foreach my $test (@cases){ push(@new_cases, $test) unless $added{$test->{name}}; } #print_testcases(@new_cases); @cases= @new_cases; #print_testcases(@cases); } } optimize_cases(\@cases); #print_testcases(@cases); return @cases; } # # Loop through all test cases # - optimize which test to run by skipping unnecessary ones # - update settings if necessary # sub optimize_cases { my ($cases)= @_; foreach my $tinfo ( @$cases ) { # Skip processing if already marked as skipped next if $tinfo->{skip}; # ======================================================= # If a special binlog format was selected with # --mysqld=--binlog-format=x, skip all test that does not # support it # ======================================================= #print "binlog_format: $binlog_format\n"; if (defined $binlog_format ) { # ======================================================= # Fixed --binlog-format=x specified on command line # ======================================================= if ( defined $tinfo->{'binlog_formats'} ) { #print "binlog_formats: ". join(", ", @{$tinfo->{binlog_formats}})."\n"; # The test supports different binlog formats # check if the selected one is ok my $supported= grep { $_ eq $binlog_format } @{$tinfo->{'binlog_formats'}}; if ( !$supported ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Doesn't support --binlog-format='$binlog_format'"; } } } else { # ======================================================= # Use dynamic switching of binlog format # ======================================================= # Get binlog-format used by this test from master_opt my $test_binlog_format; foreach my $opt ( @{$tinfo->{master_opt}} ) { $test_binlog_format= mtr_match_prefix($opt, "--binlog-format=") || $test_binlog_format; } if (defined $test_binlog_format and defined $tinfo->{binlog_formats} ) { my $supported= grep { $_ eq $test_binlog_format } @{$tinfo->{'binlog_formats'}}; if ( !$supported ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Doesn't support --binlog-format='$test_binlog_format'"; next; } } } } } # # Read options from the given opt file and append them as an array # to $tinfo->{$opt_name} # sub process_opts_file { my ($tinfo, $opt_file, $opt_name)= @_; $tinfo->{$opt_name}= []; if ( -f $opt_file ) { my $opts= opts_from_file($opt_file); foreach my $opt ( @$opts ) { my $value; # The opt file is used both to send special options to the mysqld # as well as pass special test case specific options to this # script $value= mtr_match_prefix($opt, "--timezone="); if ( defined $value ) { $tinfo->{'timezone'}= $value; next; } $value= mtr_match_prefix($opt, "--result-file="); if ( defined $value ) { # Specifies the file mysqltest should compare # output against $tinfo->{'result_file'}= "r/$value.result"; next; } $value= mtr_match_prefix($opt, "--config-file-template="); if ( defined $value) { # Specifies the configuration file to use for this test $tinfo->{'template_path'}= dirname($tinfo->{path})."/$value"; next; } # If we set default time zone, remove the one we have $value= mtr_match_prefix($opt, "--default-time-zone="); if ( defined $value ) { # Set timezone for this test case to something different $tinfo->{'timezone'}= "GMT-8"; # Fallthrough, add the --default-time-zone option } # The --restart option forces a restart even if no special # option is set. If the options are the same as next testcase # there is no need to restart after the testcase # has completed if ( $opt eq "--force-restart" ) { $tinfo->{'force_restart'}= 1; next; } # Ok, this was a real option, add it push(@{$tinfo->{$opt_name}}, $opt); } } } ############################################################################## # # Collect information about a single test case # ############################################################################## sub collect_one_test_case { my $suitedir= shift; my $testdir= shift; my $resdir= shift; my $suitename= shift; my $tname= shift; my $filename= shift; my $disabled= shift; my $suite_opts= shift; #print "collect_one_test_case\n"; #print " suitedir: $suitedir\n"; #print " testdir: $testdir\n"; #print " resdir: $resdir\n"; #print " suitename: $suitename\n"; #print " tname: $tname\n"; #print " filename: $filename\n"; # ---------------------------------------------------------------------- # Check --start-from # ---------------------------------------------------------------------- if ( $start_from ) { # start_from can be specified as [suite.].testname_prefix my ($suite, $test, $ext)= split_testname($start_from); if ( $suite and $suitename lt $suite){ return; # Skip silently } if ( $tname lt $test ){ return; # Skip silently } } # ---------------------------------------------------------------------- # Set defaults # ---------------------------------------------------------------------- my $tinfo= {}; $tinfo->{'name'}= $suitename . ".$tname"; $tinfo->{'path'}= "$testdir/$filename"; # TODO allow nonexistsing result file # in that case .test must issue "exit" otherwise test should fail by default $tinfo->{'result_file'}= "$resdir/$tname.result"; # ---------------------------------------------------------------------- # Skip some tests but include in list, just mark them as skipped # ---------------------------------------------------------------------- if ( $skip_test_reg and $tname =~ /$skip_test_reg/o ) { $tinfo->{'skip'}= 1; return $tinfo; } # ---------------------------------------------------------------------- # Check for disabled tests # ---------------------------------------------------------------------- my $marked_as_disabled= 0; if ( $disabled->{$tname} ) { # Test was marked as disabled in suites disabled.def file $marked_as_disabled= 1; $tinfo->{'comment'}= $disabled->{$tname}; } my $disabled_file= "$testdir/$tname.disabled"; if ( -f $disabled_file ) { $marked_as_disabled= 1; $tinfo->{'comment'}= mtr_fromfile($disabled_file); } if ( $marked_as_disabled ) { if ( $enable_disabled ) { # User has selected to run all disabled tests mtr_report(" - $tinfo->{name} wil be run although it's been disabled\n", " due to '$tinfo->{comment}'"); } else { $tinfo->{'skip'}= 1; $tinfo->{'disable'}= 1; # Sub type of 'skip' return $tinfo; } } # ---------------------------------------------------------------------- # Append suite extra options to both master and slave # ---------------------------------------------------------------------- push(@{$tinfo->{'master_opt'}}, @$suite_opts); push(@{$tinfo->{'slave_opt'}}, @$suite_opts); # ---------------------------------------------------------------------- # Add master opts, extra options only for master # ---------------------------------------------------------------------- process_opts_file($tinfo, "$testdir/$tname-master.opt", 'master_opt'); # ---------------------------------------------------------------------- # Add slave opts, list of extra option only for slave # ---------------------------------------------------------------------- process_opts_file($tinfo, "$testdir/$tname-slave.opt", 'slave_opt'); # ---------------------------------------------------------------------- # master sh # ---------------------------------------------------------------------- my $master_sh= "$testdir/$tname-master.sh"; if ( -f $master_sh ) { if ( IS_WIN32PERL ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "No tests with sh scripts on Windows"; return $tinfo; } else { $tinfo->{'master_sh'}= $master_sh; } } # ---------------------------------------------------------------------- # slave sh # ---------------------------------------------------------------------- my $slave_sh= "$testdir/$tname-slave.sh"; if ( -f $slave_sh ) { if ( IS_WIN32PERL ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "No tests with sh scripts on Windows"; return $tinfo; } else { $tinfo->{'slave_sh'}= $slave_sh; } } # ---------------------------------------------------------------------- # <tname>.slave-mi # ---------------------------------------------------------------------- mtr_error("$tname: slave-mi not supported anymore") if ( -f "$testdir/$tname.slave-mi"); tags_from_test_file($tinfo,"$testdir/${tname}.test"); if ( defined $default_storage_engine ) { # Different default engine is used # tag test to require that engine $tinfo->{'ndb_test'}= 1 if ( $default_storage_engine =~ /^ndb/i ); $tinfo->{'innodb_test'}= 1 if ( $default_storage_engine =~ /^innodb/i ); } if ( $tinfo->{'big_test'} and ! $::opt_big_test ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Test need 'big-test' option"; return $tinfo } if ( $tinfo->{'ndb_extra'} and ! $::opt_ndb_extra_test ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Test need 'ndb_extra' option"; return $tinfo } if ( $tinfo->{'need_debug'} && ! $::debug_compiled_binaries ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Test need debug binaries"; return $tinfo } if ( $tinfo->{'ndb_test'} ) { # This is a NDB test if ( $::opt_skip_ndbcluster == 2 ) { # Ndb is not supported, skip it $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "No ndbcluster support"; return $tinfo; } elsif ( $::opt_skip_ndbcluster ) { # All ndb test's should be skipped $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "No ndbcluster tests(--skip-ndbcluster)"; return $tinfo; } } else { # This is not a ndb test if ( $opt_with_ndbcluster_only ) { # Only the ndb test should be run, all other should be skipped $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Only ndbcluster tests"; return $tinfo; } } if ( $tinfo->{'innodb_test'} ) { # This is a test that need innodb if ( $::mysqld_variables{'innodb'} ne "TRUE" ) { # innodb is not supported, skip it $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "No innodb support"; return $tinfo; } } else { push(@{$tinfo->{'master_opt'}}, "--loose-skip-innodb"); push(@{$tinfo->{'slave_opt'}}, "--loose-skip-innodb"); } if ( $tinfo->{'need_binlog'} ) { if (grep(/^--skip-log-bin/, @::opt_extra_mysqld_opt) ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Test need binlog"; return $tinfo; } } else { # Test does not need binlog, add --skip-binlog to # the options used when starting push(@{$tinfo->{'master_opt'}}, "--loose-skip-log-bin"); push(@{$tinfo->{'slave_opt'}}, "--loose-skip-log-bin"); } if ( $tinfo->{'rpl_test'} ) { if ( $skip_rpl ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "No replication tests(--skip-rpl)"; return $tinfo; } } if ( $::opt_embedded_server ) { if ( $tinfo->{'not_embedded'} ) { $tinfo->{'skip'}= 1; $tinfo->{'comment'}= "Not run for embedded server"; return $tinfo; } } # ---------------------------------------------------------------------- # Find config file to use if not already selected in <testname>.opt file # ---------------------------------------------------------------------- if (defined $defaults_file) { # Using same config file for all tests $tinfo->{template_path}= $defaults_file; } elsif (! $tinfo->{template_path} ) { my $config= "$suitedir/my.cnf"; if (! -f $config ) { # assume default.cnf will be used $config= "include/default_my.cnf"; # Suite has no config, autodetect which one to use if ( $tinfo->{rpl_test} ){ $config= "suite/rpl/my.cnf"; if ( $tinfo->{ndb_test} ){ $config= "suite/rpl_ndb/my.cnf"; } } elsif ( $tinfo->{ndb_test} ){ $config= "suite/ndb/my.cnf"; } } $tinfo->{template_path}= $config; } # Set extra config file to use if (defined $defaults_extra_file) { $tinfo->{extra_template_path}= $defaults_extra_file; } # ---------------------------------------------------------------------- # Append mysqld extra options to both master and slave # ---------------------------------------------------------------------- push(@{$tinfo->{'master_opt'}}, @::opt_extra_mysqld_opt); push(@{$tinfo->{'slave_opt'}}, @::opt_extra_mysqld_opt); return $tinfo; } # List of tags in the .test files that if found should set # the specified value in "tinfo" my @tags= ( ["include/have_binlog_format_row.inc", "binlog_formats", ["row"]], ["include/have_binlog_format_statement.inc", "binlog_formats", ["statement"]], ["include/have_binlog_format_mixed.inc", "binlog_formats", ["mixed"]], ["include/have_binlog_format_mixed_or_row.inc", "binlog_formats", ["mixed", "row"]], ["include/have_binlog_format_mixed_or_statement.inc", "binlog_formats", ["mixed", "statement"]], ["include/have_binlog_format_row_or_statement.inc", "binlog_formats", ["row", "statement"]], ["include/have_log_bin.inc", "need_binlog", 1], ["include/have_innodb.inc", "innodb_test", 1], ["include/big_test.inc", "big_test", 1], ["include/have_debug.inc", "need_debug", 1], ["include/have_ndb.inc", "ndb_test", 1], ["include/have_multi_ndb.inc", "ndb_test", 1], ["include/have_ndb_extra.inc", "ndb_extra", 1], ["include/master-slave.inc", "rpl_test", 1], ["include/ndb_master-slave.inc", "rpl_test", 1], ["include/ndb_master-slave.inc", "ndb_test", 1], ["include/federated.inc", "federated_test", 1], ["include/not_embedded.inc", "not_embedded", 1], ); sub tags_from_test_file { my $tinfo= shift; my $file= shift; #mtr_verbose("$file"); my $F= IO::File->new($file) or mtr_error("can't open file \"$file\": $!"); while ( my $line= <$F> ) { # Skip line if it start's with # next if ( $line =~ /^#/ ); # Match this line against tag in "tags" array foreach my $tag (@tags) { if ( index($line, $tag->[0]) >= 0 ) { # Tag matched, assign value to "tinfo" $tinfo->{"$tag->[1]"}= $tag->[2]; } } # If test sources another file, open it as well if ( $line =~ /^\-\-([[:space:]]*)source(.*)$/ or $line =~ /^([[:space:]]*)source(.*);$/ ) { my $value= $2; $value =~ s/^\s+//; # Remove leading space $value =~ s/[[:space:]]+$//; # Remove ending space # Sourced file may exist relative to test or # in global location foreach my $sourced_file (dirname($file). "/$value", "$::glob_mysql_test_dir/$value") { if ( -f $sourced_file ) { # Only source the file if it exists, we may get # false positives in the regexes above if someone # writes "source nnnn;" in a test case(such as mysqltest.test) tags_from_test_file($tinfo, $sourced_file); last; } } } } } sub unspace { my $string= shift; my $quote= shift; $string =~ s/[ \t]/\x11/g; return "$quote$string$quote"; } sub envsubst { my $string= shift; if ( ! defined $ENV{$string} ) { mtr_error(".opt file references '$string' which is not set"); } return $ENV{$string}; } sub opts_from_file ($) { my $file= shift; open(FILE,"<",$file) or mtr_error("can't open file \"$file\": $!"); my @args; while ( <FILE> ) { chomp; # --set-variable=init_connect=set @a='a\\0c' s/^\s+//; # Remove leading space s/\s+$//; # Remove ending space # This is strange, but we need to fill whitespace inside # quotes with something, to remove later. We do this to # be able to split on space. Else, we have trouble with # options like # # --someopt="--insideopt1 --insideopt2" # # But still with this, we are not 100% sure it is right, # we need a shell to do it right. s/\'([^\'\"]*)\'/unspace($1,"\x0a")/ge; s/\"([^\'\"]*)\"/unspace($1,"\x0b")/ge; s/\'([^\'\"]*)\'/unspace($1,"\x0a")/ge; s/\"([^\'\"]*)\"/unspace($1,"\x0b")/ge; foreach my $arg (split(/[ \t]+/)) { $arg =~ tr/\x11\x0a\x0b/ \'\"/; # Put back real chars # The outermost quotes has to go $arg =~ s/^([^\'\"]*)\'(.*)\'([^\'\"]*)$/$1$2$3/ or $arg =~ s/^([^\'\"]*)\"(.*)\"([^\'\"]*)$/$1$2$3/; $arg =~ s/\\\\/\\/g; # Expand environment variables $arg =~ s/\$\{(\w+)\}/envsubst($1)/ge; $arg =~ s/\$(\w+)/envsubst($1)/ge; # Do not pass empty string since my_getopt is not capable to handle it. if (length($arg)) { push(@args, $arg); } } } close FILE; return \@args; } sub print_testcases { my (@cases)= @_; print "=" x 60, "\n"; foreach my $test (@cases){ print "[", $test->{name}, "]", "\n"; while ((my ($key, $value)) = each(%$test)) { print " ", $key, "= "; if (ref $value eq "ARRAY") { print "[", join(", ", @$value), "]"; } else { print $value; } print "\n"; } print "\n"; } print "=" x 60, "\n"; } 1;