NAME

DBIx::DWIW - Robust and simple DBI wrapper to Do What I Want (DWIW)

SYNOPSIS

When used directly:

use DBIx::DWIW;

my $db = DBIx::DWIW->Connect(DB   => $database,
                             User => $user,
                             Pass => $password,
                             Host => $host);

my @records = $db->Array("select * from foo");

When sub-classed for full functionality:

use MyDBI;  # class inherits from DBIx::DWIW

my $db = MyDBI->Connect('somedb') or die;

my @records = $db->Hashes("SELECT * FROM foo ORDER BY bar");

DESCRIPTION

NOTE: This module is currently specific to MySQL, but needn't be. We just haven't had a need to talk to any other database server.

DBIx::DWIW was developed (over the course of roughly 1.5 years) in Yahoo! Finance (http://finance.yahoo.com/) to suit our needs. Parts of the API may not make sense and the documentation may be lacking in some areas. We've been using it for so long (in one form or another) that these may not be readily obvious to us, so feel free to point that out. There's a reason the version number is currently < 1.0.

This module was recently extracted from Yahoo-specific code, so things may be a little strange yet while we smooth out any bumps and blemishes left over form that.

DBIx::DWIW is intended to be sub-classed. Doing so will give you all the benefits it can provide and the ability to easily customize some of its features. You can, of course, use it directly if it meets your needs as-is. But you'll be accepting its default behavior in some cases where it may not be wise to do so.

The DBIx::DWIW distribution comes with a sample sub-class in the file examples/MyDBI.pm which illustrates some of what you might want to do in your own class(es).

This module provides three main benefits:

Centralized Configuration

Rather than store the various connection parameters (username, password, hostname, port number, database name) in each and every script or application which needs them, you can easily put them in once place--or even generate them on the fly by writing a bit of custom cdoe.

If this is all you need, consider looking at Brian Aker's fine DBIx::Password module on the CPAN. It may be sufficient.

API Simplicity

Taking a lesson from Python (gasp!), this module promotes one obvious way to do most things. If you want to run a query and get the reults back as a list of hashrefs, there's one way to do that. The API may sacrifice speed in some cases, but new users can easily lean the simple and descriptive method calls. (Nobody is forcing you to use it.)

Fault Tolerance

Databases sometimes go down. Networks flake out. Bad stuff happens. Rather than have your application die, DBIx::DWIW provides a way to handle outages. You can build custom wait/retry/fail logic which does anything you might want (such as ringing your pager or sending e-mail).

DBIx::DWIW CLASS METHODS

The following methods are available from DBIx::DWIW objects. Any function or method not documented should be considered private. If you call it, your code may break someday and it will be your fault.

The methods follow the Perl tradition of returning false values when an error cocurs (an usually setting $@ with a descriptive error message).

Connect()

The Connect() constructor creates and returns a database connection object through which all database actions are conducted. On error, it will call die(), so you may want to eval {...} the call. The NoAbort option (described below) controls that behavior.

Connect() accepts ``hash-style'' key/value pairs as arguments. The arguments which is recognizes are:

Host

The name of the host to connect to. Use undef to force a socket connection on the local machine.

User

The database to user to authenticate as.

Pass

The password to authenticate with.

DB

The name of the database to use.

Socket

NOT IMPLEMENTED.

The path to the Unix socket to use.

Port

The port number to connect to.

Unique

A boolean which controls connection reuse.

If false (the default), multiple Connects with the same connection parameters (User, Pass, DB, Host) will return the same open connection. If Unique is true, it will return a connection distinct from all other connections.

If you have a process with an active connection that fork(s), be aware that you can NOT share the connection between the parent and child. Well, you can if you're REALLY CAREFUL and know what you're doing. But don't do it.

Instead, acquire a new connection in the child. Be sure to set this flag when you do, or you'll end up with the same connection and spend a lot of time pulling your hair out over why the code does mysterous things.

Verbose

Turns verbose reporting on. See Verbose().

Quiet

Turns off warning messages. See Quiet().

NoRetry

If true, the Connect() will fail immediately if it can't connect to the database. Normally, it will retry based on calls to RetryWait(). NoRetry affects only Connect, and has no effect on the fault-tolerance of the package once connected.

NoAbort

If there is an error in the arguments, or in the end the database can't be connected to, Connect() normally prints an error message and dies. If NoAbort is true, it will put the error string into $@ and return false.

There are a minimum of four components to any database connection: DB, User, Pass, and Host. If any are not provided, there may be defaults that kick in. A local configuration package, such as the MyDBI example class that comes with DBIx::DWIW, may provide appropriate default connection values for several database. In such a case, a client my be able to simply use:

my $db = MyDBI->Connect(DB => 'Finances');

to connect to the Finances database.

as a convenience, you can just give the database name:

my $db = MyDBI->Connect('Finances');

See the local configuration package appropriate to your installation for more information about what is and isn't preconfigured.

Disconnect()

Closes the connection. Upon program exit, this is called automatically on all open connections. Returns true if the open connection was closed, false if there was no connection, or there was some other error (with the error being returned in $@).

Quote(@values)

Calls the DBI quote() function on each value, returning a list of properly quoted values. As per quote(), NULL will be returned for items that are not defined.

ExecuteReturnCode()

Returns the return code from the most recently Execute()d query. This is what Execute() returns, so there's little reason to call it direclty. But it didn't used to be that way, so old code may be relying on this.

Execute($sql)

Executes the given SQL, returning true if successful, false if not (with the error in $@).

Do() is a synonym for Execute()

Prepare($sql)

Prepares the given sql statement, but does not execute it (just like DBI). Instead, it returns a statement handle $sth that you can later execute by calling its Execute() method:

my $sth = $db->Prepare("INSERT INTO foo VALUES (?, ?)");

$sth->Execute($a, $b);

The statement handle returned is not a native DBI statement handle. It's a DBIx::DWIW::Statement handle.

RecentSth()

Returns the DBI statement handle ($sth) of the most-recently successfuly executed statement.

RecentPreparedSth()

Returns the DBI statement handle ($sth) of the most-recently prepared DBI statement handle (which may or may not have already been executed).

InsertedId()

Returns the mysql_insertid associated with the most recently executed statement. Returns nothing if there is none.

Synonyms: InsertID(), LastInsertID(), and LastInsertId()

RowsAffected()

Returns the number of rows affected for the most recently executed statement. This is valid only if it was for a non-SELECT. (For SELECTs, count the return values). As per the DBI, the -1 is returned if there was an error.

RecentSql()

Returns the sql of the most recently executed statement.

PreparedSql()

Returns the sql of the most recently prepared statement. (Useful for showing sql that doesn't parse.)

Hash($sql)

A generic query routine. Pass an SQL statement that returns a single record, and it will return a hashref with all the key/value pairs of the record.

The example at the bottom of page 50 of DuBois's MySQL book would return a value similar to:

my $hashref = {
   last_name  => 'McKinley',
   first_name => 'William',
};

On error, $@ has the error text, and false is returned. If the query doesn't return a record, false is returned, but $@ is also false.

Use this routine only if the query will return a single record. Use Hashes() for queries that might return multiple records.

Hashes($sql)

A generic query routine. Given an SQL statement, returns a list of hashrefs, one per returned record, containing the key/value pairs of each record.

The example in the middle of page 50 of DuBois's MySQL would return a value similar to:

my @hashrefs = (
 { last_name => 'Tyler',    first_name => 'John',    birth => '1790-03-29' },
 { last_name => 'Buchanan', first_name => 'James',   birth => '1791-04-23' },
 { last_name => 'Polk',     first_name => 'James K', birth => '1795-11-02' },
 { last_name => 'Fillmore', first_name => 'Millard', birth => '1800-01-07' },
 { last_name => 'Pierce',   first_name => 'Franklin',birth => '1804-11-23' },
);

On error, $@ has the error text, and false is returned. If the query doesn't return a record, false is returned, but $@ is also false.

Array($sql)

Similar to Hash(), but returns a list of values from the matched record. On error, the empty list is returned and the error can be found in $@. If the query matches no records, the an empty list is returned but $@ is false.

The example at the bottom of page 50 of DuBois's MySQL would return a value similar to:

my @array = ( 'McKinley', 'William' );

Use this routine only if the query will return a single record. Use Arrays() or FlatArray() for queries that might return multiple records.

Arrays($sql)

A generic query routine. Given an SQL statement, returns a list of hashrefs, one per returned record, containing the values of each record.

The example in the middle of page 50 of DuBois's MySQL would return a value similar to:

my @arrayrefs = (
 [ 'Tyler',     'John',     '1790-03-29' ],
 [ 'Buchanan',  'James',    '1791-04-23' ],
 [ 'Polk',      'James K',  '1795-11-02' ],
 [ 'Fillmore',  'Millard',  '1800-01-07' ],
 [ 'Pierce',    'Franklin', '1804-11-23' ],
);

On error, $@ has the error text, and false is returned. If the query doesn't return a record, false is returned, but $@ is also false.

FlatArray($sql)

A generic query routine. Pass an SQL string, and all matching fields of all matching records are returned in one big list.

If the query matches a single records, FlatArray() ends up being the same as Array(). But if there are multiple records matched, the return list will contain a set of fields from each record.

The example in the middle of page 50 of DuBois's MySQL would return a value similar to:

my @items = (
    'Tyler', 'John', '1790-03-29', 'Buchanan', 'James', '1791-04-23',
    'Polk', 'James K', '1795-11-02', 'Fillmore', 'Millard',
    '1800-01-07', 'Pierce', 'Franklin', '1804-11-23'
);

FlatArray() tends to be most useful when the query returns one column per record, as with

my @names = $db->FlatArray('select distinct name from mydb');

or two records with a key/value relationship:

my %IdToName = $db->FlatArray('select id, name from mydb');

But you never know.

Verbose([boolean])

Returns the value of the verbose flag associated with the connection. If a value is provided, it is taken as the new value to install. Verbose is OFF by default. If you pass a true value, you'll get some verbose output each time a query executes.

Returns the current value.

Quiet()

When errors occur, a message will be sent to STDOUT if Quiet is true (it is by default). Pass a false value to disble it.

Returns the current value.

Safe()

Enable or disable "safe" mode (on by default). In "safe" mode, you must prefix a native DBI method call with "dbi_" in order to call it. If safe mode is off, you can call native DBI mathods using their real names.

For example, in safe mode, you'd write something like this:

$db->dbi_commit;

but in unsafe mode you could use:

$db->commit;

The rationale behind having a safe mode is that you probably don't want to mix DBIx::DWIW and DBI method calls on an object unless you know what you're doing. You need to opt-in.

Safe() returns the current value.

dbh()

Returns the real DBI database handle for the connection.

RetryWait($error)

This method is called each time there is a error (usually caused by a network outage or a server going down) which a sub-class may want to examine and decide how to continue.

If RetryWait() returns 1, the operation which was being attempted when the failure occured will be retried. If it returns 0, the action will fail.

The default implementation causes your application to emit a message to STDOUT (via a warn() call) and then sleep for 30 seconds before retrying. You probably wnat to override this so that it will eventually give up. Otherwise your application may hang forever. It does maintain a count of how many times the retry has been attempted in $self-{RetryCount}>.

Local Configuration

There are two ways to to configure DBIx::DWIW for your local databases. The simplest (but least flexible) way is to create a package like:

package MyDBI;
@ISA = 'DBIx::DWIWl';
use strict;

sub DefaultDB   { "MyDatabase"         }
sub DefaultUser { "defaultuser"        }
sub DefaultPass { "paSSw0rd"           }
sub DefaultHost { "mysql.somehost.com" }
sub DefaultPort { 3306                 }

The four routines override those in DBIx::DWIW, and explicitly provide exactly what's needed to contact the given database.

The user can then use

use MyDBI
my $db = MyDBI->Connect();

and not have to worry about the details.

A more flexible approach appropriate for multiple-database or multiple-user installations is to create a more complex package, such as the MyDBI.pm which was included in the examples sub-directory of the DBIx::DWIW distribution.

In that setup, you have quit a bit of control over what connection parameters are used. And, since it's Just Perl Code, you can do anything you need in there.

The following methods are provided to support this in sub-classes:

LocalConfig($name)

Passed a configuration name, LocalConfig() should return a list of conncetion parameters suitable for passing to Connect().

By default, LocalConfig() simply returns undef.

DefaultDB($config_name)

Returns the default database name for the given configuration. Calls LocalConfig() to get it.

DefaultUser($config_name)

Returns the default username for the given configuration. Calls LocalConfig() to get it.

DefaultPass($config_name)

Returns the default password for the given configuration. Calls LocalConfig() to get it.

DefaultHost($config_name)

Returns the default hostname for the given configuration. Calls LocalConfig() to get it.

DefaultPort($config_name)

Returns the default Port number for the given configuration. Calls LocalConfig() to get it.

The DBIx::DWIW::Statement CLASS

Calling Prepre() on a database handle returns a DBIx::DWIW::Statement object which acts like a limited DBI statement handle.

Methods

The following methods can be called on a statement object.

    Execute([@values])

    Executes the statement. If values are provided, they'll be substitued for the appropriate placeholders in the SQL.

AUTHORS

DBIx::DWIW evolved out of some Perl modules that we developed and used in Yahoo! Finance (http://finance.yahoo.com). The folowing people contributed to its development:

Jeffrey Friedl (jfriedl@yahoo.com)
Ray Goldberger (rayg@bitbaron.com)
Jeremy Zawodny (Jeremy@Zawodny.com)

Please direct comments, questions, etc to Jeremy for the time being. Thanks.

COPYRIGHT

DBIx::DWIW is Copyright (c) 2001, Yahoo! Inc. All rights reserved.

You may distribute under the same terms of the Artistic License, as specified in the Perl README file.

SEE ALSO

DBI, perl

Jeremy's presentation at the 2001 Open Source Database Summit, which introduced DBIx::DWIW is availble from:

http://jeremy.zawodny.com/mysql/

2 POD Errors

The following errors were encountered while parsing the POD:

Around line 1366:

'=item' outside of any '=over'

Around line 1510:

You can't have =items (as at line 1555) unless the first thing after the =over is an =item