1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
|
#!/usr/bin/perl
# lib.pl is the file where database specific things should live,
# whereever possible. For example, you define certain constants
# here and the like.
use strict;
use warnings;
use File::Spec;
my $testname = "output$$";
my $base_dir = File::Spec->rel2abs (File::Spec->curdir ());
my $test_dir = File::Spec->rel2abs ($testname);
my $test_dsn = $ENV{DBI_DSN} || "DBI:CSV:f_dir=$testname";
my $test_user = $ENV{DBI_USER} || "";
my $test_pass = $ENV{DBI_PASS} || "";
sub COL_NULLABLE () { 1 }
sub COL_KEY () { 2 }
my %v;
{ my @req = qw( DBI SQL::Statement Text::CSV_XS DBD::CSV );
my $req = join ";\n" => map { qq{require $_;\n\$v{"$_"} = $_->VERSION ()} } @req;
eval $req;
if ($@) {
my @missing = grep { !exists $v{$_} } @req;
print STDERR "\n\nYOU ARE MISSING REQUIRED MODULES: [ @missing ]\n\n";
exit 0;
}
}
sub AnsiTypeToDb
{
my ($type, $size) = @_;
my $uctype = uc $type;
if ($uctype eq "CHAR" || $uctype eq "VARCHAR") {
$size ||= 1;
return "$uctype ($size)";
}
$uctype eq "BLOB" || $uctype eq "REAL" || $uctype eq "INTEGER" and
return $uctype;
$uctype eq "INT" and
return "INTEGER";
warn "Unknown type $type\n";
return $type;
} # AnsiTypeToDb
# This function generates a table definition based on an input list. The input
# list consists of references, each reference referring to a single column. The
# column reference consists of column name, type, size and a bitmask of certain
# flags, namely
#
# COL_NULLABLE - true, if this column may contain NULL's
# COL_KEY - true, if this column is part of the table's primary key
sub TableDefinition
{
my ($tablename, @cols) = @_;
my @keys = ();
foreach my $col (@cols) {
$col->[2] & COL_KEY and push @keys, $col->[0];
}
my @colDefs;
foreach my $col (@cols) {
my $colDef = $col->[0] . " " . AnsiTypeToDb ($col->[1], $col->[2]);
$col->[3] & COL_NULLABLE or $colDef .= " NOT NULL";
push @colDefs, $colDef;
}
my $keyDef = @keys ? ", PRIMARY KEY (" . join (", ", @keys) . ")" : "";
my $tq = $tablename =~ m/^\w+\./ ? qq{"$tablename"} : $tablename;
return sprintf "CREATE TABLE %s (%s%s)", $tq,
join (", ", @colDefs), $keyDef;
} # TableDefinition
# This function generates a list of tables associated to a given DSN.
sub ListTables
{
my $dbh = shift or return;
my @tables = $dbh->func ("list_tables");
my $msg = $dbh->errstr || $DBI::errstr;
$msg and die "Cannot create table list: $msg";
@tables;
} # ListTables
sub DbCleanup
{
chdir $base_dir;
-d $testname or return;
chdir $testname or BAIL_OUT ("Cleanup failed");
unlink glob "*";
chdir $base_dir;
rmdir $testname;
} # DbCleanup
mkdir $testname, 0755;
END { DbCleanup (); }
# This functions generates a list of possible DSN's aka
# databases and returns a possible table name for a new
# table being created.
#
# Problem is, we have two different situations here: Test scripts
# call us by pasing a dbh, which is fine for most situations.
{ my $listTablesHook;
my $testtable = "testaa";
my $listed = 0;
my @tables;
sub FindNewTable
{
my $dbh = shift;
unless ($listed) {
if (defined $listTablesHook) {
@tables = $listTablesHook->($dbh);
}
elsif (defined &ListTables) {
@tables = ListTables ($dbh);
}
else {
die "Fatal: ListTables not implemented.\n";
}
$listed = 1;
}
# A small loop to find a free test table we can use to mangle stuff in
# and out of. This starts at testaa and loops until testaz, then testba
# - testbz and so on until testzz.
my $foundtesttable = 1;
my $table;
while ($foundtesttable) {
$foundtesttable = 0;
foreach $table (@tables) {
if ($table eq $testtable) {
$testtable++;
$foundtesttable = 1;
}
}
}
$table = $testtable;
$testtable++;
return $table;
} # FindNewTable
}
sub isSaneCase
{
my @f = glob "??????.???";
foreach my $try (qw( FrUbLl BlURgH wOngOs )) {
my $fn = "$try.csv";
grep m{^$fn$}i => @f and next;
open my $fh, ">", $fn or return 1;
close $fh;
my $sane = (-f $fn && ! -f lc $fn && ! -f uc $fn);
unlink $fn;
return $sane;
}
# Assume insane
return 0;
} # isSaneCase
sub ServerError
{
die "# Cannot connect: $DBI::errstr\n";
} # ServerError
sub Connect
{
my $attr = @_ && ref $_[-1] eq "HASH" ? pop @_ : {};
my ($dsn, $usr, $pass) = @_;
$dsn ||= $test_dsn;
$usr ||= $test_user;
$pass ||= $test_pass;
my $dbh = DBI->connect ($dsn, $usr, $pass, $attr) or ServerError;
$dbh;
} # Connect
sub DbDir
{
@_ and $test_dir = File::Spec->catdir ($base_dir, shift);
$test_dir;
} # DbDir
sub DbFile
{
my $file = shift or return;
File::Spec->catdir ($test_dir, $file);
} # DbFile
1;
|