NAME

Debuggit - A fairly simplistic debug statement handler

SYNOPSIS

use Debuggit(DEBUG => 1);

# say you have a global hashref for your site configuration
# (not to imply that global vars are good)
our $Config = get_global_config();

# now we can set some config things based on whether we're in debug mode or not
$Config->{'DB'} = DEBUG ? 'dev' : 'prod';

# maybe we need to pull our local Perl modules from our VC working copy
push @INC, $Config->{'vcdir/lib'} if DEBUG;

# basic debugging output
debuggit("only print this if debugging is on");
debuggit(3 => "only print this if debugging is level 3 or higher");

# show off our formatting
my $var1 = 6;
my $var2;
my $var3 = " leading and trailing spaces   ";
# assuming debugging is enabled ...
debuggit("var1 is", $var1);   # var1 is 6
debuggit("var2 is", $var2);   # var2 is <<undef>>
debuggit("var3 is", $var3);   # var3 is << leading and trailing spaces   >>
# note that spaces between args, as well as final newlines, are provided automatically

# use "functions" in the debugging args list
my $var4 = { complex => 'hash', with => 'lots', of => 'stuff' };
# this will call Data::Dumper::Dumper() for you
# (even if you've never loaded Data::Dumper)
debuggit("var4 is", DUMP => $var4);

# make your own function
Debuggit::add_func(CONFIG => 1, sub { my ($self, $var) = $_; return "$self var $var is $Config->{$var}" });
# and use it like so
debuggit(CONFIG => 'DB');     # CONFIG var DB is dev

DESCRIPTION

You want sophisticated, full-featured, on-demand debugging, and you don't want to take it out when you release the code because you might need it again later, but you also don't want it to take up any space or cause any slowdown of your production application. Sounds impossible? Nah. Just use Debuggit.

Quick Start

To start:

use strict;
use warnings;

use Debuggit;


my $var = 6;
debuggit(2 => "var is", $var);      # this does not print
debuggit(4 => "var is", $var);      # neither does this

Later ...

use strict;
use warnings;

use Debuggit(DEBUG => 2);


my $var = 6;
debuggit(2 => "var is", $var);      # now this prints
debuggit(4 => "var is", $var);      # but this still doesn't

That's it. Really. Everything else is just gravy.

A Note on Documentation

You know, documentation is a double-edged sword. Put too little, and no one will use your module because it's poorly documented. Put too much, and no one will use your module because it must be too complex ... I mean, just look at how much documentation it takes to describe it!

Well, I happen to actually like writing documentation, so there's a healthy chunk of it here. But let me assure you that this is just because I'm being thorough, not because this module is complex. It's actually almost ridiculously simple at its core. (In fact, about 80% of the module is this documentation.) Start with the "Quick Start" above. After you get bored with how simple Debuggit is to use like that, and want to explore more options, come back and read the rest. Because everything thing else is completely optional.

A Note on Version Numbers

I know it's very fashionable to mark any module you put on CPAN for the first time as version 0.000000001, and then take about eight years before daring to release a 1.0. However, this module (in earlier incarnations) has been used in production code for over 10 years now, so I don't really think it makes much sense to call it 0.01. I've chosen to start its CPAN life at 2.01, as this is its 3rd major rewrite. If it makes you feel better, just subtract 2 from the version number and you'll get a fairly accurate idea of its "true" CPAN version. Just be aware that it has had a pretty full life outside CPAN as well.

The Changes file details this life fairly accurately. The version numbers are assigned using 20/20 hindsight, and I've filled in a few gaps in historical notes, but all the dates and most of the commit messages are fully accurate, thanks to the wonders of version control.

EXPORTS

DEBUG

DEBUG is a constant (similar to use constant) which holds your current debugging level. Because it's implemented using constant folding, any conditional based on it will actually be removed during compile-time if the debugging level isn't high enough (or turned off completely). For instance, this code:

calculate_complex_stuff(1..10_000) if DEBUG >= 2;

would disappear entirely if DEBUG is set to 0, or to 1. Of course, being a constant has its own foibles: you can't interpolate it into double-quoted strings, and you can't put it in front of a fat comma. See constant for full details.

You "set" DEBUG when you use Debuggit:

use Debuggit(DEBUG => 2);

Once it's set, you can't change it. You also can't set it from outside the code (like, from an environment variable), but that's a feature I'd consider adding if people thought it was useful.

If you want your debugging turned off altogether, you can do this:

use Debuggit(DEBUG => 0);

Or you can do this:

use Debuggit;

But there's a subtle difference between those last two. In your top-level script, there's actually no difference at all. But in a module, not setting DEBUG doesn't mean DEBUG should be zero; rather it means it should "inherit" the value of DEBUG from the top-level script; or, to look at it another way, the value of DEBUG from the top-level script "falls through" to whatever modules are included by it.

The definition of "top-level script" (to Debuggit, anyway) is "first 'use Debuggit' statement that is executed by the Perl compiler." Which means you should always put use Debuggit before the use statements for other modules (or at least other modules of yours).

So typically what you would do is just put use Debuggit at the top of all your code until you're ready to debug. When you hit a problem, change to use Debuggit(DEBUG => 3) (or whatever level you feel is appropriate), but only in the top-level script. Then you get debugging info from all your modules with one simple change. Or, if you just know the problem is in module X, you can enable debugging in that module only. Convenient, right?

This may not work as well as you'd like if you can't figure out where your top-level script actually is (one great example of that is a mod_perl environment). But you can still enable the debugging in each module, so it isn't tragic. Just not as convenient.

FUNCTIONS

debuggit([level =>] arg[, ...])

debuggit() is the only exported function of Debuggit (except for the constant, which is technically a function, but you know what I mean). When DEBUG is set to 0, you are guaranteed that debuggit is an empty function. When DEBUG is non-zero, you are guaranteed that debuggit will do the equivalent of this:

sub debuggit
{
    my $level = $_[0] =~ /^\d+$/ ? shift : 1;
    if (DEBUG >= $level)
    {
        @_ = process_funcs(@_);
        my $msg = $formatter->(@_);
        $output->($msg);
    }
}

meaning that you can override both $formatter and $output, and you can add to or subtract from the functions handled by process_funcs().

[local] $Debuggit::formatter = coderef

The $formatter variable allows you to override Debuggit's internal format function (see above). For instance, something like this:

$Debuggit::formatter = sub { shift . "\n" . join('', map { "\t$_\n" } @_) };

my @list = qw< fred sue joe charlie >;
debuggit("Names:", @list);

outputs something like this:

Names:
    fred
    sue
    joe
    charlie

although that seems less useful than the default formatter, in general. Or maybe you don't like how Debuggit provides spaces and newlines for you:

$Debuggit::formatter = sub { join('', @_) };

But don't forget that now you've also lost the special handling of undef and strings with leading or trailing spaces.

Happily, since $formatter is a variable, you can use local to restore the previous value at the end of the enclosing block. Better yet, you can take advantage of the fact that the default formatter is stored as Debuggit::default_formatter() to do clever things like this:

# add timestamp to debugging (at least for this function/module/whatever)
local $Debuggit::formatter = sub
{
    return scalar(localtime) . ': ' . Debuggit::default_formatter(@_);
};

or even this:

# all debugging statements in the current context will show function name
local $Debuggit::formatter = sub
{
    # note that caller(0) would be this formatter sub, and
    # caller(1) would be debuggit(), so caller(2) is what we want
    # element 3 is the subroutine name (which includes the package name)
    return (caller(2))[3] . ': ' . Debuggit::default_formatter(@_);
};

Note that that last example only handles the simple cases--if your debuggit() calls get stuck inside eval's or coderef's or anything like that, this breaks down. But often the simple case is close enough.

Functions are handled before the formatter is called (see "DEBUGGING FUNCTIONS"), so any replacement formatter you create doesn't have to worry about those.

The default_formatter() looks mostly like this:

$default_formatter = sub
{
    return join(' ', map { defined $_ ? $_ : '<<undef>>' } @_) . "\n";
};

but see below for full details.

Before you try to get too terribly fancy with the formatter, you may wish to investigate add_func(). A good rule of thumb when trying to decide whether you want a new function or a new formatter is this: imagine that debuggit() takes a bunch of chunks of text and produces a line (which is pretty much what is does). If you want to fiddle with how one of those chunks looks, you want a function. If you need to change the look of the whole line, though, you need a new formatter.

If you're more interested in where the formatted output gets sent to, look at $output.

Debuggit::default_formatter(@_)

This function is what $formatter (see above) is set to unless (and until) you change it. It can also be called from your own formatter function (again, see above). Its purpose is to turn the arguments you pass debuggit() into a formatted line. This line is then sent to the function stored in $output.

The default formatter provides the following conveniences:

  • A single space is put between separate arguments.

  • An undefined argument is replaced with the string '<<undef>>' (distinguishes undef from an empty string, and avoids unsightly "uninitialized variable" warnings).

  • Any value which has leading or trailing spaces (that is, / +/, not /\s+/) has '<<' prepended to it and '>>' appended to it. This allows you to easily see (and hopefully accurately count) any such extra spaces.

  • A newline is appended to the formatted line.

All these features are demonstrated in the "SYNOPSIS".

[local] $Debuggit::output = coderef

The $output variable allows you to override Debuggit's internal output function (see debuggit()). For instance, something like this:

local $Debuggit::output = sub { print @_ };

allows you to print debugging messages to stdout rather than stderr (although I'm not sure why you'd want to). Like with $formatter, the use of local allows you to change the output function temporarily (i.e. until the end of the enclosing block).

Perhaps you want a log file:

my $log = '/tmp/debug.log';
$Debuggit::output = sub
{
    open(LOG, ">>$log") or return;
    print LOG @_;
    close(LOG);
};

(Notice how you have to append to the file, else multiple debuggit() calls will just overwrite each other.)

Note that you don't have to append a newline ($formatter does that). And finally note that not using local sometimes has its advantages: in this case, you might put such code in a common header file that all your Perl modules call, and the output will be adjusted for all parts of your program, regardless of scope. (See "POLICY MODULES" for the best way to accomplish that.)

You could also save to a string:

our $log_msg;
local $Debuggit::output = sub { $log_msg .= join('', @_) };

Again, we're appending. We join all the args together (although most formatters will return only one value, probably best not to assume), but use no separator.

The default output function is merely:

sub { print STDERR @_ };

Note that this is subtly different from:

sub { warn @_ };

in the presence of $SIG{__WARN__} handlers and/or mod_perl.

DEBUGGING FUNCTIONS

When writing debugging statements, you may find yourself doing the same operations over and over again. For instance, imagine that you have a set of objects that can belong to one of several subclasses. Internally, these are stored as hashes (as many objects in Perl are), and each hash contains a '_data' key whose value is a hashref, which itself contains all the interesting bits of data for the object. For debugging purposes, you often need to print out the exact type of a given object along with a particular data value. You may find yourself writing something like this over and over again:

debuggit("after bmoogling", ref($obj) . '->foo =', $obj->{'_data'}->{'foo'});

By the time you've typed that exact pattern 20 or 30 times, you may be getting tired of it. What if you could do something like this instead?

debuggit("after bmoogling", OBJDATA => ($obj, 'foo'));

(Note that the fat comma is not required (see "STYLE"), nor are the extra parends around $obj and 'foo'. But they make it more obvious what's going on here, in your author's humble opinion.)

If you could do that, that would be much nicer, yes? Well, you can:

Debuggit::add_func(OBJDATA => 2, sub
{
    my ($self, $obj, $data_name) = @_;

    return (ref($obj) . "->$data_name =", $obj->{'_data'}->{$data_name});
});

What that's saying is this: Any time debuggit() comes across an argument consisting of the string 'OBJDATA', it should remove it, plus the next 2 arguments, from its argument list; call the coderef given, passing it the arguments that were removed; and replace the args it removed with the return value of the coderef. This is called a "debugging function", or just "function" for short.

Note that this function returns a two-element list, rather than just concatenating it all into one big string. This is so that, if the data value happens to be undefined, it will be handled correctly by the formatter (see default_formatter(), above).

A function doesn't have to take any arguments, nor does it have to return any. For instance, you could replace this:

debuggit('=' x 40);
debuggit("new code section starts here");

with this:

debuggit(SEPARATOR => "new code section starts here");

by defining your function thus:

Debuggit::add_func(SEPARATOR => 0, sub
{
    $Debuggit::output->('=' x 40);
    return ();
});

Many other clever things can be done. Remember the difference between functions and formatters, which is covered above.

Default Functions

At present, there is only one debugging function that Debuggit provides for you by default:

debuggit("my hash:", DUMP => \%my_hash);

This is basically the same as:

use Data::Dumper;
debuggit("my hash:", Dumper(\%my_hash));

with one important exception: instead of loading Data::Dumper via a use statement, the DUMP debugging function loads it via a require statement, with the happy side-effect that, if debugging is not enabled, Data::Dumper is never loaded. Which undoubtedly you don't want it to be in your production code (as it can add anywhere from 300Kb to nearly 3Mb to your memory footprint).

Debuggit::add_func(FUNC_NAME => #, sub { ... });

This adds a new debugging function to the table that Debuggit keeps. The first argument is the name of the function; if you pass the name of an existing function, it is replaced silently. The second argument is the number of arguments that the function takes. The final argument is the coderef for the function itself.

Any time debuggit() finds an argument which exactly matches a function name, it removes that argument, and a number of following arguments matching the number passed to add_func(). If that number of args exceeds the number remaining in debuggit()'s argument list, it will happily fill any gaps with undef values without notifying you (or even noticing, for that matter). It then passes the total list of arguments removed (including the function name!) to the coderef passed to add_func(), calling it in list context. Finally, it takes the list returned from the coderef and inserts it back into debuggit()'s argument list at the point at which the arguments were removed. Basically, inside debuggit(), it does the equivalent of this:

$n = $func_name_being_checked_for;
$i = $point_at_which_func_name_found;
splice @_, $i, $funcs{$n}->{'num_args'} + 1, $funcs{$n}->{'coderef'}->(@_[$i..$i+$funcs{$n}->{'num_args'}]);

except hopefully more efficiently.

The name of the function is passed in so that you can do excessively clever things such as:

my $print_config = sub
{
    my ($self, $value) = @_;
    return ("Config ${self}->$value is", $CONFIG->{$self}->{$value});
};
Debuggit::add_func($_ => 1, $print_config) foreach qw< FOO BAR BAZ BMOOGLE >;

But do remember that excessive cleverness often leads to nightmarish maintenance, so caveat codor.

Since debuggit() is just doing a simple string comparison on its arguments to find functions, this means that you can't actually print out that string unless you embed it within another argument. So, assuming the default functions are still in place:

use Debuggit (DEBUG => 2);
my $test = {};
debuggit(2 => "test is", DUMP => $test);        # calls function, as expected
debuggit(2 => "i like to", 'DUMP', "stuff");    # calls function (possibly not expected)
debuggit(2 => "i like to", 'DUMP ', "stuff");   # doesn't call function, but prints "<<DUMP >>"
debuggit(2 => "i like to DUMP stuff");          # no issues here
my $value = 'DUMP';
debuggit(2 => "value is", $value, "in foo()");  # calls function(!!!)

That last one is particularly worrisome, but there's not much to be done about it, except to try to choose names for functions that you feel confident aren't going to show up as arguments to debuggit(), or else don't use debugging functions at all. Personally I find that as long as I use all caps for function names, and implement only the most necessary functions, it really isn't a problem.

Debuggit::remove_func('FUNC_NAME');

This just removes the given debugging function. Default functions are not special in any way, so those can be removed just as others can:

Debuggit::remove_func('DUMP');

POLICY MODULES

So, let's say you've started using some of Debuggit's more advanced features, such as setting formatters, or adding debugging functions, except that now you're putting the same lines of code at the top of every one of your Perl modules:

use Debuggit;
$Debuggit::formatter = sub { return scalar(localtime) . ': ' . Debuggit::default_formatter(@_) };
$Debuggit::output = sub { warn @_ };        # because I use $SIG{__WARN__}
Debuggit::add_func(CONFIG => 1, sub { my ($self, $var) = $_; return "$self var $var is $Config->{$var}" });

Whew! A bit verbose, eh? Would be nice if we could centralize that somehow.

Okay, try this:

package MyDebuggit;

use Debuggit ();            # don't let Debuggit import here, or you'll get redeclaration errors

$Debuggit::formatter = sub { return scalar(localtime) . ': ' . Debuggit::default_formatter(@_) };
$Debuggit::output = sub { warn @_ };        # because I use $SIG{__WARN__}
Debuggit::add_func(CONFIG => 1, sub { my ($self, $var) = $_; return "$self var $var is $Config->{$var}" });

sub import
{
    my $class = shift;
    Debuggit->import(PolicyModule => 1, @_);
}

The 'PolicyModule' argument to Debuggit::import() just tells it to install DEBUG and debuggit() one level higher than usual, so that your caller (not you) gets all that debuggity goodness. Now you can just:

use MyDebuggit;

or, similarly:

use MyDebuggit(DEBUG => 2);

and you're all set.

Note how the import passes @_ on to Debuggit::import. This is what makes that second use MyDebuggit example work. You don't have to do that, of course. For instance, if you're writing a module for all your test scripts to include, you might wish to force DEBUG to always be 1. That's easy too:

use Debuggit ();
$Debuggit::output = sub { diag @_ };        # nicer with Test::More et al
sub import
{
    Debuggit->import(PolicyModule => 1, DEBUG => 1);
}

STYLE

This is a pretty simple module, but there are still a couple of different ways to do things. Here are my personal thoughts as to the pluses and minuses of the following alternative styles. You, of course, may feel free to disagree: that's what keeps the world a wonderful place.

First, there's the difference between these two:

debuggit("here I am!") if DEBUG >= 2;
debuggit(2 => "here I am!");

Personally I prefer #2, but please see important information below under "PERFORMANCE". Functionally, they are the same ... when debugging is on. However, here's an interesting thing that tripped me up recently:

debuggit(4 => "row is", join(':', @$row));

As you might guess from the names, this was in a tight loop that processed each row coming back from a database. What I hadn't considered was that, even when debugging was totally off, it was still doing that join() call for every row of data, then passing the results to an empty function. In this case, the equivalent:

debuggit("row is", join(':', @$row)) if DEBUG >= 4;

really was significantly better (again, see "PERFORMANCE" for why).

Assuming you went with #2 above, you then have to decide between these two:

debuggit(2 => "here I am!");
debuggit(2, "here I am!");

I strongly recommend the first one. To me, #2 just looks like it will print "2 here I am!", which it won't. #1 is using the fat comma to offset the debugging level from the debugging arguments, and that seems to me to be a Good Thing(tm).

How about a similar choice for functions?

debuggit("here's my big structure", DUMP => $struct);
debuggit("here's my big structure", 'DUMP', $struct);

My objections to #2 are the same: it looks like "DUMP" is part of the debugging output, and it isn't. For me, the fat comma in a debuggit arg list is basically an indication that whatever precedes it is not something to be printed, but rather some message to debuggit itself to do something special.

On the other hand, don't fall into the trap of thinking that every time you use a fat comma debuggit() is going to know that you don't want to print the thing that precedes it. For instance, this:

debuggit("this is not a func", hey => "even though it looks like one");
# unless you defined a func named 'hey', of course
# but don't do that; you should use all caps for func names

Remember, the fat comma is still just a comma; debuggit() has no way to tell from its argument list whether you used a fat comma or not. Use => as a sign to your readers that you're using a debugging level or a debugging function, not as a sign to debuggit() itself.

The last thing you have to decide is how to define your "levels" of debugging. You don't have to, of course. You can just have one level, effectively, and have your debugging be either on or off. But you will probably find that it's convenient to gradually crank up the debugging level when you're trying to find that elusive problem. The lower level that you can set it to, the less debugging crap you have to wade through to find what you're looking for.

So it makes sense to have various levels of debugging, and it makes sense to have them make sense. Decide on what's best for your project (which may just be what's best for you, or might involve coming to a concensus with your coworkers) and publish that in a comment somewhere so everyone has the same expectations. And then be consistent.

How many levels should you use? Well, the quite excellent Log::Log4perl has 6, and they're named instead of numbered, so that you know what to use each level for. It also contains this very curious statement:

Neither does anyone need more logging levels than these predefined ones.
If you think you do, I would suggest you look into steering your logging
behaviour via the category mechanism.

No offense to Log4perl's author, but I always found this statement to be a bit ... well, snooty, to put it mildly. My personal view is, who am I to say how many levels you need? or what you want to use them for? Consequently, I have given you the range of positive integers to play with, and you can assign whatever meanings you like to them. But with great power comes great responsibility, and if you don't define what your levels are somewhere in your code, those who come after you will inevitably curse your name.

One last caution: You may want to define constants for your debugging levels, like so:

use constant QUIET => 1;
use constant SOFTER => 2;
use constant LOUDER => 3;
use constant LITTLE_BIT_LOUDER_NOW => 4;
# and so forth

And then you may think you're going to use them like so:

debuggit(LOUDER => "this is not going to print what you think");

(Unless you think it's going to print "LOUDER this is not going to print what you think", in which case you'd be absolutely right.) Remember that the fat comma autoquotes whatever comes before it, which deconstantifies your identifier there. You'll have to settle on one of these:

debuggit(LOUDER, "this works fine");
debuggit(LOUDER() => "as does this");

Or, alternatively, don't use constant and use something like Const::Fast instead:

const our $QUIET => 1;
const our $SOFTER => 2;
const our $LOUDER => 3;
const our $LITTLE_BIT_LOUDER_NOW => 4;
# and so forth

debuggit($LOUDER => "this one works fine too");

Personally your humble author, while preferring to use constants most of the time, doesn't actually use them for debugging levels. Possibly because the levels are already abstract representations as opposed to actual numbers.

PERFORMANCE

So is calling debuggit completely free? Well, yes and no.

If you use this style:

debuggit("here I am!") if DEBUG >= 2;

then, assuming DEBUG is set to 0 (or 1, even), it is indeed 100% free. In fact, the test suite actually uses B::Deparse to insure that the above statement produces no actual code when DEBUG == 0, and if you happen to have GTop installed (which I believe would mean that you would have to happen to be running under Linux), the test suite will also verify that use Debuggit does not add anything to your program's memory footprint.

This style, however:

debuggit(2 => "here I am!");

is slightly more problematic. Unfortunately, without using a source filter (which is a possibility for a future version, although it would be strictly optional), there just isn't any way that I can see to eliminate that call. (Unless maybe it could be done with something like Devel::Declare or optimizer, but I fear that may be beyond my meager Perl hacking ability ... patches welcome!)

So if you prefer that second style (as does your humble author), then what you end up with is a guarantee that your debuggit calls will resolve to calls to empty functions, which take a very small (but positive) amount of time. Probably you will never notice them, as whatever actual work you are doing will certainly overwhelm any time spent on calling empty functions, but I definitely can't state with confidence that it will never have any impact on your application. And don't forget that Perl still has to process your arguments in order to call the empty function: if one of your args to debuggit is a function call, it gets called even when debugging is off. So, if any of that worries you, don't do that. Use the first style and then you're covered.

So the short answer is, the second style is more compact and potentially more legible. But the first style is safer in terms of minimizing performance impact. However, I do hope that one day I can update this module with further options which can make the second style just as efficient. Hopefully this gives you the information you need to choose what's right for you.

COMPARISON

How does Debuggit compare with similar modules?

Probably its most well-known competitor would be Log::Log4perl. However, Log4perl is really a full-featured logger, which handles errors, warnings, and much much more ... debugging is only a small part of what Log4perl does. If you need Log4perl, you may well want to stick with that and ignore Debuggit. Debuggit is mainly for when you need much less than what Log4perl provides. That having been said, one advantage of Debuggit over Log4perl is that Log4perl provides only 2 levels of debugging (debug and trace), while Debuggit provides as many as you like. Of course, some may consider that a disadvantage, but I mention it for completeness.

Log4perl is also designed to actually run in your code even in production mode, whereas Debuggit is designed to disappear after debugging is over. For that reason, you may actually find a use for both alongside each other. Go for it, you crazy kids.

More similar to Debuggit are debug (by the author of the quite excellent Moose), Debug, Debug::Message, and Debug::EchoMessage. All these have similar features to Debuggit, but none have as many. To be fair, some have features that Debuggit doesn't. I've put together a comparison matrix for you, but please remember that this is based on my reading of the documentation for these modules. I have neither used any of them nor looked at their source code extensively, so my comparison could be incomplete.

I've included a few other debugging modules that I ran across as well. Several of these latter modules are designed to be used as part of a larger distribution, but could be used separately, and offer similar functionality to Debuggit, so I threw them in there. What the heck.

d   == debug
D   == Debug
DM  == Debug::Message
DEM == Debug::EchoMessage
LCD == LEOCHARRE::DEBUG
PTD == PTools::Debug
KD  == Konstrukt::Debug
BD  == Blosxom::Debug
NXD == Net::XMPP::Debug

Feature                                    | Debuggit | d | D | DM | DEM | LCD | PTD | BD | KD | NXD |
-------------------------------------------|----------|---|---|----|-----|-----|-----|----|----|-----|
DEBUG constant                             |     X    | X |   |    |     |  X  |     |    |    |     |
output function                            |     X    | X | X | X  |  X  |  X  |  X  | X  | X  |  X  |
  override formatting                      |     X    | X |   |    |     |     |     |    |    |     |
  override where output goes               |     X    | X |   | X  |     |     |     |    |    |     |
  override them separately                 |     X    |   |   |    |     |     |     |    |    |     |
  handles undefined values                 |     X    |   |   |    |     |     |     |    |    |     |
  handles vars w/ leading/trailing spaces  |     X    |   |   |    |     |     |     |    |    |     |
  can print color messages                 |          |   |   | X  |     |  X  |     |    |    |     |
  can specify indent level                 |          |   |   | X  |  X  |     |  X  |    |    |     |
  custom formatting functions              |     X    |   |   |    |     |     |     |    |    |     |
multiple debugging levels                  |     X    |   |   | X  |  X  |  X  |  X  | X  | X  |  X  |
  levels are effectively unlimited         |     X    |   |   | X  |  X  |  X  |  X  | X  |    |  X  |
  can specify levels as arbitrary strings  |          |   |   |    |     |  X  |     |    |    |     |
has OO interface                           |          | X | X | X  |  X  |     |  X  |    | X  |  X  |
  OO interface is optional                 |          | X |   |    |     |     |     |    |    |     |
is self-contained (no dependencies)        |     X    | X | X |    |  X  |  X  |     | X  |    |     |
  doesn't come bundled with other modules  |     X    | X | X | X  |  X  |  X  |     | X  |    |     |
control from outside module to be debugged |     X    | X |   |    |     |  X  |     |    |    |     |
  fallthrough from top level script        |     X    |   |   |    |     |     |     |    |    |     |
  arbitrary control by module name         |          | X |   |    |     |     |     |    |    |     |
  arbitrary control by package variable    |          |   |   |    |     |  X  |     |    |    |     |
-------------------------------------------+----------+---+---+----+-----+-----+-----+----+----+-----+

There are, of course, additional considerations in terms of coding style, which may or may not be important to you. Also, at least one (Blosxom::Debug) uses source filtering, which you may or may not object to.

BUGS

None that I know of. However, lacking omniscience, I welcome bug reports.

SUPPORT

Debuggit is on GitHub at barefootcoder/debuggit. Feel free to fork and submit patches. Please note that I develop via TDD (Test-Driven Development), so a patch that includes a failing test is much more likely to get accepted (or least accepted more quickly).

If you just want to report a problem or suggest a feature, that's okay too. You can create an issue on GitHub, or a bug in CPAN's RT (at http://rt.cpan.org). Or just send an email to bug-Debuggit@rt.cpan.org.

AUTHOR

Buddy Burden
CPAN ID: BAREFOOT
Barefoot Software
barefootcoder@gmail.com

COPYRIGHT

This program is free software licensed under

The Artistic License

The full text of the license can be found in the LICENSE file included with this module.

This module is copyright (c) 2008-2011, Barefoot Software. It has many venerable ancestors (some more direct than others), including but not limited to:

  • Barefoot::debug, (c) 2000-2006 Barefoot Software, 2004-2006 ThinkGeek

  • Barefoot::base, (c) 2001-2006 Barefoot Software

  • Geek::Dev::Debug, (c) 2004 ThinkGeek

  • VCtools::Base, (c) 2004-2008 Barefoot Software, 2004 ThinkGeek

  • Barefoot, (c) 2006-2009 Barefoot Software

  • Company::Debug, (c) 2008 Rent.com

SEE ALSO

perl(1), Log::Log4perl, debug, Debug, Debug::Message, Debug::EchoMessage.