Rev 1295 | Blame | Compare with Previous | Last modification | View Log | RSS feed
################################################################################ Codestriker: Copyright (c) 2001, 2002 David Sitsky. All rights reserved.# sits@users.sourceforge.net## This program is free software; you can redistribute it and modify it under# the terms of the GPL.package Codestriker::DB::Database;use strict;use DBI;use Codestriker;use Codestriker::DB::Column;use Codestriker::DB::Index;use Codestriker::DB::PostgreSQL;use Codestriker::DB::MySQL;use Codestriker::DB::Oracle;use Codestriker::DB::ODBC;use Codestriker::DB::SQLite;# Print out creation statements before executing them if this is true.my $_DEBUG = 0;# Base object for all database objects.sub new {my $type = shift;my $self = {};return bless $self, $type;}# Factory object for retrieving a database object based on what is set# in the configuration.sub get_database {my $type = shift;if ($Codestriker::db =~ /^DBI:mysql/i) {return Codestriker::DB::MySQL->new();} elsif ($Codestriker::db =~ /^DBI:Pg/i) {return Codestriker::DB::PostgreSQL->new($Codestriker::db);} elsif ($Codestriker::db =~ /^DBI:Odbc/i) {return Codestriker::DB::ODBC->new();} elsif ($Codestriker::db =~ /^DBI:Oracle/i) {return Codestriker::DB::Oracle->new();} elsif ($Codestriker::db =~ /^DBI:SQLite/i) {return Codestriker::DB::SQLite->new();} else {die "Unsupported database type: $Codestriker::db\n";}}# Create a new database connection with the specified auto_commit and# raise_error properties. If an active connection is already associated# with the database, return that.sub _get_connection {my ($self, $auto_commit, $raise_error) = @_;# If a connection has already been created, return it.return $self->{dbh} if (exists $self->{dbh});$self->{dbh} = DBI->connect($Codestriker::db, $Codestriker::dbuser,$Codestriker::dbpasswd,{AutoCommit=>$auto_commit,RaiseError=>$raise_error,LongReadLen=>10240000});# Return the new connection.return $self->{dbh};}# Release the connection associated with the database, and either commit or# rollback it depending on the value of $commit.sub release_connection {my ($self) = @_;# Check there is an active connection.if (! defined $self->{dbh}) {die "Cannot release connection on database as no active connection\n";}# Disconnect the connection.$self->{dbh}->disconnect();$self->{dbh} = undef;}# Create the table in the database for the specified table, and with the# provided type mappings.sub create_table {my ($self, $table) = @_;# Create the initial table entry.my $stmt = "CREATE TABLE " . $table->get_name() . "(\n";# For each column, add the appropriate statement.my @pk = ();my $first_column = 1;foreach my $column (@{$table->get_columns()}) {push @pk, $column->get_name() if $column->is_primarykey();# Add the comma for the start of the next field if necessary.if ($first_column) {$first_column = 0;} else {$stmt .= ",\n";}# Add in the basic field definition.$stmt .= $column->get_name() . " " .$self->_map_type($column->get_type());# Check if the length constraint is required for a varchar expression.if ($column->get_type() == $Codestriker::DB::Column::TYPE->{VARCHAR}) {$stmt .= "(" . $column->get_length() . ")";}# Add the "NOT NULL" constraint if the column is mandatory.$stmt .= " NOT NULL" if $column->is_mandatory();# Add any autoincrement field decorations if required.if ($column->is_autoincrement()) {$stmt .= " " . $self->_get_autoincrement_type();}}# Now add in the primary definition if required.if (scalar(@pk) > 0) {$stmt .= ",\nPRIMARY KEY (" . (join ', ', @pk) . ")\n";}# Close off the statement.$stmt .= ")\n";print STDERR "Statement is: $stmt\n" if $_DEBUG;eval {# Now create the table.$self->{dbh}->do($stmt);# Now create the indexes for this table.foreach my $index (@{$table->get_indexes()}) {my $index_stmt = "CREATE INDEX " . $index->get_name . " ON " .$table->get_name() . "(";$index_stmt .= (join ', ', @{$index->get_column_names()}) . ")";print STDERR "Index statement is: $index_stmt\n" if $_DEBUG;# Now execute the statement to create the index.$self->{dbh}->do($index_stmt);}# Commit the table creation.$self->commit();};if ($@) {eval { $self->rollback() };die "Unable to create table/indexes.\n";}}# Method for retrieving the list of current tables attached to the database.# For most DBI implementations, this implementation works fine.sub get_tables() {my $self = shift;# Remove any tables that end in a period, or have backticks. Recent# versions of MySQL are now using backticks around the table name.my @tables = $self->{dbh}->tables;@tables = map { $_ =~ s/.*\.//; $_ } @tables;@tables = map { $_ =~ s/\`//g; $_ } @tables;return @tables;}# Add a field to a specific table. If the field already exists, then catch# the error and continue silently.sub add_field {my ($self, $table, $field, $definition) = @_;my $dbh = $self->{dbh};my $rc = 0;eval {$dbh->{PrintError} = 0;my $field_type = $self->_map_type($definition);$dbh->do("ALTER TABLE $table ADD COLUMN $field $field_type");print "Added new field $field to table $table.\n";$rc = 1;$self->commit();};if ($@) {eval { $self->rollback() };}$dbh->{PrintError} = 1;return $rc;}# Check if the specified column exists in the specified table.sub column_exists {my ($self, $tablename, $columnname) = @_;my $dbh = $self->{dbh};my $rc = 0;eval {$dbh->{PrintError} = 0;my $stmt = $dbh->prepare_cached("SELECT COUNT($columnname) " ."FROM $tablename");$rc = defined $stmt && $stmt->execute() ? 1 : 0;$stmt->finish();$self->commit();};if ($@) {eval { $self->rollback() };}$dbh->{PrintError} = 1;return $rc;}# Method for moving a database table to another name in a safe manner.sub move_table {my ($self, $old_tablename, $new_tablename) = @_;my $dbh = $self->{dbh};my $rc = 0;eval {$dbh->{PrintError} = 0;my $stmt =$dbh->prepare_cached("ALTER TABLE $old_tablename RENAME TO " ."$new_tablename");my $rc = defined $stmt && $stmt->execute() ? 1 : 0;$stmt->finish() if (defined $stmt);$self->commit();};if ($@) {eval { $self->rollback() };}$dbh->{PrintError} = 1;return $rc;}# Simple method for committing the current database transaction.sub commit {my ($self) = @_;$self->{dbh}->commit();}# Simple method for rolling back the current database transaction.sub rollback {my ($self) = @_;$self->{dbh}->rollback();}1;