|
NAMEDBIx::Log4perl - Perl extension for DBI to selectively log DBI methods, SQL, parameters, result-sets, transactions etc to a Log::Log4perl handle.SYNOPSISuse Log::Log4perl; use DBIx::Log4perl; Log::Log4perl->init("/etc/mylog.conf"); my $dbh = DBIx::Log4perl->connect('DBI:odbc:mydsn', $user, $pass); $dbh->DBI_METHOD(args); or use DBIx::Log4perl; my $dbh = DBIx::Log4perl->connect('dbi:ODBC:mydsn', $user, $pass, {dbix_l4p_init => "/etc/mylog.conf", dbix_l4p_class => "My::Package"}); $dbh->DBI_METHOD(args); DESCRIPTIONNOTE: The names of DBIx::Log4perl "ATTRIBUTES" have changed in version 0.18. They are now all lowercased as per the DBI specification."DBIx::Log4perl" is a wrapper over DBI which adds logging of your DBI activity via a Log::Log4perl handle. Log::Log4perl has many advantages for logging but the ones probably most attractive are: The ability to turn logging on or off or change the logging you see without changing your code (or even without restarting your programs if you use "init_and_watch"). Different log levels allowing you to separate warnings, errors and fatals to different files. The ability to capture all the information available via DBI when an error occurs. METHODSDBIx::Log4perl adds the following methods over DBI.dbix_l4p_getattr$h->dbxi_l4p_getattr('dbix_l4p_logmask'); Returns the value for a DBIx::Log4perl attribute (see "ATTRIBUTES"). dbix_l4p_setattr$h->dbix_l4p_setattr('dbix_l4p_logmask', 1); Set the value of the specified DBIx::Log4perl attribute (see "ATTRIBUTES"). dbix_l4p_logdie$h->dbix_l4p_logdie($message); Calls the internal _error_handler method with the message $message then dies with Carp::confess. The internal error handler is inserted into DBI's HandleError if "DBIX_L4P_LOG_ERRCAPTURE" is enabled. It attempts to log as much information about the SQL you were executing, parameters etc. As an example, you might be checking a $dbh->do which attempts to update a row really does update a row and want to die with all possible information about the problem if the update fails. Failing to update a row would not ordinarily cause DBI's error handler to be called. $affected = $dbh->do(q/update table set column = 1 where column = 2/); $dbh->dbix_logdie("Update failed") if ($affected != 1); GLOBAL VARIABLESDBIx::Log4perl::LogMaskThis variable controls the amount of logging logged to the Log::Log4perl handle. There are a number of constants defined which may be ORed together to obtain the logging level you require:CONSTANTSThe following constants may be imported via the ":masks" groupuse DBIx::Log4perl qw(:masks);
ATTRIBUTESWhen you call connect you may add "DBIx::Log4perl" attributes to those which you are passing to DBI. You may also get and set attributes after connect using "dbix_l4p_getattr()" and "dbix_l4p_setattr()". "DBIx::Log4perl" supports the following attributes:
Although these attributes are supported the recommended way to use DBIx::Log4perl it to use Log::Log4perl in your application and call the "Log::Log4Perl->init" to define your log4perl configuration file. DBIx::Log4perl will then call "Log::Log4perl->get_logger("DBIx::Log4perl")" (as was intended by the authors of Log::Log4perl) and all you need is a "log4perl.logger.DBIx.Log4perl" entry in your configuration file. Log::Log4perl CONFIGURATION FILEPlease see Log::Log4perl for full details of the configuration file and appenders. DBIx::Log4perl contains a sample configuration file you may use to get started. It looks like this:log4perl.logger = FATAL, LOGFILE log4perl.appender.LOGFILE=Log::Log4perl::Appender::File log4perl.appender.LOGFILE.filename=/tmp/log log4perl.appender.LOGFILE.mode=append log4perl.appender.LOGFILE.Threshold = ERROR log4perl.appender.LOGFILE.layout=PatternLayout log4perl.appender.LOGFILE.layout.ConversionPattern=[%r] %F %L %c - %m%n log4perl.logger.DBIx.Log4perl=DEBUG, A1 log4perl.appender.A1=Log::Log4perl::Appender::File log4perl.appender.A1.filename=/tmp/xlog log4perl.appender.A1.mode=append log4perl.appender.A1.layout=Log::Log4perl::Layout::SimpleLayout This is perhaps the most simple configuration. It says fatal errors go to /tmp/log and debug and above go to /tmp/xlog. It also uses the SimpleLayout which prefixes each line with the log level. You can use: log4perl.appender.A1.layout=Log::Log4perl::Layout::PatternLayout log4perl.appender.A1.layout.ConversionPattern=%d %p> %F{1}:%L %M - %m%n to make Log::Log4perl prefix the line with a timestamp, module name and filename. DBIx::Log4perl sets $Log::Log4perl::caller_depth in each method so when Log4perl outputs the module/file DBIx::Log4perl is ignored. This is extremely useful if you need to see where a DBI method is called from. FORMAT OF LOGExample outputFor a connect the log will contain something like:DEBUG - connect(0): DBI:mysql:mjetest, bet INFO - DBI: 1.50, DBIx::Log4perl: 0.01, Driver: mysql(3.0002_4) For $sth = $dbh->prepare('insert into mytest values (?,?)'); $sth->execute(1, 'one'); you will get: DEBUG - prepare(0.1): 'insert into mytest values (?,?)' DEBUG - $execute(0.1) (insert into mytest values (?,?)) = [1,'one']; In this latter case the SQL is repeated for convenience but this only occurs if "execute" is called with parameters. If "execute" is called without any arguments the SQL is not repeated in the "execute". Also note the output will include bind_param calls if you bound parameters seperately but how this is logged depends on "DBIX_L4P_LOG_DELAYBINDPARAM". The numbers in the () after a method name indicate which connection or statement handle the operation was performed on. The first connection your application makes will be connection 0 (see "connect(0)" above). Each statement method will show the connection number followed by a '.' and the statement number (e.g., "prepare(0.1)" above is the second statement handle on the first connection). NOTE: Some DBI methods are combinations of various methods e.g. selectrow_* methods. For some of these methods DBI does not actually call all the lower methods because the driver implements selectrow_* methods in C. For these cases, DBIx::Log4perl will only be able to log the selectrow_* method, the SQL, any parameters and any returned result-set and you will not necessarily see a prepare, execute and fetch in the log. e.g., $dbh->selectrow_array('select b from mytest where a = ?',undef,1); results in: DEBUG - $selectrow_array = ['select b from mytest where a = ?',undef,1]; with no evidence prepare/execute/fetch was called. If "DBIX_L4P_LOG_ERRCAPTURE" is set all possible information about an error is written to the log by the error handler. In addition a few method calls will attempt to write a separate log entry containing information which may not be available in the error handler e.g. $sth = $dbh->prepare(q/insert into mytest values (?,?)/); $sth->bind_param_array(1, [51,1,52,53]); $sth->bind_param_array(2, ['fiftyone', 'one', 'fiftythree', 'fiftytwo']); $inserted = $sth->execute_array( { ArrayTupleStatus => \@tuple_status } ); when the mytest table has a primary key on the first column and a row with 1 already exists will result in: ERROR - $Error = [1062,'Duplicate entry \'1\' for key 1','S1000']; ERROR - for 1,fiftytwo because the @tuple_status is not available in the error handler. In this output 1062 is the native database error number, the second argument is the error text, the third argument the state and the additional lines attempt to highlight the parameters which caused the problem. Example captured errorBy default, DBIx::Log4perl replaces any DBI error handler you have with its own error handler which first logs all possible information about the SQL that was executing when the error occurred, the parameters involved, the statement handle and a stack dump of where the error occurred. Once DBIx::Log4perl's error handler is executed it continues to call any error handler you have specifically set in you Perl DBI code.Assuming you'd just run the following script: use Log::Log4perl qw(get_logger :levels); Log::Log4perl->init_and_watch("example.conf"); my $dbh = DBIx::Log4perl->connect('dbi:Oracle:XE', 'user', 'password) or die "$DBD::errstr"; $dbh->do("insert into mytable values(?, ?)", undef, 1, 'string too long for column - will be truncated which is an error'); $dbh->disconnect; but the string argument to the insert is too big for the column then DBIx::Log4perl would provide error output similar to the following: FATAL - ============================================================ DBD::Oracle::db do failed: ORA-12899: value too large for column "BET"."MYTABLE"."B" (actual: 64, maximum: 10) (DBD ERROR: error possibly near <*> indicator at char 32 in 'insert into martin values(:p1, :<*>p2)') [for Statement "insert into martin values(?, ?)"] lasth Statement (DBIx::Log4perl::db=HASH(0x974cf64)): insert into martin values(?, ?) DB: XE, Username: user handle type: db SQL: Possible SQL: /insert into mytable values(?, ?)/ db Kids=0, ActiveKids=0 DB errstr: ORA-12899: value too large for column "BET"."MYTABLE"."B" (actual: 64, maximum: 10) (DBD ERROR: error possibly near <*> indicator at char 32 in 'insert into mytable values(:p1, :<*>p2)') ParamValues captured in HandleSetErr: 1,'string too long for column - will be truncated which is an error', 0 sub statements: DBI error trap at /usr/lib/perl5/site_perl/5.8.8/DBIx/Log4perl/db.pm line 32 DBIx::Log4perl::db::do('DBIx::Log4perl::db=HASH(0x97455d8)', 'insert into mytable values(?, ?)', 'undef', 1, 'string too long for column - will be truncated which is an error') called at errors.pl line 12 ============================================================ What this shows is: o the error reported by the DBD and the method called (do in this case). o the last handle used and the SQL for the last statement executed o the connection the error occurred in o the handle type the error occurred on, db or stmt (db in this case) o Other possible SQL that may be in error under this db connection e.g. if you were executing multiple statements on a single db connection o the Kids and ActiveKids value for this db - (see DBI docs) o the error message text in "DBI::errstr" o any sql parameters passed to DBI (see DBI for ParamValues) o a trace of where the problem occurred In this case the final problem was in db.pm but as this is DBIx::Log4perl's do method, the real issue was in the stack element below this which was errors.pl line 12. Use of Data::DumperDBIx::log4perl makes extensive use of Data::Dumper to output arguments passed to DBI methods. In some cases it combines the method called with the data it is logging e.g.DEBUG - $execute = [2,'two']; This means the execute method was called with placeholder arguments of 2 and 'two'. The '$' prefixing execute is because Data::Dumper was called like this: Data::Dumper->dump( [ \@execute_args ], [ 'execute'] ) so Data::Dumper believes it is dumping $execute. DBIx::Log4perl uses this method extensively to log the method and arguments - just ignore the leading '$' in the log. NOTESDuring the development of this module I came across of large number of issues in DBI and various DBDs. I've tried to list them here but in some cases I cannot give the version the problem was fixed in because it was not released at the time of writing.DBI and $h->{Username}If you get an error like:Can't get DBI::dr=HASH(0x83cbbc4)->{Username}: unrecognised attribute name in the error handler it is because it was missing from DBI's XS code. This is fixed in DBI 1.51. DBI and $h->{ParamArrays}This is the same issue as above for $h->{Username}.DBD::ODBC and ParamValuesIn DBD::ODBC 1.13 you cannot obtain ParamValues after an execute has failed. I believe this is because DBD::ODBC insists on describing a result-set before returning ParamValues and that is not necessary for ParamValues.Fixed in 1.14. DBD::mysql and ParamArraysDBD::mysql 3.002_4 does not support ParamArrays.I had to add the following to dbdimp.c to make it work: case 'P': if (strEQ(key, "PRECISION")) retsv= ST_FETCH_AV(AV_ATTRIB_PRECISION); /* + insert the following block */ if (strEQ(key, "ParamValues")) { HV *pvhv = newHV(); if (DBIc_NUM_PARAMS(imp_sth)) { unsigned int n; SV *sv; char key[100]; I32 keylen; for (n = 0; n < DBIc_NUM_PARAMS(imp_sth); n++) { keylen = sprintf(key, "%d", n); hv_store(pvhv, key, keylen, newSVsv(imp_sth->params[n].value), 0); } } retsv = newRV_noinc((SV*)pvhv); } /* - end of inserted block */ break; I believe this code is now added in DBD::mysql 3.0003_1. ContributingThere are six main ways you may help with the development and maintenance of this module:
CPAN Testers ReportingPlease, please, please (is that enough), consider installing CPAN::Reporter so that when you install perl modules a report of the installation success or failure can be sent to cpan testers. In this way module authors 1) get feedback on the fact that a module is being installed 2) get to know if there are any installation problems. Also other people like you may look at the test reports to see how successful they are before choosing the version of a module to install.CPAN::Reporter is easy to install and configure like this: perl -MCPAN -e shell cpan> install CPAN::Reporter cpan> reload cpan cpan> o conf init test_report Simply answer the questions to configure CPAN::Reporter. You can find the CPAN testers wiki at <http://wiki.cpantesters.org/> and the installation guide for CPAN::Reporter at <http://wiki.cpantesters.org/wiki/CPANInstall>. TO_DO
REQUIREMENTSYou will need at least Log::Log4perl 1.04 and DBI 1.50.DBI-1.51 contains the changes listed under "NOTES". Versions of Log::Log4perl before 1.04 work but unfortunately you will get code references in some of the log output where DBIx::Log4perl does: $log->logwarn(sub {Data::Dumper->Dump(something)}) The same applies to logdie. See the Log4perl mailing list for details. SEE ALSODBILog::Log4perl AUTHORM. J. Evans, <mjevans@cpan.org>COPYRIGHT AND LICENSECopyright (C) 2006 - 2012 by M. J. EvansThis library is free software; you can redistribute it and/or modify it under the same terms as Perl itself, either Perl version 5.8.7 or, at your option, any later version of Perl 5 you may have available.
Visit the GSP FreeBSD Man Page Interface. |