package DBD::myDriver;
    use base qw(DBI::DBD::SqlEngine);
    sub driver
    {
        ...
        my $drh = $proto->SUPER::driver($attr);
        ...
        return $drh->{class};
    }
    sub CLONE { ... }
    package DBD::myDriver::dr;
    @ISA = qw(DBI::DBD::SqlEngine::dr);
    sub data_sources { ... }
    ...
    package DBD::myDriver::db;
    @ISA = qw(DBI::DBD::SqlEngine::db);
    sub init_valid_attributes { ... }
    sub init_default_attributes { ... }
    sub set_versions { ... }
    sub validate_STORE_attr { my ($dbh, $attrib, $value) = @_; ... }
    sub validate_FETCH_attr { my ($dbh, $attrib) = @_; ... }
    sub get_myd_versions { ... }
    sub get_avail_tables { ... }
    package DBD::myDriver::st;
    @ISA = qw(DBI::DBD::SqlEngine::st);
    sub FETCH { ... }
    sub STORE { ... }
    package DBD::myDriver::Statement;
    @ISA = qw(DBI::DBD::SqlEngine::Statement);
    sub open_table { ... }
    package DBD::myDriver::Table;
    @ISA = qw(DBI::DBD::SqlEngine::Table);
    my %reset_on_modify = (
                            myd_abc => "myd_foo",
                            myd_mno => "myd_bar",
                          );
    __PACKAGE__->register_reset_on_modify( \%reset_on_modify );
    my %compat_map = (
                       abc => 'foo_abc',
                       xyz => 'foo_xyz',
                     );
    __PACKAGE__->register_compat_map( \%compat_map );
    sub bootstrap_table_meta { ... }
    sub init_table_meta { ... }
    sub table_meta_attr_changed { ... }
    sub open_data { ... }
    sub new { ... }
    sub fetch_row { ... }
    sub push_row { ... }
    sub push_names { ... }
    sub seek { ... }
    sub truncate { ... }
    sub drop { ... }
    # optimize the SQL engine by add one or more of
    sub update_current_row { ... }
    # or
    sub update_specific_row { ... }
    # or
    sub update_one_row { ... }
    # or
    sub insert_new_row { ... }
    # or
    sub delete_current_row { ... }
    # or
    sub delete_one_row { ... }
 
  DBI->connect ('DBI:DBM:', undef, undef, {})
  # invokes
  package DBD::DBM::dr;
  @DBD::DBM::dr::ISA = qw(DBI::DBD::SqlEngine::dr);
  sub connect ($$;$$$)
  {
      ...
  }
Similar for "data_sources ()" and "disconnect_all()".
Pure Perl DBI drivers derived from DBI::DBD::SqlEngine usually don't need to override any of the methods provided through the DBD::XXX::dr package. However if you need additional initialization not fitting in "init_valid_attributes()" and "init_default_attributes()" of you're ::db class, the connect method might be the final place to be modified.
  $sth = $dbh->prepare ("select * from foo");
  # returns the f_encoding setting for table foo
  $dbh->csv_get_meta ("foo", "f_encoding");
DBI::DBD::SqlEngine provides the typical methods required here. Developers who write DBI drivers based on DBI::DBD::SqlEngine need to override the methods "set_versions" and "init_valid_attributes".
$sth->execute () or die $sth->errstr;
  $dbh->{sql_table_source} = "DBD::Foo::TableSource";
  $dbh->{sql_data_source} = "DBD::Foo::DataSource";
  package DBD::DBM;
  use base qw( DBI::DBD::SqlEngine );
  sub driver
  {
      my ( $class, $attr ) = @_;
      ...
      my $drh = $class->SUPER::driver( $attr );
      ...
      return $drh;
  }
It is not necessary to implement your own driver method as long as additional initialization (e.g. installing more private driver methods) is not required. You do not need to call "setup_driver" as DBI::DBD::SqlEngine takes care of it.
DBI::DBD::SqlEngine based DBI drivers usually do not need to implement anything here, it is enough to do the basic initialization:
package DBD:XXX::dr; @DBD::XXX::dr::ISA = qw (DBI::DBD::SqlEngine::dr); $DBD::XXX::dr::imp_data_size = 0; $DBD::XXX::dr::data_sources_attr = undef; $DBD::XXX::ATTRIBUTION = "DBD::XXX $DBD::XXX::VERSION by Hans Mustermann";
Methods provided by "DBI::DBD::SqlEngine::dr":
  DBI->connect( "dbi:Foo", , , { ... } );
First it instantiates a new driver using "DBI::_new_dbh". After that, initial bootstrap of the newly instantiated driver is done by
$dbh->func( 0, "init_default_attributes" );
The first argument (0) signals that this is the very first call to "init_default_attributes". Modern drivers understand that and do early stage setup here after calling
  package DBD::Foo::db;
  our @DBD::Foo::db::ISA = qw(DBI::DBD::SqlEngine::db);
  
  sub init_default_attributes
  {
    my ($dbh, $phase) = @_;
    $dbh->SUPER::init_default_attributes($phase);
    ...; # own setup code, maybe separated by phases
  }
When the $phase argument is passed down until "DBI::DBD::SqlEngine::db::init_default_attributes", "connect()" recognizes a modern driver and initializes the attributes from DSN and $attr arguments passed via "DBI->connect( $dsn, $user, $pass, \%attr )".
At the end of the attribute initialization after phase 0, "connect()" invoked "init_default_attributes" again for phase 1:
$dbh->func( 1, "init_default_attributes" );
@ary = DBI->data_sources($driver); @ary = DBI->data_sources($driver, \%attr);
Methods provided by "DBI::DBD::SqlEngine::db":
    return $validated_attribute_name;
In case of validation fails (e.g. accessing private attribute or similar), "validate_FETCH_attr" is permitted to throw an exception.
The driver prefix is extracted from the attribute name and verified against "$dbh->{ $drv_prefix . "valid_attrs" }" (when it exists). If the requested attribute value is not listed as a valid attribute, this method croaks. If the attribute is valid and readonly (listed in "$dbh->{ $drv_prefix . "readonly_attrs" }" when it exists), a real copy of the attribute value is returned. So it's not possible to modify "f_valid_attrs" from outside of DBI::DBD::SqlEngine::db or a derived class.
    return ($validated_attribute_name, $validated_attribute_value);
In case of validation fails (e.g. accessing private attribute or similar), "validate_STORE_attr" is permitted to throw an exception ("DBI::DBD::SqlEngine::db::validate_STORE_attr" throws an exception when someone tries to assign value other than "SQL_IC_UPPER .. SQL_IC_MIXED" to "$dbh->{sql_identifier_case}" or "$dbh->{sql_quoted_identifier_case}").
An example of a valid attributes list can be found in "DBI::DBD::SqlEngine::db::init_valid_attributes".
This method is called at the end of the "connect ()" phase.
When overriding this method, do not forget to invoke the superior one.
"DBI::DBD::SqlEngine::db::init_valid_attributes" initializes the attributes "sql_valid_attrs" and "sql_readonly_attrs".
When overriding this method, do not forget to invoke the superior one, preferably before doing anything else.
"DBI::DBD::SqlEngine::db::init_default_attributes" initializes the attributes "sql_identifier_case", "sql_quoted_identifier_case", "sql_handler", "sql_init_order", "sql_meta", "sql_engine_version", "sql_nano_version" and "sql_statement_version" when SQL::Statement is available.
It sets "sql_init_order" to the given $phase.
When the derived implementor class provides the attribute to validate attributes (e.g. "$dbh->{dbm_valid_attrs} = {...};") or the attribute containing the immutable attributes (e.g. "$dbh->{dbm_readonly_attrs} = {...};"), the attributes "drv_valid_attrs", "drv_readonly_attrs" and "drv_version" are added (when available) to the list of valid and immutable attributes (where "drv_" is interpreted as the driver prefix).
The DBI::DBD::SqlEngine implementation returns all version information known by DBI::DBD::SqlEngine (e.g. DBI version, Perl version, DBI::DBD::SqlEngine version and the SQL handler version).
"get_versions" takes the $dbh as the first argument and optionally a second argument containing a table name. The second argument is not evaluated in "DBI::DBD::SqlEngine::db::get_versions" itself - but might be in the future.
If the derived implementor class provides a method named "get_${drv_prefix}versions", this is invoked and the return value of it is associated to the derived driver name:
    if (my $dgv = $dbh->{ImplementorClass}->can ("get_" . $drv_prefix . "versions") {
        (my $derived_driver = $dbh->{ImplementorClass}) =~ s/::db$//;
        $versions{$derived_driver} = &$dgv ($dbh, $table);
    }
Override it to add more version information about your module, (e.g. some kind of parser version in case of DBD::CSV, ...), if one line is not enough room to provide all relevant information.
It is not recommended to override this method.
Attributes used by "DBI::DBD::SqlEngine::db":
This section describes attributes which are important to developers of DBI Database Drivers derived from "DBI::DBD::SqlEngine".
"DBI::DBD::SqlEngine" initializes following attributes:
  $dbh->{sql_init_order} = {
       0 => [qw( Profile RaiseError PrintError AutoCommit )],
      90 => [ "sql_meta", $dbh->{$drv_pfx_meta} ? $dbh->{$drv_pfx_meta} : () ]
  }
The default priority of not listed attribute keys is 50. It is well known that a lot of attributes needed to be set before some table settings are initialized. For example, for DBD::DBM, when using
  my $dbh = DBI->connect( "dbi:DBM:", undef, undef, {
      f_dir => "/path/to/dbm/databases",
      dbm_type => "BerkeleyDB",
      dbm_mldbm => "JSON", # use MLDBM::Serializer::JSON
      dbm_tables => {
          quick => {
              dbm_type => "GDBM_File",
              dbm_MLDBM => "FreezeThaw"
          }
      }
  });
This defines a known table "quick" which uses the GDBM_File backend and FreezeThaw as serializer instead of the overall default BerkeleyDB and JSON. But all files containing the table data have to be searched in "$dbh->{f_dir}", which requires "$dbh->{f_dir}" must be initialized before "$dbh->{sql_meta}->{quick}" is initialized by "bootstrap_table_meta" method of ``DBI::DBD::SqlEngine::Table'' to get "$dbh->{sql_meta}->{quick}->{f_dir}" being initialized properly.
See ``DBI::DBD::SqlEngine::TableSource'' for details.
See ``DBI::DBD::SqlEngine::DataSource'' for details.
* ANSI * CSV * AnyData
Defaults to ``CSV''. Because an SQL::Parser is instantiated only once and SQL::Parser doesn't allow to modify the dialect once instantiated, it's strongly recommended to set this flag before any statement is executed (best place is connect attribute hash).
This method usually requires extending in a derived implementation. See DBD::CSV or DBD::DBM for some example.
  package DBI::DBD::SqlEngine::TableSource;
  sub data_sources ($;$)
  {
    my ( $class, $drh, $attrs ) = @_;
    ...
  }
  sub avail_tables
  {
    my ( $class, $drh ) = @_;
    ...
  }
The "data_sources" method is called when the user invokes any of the following:
@ary = DBI->data_sources($driver); @ary = DBI->data_sources($driver, \%attr); @ary = $dbh->data_sources(); @ary = $dbh->data_sources(\%attr);
The "avail_tables" method is called when the user invokes any of the following:
@names = $dbh->tables( $catalog, $schema, $table, $type ); $sth = $dbh->table_info( $catalog, $schema, $table, $type ); $sth = $dbh->table_info( $catalog, $schema, $table, $type, \%attr ); $dbh->func( "list_tables" );
Every time where an "\%attr" argument can be specified, this "\%attr" object's "sql_table_source" attribute is preferred over the $dbh attribute or the driver default.
Derived classes shall be restricted to similar functionality, too (e.g. opening streams from an archive, transparently compress/uncompress log files before parsing them,
  package DBI::DBD::SqlEngine::DataSource;
  sub complete_table_name ($$;$)
  {
    my ( $self, $meta, $table, $respect_case ) = @_;
    ...
  }
The method "complete_table_name" is called when first setting up the meta information for a table:
"SELECT user.id, user.name, user.shell FROM user WHERE ..."
results in opening the table "user". First step of the table open process is completing the name. Let's imagine you're having a DBD::CSV handle with following settings:
  $dbh->{sql_identifier_case} = SQL_IC_LOWER;
  $dbh->{f_ext} = '.lst';
  $dbh->{f_dir} = '/data/web/adrmgr';
Those settings will result in looking for files matching "[Uu][Ss][Ee][Rr](\.lst)?$" in "/data/web/adrmgr/". The scanning of the directory "/data/web/adrmgr/" and the pattern match check will be done in "DBD::File::DataSource::File" by the "complete_table_name" method.
If you intend to provide other sources of data streams than files, in addition to provide an appropriate "complete_table_name" method, a method to open the resource is required:
  package DBI::DBD::SqlEngine::DataSource;
  sub open_data ($)
  {
    my ( $self, $meta, $attrs, $flags ) = @_;
    ...
  }
After the method "open_data" has been run successfully, the table's meta information are in a state which allows the table's data accessor methods will be able to fetch/store row information. Implementation details heavily depends on the table implementation, whereby the most famous is surely DBD::File::Table.
You should consult the documentation of "SQL::Eval::Table" (see SQL::Eval) to get more information about the abstract methods of the table's base class you have to override and a description of the table meta information expected by the SQL engines.
It copies the following attributes from the database into the table meta data "$dbh->{ReadOnly}" into "$meta->{readonly}", "sql_identifier_case" and "sql_data_source" and makes them sticky to the table.
This method should be called before you attempt to map between file name and table name to ensure the correct directory, extension etc. are used.
If the modified attribute requires to reset a calculated attribute, the calculated attribute is reset (deleted from meta data structure) and the initialized flag is removed, too. The decision is made based on %register_reset_on_modify.
If your DBD has calculated values in the meta data area, then call "register_reset_on_modify":
    my %reset_on_modify = ( "xxx_foo" => "xxx_bar" );
    __PACKAGE__->register_reset_on_modify( \%reset_on_modify );
    # from DBD::DBM
    my %compat_map = ( "dbm_ext" => "f_ext" );
    __PACKAGE__->register_compat_map( \%compat_map );
After this is done, a derived class might add more steps in an overridden "open_file" method.
1. get the table meta data 2. open the data file 3. bless the table data structure using inherited constructor new
It is not recommended to override the constructor of the table class. Find a reasonable place to add you extensions in one of the above four methods.
H.Merijn Brand < h.m.brand at xs4all.nl > and Jens Rehsack < rehsack at googlemail.com >
All rights reserved.
You may freely distribute and/or modify this module under the terms of either the GNU General Public License (GPL) or the Artistic License, as specified in the Perl README file.