2000-08-17 01:05:02 +03:00
#!@PERL@
2000-07-31 21:29:14 +02:00
# Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Library General Public
# License as published by the Free Software Foundation; either
# version 2 of the License, or (at your option) any later version.
#
# This library 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
# Library General Public License for more details.
#
# You should have received a copy of the GNU Library General Public
# License along with this library; if not, write to the Free
# Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
# MA 02111-1307, USA
#
# The configuration file for the DBI/DBD tests on different databases ....
# You will need the DBD module for the database you are running.
# Monty made this bench script and I (Luuk de Boer) rewrote it to DBI/DBD.
# Monty rewrote this again to use packages.
#
# Each database has a different package that has 3 functions:
# new Creates a object with some standard slot
# version Version number of the server
# create Generates commands to create a table
#
#
# First some global functions that help use the packages:
#
sub get_server
{
my ( $name ,$host ,$database ,$odbc ,$machine ) = @_;
my ( $server ) ;
if ( $name = ~ /mysql/i)
{ $server = new db_MySQL( $host , $database , $machine ) ; }
elsif ( $name = ~ /pg/i)
{ $server = new db_Pg( $host ,$database ) ; }
elsif ( $name = ~ /msql/i)
{ $server = new db_mSQL( $host ,$database ) ; }
elsif ( $name = ~ /solid/i)
{ $server = new db_Solid( $host ,$database ) ; }
elsif ( $name = ~ /Empress/i)
{ $server = new db_Empress( $host ,$database ) ; }
elsif ( $name = ~ /Oracle/i)
{ $server = new db_Oracle( $host ,$database ) ; }
elsif ( $name = ~ /Access/i)
{ $server = new db_access( $host ,$database ) ; }
elsif ( $name = ~ /Informix/i)
{ $server = new db_Informix( $host ,$database ) ; }
elsif ( $name = ~ /ms-sql/i)
{ $server = new db_ms_sql( $host ,$database ) ; }
elsif ( $name = ~ /sybase/i)
{ $server = new db_sybase( $host ,$database ) ; }
elsif ( $name = ~ /Adabas/i) # Adabas has two drivers
{
$server = new db_Adabas( $host ,$database ) ;
if ( $name = ~ /AdabasD/i)
{
$server ->{ 'data_source' } = ~ s/:Adabas:/:AdabasD:/;
}
}
elsif ( $name = ~ /DB2/i)
{ $server = new db_db2( $host ,$database ) ; }
elsif ( $name = ~ /Mimer/i)
{ $server = new db_Mimer( $host ,$database ) ; }
elsif ( $name = ~ /interBase/i)
{ $server = new db_interbase( $host ,$database ) ; }
else
{
die " Unknown sql server name used: $name \nUse one of: Access, Adabas, AdabasD, Empress, Oracle, Informix, DB2, mSQL, Mimer, MS-SQL, MySQL, Pg, Solid or Sybase.\nIf the connection is done trough ODBC the name must end with _ODBC\n " ;
}
if ( $name = ~ /_ODBC$/i || defined( $odbc ) && $odbc )
{
if ( ! ( $server ->{ 'data_source' } = ~ /^( [ ^:] *) :( [ ^:] +) :( [ ^:] *) / ) )
{
die "Can't find databasename in data_source: '" .
$server ->{ 'data_source' } . "'\n" ;
}
if ( $3 ) {
$server ->{ 'data_source' } = " $1 :ODBC: $3 " ;
} else {
$server ->{ 'data_source' } = " $1 :ODBC: $database " ;
}
}
return $server ;
}
sub all_servers
{
return [ "Access" , "Adabas" , "DB2" , "Empress" , "Oracle" , "Informix" ,
"InterBase" , "Mimer" , "mSQL" , "MS-SQL" , "MySQL" , "Pg" , "Solid" ,
"Sybase" ] ;
}
#############################################################################
# First the configuration for MySQL off course :-)
#############################################################################
package db_MySQL;
sub new
{
my ( $type ,$host ,$database ,$machine ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "mysql" ;
$self ->{ 'data_source' } = " DBI:mysql: $database : $host " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "blob" ;
$self ->{ 'text' } = "text" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
2000-09-13 01:08:34 +03:00
$self ->{ 'vacuum' } = 1; # When using with --fast
2000-07-31 21:29:14 +02:00
$limits { 'max_conditions' } = 9999; # (Actually not a limit)
$limits { 'max_columns' } = 2000; # Max number of columns in table
# Windows can't handle that many files in one directory
$limits { 'max_tables' } = ( $machine = ~ "^win" ) ? 5000 : 65000;
$limits { 'max_text_size' } = 65000; # Max size with default buffers.
$limits { 'query_size' } = 1000000; # Max size with default buffers.
$limits { 'max_index' } = 16; # Max number of keys
$limits { 'max_index_parts' } = 16; # Max segments/key
$limits { 'max_column_name' } = 64; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 1; # Has load data infile
$limits { 'lock_tables' } = 1; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 1; # Can do 'select 1';
$limits { 'multi_drop' } = 1; # Drop table can take many tables
$limits { 'subqueries' } = 0; # Doesn't support sub-queries.
$limits { 'left_outer_join' } = 1; # Supports left outer joins
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 1; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can use group functions in HAVING
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 1; # Can use 'GROUP BY 1'
$limits { 'alter_table' } = 1; # Have ALTER TABLE
$limits { 'alter_add_multi_col' } = 1; #Have ALTER TABLE t add a int,add b int;
$limits { 'alter_table_dropcol' } = 1; # Have ALTER TABLE DROP column
$limits { 'multi_value_insert' } = 1; # Have INSERT ... values (1,2),(3,4)
$limits { 'group_func_extra_std' } = 1; # Have group function std().
$limits { 'func_odbc_mod' } = 1; # Have function mod.
$limits { 'func_extra_%' } = 1; # Has % as alias for mod()
$limits { 'func_odbc_floor' } = 1; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 1; # Have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 1; # Has function in
$limits { 'limit' } = 1; # supports the limit attribute
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
$smds { 'time' } = 1;
$smds { 'q1' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q2' } = 'b' ;
$smds { 'q3' } = 'b' ; # with time ('')
$smds { 'q4' } = 'c' ; # with time not supp by mysql (d)
$smds { 'q5' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q6' } = 'c' ; # with time not supp by mysql ('')
$smds { 'q7' } = 'c' ;
$smds { 'q8' } = 'f' ;
$smds { 'q9' } = 'c' ;
$smds { 'q10' } = 'b' ;
$smds { 'q11' } = 'b' ;
$smds { 'q12' } = 'd' ;
$smds { 'q13' } = 'c' ;
$smds { 'q14' } = 'd' ;
$smds { 'q15' } = 'd' ;
$smds { 'q16' } = 'a' ;
$smds { 'q17' } = 'c' ;
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
my ( $dbh ,$sth ,$version ,@row) ;
$dbh = $self ->connect( ) ;
$sth = $dbh ->prepare( "select VERSION()" ) or die $DBI ::errstr;
$version = "MySQL 3.20.?" ;
if ( $sth ->execute && ( @row = $sth ->fetchrow_array) )
{
$row [ 0] = ~ s/-/ /g; # To get better tables with long names
$version = " MySQL $row [0] " ;
}
$sth ->finish;
$dbh ->disconnect;
return $version ;
}
#
# Connection with optional disabling of logging
#
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
$dbh ->do( "SET OPTION LOG_OFF=1,UPDATE_LOG=0" ) ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
# If one uses $main::opt_fast then one is allowed to use
# non standard types to get better speed.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ,$options ) = @_;
my( $query ,@queries) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/ decimal/ double( 10,2) /i;
$field = ~ s/ big_decimal/ double( 10,2) /i;
$field = ~ s/ date/ int/i; # Because of tcp ?
$query .= $field . ',' ;
}
foreach $index ( @$index )
{
$query .= $index . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
$query .= " $options " if ( defined( $options ) ) ;
$query .= " $main ::opt_create_options " if ( defined( $main ::opt_create_options) ) ;
push( @queries,$query ) ;
return @queries;
}
sub insert_file {
my ( $self ,$dbname , $file , $dbh ) = @_;
my ( $command , $sth ) ;
$file = ~ s| \\ | /| g; # Change Win32 names to Unix syntax
$command = " load data infile ' $file ' into table $dbname columns optionally enclosed by '\\'' terminated by ',' " ;
# print "$command\n";
$sth = $dbh ->do( $command ) or die $DBI ::errstr;
return $sth ; # Contains number of rows
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index {
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index ON $table " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 0;
}
#
# This should return 1 if we to do disconnect / connect when doing
# big batches
#
sub small_rollback_segment
{
return 0;
}
#
# reconnect on errors (needed mainly be crash-me)
#
sub reconnect_on_errors
{
return 0;
}
2000-09-13 01:08:34 +03:00
#
# Optimize tables for better performance
#
sub vacuum
{
my ( $self ,$full_vacuum ,$dbh_ref ,@tables) = @_;
my ( $loop_time ,$end_time ,$dbh ) ;
if ( $# tables >= 0)
{
$dbh = $$ dbh_ref;
$loop_time = new Benchmark;
$dbh ->do( "OPTIMIZE TABLE " . join( ',' ,@tables) ) || die " Got error: $DBI ::errstr when executing 'OPTIMIZE TABLE'\n " ;
$end_time = new Benchmark;
print "Time for book-keeping (1): " .
Benchmark::timestr( Benchmark::timediff( $end_time , $loop_time ) ,"all" ) . "\n\n" ;
}
}
2000-07-31 21:29:14 +02:00
#############################################################################
# Definitions for mSQL
#############################################################################
package db_mSQL;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "msql" ;
$self ->{ 'data_source' } = " DBI:mSQL: $database : $main :: $opt_host " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'double_quotes' } = 0;
$limits { 'max_conditions' } = 74;
$limits { 'max_columns' } = 75;
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 32000;
$limits { 'query_size' } = 65535;
$limits { 'max_index' } = 5;
$limits { 'max_index_parts' } = 10;
$limits { 'max_column_name' } = 35;
$limits { 'join_optimizer' } = 0; # Can't optimize FROM tables
$limits { 'load_data_infile' } = 0;
$limits { 'lock_tables' } = 0;
$limits { 'functions' } = 0;
$limits { 'group_functions' } = 0;
$limits { 'group_distinct_functions' } = 0; # Have count(distinct)
$limits { 'multi_drop' } = 0;
$limits { 'select_without_from' } = 0;
$limits { 'subqueries' } = 0;
$limits { 'left_outer_join' } = 0;
$limits { 'table_wildcard' } = 0;
$limits { 'having_with_alias' } = 0;
$limits { 'having_with_group' } = 0;
$limits { 'like_with_column' } = 1;
$limits { 'order_by_position' } = 1;
$limits { 'group_by_position' } = 1;
$limits { 'alter_table' } = 0;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 0;
$limits { 'group_func_extra_std' } = 0;
$limits { 'limit' } = 1; # supports the limit attribute
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
$limits { 'func_odbc_mod' } = 0;
$limits { 'func_extra_%' } = 0;
$limits { 'func_odbc_floor' } = 0;
$limits { 'func_extra_if' } = 0;
$limits { 'column_alias' } = 0;
$limits { 'NEG' } = 0;
$limits { 'func_extra_in_num' } = 0;
$self ->{ 'blob' } = "text(" . $limits { 'max_text_size' } .")" ;
$self ->{ 'text' } = "text(" . $limits { 'max_text_size' } .")" ;
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $tmp ,$dir ) ;
foreach $dir ( "/usr/local/Hughes" , "/usr/local/mSQL" ,"/my/local/mSQL" ,
"/usr/local" )
{
if ( -x " $dir /bin/msqladmin " )
{
$tmp = ` $dir /bin/msqladmin version | grep server` ;
if ( $tmp = ~ /^\s *( .*\w ) \s *$/)
{ # Strip pre- and endspace
$tmp = $1 ;
$tmp = ~ s/\s +/ /g; # Remove unnecessary spaces
return $tmp ;
}
}
}
return "mSQL version ???" ;
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Can't handle many field types, so we map everything to int and real.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$name ,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/varchar/char/i; # mSQL doesn't have VARCHAR()
# mSQL can't handle more than the real basic int types
$field = ~ s/tinyint| smallint| mediumint| integer/int/i;
# mSQL can't handle different visual lengths
$field = ~ s/int\( \d *\) /int/i;
2000-08-16 04:17:40 -05:00
# mSQL doesn't have float, change it to real
2000-07-31 21:29:14 +02:00
$field = ~ s/float( \( \d *,\d *\) ) { 0,1} /real/i;
$field = ~ s/double( \( \d *,\d *\) ) { 0,1} /real/i;
# mSQL doesn't have blob, it has text instead
if ( $field = ~ / blob/i)
{
$name = $self ->{ 'blob' } ;
$field = ~ s/ blob/ $name /;
}
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 0;
# Prepend table_name to index name because the the name may clash with
# a field name. (Should be diffent name space, but this is mSQL...)
foreach $index ( @$index )
{
# Primary key is unique index in mSQL
$index = ~ s/primary key/unique index primary/i;
if ( $index = ~ /^unique\s *\( ( [ ^\( ] *) \) $/i)
{
$nr ++;
push( @queries," create unique index ${ table_name } _ $nr on $table_name ( $1 ) " ) ;
}
else
{
if ( !( $index = ~ /^( .*index) \s +( \w *) \s +( \( .*\) ) $/i) )
{
die " Can't parse index information in ' $index '\n " ;
}
push( @queries," create $1 ${ table_name } _ $2 on $table_name $3 " ) ;
}
}
return @queries;
}
sub insert_file {
my( $self ,$dbname , $file ) = @_;
print "insert an ascii file isn't supported by mSQL\n" ;
return 0;
}
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index FROM $table " ;
}
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Definitions for PostgreSQL #
#############################################################################
package db_Pg;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "pg" ;
$self ->{ 'data_source' } = " DBI:Pg:dbname= $database " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "text" ;
$self ->{ 'text' } = "text" ;
$self ->{ 'double_quotes' } = 1;
$self ->{ "vacuum" } = 1;
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Is this true ?
$limits { 'NEG' } = 1; # Can't handle -id
$limits { 'alter_table' } = 1; # alter ??
$limits { 'alter_add_multi_col' } = 0; # alter_add_multi_col ?
$limits { 'alter_table_dropcol' } = 0; # alter_drop_col ?
$limits { 'column_alias' } = 1;
$limits { 'func_extra_%' } = 1;
$limits { 'func_extra_if' } = 0;
$limits { 'func_extra_in_num' } = 1;
$limits { 'func_odbc_floor' } = 1;
$limits { 'func_odbc_mod' } = 1; # Has %
$limits { 'functions' } = 1;
$limits { 'group_by_position' } = 1;
$limits { 'group_func_extra_std' } = 0;
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_functions' } = 1;
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'having_with_alias' } = 0;
2000-08-15 20:09:37 +03:00
$limits { 'having_with_group' } = 1;
2000-07-31 21:29:14 +02:00
$limits { 'left_outer_join' } = 0;
$limits { 'like_with_column' } = 1;
$limits { 'lock_tables' } = 0; # in ATIS gives this a problem
$limits { 'multi_drop' } = 1;
$limits { 'order_by_position' } = 1;
$limits { 'select_without_from' } = 1;
$limits { 'subqueries' } = 1;
$limits { 'table_wildcard' } = 1;
2000-08-15 20:09:37 +03:00
$limits { 'max_column_name' } = 32; # Is this true
$limits { 'max_columns' } = 1000; # 500 crashes pg 6.3
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_conditions' } = 30; # This makes Pg real slow
$limits { 'max_index' } = 64; # Is this true ?
$limits { 'max_index_parts' } = 16; # Is this true ?
$limits { 'max_text_size' } = 7000; # 8000 crashes pg 6.3
2000-08-17 01:05:02 +03:00
$limits { 'query_size' } = 16777216;
2000-08-22 00:18:32 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
# the different cases per query ...
$smds { 'q1' } = 'b' ; # with time
$smds { 'q2' } = 'b' ;
$smds { 'q3' } = 'b' ; # with time
$smds { 'q4' } = 'c' ; # with time
$smds { 'q5' } = 'b' ; # with time
$smds { 'q6' } = 'c' ; # strange error ....
$smds { 'q7' } = 'c' ;
$smds { 'q8' } = 'f' ; # needs 128M to execute - can't do insert ...group by
$smds { 'q9' } = 'c' ;
$smds { 'q10' } = 'b' ;
$smds { 'q11' } = 'b' ; # can't do float8 * int4 - create operator
$smds { 'q12' } = 'd' ; # strange error???
$smds { 'q13' } = 'c' ;
$smds { 'q14' } = 'd' ; # strange error???
$smds { 'q15' } = 'd' ; # strange error???
$smds { 'q16' } = 'a' ;
$smds { 'q17' } = 'c' ;
$smds { 'time' } = 1; # the use of the time table -> 1 is on.
# when 0 then the date field must be a
# date field not a int field!!!
return $self ;
}
# couldn't find the option to get the version number
sub version
{
my ( $version ,$dir ) ;
foreach $dir ( $ENV { 'PGDATA' } ,"/usr/local/pgsql/data" , "/my/local/pgsql/" )
{
if ( $dir && -e " $dir /PG_VERSION " )
{
$version = ` cat $dir /PG_VERSION` ;
if ( $? = = 0)
{
chomp( $version ) ;
return " PostgreSQL $version " ;
}
}
}
return "PostgreSQL version ???" ;
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$name ,$in ,$indfield ,$table ,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
if ( $main ::opt_fast)
{
# Allow use of char2, char4, char8 or char16
$field = ~ s/char( 2| 4| 8| 16) /char$1 /;
}
# Pg can't handle more than the real basic int types
$field = ~ s/tinyint| smallint| mediumint| integer/int/;
# Pg can't handle different visual lengths
$field = ~ s/int\( \d *\) /int/;
$field = ~ s/float\( \d *,\d *\) /float/;
$field = ~ s/ double/ float/;
$field = ~ s/ decimal/ float/i;
$field = ~ s/ big_decimal/ float/i;
$field = ~ s/ date/ int/i;
# Pg doesn't have blob, it has text instead
$field = ~ s/ blob/ text/;
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
foreach $index ( @$index )
{
$index = ~ s/primary key/unique index primary_key/i;
if ( $index = ~ /^unique.*\( ( [ ^\( ] *) \) $/i)
{
2000-08-17 01:05:02 +03:00
# original: $indfield="using btree (" .$1.")";
# using btree doesn<73> t seem to work with Postgres anymore; it creates
# the table and adds the index, but it isn<73> t unique
$indfield = " (" .$1 .")" ;
2000-07-31 21:29:14 +02:00
$in = "unique index" ;
$table = " index_ $nr " ; $nr ++;
}
elsif ( $index = ~ /^( .*index) \s +( \w *) \s +( \( .*\) ) $/i)
{
2000-08-17 01:05:02 +03:00
# original: $indfield="using btree (" .$1.")";
$indfield = " " .$3 ;
2000-07-31 21:29:14 +02:00
$in = "index" ;
$table = " index_ $nr " ; $nr ++;
}
else
{
die " Can't parse index information in ' $index '\n " ;
}
push( @queries," create $in ${ table_name } _ $table on $table_name $indfield " ) ;
}
$queries [ 0] = $query ;
return @queries;
}
sub insert_file {
my ( $self ,$dbname , $file , $dbh ) = @_;
my ( $command , $sth ) ;
# Syntax:
# copy [binary] <class_name> [with oids]
# {to|from} {<filename>|stdin|stdout} [using delimiters <delim>]
print "The ascii files aren't correct for postgres ....!!!\n" ;
$command = " copy $dbname from ' $file ' using delimiters ',' " ;
print " $command \n " ;
$sth = $dbh ->do( $command ) or die $DBI ::errstr;
return $sth ;
}
#
# As postgreSQL wants A % B instead of standard mod(A,B) we have to map
# This will not handle all cases, but as the benchmarks doesn't use functions
# inside MOD() the following should work
#
# PostgreSQL cant handle count(*) or even count(1), but it can handle
# count(1+1) sometimes. ==> this is solved in PostgreSQL 6.3
#
# PostgreSQL 6.5 is supporting MOD.
sub query {
my( $self ,$sql ) = @_;
my( @select,$change ) ;
# if you use PostgreSQL 6.x and x is lower as 5 then uncomment the line below.
# $sql =~ s/mod\(([^,]*),([^\)]*)\)/\($1 % $2\)/gi;
#
# if you use PostgreSQL 6.1.x uncomment the lines below
# if ($sql =~ /select\s+count\(\*\)\s+from/i) {
# }
# elsif ($sql =~ /count\(\*\)/i)
# {
# if ($sql =~ /select\s+(.*)\s+from/i)
# {
# @select = split(/,/,$1);
# if ($select[0] =~ /(.*)\s+as\s+\w+$/i)
# {
# $change = $1;
# }
# else
# {
# $change = $select[0];
# }
# }
# if (($change =~ /count/i) || ($change eq "")) {
# $change = "1+1";
# }
# $sql =~ s/count\(\*\)/count($change)/gi;
# }
# till here.
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
sub abort_if_fatal_error
{
return 1 if ( $DBI ::errstr = ~ /sent to backend, but backend closed/i) ;
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
sub vacuum
{
my ( $self ,$full_vacuum ,$dbh_ref ) = @_;
my ( $loop_time ,$end_time ,$dbh ) ;
if ( defined( $full_vacuum ) )
{
$$ dbh_ref->disconnect; $$ dbh_ref = $self ->connect( ) ;
}
$dbh = $$ dbh_ref;
$loop_time = new Benchmark;
$dbh ->do( "vacuum" ) || die " Got error: $DBI ::errstr when executing 'vacuum'\n " ;
$dbh ->do( "vacuum pg_attributes" ) || die " Got error: $DBI ::errstr when executing 'vacuum'\n " ;
$dbh ->do( "vacuum pg_index" ) || die " Got error: $DBI ::errstr when executing 'vacuum'\n " ;
$dbh ->do( "vacuum analyze" ) || die " Got error: $DBI ::errstr when executing 'vacuum'\n " ;
$end_time = new Benchmark;
print "Time for book-keeping (1): " .
Benchmark::timestr( Benchmark::timediff( $end_time , $loop_time ) ,"all" ) . "\n\n" ;
$dbh ->disconnect; $$ dbh_ref = $self ->connect( ) ;
}
#############################################################################
# Definitions for Solid
#############################################################################
package db_Solid;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "solid" ;
$self ->{ 'data_source' } = "DBI:Solid:" ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "long varchar" ;
$self ->{ 'text' } = "long varchar" ;
$self ->{ 'double_quotes' } = 1;
$limits { 'max_conditions' } = 9999; # Probably big enough
$limits { 'max_columns' } = 2000; # From crash-me
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 65492; # According to tests
$limits { 'query_size' } = 65535; # Probably a limit
$limits { 'max_index' } = 64; # Probably big enough
$limits { 'max_index_parts' } = 64;
$limits { 'max_column_name' } = 80;
$limits { 'join_optimizer' } = 1;
$limits { 'load_data_infile' } = 0;
$limits { 'lock_tables' } = 0;
$limits { 'functions' } = 1;
$limits { 'group_functions' } = 1;
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 0; # Can do 'select 1' ?;
$limits { 'multi_drop' } = 0;
$limits { 'subqueries' } = 1;
$limits { 'left_outer_join' } = 1;
$limits { 'table_wildcard' } = 1;
$limits { 'having_with_alias' } = 0;
$limits { 'having_with_group' } = 1;
$limits { 'like_with_column' } = 1;
$limits { 'order_by_position' } = 0; # 2.30.0018 can this
$limits { 'group_by_position' } = 0;
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 0;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 1;
$limits { 'func_extra_%' } = 0;
$limits { 'func_odbc_floor' } = 1;
$limits { 'column_alias' } = 1;
$limits { 'NEG' } = 1;
$limits { 'func_extra_in_num' } = 1;
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
# for the smds small benchmark test ....
# the different cases per query ...
$smds { 'q1' } = 'a' ;
$smds { 'q2' } = '' ;
$smds { 'q3' } = 'b' ; #doesn't work -> strange error about column -fixed
$smds { 'q4' } = 'a' ;
$smds { 'q5' } = 'b' ;
$smds { 'q6' } = 'c' ;
$smds { 'q7' } = 'b' ;
$smds { 'q8' } = 'f' ;
$smds { 'q9' } = 'b' ;
$smds { 'q10' } = 'b' ;
$smds { 'q11' } = '' ;
$smds { 'q12' } = 'd' ;
$smds { 'q13' } = 'b' ;
$smds { 'q14' } = 'd' ;
$smds { 'q15' } = 'd' ;
$smds { 'q16' } = '' ;
$smds { 'q17' } = '' ;
$smds { 'time' } = 1; # the use of the time table -> 1 is on.
# when 0 then the date field must be a
# date field not a int field!!!
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $version ,$dir ) ;
foreach $dir ( $ENV { 'SOLIDDIR' } ,"/usr/local/solid" , "/my/local/solid" )
{
if ( $dir && -e " $dir /bin/solcon " )
{
$version = ` $dir /bin/solcon -e"ver" $main ::opt_user $main ::opt_password | grep Server | head -1` ;
if ( $? = = 0)
{
chomp( $version ) ;
return $version ;
}
}
}
return "Solid version ???" ;
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/mediumint/integer/i;
$field = ~ s/ double/ float/i;
# Solid doesn't have blob, it has long varchar
$field = ~ s/ blob/ long varchar/;
$field = ~ s/ decimal/ float/i;
$field = ~ s/ big_decimal/ float/i;
$field = ~ s/ date/ int/i;
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 0;
foreach $index ( @$index )
{
if ( $index = ~ /^primary key/i || $index = ~ /^unique/i)
{ # Add to create statement
substr( $queries [ 0] ,-1,0) = "," . $index ;
}
else
{
$index = ~ /^( .*) \s +( \( .*\) ) $/;
push( @queries," create ${ 1 } $nr on $table_name $2 " ) ;
$nr ++;
}
}
return @queries;
}
# there is no sql statement in solid which can do the load from
# an ascii file in the db ... but there is the speedloader program
# an external program which can load the ascii file in the db ...
# the server must be down before using speedloader !!!!
# (in the standalone version)
# it works also with a control file ... that one must be made ....
sub insert_file {
my ( $self , $dbname , $file ) = @_;
my ( $speedcmd ) ;
$speedcmd = '/usr/local/solid/bin/solload' ;
print "At this moment not supported - solid server must go down \n" ;
return 0;
}
# solid can't handle an alias in a having statement so
# select test as foo from tmp group by foo having foor > 2
# becomes
# select test as foo from tmp group by foo having test > 2
#
sub query {
my( $self ,$sql ) = @_;
my( @select,$tmp ,$newhaving ,$key ,%change) ;
if ( $sql = ~ /having\s +/i)
{
if ( $sql = ~ /select ( .*) from/i)
{
( @select) = split( /,\s */, $1 ) ;
foreach $tmp ( @select)
{
if ( $tmp = ~ /( .*) \s +as\s +( \w +) /)
{
$change { $2 } = $1 ;
}
}
}
if ( $sql = ~ /having\s +( \w +) /i)
{
$newhaving = $1 ;
foreach $key ( sort { $a cmp $b } keys %change)
{
if ( $newhaving eq $key )
{
$newhaving = ~ s/$key /$change { $key } /g;
}
}
}
$sql = ~ s/( having) \s +( \w +) /$1 $newhaving /i;
}
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Definitions for Empress
#
# at this moment DBI:Empress can only handle 200 prepare statements ...
# so Empress can't be tested with the benchmark test :(
#############################################################################
package db_Empress;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "empress" ;
$self ->{ 'data_source' } = " DBI:EmpressNet:SERVER= $main :: $opt_host ;Database=/usr/local/empress/rdbms/bin/ $database " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "text" ;
$self ->{ 'text' } = "text" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$limits { 'max_conditions' } = 1258;
$limits { 'max_columns' } = 226; # server is disconnecting????
# above this value .... but can handle 2419 columns
# maybe something for crash-me ... but how to check ???
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 4095; # max returned ....
$limits { 'query_size' } = 65535; # Not a limit, big enough
$limits { 'max_index' } = 64; # Big enough
$limits { 'max_index_parts' } = 64; # Big enough
$limits { 'max_column_name' } = 31;
$limits { 'join_optimizer' } = 1;
$limits { 'load_data_infile' } = 0;
$limits { 'lock_tables' } = 1;
$limits { 'functions' } = 1;
$limits { 'group_functions' } = 1;
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 0;
$limits { 'multi_drop' } = 0;
$limits { 'subqueries' } = 1;
$limits { 'table_wildcard' } = 0;
$limits { 'having_with_alias' } = 0; # AS isn't supported in a select
$limits { 'having_with_group' } = 1;
$limits { 'like_with_column' } = 1;
$limits { 'order_by_position' } = 1;
$limits { 'group_by_position' } = 0;
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 0;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 0;
$limits { 'func_extra_%' } = 1;
$limits { 'func_odbc_floor' } = 1;
$limits { 'func_extra_if' } = 0;
$limits { 'column_alias' } = 0;
$limits { 'NEG' } = 1;
$limits { 'func_extra_in_num' } = 0;
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
# for the smds small benchmark test ....
# the different cases per query ... EMPRESS
$smds { 'q1' } = 'a' ;
$smds { 'q2' } = '' ;
$smds { 'q3' } = 'a' ;
$smds { 'q4' } = 'a' ;
$smds { 'q5' } = 'a' ;
$smds { 'q6' } = 'a' ;
$smds { 'q7' } = 'b' ;
$smds { 'q8' } = 'd' ;
$smds { 'q9' } = 'b' ;
$smds { 'q10' } = 'a' ;
$smds { 'q11' } = '' ;
$smds { 'q12' } = 'd' ;
$smds { 'q13' } = 'b' ;
$smds { 'q14' } = 'b' ;
$smds { 'q15' } = 'a' ;
$smds { 'q16' } = '' ;
$smds { 'q17' } = '' ;
$smds { 'time' } = 1; # the use of the time table -> 1 is on.
# when 0 then the date field must be a
# date field not a int field!!!
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ,$dbh ) = @_;
my ( $version ) ;
$version = "" ;
if ( -x "/usr/local/empress/rdbms/bin/empvers" )
{
$version = ` /usr/local/empress/rdbms/bin/empvers | grep Version` ;
}
if ( $version )
{
chomp( $version ) ;
}
else
{
$version = "Empress version ???" ;
}
return $version ;
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
sub insert_file {
my( $self ,$dbname , $file ) = @_;
my( $command ,$sth ) ;
$command = " insert into $dbname from ' $file ' " ;
print " $command \n " if ( $opt_debug ) ;
$sth = $dbh ->do( $command ) or die $DBI ::errstr;
return $sth ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/mediumint/int/i;
$field = ~ s/tinyint/int/i;
$field = ~ s/smallint/int/i;
$field = ~ s/longint/int/i;
$field = ~ s/integer/int/i;
$field = ~ s/ double/ longfloat/i;
# Solid doesn't have blob, it has long varchar
# $field =~ s/ blob/ text(65535,65535,65535,65535)/;
$field = ~ s/ blob/ text/;
$field = ~ s/ varchar\( ( \d +) \) / char( $1 ,3) /;
$field = ~ s/ char\( ( \d +) \) / char( $1 ,3) /;
$field = ~ s/ decimal/ float/i;
$field = ~ s/ big_decimal/ longfloat/i;
$field = ~ s/ date/ int/i;
$field = ~ s/ float( .*) / float/i;
if ( $field = ~ / int\( ( \d +) \) /) {
if ( $1 > 4) {
$field = ~ s/ int\( \d +\) / longinteger/i;
} else {
$field = ~ s/ int\( \d +\) / longinteger/i;
}
} else {
$field = ~ s/ int/ longinteger/i;
}
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 1;
foreach $index ( @$index )
{
# Primary key is unique index in Empress
$index = ~ s/primary key/unique index/i;
if ( $index = ~ /^unique.*\( ( [ ^\( ] *) \) $/i)
{
$nr ++;
push( @queries," create unique index ${ table_name } _ $nr on $table_name ( $1 ) " ) ;
}
else
{
if ( !( $index = ~ /^( .*index) \s +( \w *) \s +( \( .*\) ) $/i) )
{
die " Can't parse index information in ' $index '\n " ;
}
push( @queries," create $1 ${ table_name } _ $2 on $table_name $3 " ) ;
}
}
return @queries;
}
# empress can't handle an alias and but can handle the number of the
# columname - so
# select test as foo from tmp order by foo
# becomes
# select test from tmp order by 1
#
sub query {
my( $self ,$sql ) = @_;
my( @select,$i ,$tmp ,$newselect ,$neworder ,@order,$key ,%change) ;
my( $tmp1 ,$otmp ,$tmp2 ) ;
if ( $sql = ~ /\s +as\s +/i)
{
if ( $sql = ~ /select\s +( .*) \s +from/i) {
$newselect = $1 ;
( @select) = split( /,\s */, $1 ) ;
$i = 1;
foreach $tmp ( @select) {
if ( $tmp = ~ /\s +as\s +( \w +) /) {
$change { $1 } = $i ;
}
$i ++;
}
}
$newselect = ~ s/\s +as\s +( \w +) //gi;
$tmp2 = 0;
if ( $sql = ~ /order\s +by\s +( .*) $/i) {
( @order) = split( /,\s */, $1 ) ;
foreach $otmp ( @order) {
foreach $key ( sort { $a cmp $b } keys %change) {
if ( $otmp eq $key ) {
$neworder .= " $tmp1 " ." $change { $key } " ;
$tmp1 = ", " ;
$tmp2 = 1;
} elsif ( $otmp = ~ /( \w +) \s +( .+) $/) {
if ( $key eq $1 ) {
$neworder .= " $tmp1 " ." $change { $key } $2 " ;
$tmp2 = 1;
}
}
}
if ( $tmp2 = = 0) {
$neworder .= " $tmp1 " ." $otmp " ;
}
$tmp2 = 0;
$tmp1 = ", " ;
}
}
$sql = ~ s/( select ) \s +( .*) \s +( from) /$1 $newselect $3 /i;
$sql = ~ s/( order\s +by) \s +( .*) $/$1 $neworder /i;
}
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
# This is a because of the 200 statement problem with DBI-Empress
sub abort_if_fatal_error
{
if ( $DBI ::errstr = ~ /Overflow of table of prepared statements/i)
{
print "Overflow of prepared statements ... killing the process\n" ;
exit 1;
}
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Definitions for Oracle
#############################################################################
package db_Oracle;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "Oracle" ;
$self ->{ 'data_source' } = " DBI:Oracle: $database " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "long" ;
$self ->{ 'text' } = "long" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$self ->{ "vacuum" } = 1;
$limits { 'max_conditions' } = 9999; # (Actually not a limit)
$limits { 'max_columns' } = 254; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 2000; # Limit for blob test-connect
$limits { 'query_size' } = 65525; # Max size with default buffers.
$limits { 'max_index' } = 16; # Max number of keys
$limits { 'max_index_parts' } = 16; # Max segments/key
$limits { 'max_column_name' } = 32; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 0;
$limits { 'multi_drop' } = 0;
$limits { 'subqueries' } = 1;
$limits { 'left_outer_join' } = 0; # This may be fixed in the query module
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can't use group functions in HAVING
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 0;
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 0;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 0; # Oracle has problem with mod()
$limits { 'func_extra_%' } = 0; # Has % as alias for mod()
$limits { 'func_odbc_floor' } = 1; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 1; # Has function in
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
$smds { 'time' } = 1;
$smds { 'q1' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q2' } = 'b' ;
$smds { 'q3' } = 'b' ; # with time ('')
$smds { 'q4' } = 'c' ; # with time not supp by mysql (d)
$smds { 'q5' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q6' } = 'c' ; # with time not supp by mysql ('')
$smds { 'q7' } = 'c' ;
$smds { 'q8' } = 'f' ;
$smds { 'q9' } = 'c' ;
$smds { 'q10' } = 'b' ;
$smds { 'q11' } = 'b' ;
$smds { 'q12' } = 'd' ;
$smds { 'q13' } = 'c' ;
$smds { 'q14' } = 'd' ;
$smds { 'q15' } = 'd' ;
$smds { 'q16' } = 'a' ;
$smds { 'q17' } = 'c' ;
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
my ( $dbh ,$sth ,$version ,@row) ;
$dbh = $self ->connect( ) ;
$sth = $dbh ->prepare( "select VERSION from product_component_version WHERE PRODUCT like 'Oracle%'" ) or die $DBI ::errstr;
$version = "Oracle 7.x" ;
if ( $sth ->execute && ( @row = $sth ->fetchrow_array) )
{
$version = " Oracle $row [0] " ;
}
$sth ->finish;
$dbh ->disconnect;
return $version ;
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
# If one uses $main::opt_fast then one is allowed to use
# non standard types to get better speed.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$ind ,@keys) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/ character\( ( \d +) \) / char\( $1 \) /i;
$field = ~ s/ character varying\( ( \d +) \) / varchar\( $1 \) /i;
$field = ~ s/ char varying\( ( \d +) \) / varchar\( $1 \) /i;
$field = ~ s/ integer/ number\( 38\) /i;
$field = ~ s/ int/ number\( 38\) /i;
$field = ~ s/ tinyint/ number\( 38\) /i;
$field = ~ s/ smallint/ number\( 38\) /i;
$field = ~ s/ mediumint/ number\( 38\) /i;
$field = ~ s/ tinynumber\( ( \d +) \) \( ( \d +) \) / number\( $1 ,$2 \) /i;
$field = ~ s/ smallnumber\( ( \d +) \) \( ( \d +) \) / number\( $1 ,$2 \) /i;
$field = ~ s/ mediumnumber\( ( \d +) \) \( ( \d +) \) / number\( $1 ,$2 \) /i;
$field = ~ s/ number\( ( \d +) \) \( ( \d +) \) / number\( $1 ,$2 \) /i;
$field = ~ s/ numeric\( ( \d +) \) \( ( \d +) \) / number\( $1 ,$2 \) /i;
$field = ~ s/ decimal\( ( \d +) \) \( ( \d +) \) / number\( $1 ,$2 \) /i;
$field = ~ s/ dec\( ( \d +) \) \( ( \d +) \) / number\( $1 ,$2 \) /i;
$field = ~ s/ float/ number/;
$field = ~ s/ real/ number/;
$field = ~ s/ double precision/ number/;
$field = ~ s/ double/ number/;
$field = ~ s/ blob/ long/;
$query .= $field . ',' ;
}
foreach $ind ( @$index )
{
my @index;
if ( $ind = ~ /\b KEY\b /i ) {
push( @keys," ALTER TABLE $table_name ADD $ind " ) ;
} else {
my @fields = split( ' ' ,$index ) ;
my $query = " CREATE INDEX $fields [1] ON $table_name $fields [2] " ;
push( @index,$query ) ;
}
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ,@keys,@index) ;
#print "query:$query\n";
return @queries;
}
sub insert_file {
my( $self ,$dbname , $file ) = @_;
print "insert an ascii file isn't supported by Oracle (?)\n" ;
return 0;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 1;
}
sub reconnect_on_errors
{
return 0;
}
#
# optimize the tables ....
#
sub vacuum
{
my ( $self ,$full_vacuum ,$dbh_ref ) = @_;
my ( $loop_time ,$end_time ,$sth ,$dbh ) ;
if ( defined( $full_vacuum ) )
{
$$ dbh_ref->disconnect; $$ dbh_ref = $self ->connect( ) ;
}
$dbh = $$ dbh_ref;
$loop_time = new Benchmark;
# first analyze all tables
$sth = $dbh ->prepare( "select table_name from user_tables" ) || die " Got error: $DBI ::errstr " ;
$sth ->execute || die " Got error: $DBI ::errstr when select user_tables " ;
while ( my @r = $sth ->fetchrow_array)
{
$dbh ->do( " analyze table $r [0] compute statistics " ) || die " Got error: $DBI ::errstr when executing 'analyze table'\n " ;
}
# now analyze all indexes ...
$sth = $dbh ->prepare( "select index_name from user_indexes" ) || die " Got error: $DBI ::errstr " ;
$sth ->execute || die " Got error: $DBI ::errstr when select user_indexes " ;
while ( my @r1 = $sth ->fetchrow_array)
{
$dbh ->do( " analyze index $r1 [0] compute statistics " ) || die " Got error: $DBI ::errstr when executing 'analyze index $r1 [0]'\n " ;
}
$end_time = new Benchmark;
print "Time for book-keeping (1): " .
Benchmark::timestr( Benchmark::timediff( $end_time , $loop_time ) ,"all" ) . "\n\n" ;
$dbh ->disconnect; $$ dbh_ref = $self ->connect( ) ;
}
#############################################################################
# Definitions for Informix
#############################################################################
package db_Informix;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "Informix" ;
$self ->{ 'data_source' } = " DBI:Informix: $database " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "byte in table" ;
$self ->{ 'text' } = "byte in table" ;
$self ->{ 'double_quotes' } = 0; # Can handle: 'Walker''s'
$self ->{ 'host' } = $main ::opt_host;
$limits { 'NEG' } = 1; # Supports -id
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 1;
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'func_extra_%' } = 0; # Has % as alias for mod()
$limits { 'func_extra_if' } = 0; # Have function if.
$limits { 'func_extra_in_num' } = 0; # Has function in
$limits { 'func_odbc_floor' } = 0; # Has func_odbc_floor function
$limits { 'func_odbc_mod' } = 1; # Have function mod.
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_by_position' } = 1; # Can use 'GROUP BY 1'
$limits { 'group_by_alias' } = 0; # Can use 'select a as ab from x GROUP BY ab'
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can't use group functions in HAVING
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables (always 1 only for msql)
$limits { 'left_outer_join' } = 0; # Supports left outer joins (ANSI)
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 1; # Has lock tables
$limits { 'max_conditions' } = 1214; # (Actually not a limit)
$limits { 'max_column_name' } = 18; # max table and column name
$limits { 'max_columns' } = 994; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_index' } = 64; # Max number of keys
$limits { 'max_index_parts' } = 15; # Max segments/key
$limits { 'max_text_size' } = 65535; # Max size with default buffers. ??
$limits { 'multi_drop' } = 0; # Drop table can take many tables
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'query_size' } = 32766; # Max size with default buffers.
$limits { 'select_without_from' } = 0; # Can do 'select 1';
$limits { 'subqueries' } = 1; # Doesn't support sub-queries.
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
my ( $dbh ,$sth ,$version ,@row) ;
$ENV { 'INFORMIXSERVER' } = $self ->{ 'host' } ;
$dbh = $self ->connect( ) ;
$sth = $dbh ->prepare( "SELECT owner FROM systables WHERE tabname = ' VERSION'" )
or die $DBI ::errstr;
$version = 'Informix unknown' ;
if ( $sth ->execute && ( @row = $sth ->fetchrow_array) )
{
$version = " Informix $row [0] " ;
}
$sth ->finish;
$dbh ->disconnect;
return $version ;
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Create table
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$name ,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
# $field =~ s/\btransport_description\b/transport_desc/;
# to overcome limit 18 chars
$field = ~ s/tinyint/smallint/i;
$field = ~ s/tinyint\( \d +\) /smallint/i;
$field = ~ s/mediumint/integer/i;
$field = ~ s/mediumint\( \d +\) /integer/i;
$field = ~ s/smallint\( \d +\) /smallint/i;
$field = ~ s/integer\( \d +\) /integer/i;
$field = ~ s/int\( \d +\) /integer/i;
# $field =~ s/\b(?:small)?int(?:eger)?\((\d+)\)/decimal($1)/i;
# $field =~ s/float(\(\d*,\d*\)){0,1}/real/i;
$field = ~ s/( float| double) ( \( .*?\) ) ?/float/i;
if ( $field = ~ / blob/i)
{
$name = $self ->{ 'blob' } ;
$field = ~ s/ blob/ $name /;
}
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 0;
foreach $index ( @$index )
{
# Primary key is unique index in Informix
$index = ~ s/primary key/unique index primary/i;
if ( $index = ~ /^unique\s *\( ( [ ^\( ] *) \) $/i)
{
$nr ++;
push( @queries," create unique index ${ table_name } _ $nr on $table_name ( $1 ) " ) ;
}
else
{
if ( !( $index = ~ /^( .*index) \s +( \w *) \s +( \( .*\) ) $/i) )
{
die " Can't parse index information in ' $index '\n " ;
}
### push(@queries,"create $1 ${table_name}_$2 on $table_name $3");
$nr ++;
push( @queries," create $1 ${ table_name } _ $nr on $table_name $3 " ) ;
}
}
return @queries;
}
#
# Some test needed this
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Configuration for Access
#############################################################################
package db_access;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "access" ;
$self ->{ 'data_source' } = " DBI:ODBC: $database " ;
if ( defined( $opt_host ) && $opt_host ne "" )
{
$self ->{ 'data_source' } .= " : $host " ;
}
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "blob" ;
$self ->{ 'text' } = "blob" ; # text ?
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$limits { 'max_conditions' } = 97; # We get 'Query is too complex'
$limits { 'max_columns' } = 255; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 255; # Max size with default buffers.
$limits { 'query_size' } = 65535; # Not a limit, big enough
$limits { 'max_index' } = 32; # Max number of keys
$limits { 'max_index_parts' } = 10; # Max segments/key
$limits { 'max_column_name' } = 64; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 0; # Have count(distinct)
$limits { 'select_without_from' } = 1; # Can do 'select 1';
$limits { 'multi_drop' } = 0; # Drop table can take many tables
$limits { 'subqueries' } = 1; # Supports sub-queries.
$limits { 'left_outer_join' } = 1; # Supports left outer joins
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can use group functions in HAVING
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 0; # Can use 'GROUP BY 1'
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 2; #Have ALTER TABLE t add a int, b int;
$limits { 'alter_table_dropcol' } = 1;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 0; # Have function mod.
$limits { 'func_extra_%' } = 0; # Has % as alias for mod()
$limits { 'func_odbc_floor' } = 0; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 1; # Has function in
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
return "Access 2000" ; #DBI/ODBC can't return the server version
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/mediumint/integer/i;
$field = ~ s/tinyint/smallint/i;
$field = ~ s/float\( \d +,\d +\) /float/i;
$field = ~ s/integer\( \d +\) /integer/i;
$field = ~ s/smallint\( \d +\) /smallint/i;
$field = ~ s/int\( \d +\) /integer/i;
$field = ~ s/blob/text/i;
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 0;
foreach $index ( @$index )
{
$ext = "WITH DISALLOW NULL" ;
if ( ( $index = ~ s/primary key/unique index primary_key/i) )
{
$ext = "WITH PRIMARY;"
}
if ( $index = ~ /^unique.*\( ( [ ^\( ] *) \) $/i)
{
$nr ++;
$index = " unique index ${ table_name } _ $nr ( $1 ) " ;
}
$index = ~ /^( .*) \s +( \( .*\) ) $/;
push( @queries," create ${ 1 } on $table_name $2 " ) ;
}
return @queries;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index ON $table " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 1 if ( ( $DBI ::errstr = ~ /The database engine couldn\' t lock table/i) ||
( $DBI ::errstr = ~ /niet vergrendelen. De tabel is momenteel in gebruik /i) ||
( $DBI ::errstr = ~ /Den anv.* redan av en annan/i) ||
( $DBI ::errstr = ~ /non-exclusive access/) ) ;
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 1;
}
#############################################################################
# Configuration for Microsoft SQL server
#############################################################################
package db_ms_sql;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "ms-sql" ;
$self ->{ 'data_source' } = " DBI:ODBC: $database " ;
if ( defined( $opt_host ) && $opt_host ne "" )
{
$self ->{ 'data_source' } .= " : $host " ;
}
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "text" ;
$self ->{ 'text' } = "text" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$limits { 'max_conditions' } = 1030; # We get 'Query is too complex'
$limits { 'max_columns' } = 250; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 9830; # Max size with default buffers.
$limits { 'query_size' } = 9830; # Max size with default buffers.
$limits { 'max_index' } = 64; # Max number of keys
$limits { 'max_index_parts' } = 15; # Max segments/key
$limits { 'max_column_name' } = 30; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 1; # Can do 'select 1';
$limits { 'multi_drop' } = 1; # Drop table can take many tables
$limits { 'subqueries' } = 1; # Supports sub-queries.
$limits { 'left_outer_join' } = 1; # Supports left outer joins
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can't use group functions in HAVING
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 0; # Can use 'GROUP BY 1'
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 0;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 0; # Have function mod.
$limits { 'func_extra_%' } = 1; # Has % as alias for mod()
$limits { 'func_odbc_floor' } = 1; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 0; # Has function in
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
my( $sth ,@row) ;
$dbh = $self ->connect( ) ;
$sth = $dbh ->prepare( "SELECT \@\@VERSION" ) or die $DBI ::errstr;
$sth ->execute or die $DBI ::errstr;
@row = $sth ->fetchrow_array;
if ( $row [ 0] ) {
@server = split( /\n /,$row [ 0] ) ;
chomp( @server) ;
return " $server [0] " ;
} else {
return "Microsoft SQL server ?" ;
}
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/mediumint/integer/i;
$field = ~ s/float\( \d +,\d +\) /float/i;
$field = ~ s/double\( \d +,\d +\) /float/i;
$field = ~ s/double/float/i;
$field = ~ s/integer\( \d +\) /integer/i;
$field = ~ s/int\( \d +\) /integer/i;
$field = ~ s/smallint\( \d +\) /smallint/i;
$field = ~ s/smallinteger/smallint/i;
$field = ~ s/tinyint\( \d +\) /tinyint/i;
$field = ~ s/tinyinteger/tinyint/i;
$field = ~ s/blob/text/i;
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 0;
foreach $index ( @$index )
{
$ext = "WITH DISALLOW NULL" ;
if ( ( $index = ~ s/primary key/unique index primary_key/i) )
{
$ext = "WITH PRIMARY;"
}
if ( $index = ~ /^unique.*\( ( [ ^\( ] *) \) $/i)
{
$nr ++;
$index = " unique index ${ table_name } _ $nr ( $1 ) " ;
}
$index = ~ /^( .*) \s +( \( .*\) ) $/;
push( @queries," create ${ 1 } on $table_name $2 " ) ;
}
return @queries;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $table . $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Configuration for Sybase
#############################################################################
package db_sybase;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "sybase" ;
$self ->{ 'data_source' } = " DBI:ODBC: $database " ;
if ( defined( $opt_host ) && $opt_host ne "" )
{
$self ->{ 'data_source' } .= " : $host " ;
}
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "text" ;
$self ->{ 'text' } = "text" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$self ->{ "vacuum" } = 1;
$limits { 'max_conditions' } = 1030; # We get 'Query is too complex'
$limits { 'max_columns' } = 250; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 9830; # Max size with default buffers.
$limits { 'query_size' } = 9830; # Max size with default buffers.
$limits { 'max_index' } = 64; # Max number of keys
$limits { 'max_index_parts' } = 15; # Max segments/key
$limits { 'max_column_name' } = 30; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 1; # Can do 'select 1';
$limits { 'multi_drop' } = 1; # Drop table can take many tables
$limits { 'subqueries' } = 1; # Supports sub-queries.
$limits { 'left_outer_join' } = 1; # Supports left outer joins
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can't use group functions in HAVING
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 0; # Can use 'GROUP BY 1'
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 0;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 0; # Have function mod.
$limits { 'func_extra_%' } = 1; # Has % as alias for mod()
$limits { 'func_odbc_floor' } = 1; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 0; # Has function in
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
return "Sybase enterprise 11.5 NT" ; #DBI/ODBC can't return the server version
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/mediumint/integer/i;
$field = ~ s/float\( \d +,\d +\) /float/i;
$field = ~ s/int\( \d +\) /int/i;
$field = ~ s/double/float/i;
$field = ~ s/integer\( \d +\) /integer/i;
$field = ~ s/smallint\( \d +\) /smallint/i;
$field = ~ s/tinyint\( \d +\) /tinyint/i;
$field = ~ s/blob/text/i;
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 0;
foreach $index ( @$index )
{
# $ext="WITH DISALLOW NULL";
if ( ( $index = ~ s/primary key/unique index primary_key/i) )
{
# $ext="WITH PRIMARY;"
}
if ( $index = ~ /^unique.*\( ( [ ^\( ] *) \) $/i)
{
$nr ++;
$index = " unique index ${ table_name } _ $nr ( $1 ) " ;
}
$index = ~ /^( .*) \s +( \( .*\) ) $/;
push( @queries," create ${ 1 } on $table_name $2 " ) ;
}
return @queries;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $table . $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#
# optimize the tables ....
#
sub vacuum
{
my ( $self ,$full_vacuum ,$dbh_ref ) = @_;
my ( $loop_time ,$end_time ,$dbh ) ;
if ( defined( $full_vacuum ) )
{
$$ dbh_ref->disconnect; $$ dbh_ref = $self ->connect( ) ;
}
$dbh = $$ dbh_ref;
$loop_time = new Benchmark;
$dbh ->do( "analyze table ?? compute statistics" ) || die " Got error: $DBI ::errstr when executing 'vacuum'\n " ;
$end_time = new Benchmark;
print "Time for book-keeping (1): " .
Benchmark::timestr( Benchmark::timediff( $end_time , $loop_time ) ,"all" ) . "\n\n" ;
$dbh ->disconnect; $$ dbh_ref = $self ->connect( ) ;
}
#############################################################################
# Definitions for Adabas
#############################################################################
package db_Adabas;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "Adabas" ;
$self ->{ 'data_source' } = " DBI:Adabas: $database " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "long" ;
$self ->{ 'text' } = "long" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$limits { 'max_conditions' } = 50; # (Actually not a limit)
$limits { 'max_columns' } = 254; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 2000; # Limit for blob test-connect
$limits { 'query_size' } = 65525; # Max size with default buffers.
$limits { 'max_index' } = 16; # Max number of keys
$limits { 'max_index_parts' } = 16; # Max segments/key
$limits { 'max_column_name' } = 32; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 0;
$limits { 'multi_drop' } = 0;
$limits { 'subqueries' } = 1;
$limits { 'left_outer_join' } = 0; # This may be fixed in the query module
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can't use group functions in HAVING
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 1;
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 2; #Have ALTER TABLE t add a int, b int;
$limits { 'alter_table_dropcol' } = 1;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 0; # Oracle has problem with mod()
$limits { 'func_extra_%' } = 0; # Has % as alias for mod()
$limits { 'func_odbc_floor' } = 1; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 1; # Has function in
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
$smds { 'time' } = 1;
$smds { 'q1' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q2' } = 'b' ;
$smds { 'q3' } = 'b' ; # with time ('')
$smds { 'q4' } = 'c' ; # with time not supp by mysql (d)
$smds { 'q5' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q6' } = 'c' ; # with time not supp by mysql ('')
$smds { 'q7' } = 'c' ;
$smds { 'q8' } = 'f' ;
$smds { 'q9' } = 'c' ;
$smds { 'q10' } = 'b' ;
$smds { 'q11' } = 'b' ;
$smds { 'q12' } = 'd' ;
$smds { 'q13' } = 'c' ;
$smds { 'q14' } = 'd' ;
$smds { 'q15' } = 'd' ;
$smds { 'q16' } = 'a' ;
$smds { 'q17' } = 'c' ;
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
my ( $dbh ,$sth ,$version ,@row) ;
$dbh = $self ->connect( ) ;
$sth = $dbh ->prepare( "SELECT KERNEL FROM VERSIONS" ) or die $DBI ::errstr;
$version = "Adabas (unknown)" ;
if ( $sth ->execute && ( @row = $sth ->fetchrow_array)
&& $row [ 0] = ~ /( [ \d \. ] +) /)
{
$version = " Adabas $1 " ;
}
$sth ->finish;
$dbh ->disconnect;
return $version ;
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
# If one uses $main::opt_fast then one is allowed to use
# non standard types to get better speed.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$ind ,@keys) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/CHARACTER\s +VARYING/VARCHAR/i;
$field = ~ s/TINYINT/SMALLINT/i;
$field = ~ s/MEDIUMINT/INT/i;
$field = ~ s/SMALLINT\s *\( \d +\) /SMALLINT/i;
$field = ~ s/INT\s *\( \d +\) /INT/i;
$field = ~ s/BLOB/LONG/i;
$field = ~ s/INTEGER\s *\( \d +\) /INTEGER/i;
$field = ~ s/FLOAT\s *\( ( \d +) ,\d +\) /FLOAT\( $1 \) /i;
$field = ~ s/DOUBLE/FLOAT\( 38\) /i;
$field = ~ s/DOUBLE\s +PRECISION/FLOAT\( 38\) /i;
$query .= $field . ',' ;
}
foreach $ind ( @$index )
{
my @index;
if ( $ind = ~ /\b KEY\b /i ) {
push( @keys," ALTER TABLE $table_name ADD $ind " ) ;
} else {
my @fields = split( ' ' ,$index ) ;
my $query = " CREATE INDEX $fields [1] ON $table_name $fields [2] " ;
push( @index,$query ) ;
}
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ,@keys,@index) ;
#print "query:$query\n";
return @queries;
}
sub insert_file {
my( $self ,$dbname , $file ) = @_;
print "insert an ascii file isn't supported by Oracle (?)\n" ;
return 0;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Configuration for IBM DB2
#############################################################################
package db_db2;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "DB2" ;
$self ->{ 'data_source' } = " DBI:ODBC: $database " ;
if ( defined( $opt_host ) && $opt_host ne "" )
{
$self ->{ 'data_source' } .= " : $host " ;
}
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "varchar(255)" ;
$self ->{ 'text' } = "varchar(255)" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$limits { 'max_conditions' } = 418; # We get 'Query is too complex'
$limits { 'max_columns' } = 500; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 254; # Max size with default buffers.
$limits { 'query_size' } = 254; # Max size with default buffers.
$limits { 'max_index' } = 48; # Max number of keys
$limits { 'max_index_parts' } = 15; # Max segments/key
$limits { 'max_column_name' } = 18; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1;
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 0; # Can do 'select 1';
$limits { 'multi_drop' } = 0; # Drop table can take many tables
$limits { 'subqueries' } = 1; # Supports sub-queries.
$limits { 'left_outer_join' } = 1; # Supports left outer joins
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can't use group functions in HAVING
$limits { 'like_with_column' } = 0; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 0; # Can use 'GROUP BY 1'
$limits { 'alter_table' } = 1;
$limits { 'alter_add_multi_col' } = 0;
$limits { 'alter_table_dropcol' } = 0;
$limits { 'group_func_extra_std' } = 0; # Have group function std().
$limits { 'func_odbc_mod' } = 1; # Have function mod.
$limits { 'func_extra_%' } = 0; # Has % as alias for mod()
$limits { 'func_odbc_floor' } = 1; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 0; # Has function in
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
return "IBM DB2 5" ; #DBI/ODBC can't return the server version
}
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user, $main ::opt_password) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ) = @_;
my( $query ,@queries,$nr ) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/mediumint/integer/i;
$field = ~ s/float\( \d +,\d +\) /float/i;
$field = ~ s/integer\( \d +\) /integer/i;
$field = ~ s/int\( \d +\) /integer/i;
$field = ~ s/tinyint\( \d +\) /smallint/i;
$field = ~ s/tinyint/smallint/i;
$field = ~ s/smallint\( \d +\) /smallint/i;
$field = ~ s/smallinteger/smallint/i;
$field = ~ s/blob/varchar( 256) /i;
$query .= $field . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
push( @queries,$query ) ;
$nr = 0;
foreach $index ( @$index )
{
$ext = "WITH DISALLOW NULL" ;
if ( ( $index = ~ s/primary key/unique index primary_key/i) )
{
$ext = "WITH PRIMARY;"
}
if ( $index = ~ /^unique.*\( ( [ ^\( ] *) \) $/i)
{
$nr ++;
$index = " unique index ${ table_name } _ $nr ( $1 ) " ;
}
$index = ~ /^( .*) \s +( \( .*\) ) $/;
push( @queries," create ${ 1 } on $table_name $2 " ) ;
}
return @queries;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index
{
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $table . $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 0;
}
sub small_rollback_segment
{
return 1;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Configuration for MIMER
#############################################################################
package db_Mimer;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "mimer" ;
$self ->{ 'data_source' } = " DBI:mimer: $database : $host " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "binary varying(15000)" ;
$self ->{ 'text' } = "character varying(15000)" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$self ->{ 'char_null' } = "cast(NULL as char(1))" ;
$self ->{ 'numeric_null' } = "cast(NULL as int)" ;
$limits { 'max_conditions' } = 9999; # (Actually not a limit)
$limits { 'max_columns' } = 252; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 15000; # Max size with default buffers.
$limits { 'query_size' } = 1000000; # Max size with default buffers.
$limits { 'max_index' } = 32; # Max number of keys
$limits { 'max_index_parts' } = 16; # Max segments/key
$limits { 'max_column_name' } = 128; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 1; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 0; # Cannot do 'select 1';
$limits { 'multi_drop' } = 0; # Drop table cannot take many tables
$limits { 'subqueries' } = 1; # Supports sub-queries.
$limits { 'left_outer_join' } = 1; # Supports left outer joins
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 1; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can use group functions in HAVING
$limits { 'like_with_column' } = 1; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 0; # Cannot use 'GROUP BY 1'
$limits { 'alter_table' } = 1; # Have ALTER TABLE
$limits { 'alter_add_multi_col' } = 0; # Have ALTER TABLE t add a int,add b int;
$limits { 'alter_table_dropcol' } = 1; # Have ALTER TABLE DROP column
$limits { 'multi_value_insert' } = 0; # Does not have INSERT ... values (1,2),(3,4)
$limits { 'group_func_extra_std' } = 0; # Does not have group function std().
$limits { 'func_odbc_mod' } = 1; # Have function mod.
$limits { 'func_extra_%' } = 0; # Does not have % as alias for mod()
$limits { 'func_odbc_floor' } = 1; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Does not have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 1; # Supports -id
$limits { 'func_extra_in_num' } = 1; # Has function in
$limits { 'limit' } = 0; # Does not support the limit attribute
2000-08-17 01:05:02 +03:00
$limits { 'unique_index' } = 1; # Unique index works or not
2000-07-31 21:29:14 +02:00
$smds { 'time' } = 1;
$smds { 'q1' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q2' } = 'b' ;
$smds { 'q3' } = 'b' ; # with time ('')
$smds { 'q4' } = 'c' ; # with time not supp by mysql (d)
$smds { 'q5' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q6' } = 'c' ; # with time not supp by mysql ('')
$smds { 'q7' } = 'c' ;
$smds { 'q8' } = 'f' ;
$smds { 'q9' } = 'c' ;
$smds { 'q10' } = 'b' ;
$smds { 'q11' } = 'b' ;
$smds { 'q12' } = 'd' ;
$smds { 'q13' } = 'c' ;
$smds { 'q14' } = 'd' ;
$smds { 'q15' } = 'd' ;
$smds { 'q16' } = 'a' ;
$smds { 'q17' } = 'c' ;
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
my ( $dbh ,$sth ,$version ,@row) ;
$dbh = $self ->connect( ) ;
#
# Pick up SQLGetInfo option SQL_DBMS_VER (18)
#
$version = $dbh ->func( 18, GetInfo) ;
$dbh ->disconnect;
return $version ;
}
#
# Connection with optional disabling of logging
#
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
$dbh ->do( "SET OPTION LOG_OFF=1,UPDATE_LOG=0" ) ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
# If one uses $main::opt_fast then one is allowed to use
# non standard types to get better speed.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ,$options ) = @_;
my( $query ,@queries) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/ decimal/ double( 10,2) /i;
$field = ~ s/ big_decimal/ double( 10,2) /i;
$field = ~ s/ date/ int/i; # Because of tcp ?
$query .= $field . ',' ;
}
foreach $index ( @$index )
{
$query .= $index . ',' ;
}
substr( $query ,-1) = ")" ; # Remove last ',';
$query .= " $options " if ( defined( $options ) ) ;
push( @queries,$query ) ;
return @queries;
}
sub insert_file {
my( $self ,$dbname , $file ) = @_;
print "insert of an ascii file isn't supported by Mimer\n" ;
return 0;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index {
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 1 if ( $DBI ::errstr = ~ /Table locked by another cursor/) ;
return 0;
}
sub small_rollback_segment
{
return 0;
}
sub reconnect_on_errors
{
return 0;
}
#############################################################################
# Configuration for InterBase
#############################################################################
package db_interbase;
sub new
{
my ( $type ,$host ,$database ) = @_;
my $self = { } ;
my %limits;
bless $self ;
$self ->{ 'cmp_name' } = "interbase" ;
$self ->{ 'data_source' } = " DBI:InterBase:database= $database " ;
$self ->{ 'limits' } = \% limits;
$self ->{ 'smds' } = \% smds;
$self ->{ 'blob' } = "blob" ;
$self ->{ 'text' } = "" ;
$self ->{ 'double_quotes' } = 1; # Can handle: 'Walker''s'
$self ->{ 'char_null' } = "" ;
$self ->{ 'numeric_null' } = "" ;
$limits { 'max_conditions' } = 9999; # (Actually not a limit)
$limits { 'max_columns' } = 252; # Max number of columns in table
$limits { 'max_tables' } = 65000; # Should be big enough
$limits { 'max_text_size' } = 15000; # Max size with default buffers.
$limits { 'query_size' } = 1000000; # Max size with default buffers.
$limits { 'max_index' } = 31; # Max number of keys
$limits { 'max_index_parts' } = 8; # Max segments/key
$limits { 'max_column_name' } = 128; # max table and column name
$limits { 'join_optimizer' } = 1; # Can optimize FROM tables
$limits { 'load_data_infile' } = 0; # Has load data infile
$limits { 'lock_tables' } = 0; # Has lock tables
$limits { 'functions' } = 1; # Has simple functions (+/-)
$limits { 'group_functions' } = 1; # Have group functions
$limits { 'group_func_sql_min_str' } = 1; # Can execute MIN() and MAX() on strings
$limits { 'group_distinct_functions' } = 1; # Have count(distinct)
$limits { 'select_without_from' } = 0; # Cannot do 'select 1';
$limits { 'multi_drop' } = 0; # Drop table cannot take many tables
$limits { 'subqueries' } = 1; # Supports sub-queries.
$limits { 'left_outer_join' } = 1; # Supports left outer joins
$limits { 'table_wildcard' } = 1; # Has SELECT table_name.*
$limits { 'having_with_alias' } = 0; # Can use aliases in HAVING
$limits { 'having_with_group' } = 1; # Can use group functions in HAVING
$limits { 'like_with_column' } = 0; # Can use column1 LIKE column2
$limits { 'order_by_position' } = 1; # Can use 'ORDER BY 1'
$limits { 'group_by_position' } = 0; # Cannot use 'GROUP BY 1'
$limits { 'alter_table' } = 1; # Have ALTER TABLE
$limits { 'alter_add_multi_col' } = 1; # Have ALTER TABLE t add a int,add b int;
$limits { 'alter_table_dropcol' } = 1; # Have ALTER TABLE DROP column
$limits { 'multi_value_insert' } = 0; # Does not have INSERT ... values (1,2),(3,4)
$limits { 'group_func_extra_std' } = 0; # Does not have group function std().
$limits { 'func_odbc_mod' } = 0; # Have function mod.
$limits { 'func_extra_%' } = 0; # Does not have % as alias for mod()
$limits { 'func_odbc_floor' } = 0; # Has func_odbc_floor function
$limits { 'func_extra_if' } = 0; # Does not have function if.
$limits { 'column_alias' } = 1; # Alias for fields in select statement.
$limits { 'NEG' } = 0; # Supports -id
$limits { 'func_extra_in_num' } = 0; # Has function in
$limits { 'limit' } = 0; # Does not support the limit attribute
$smds { 'time' } = 1;
$smds { 'q1' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q2' } = 'b' ;
$smds { 'q3' } = 'b' ; # with time ('')
$smds { 'q4' } = 'c' ; # with time not supp by mysql (d)
$smds { 'q5' } = 'b' ; # with time not supp by mysql ('')
$smds { 'q6' } = 'c' ; # with time not supp by mysql ('')
$smds { 'q7' } = 'c' ;
$smds { 'q8' } = 'f' ;
$smds { 'q9' } = 'c' ;
$smds { 'q10' } = 'b' ;
$smds { 'q11' } = 'b' ;
$smds { 'q12' } = 'd' ;
$smds { 'q13' } = 'c' ;
$smds { 'q14' } = 'd' ;
$smds { 'q15' } = 'd' ;
$smds { 'q16' } = 'a' ;
$smds { 'q17' } = 'c' ;
return $self ;
}
#
# Get the version number of the database
#
sub version
{
my ( $self ) = @_;
my ( $dbh ,$sth ,$version ,@row) ;
$dbh = $self ->connect( ) ;
# $sth = $dbh->prepare("show version");
# $sth->execute;
# @row = $sth->fetchrow_array;
# $version = $row[0];
# $version =~ s/.*version \"(.*)\"$/$1/;
$dbh ->disconnect;
$version = "6.0Beta" ;
return $version ;
}
#
# Connection with optional disabling of logging
#
sub connect
{
my ( $self ) = @_;
my ( $dbh ) ;
$dbh = DBI->connect( $self ->{ 'data_source' } , $main ::opt_user,
$main ::opt_password,{ PrintError = > 0, AutoCommit = > 1} ) ||
die " Got error: ' $DBI ::errstr' when connecting to " . $self ->{ 'data_source' } ." with user: ' $main ::opt_user' password: ' $main ::opt_password'\n " ;
return $dbh ;
}
#
# Returns a list of statements to create a table
# The field types are in ANSI SQL format.
#
# If one uses $main::opt_fast then one is allowed to use
# non standard types to get better speed.
#
sub create
{
my( $self ,$table_name ,$fields ,$index ,$options ) = @_;
my( $query ,@queries) ;
$query = " create table $table_name ( " ;
foreach $field ( @$fields )
{
$field = ~ s/ big_decimal/ float/i;
$field = ~ s/ double/ float/i;
$field = ~ s/ tinyint/ smallint/i;
$field = ~ s/ mediumint/ int/i;
$field = ~ s/ integer/ int/i;
$field = ~ s/ float\( \d ,\d \) / float/i;
$field = ~ s/ date/ int/i; # Because of tcp ?
$field = ~ s/ smallint\( \d \) / smallint/i;
$field = ~ s/ int\( \d \) / int/i;
$query .= $field . ',' ;
}
foreach $ind ( @$index )
{
my @index;
if ( $ind = ~ /\b KEY\b /i ) {
push( @keys," ALTER TABLE $table_name ADD $ind " ) ;
} else {
my @fields = split( ' ' ,$index ) ;
my $query = " CREATE INDEX $fields [1] ON $table_name $fields [2] " ;
push( @index,$query ) ;
}
}
substr( $query ,-1) = ")" ; # Remove last ',';
$query .= " $options " if ( defined( $options ) ) ;
push( @queries,$query ) ;
return @queries;
}
sub insert_file {
my( $self ,$dbname , $file ) = @_;
print "insert of an ascii file isn't supported by InterBase\n" ;
return 0;
}
#
# Do any conversions to the ANSI SQL query so that the database can handle it
#
sub query {
my( $self ,$sql ) = @_;
return $sql ;
}
sub drop_index {
my ( $self ,$table ,$index ) = @_;
return " DROP INDEX $index " ;
}
#
# Abort if the server has crashed
# return: 0 if ok
# 1 question should be retried
#
sub abort_if_fatal_error
{
return 1 if ( $DBI ::errstr = ~ /Table locked by another cursor/) ;
return 0;
}
sub small_rollback_segment
{
return 1;
}
sub reconnect_on_errors
{
return 1;
}
1;